diff --git a/blendish.d b/blendish.d new file mode 100644 index 0000000..79d2f34 --- /dev/null +++ b/blendish.d @@ -0,0 +1,2254 @@ +/* +Blendish - Blender 2.5 UI based theming functions for NanoVega + +Copyright (c) 2014 Leonard Ritter + +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. +*/ +// Fork developement, feature integration and new bugs: +// Ketmar // Invisible Vector +/** + +Revision 6 (2014-09-21) + +Summary + +Blendish is a small collection of drawing functions for NanoVega, designed to +replicate the look of the Blender 2.5+ User Interface. You can use these +functions to theme your UI library. Several metric constants for faithful +reproduction are also included. + +Blendish supports the original Blender icon sheet; As the licensing of Blenders +icons is unclear, they are not included in Blendishes repository, but a SVG +template, "icons_template.svg" is provided, which you can use to build your own +icon sheet. + +To use icons, you must first load the icon sheet using one of the +`nvgCreateImage*()` functions and then pass the image handle to `bndSetIconImage()`; +otherwise, no icons will be drawn. See `bndSetIconImage()` for more information. + +Blendish will not render text until a suitable UI font has been passed to +`bndSetFont()` has been called. See `bndSetFont()` for more information. + + +Drawbacks + +There is no support for varying dpi resolutions yet. The library is hardcoded +to the equivalent of 72 dpi in the Blender system settings. + +Support for label truncation is missing. Text rendering breaks when widgets are +too short to contain their labels. +*/ +module arsd.blendish; +private: + +import arsd.nanovega; +version(aliced) { + import iv.meta; +} else { + private alias usize = size_t; + // i fear phobos! + private template Unqual(T) { + static if (is(T U == immutable U)) alias Unqual = U; + else static if (is(T U == shared inout const U)) alias Unqual = U; + else static if (is(T U == shared inout U)) alias Unqual = U; + else static if (is(T U == shared const U)) alias Unqual = U; + else static if (is(T U == shared U)) alias Unqual = U; + else static if (is(T U == inout const U)) alias Unqual = U; + else static if (is(T U == inout U)) alias Unqual = U; + else static if (is(T U == const U)) alias Unqual = U; + else alias Unqual = T; + } + private template isAnyCharType(T, bool unqual=false) { + static if (unqual) private alias UT = Unqual!T; else private alias UT = T; + enum isAnyCharType = is(UT == char) || is(UT == wchar) || is(UT == dchar); + } + private template isWideCharType(T, bool unqual=false) { + static if (unqual) private alias UT = Unqual!T; else private alias UT = T; + enum isWideCharType = is(UT == wchar) || is(UT == dchar); + } +} + +nothrow @trusted @nogc: + + +/** describes the theme used to draw a single widget or widget box; + * these values correspond to the same values that can be retrieved from + * the Theme panel in the Blender preferences */ +public struct BNDwidgetTheme { + /// theme name + string name; + /// color of widget box outline + NVGColor outlineColor; + /// color of widget item (meaning changes depending on class) + NVGColor itemColor; + /// fill color of widget box + NVGColor innerColor; + /// fill color of widget box when active + NVGColor innerSelectedColor; + /// color of text label + NVGColor textColor; + /// color of text label when active + NVGColor textSelectedColor; + /// delta modifier for upper part of gradient (-100 to 100) + int shadeTop; + /// delta modifier for lower part of gradient (-100 to 100) + int shadeDown; + /// color of hovered text (if transparent, use `textSelectedColor`) + NVGColor textHoverColor; + /// color of caret for text field (if transparent, use `textColor`) + NVGColor textCaretColor; +} + +/// describes the theme used to draw nodes +public struct BNDnodeTheme { + /// theme name + string name; + /// inner color of selected node (and downarrow) + NVGColor nodeSelectedColor; + /// outline of wires + NVGColor wiresColor; + /// color of text label when active + NVGColor textSelectedColor; + + /// inner color of active node (and dragged wire) + NVGColor activeNodeColor; + /// color of selected wire + NVGColor wireSelectColor; + /// color of background of node + NVGColor nodeBackdropColor; + + /// how much a noodle curves (0 to 10) + int noodleCurving; +} + +/// describes the theme used to draw widgets +public struct BNDtheme { + /// theme name + string name; + /// the background color of panels and windows + NVGColor backgroundColor; + /// theme for labels + BNDwidgetTheme regularTheme; + /// theme for tool buttons + BNDwidgetTheme toolTheme; + /// theme for radio buttons + BNDwidgetTheme radioTheme; + /// theme for text fields + BNDwidgetTheme textFieldTheme; + /// theme for option buttons (checkboxes) + BNDwidgetTheme optionTheme; + /// theme for choice buttons (comboboxes) + /// Blender calls them "menu buttons" + BNDwidgetTheme choiceTheme; + /// theme for number fields + BNDwidgetTheme numberFieldTheme; + /// theme for slider controls + BNDwidgetTheme sliderTheme; + /// theme for scrollbars + BNDwidgetTheme scrollBarTheme; + /// theme for tooltips + BNDwidgetTheme tooltipTheme; + /// theme for menu backgrounds + BNDwidgetTheme menuTheme; + /// theme for menu items + BNDwidgetTheme menuItemTheme; + /// theme for nodes + BNDnodeTheme nodeTheme; +} + +/// how text on a control is aligned +public alias BNDtextAlignment = int; +/// how text on a control is aligned (values) +public enum /*BNDtextAlignment*/ : int { + BND_LEFT = 0, /// left + BND_CENTER, /// center + BND_RIGHT, /// right +} + +/// states altering the styling of a widget +public alias BNDwidgetState = int; +/// states altering the styling of a widget (values) +public enum /*BNDwidgetState*/ : int { + /// not interacting + BND_DEFAULT = 0, + /// the mouse is hovering over the control + BND_HOVER, + /// the widget is activated (pressed) or in an active state (toggled) + BND_ACTIVE, +} + +/// flags indicating which corners are sharp (for grouping widgets) +public alias BNDcornerFlags = int; +public enum /*BNDcornerFlags*/ : int { + /// all corners are round + BND_CORNER_NONE = 0, + /// sharp top left corner + BND_CORNER_TOP_LEFT = 1, + /// sharp top right corner + BND_CORNER_TOP_RIGHT = 2, + /// sharp bottom right corner + BND_CORNER_DOWN_RIGHT = 4, + /// sharp bottom left corner + BND_CORNER_DOWN_LEFT = 8, + /// all corners are sharp; you can invert a set of flags using ^= BND_CORNER_ALL + BND_CORNER_ALL = 0xF, + /// top border is sharp + BND_CORNER_TOP = 3, + /// bottom border is sharp + BND_CORNER_DOWN = 0xC, + /// left border is sharp + BND_CORNER_LEFT = 9, + /// right border is sharp + BND_CORNER_RIGHT = 6, +} + +/** build an icon ID from two coordinates into the icon sheet, where + * (0, 0) designates the upper-leftmost icon, (1, 0) the one right next to it, + * and so on. */ +public enum BND_ICONID(int x, int y) = ((x)|((y)<<8)); + +/// alpha of disabled widget groups; can be used in conjunction with nvgGlobalAlpha() +public __gshared float BND_DISABLED_ALPHA = 0.5; + +public __gshared { + /// default widget height + int BND_WIDGET_HEIGHT = 21; + /// default toolbutton width (if icon only) + int BND_TOOL_WIDTH = 20; + + /// default radius of node ports + int BND_NODE_PORT_RADIUS = 5; + /// top margin of node content + int BND_NODE_MARGIN_TOP = 25; + /// bottom margin of node content + int BND_NODE_MARGIN_DOWN = 5; + /// left and right margin of node content + int BND_NODE_MARGIN_SIDE = 10; + /// height of node title bar + int BND_NODE_TITLE_HEIGHT = 20; + /// width of node title arrow click area + int BND_NODE_ARROW_AREA_WIDTH = 20; + + /// size of splitter corner click area + int BND_SPLITTER_AREA_SIZE = 12; + + /// width of vertical scrollbar + int BND_SCROLLBAR_WIDTH = 13; + /// height of horizontal scrollbar + int BND_SCROLLBAR_HEIGHT = 14; + + /// default vertical spacing + int BND_VSPACING = 1; + /// default vertical spacing between groups + int BND_VSPACING_GROUP = 8; + /// default horizontal spacing + int BND_HSPACING = 8; +} + +public alias BNDicon = int; +public enum /*BNDicon*/ { + BND_ICON_NONE = BND_ICONID!(0, 29), + BND_ICON_QUESTION = BND_ICONID!(1, 29), + BND_ICON_ERROR = BND_ICONID!(2, 29), + BND_ICON_CANCEL = BND_ICONID!(3, 29), + BND_ICON_TRIA_RIGHT = BND_ICONID!(4, 29), + BND_ICON_TRIA_DOWN = BND_ICONID!(5, 29), + BND_ICON_TRIA_LEFT = BND_ICONID!(6, 29), + BND_ICON_TRIA_UP = BND_ICONID!(7, 29), + BND_ICON_ARROW_LEFTRIGHT = BND_ICONID!(8, 29), + BND_ICON_PLUS = BND_ICONID!(9, 29), + BND_ICON_DISCLOSURE_TRI_DOWN = BND_ICONID!(10, 29), + BND_ICON_DISCLOSURE_TRI_RIGHT = BND_ICONID!(11, 29), + BND_ICON_RADIOBUT_OFF = BND_ICONID!(12, 29), + BND_ICON_RADIOBUT_ON = BND_ICONID!(13, 29), + BND_ICON_MENU_PANEL = BND_ICONID!(14, 29), + BND_ICON_BLENDER = BND_ICONID!(15, 29), + BND_ICON_GRIP = BND_ICONID!(16, 29), + BND_ICON_DOT = BND_ICONID!(17, 29), + BND_ICON_COLLAPSEMENU = BND_ICONID!(18, 29), + BND_ICON_X = BND_ICONID!(19, 29), + BND_ICON_GO_LEFT = BND_ICONID!(21, 29), + BND_ICON_PLUG = BND_ICONID!(22, 29), + BND_ICON_UI = BND_ICONID!(23, 29), + BND_ICON_NODE = BND_ICONID!(24, 29), + BND_ICON_NODE_SEL = BND_ICONID!(25, 29), +} +public enum /*BNDicon*/ { + BND_ICON_FULLSCREEN = BND_ICONID!(0, 28), + BND_ICON_SPLITSCREEN = BND_ICONID!(1, 28), + BND_ICON_RIGHTARROW_THIN = BND_ICONID!(2, 28), + BND_ICON_BORDERMOVE = BND_ICONID!(3, 28), + BND_ICON_VIEWZOOM = BND_ICONID!(4, 28), + BND_ICON_ZOOMIN = BND_ICONID!(5, 28), + BND_ICON_ZOOMOUT = BND_ICONID!(6, 28), + BND_ICON_PANEL_CLOSE = BND_ICONID!(7, 28), + BND_ICON_COPY_ID = BND_ICONID!(8, 28), + BND_ICON_EYEDROPPER = BND_ICONID!(9, 28), + BND_ICON_LINK_AREA = BND_ICONID!(10, 28), + BND_ICON_AUTO = BND_ICONID!(11, 28), + BND_ICON_CHECKBOX_DEHLT = BND_ICONID!(12, 28), + BND_ICON_CHECKBOX_HLT = BND_ICONID!(13, 28), + BND_ICON_UNLOCKED = BND_ICONID!(14, 28), + BND_ICON_LOCKED = BND_ICONID!(15, 28), + BND_ICON_UNPINNED = BND_ICONID!(16, 28), + BND_ICON_PINNED = BND_ICONID!(17, 28), + BND_ICON_SCREEN_BACK = BND_ICONID!(18, 28), + BND_ICON_RIGHTARROW = BND_ICONID!(19, 28), + BND_ICON_DOWNARROW_HLT = BND_ICONID!(20, 28), + BND_ICON_DOTSUP = BND_ICONID!(21, 28), + BND_ICON_DOTSDOWN = BND_ICONID!(22, 28), + BND_ICON_LINK = BND_ICONID!(23, 28), + BND_ICON_INLINK = BND_ICONID!(24, 28), + BND_ICON_PLUGIN = BND_ICONID!(25, 28), +} +public enum /*BNDicon*/ { + BND_ICON_HELP = BND_ICONID!(0, 27), + BND_ICON_GHOST_ENABLED = BND_ICONID!(1, 27), + BND_ICON_COLOR = BND_ICONID!(2, 27), + BND_ICON_LINKED = BND_ICONID!(3, 27), + BND_ICON_UNLINKED = BND_ICONID!(4, 27), + BND_ICON_HAND = BND_ICONID!(5, 27), + BND_ICON_ZOOM_ALL = BND_ICONID!(6, 27), + BND_ICON_ZOOM_SELECTED = BND_ICONID!(7, 27), + BND_ICON_ZOOM_PREVIOUS = BND_ICONID!(8, 27), + BND_ICON_ZOOM_IN = BND_ICONID!(9, 27), + BND_ICON_ZOOM_OUT = BND_ICONID!(10, 27), + BND_ICON_RENDER_REGION = BND_ICONID!(11, 27), + BND_ICON_BORDER_RECT = BND_ICONID!(12, 27), + BND_ICON_BORDER_LASSO = BND_ICONID!(13, 27), + BND_ICON_FREEZE = BND_ICONID!(14, 27), + BND_ICON_STYLUS_PRESSURE = BND_ICONID!(15, 27), + BND_ICON_GHOST_DISABLED = BND_ICONID!(16, 27), + BND_ICON_NEW = BND_ICONID!(17, 27), + BND_ICON_FILE_TICK = BND_ICONID!(18, 27), + BND_ICON_QUIT = BND_ICONID!(19, 27), + BND_ICON_URL = BND_ICONID!(20, 27), + BND_ICON_RECOVER_LAST = BND_ICONID!(21, 27), + BND_ICON_FULLSCREEN_ENTER = BND_ICONID!(23, 27), + BND_ICON_FULLSCREEN_EXIT = BND_ICONID!(24, 27), + BND_ICON_BLANK1 = BND_ICONID!(25, 27), +} +public enum /*BNDicon*/ { + BND_ICON_LAMP = BND_ICONID!(0, 26), + BND_ICON_MATERIAL = BND_ICONID!(1, 26), + BND_ICON_TEXTURE = BND_ICONID!(2, 26), + BND_ICON_ANIM = BND_ICONID!(3, 26), + BND_ICON_WORLD = BND_ICONID!(4, 26), + BND_ICON_SCENE = BND_ICONID!(5, 26), + BND_ICON_EDIT = BND_ICONID!(6, 26), + BND_ICON_GAME = BND_ICONID!(7, 26), + BND_ICON_RADIO = BND_ICONID!(8, 26), + BND_ICON_SCRIPT = BND_ICONID!(9, 26), + BND_ICON_PARTICLES = BND_ICONID!(10, 26), + BND_ICON_PHYSICS = BND_ICONID!(11, 26), + BND_ICON_SPEAKER = BND_ICONID!(12, 26), + BND_ICON_TEXTURE_SHADED = BND_ICONID!(13, 26), +} +public enum /*BNDicon*/ { + BND_ICON_VIEW3D = BND_ICONID!(0, 25), + BND_ICON_IPO = BND_ICONID!(1, 25), + BND_ICON_OOPS = BND_ICONID!(2, 25), + BND_ICON_BUTS = BND_ICONID!(3, 25), + BND_ICON_FILESEL = BND_ICONID!(4, 25), + BND_ICON_IMAGE_COL = BND_ICONID!(5, 25), + BND_ICON_INFO = BND_ICONID!(6, 25), + BND_ICON_SEQUENCE = BND_ICONID!(7, 25), + BND_ICON_TEXT = BND_ICONID!(8, 25), + BND_ICON_IMASEL = BND_ICONID!(9, 25), + BND_ICON_SOUND = BND_ICONID!(10, 25), + BND_ICON_ACTION = BND_ICONID!(11, 25), + BND_ICON_NLA = BND_ICONID!(12, 25), + BND_ICON_SCRIPTWIN = BND_ICONID!(13, 25), + BND_ICON_TIME = BND_ICONID!(14, 25), + BND_ICON_NODETREE = BND_ICONID!(15, 25), + BND_ICON_LOGIC = BND_ICONID!(16, 25), + BND_ICON_CONSOLE = BND_ICONID!(17, 25), + BND_ICON_PREFERENCES = BND_ICONID!(18, 25), + BND_ICON_CLIP = BND_ICONID!(19, 25), + BND_ICON_ASSET_MANAGER = BND_ICONID!(20, 25), +} +public enum /*BNDicon*/ { + BND_ICON_OBJECT_DATAMODE = BND_ICONID!(0, 24), + BND_ICON_EDITMODE_HLT = BND_ICONID!(1, 24), + BND_ICON_FACESEL_HLT = BND_ICONID!(2, 24), + BND_ICON_VPAINT_HLT = BND_ICONID!(3, 24), + BND_ICON_TPAINT_HLT = BND_ICONID!(4, 24), + BND_ICON_WPAINT_HLT = BND_ICONID!(5, 24), + BND_ICON_SCULPTMODE_HLT = BND_ICONID!(6, 24), + BND_ICON_POSE_HLT = BND_ICONID!(7, 24), + BND_ICON_PARTICLEMODE = BND_ICONID!(8, 24), + BND_ICON_LIGHTPAINT = BND_ICONID!(9, 24), +} +public enum /*BNDicon*/ { + BND_ICON_SCENE_DATA = BND_ICONID!(0, 23), + BND_ICON_RENDERLAYERS = BND_ICONID!(1, 23), + BND_ICON_WORLD_DATA = BND_ICONID!(2, 23), + BND_ICON_OBJECT_DATA = BND_ICONID!(3, 23), + BND_ICON_MESH_DATA = BND_ICONID!(4, 23), + BND_ICON_CURVE_DATA = BND_ICONID!(5, 23), + BND_ICON_META_DATA = BND_ICONID!(6, 23), + BND_ICON_LATTICE_DATA = BND_ICONID!(7, 23), + BND_ICON_LAMP_DATA = BND_ICONID!(8, 23), + BND_ICON_MATERIAL_DATA = BND_ICONID!(9, 23), + BND_ICON_TEXTURE_DATA = BND_ICONID!(10, 23), + BND_ICON_ANIM_DATA = BND_ICONID!(11, 23), + BND_ICON_CAMERA_DATA = BND_ICONID!(12, 23), + BND_ICON_PARTICLE_DATA = BND_ICONID!(13, 23), + BND_ICON_LIBRARY_DATA_DIRECT = BND_ICONID!(14, 23), + BND_ICON_GROUP = BND_ICONID!(15, 23), + BND_ICON_ARMATURE_DATA = BND_ICONID!(16, 23), + BND_ICON_POSE_DATA = BND_ICONID!(17, 23), + BND_ICON_BONE_DATA = BND_ICONID!(18, 23), + BND_ICON_CONSTRAINT = BND_ICONID!(19, 23), + BND_ICON_SHAPEKEY_DATA = BND_ICONID!(20, 23), + BND_ICON_CONSTRAINT_BONE = BND_ICONID!(21, 23), + BND_ICON_CAMERA_STEREO = BND_ICONID!(22, 23), + BND_ICON_PACKAGE = BND_ICONID!(23, 23), + BND_ICON_UGLYPACKAGE = BND_ICONID!(24, 23), +} +public enum /*BNDicon*/ { + BND_ICON_BRUSH_DATA = BND_ICONID!(0, 22), + BND_ICON_IMAGE_DATA = BND_ICONID!(1, 22), + BND_ICON_FILE = BND_ICONID!(2, 22), + BND_ICON_FCURVE = BND_ICONID!(3, 22), + BND_ICON_FONT_DATA = BND_ICONID!(4, 22), + BND_ICON_RENDER_RESULT = BND_ICONID!(5, 22), + BND_ICON_SURFACE_DATA = BND_ICONID!(6, 22), + BND_ICON_EMPTY_DATA = BND_ICONID!(7, 22), + BND_ICON_SETTINGS = BND_ICONID!(8, 22), + BND_ICON_RENDER_ANIMATION = BND_ICONID!(9, 22), + BND_ICON_RENDER_STILL = BND_ICONID!(10, 22), + BND_ICON_BOIDS = BND_ICONID!(12, 22), + BND_ICON_STRANDS = BND_ICONID!(13, 22), + BND_ICON_LIBRARY_DATA_INDIRECT = BND_ICONID!(14, 22), + BND_ICON_GREASEPENCIL = BND_ICONID!(15, 22), + BND_ICON_LINE_DATA = BND_ICONID!(16, 22), + BND_ICON_GROUP_BONE = BND_ICONID!(18, 22), + BND_ICON_GROUP_VERTEX = BND_ICONID!(19, 22), + BND_ICON_GROUP_VCOL = BND_ICONID!(20, 22), + BND_ICON_GROUP_UVS = BND_ICONID!(21, 22), + BND_ICON_RNA = BND_ICONID!(24, 22), + BND_ICON_RNA_ADD = BND_ICONID!(25, 22), +} +public enum /*BNDicon*/ { + BND_ICON_OUTLINER_OB_EMPTY = BND_ICONID!(0, 20), + BND_ICON_OUTLINER_OB_MESH = BND_ICONID!(1, 20), + BND_ICON_OUTLINER_OB_CURVE = BND_ICONID!(2, 20), + BND_ICON_OUTLINER_OB_LATTICE = BND_ICONID!(3, 20), + BND_ICON_OUTLINER_OB_META = BND_ICONID!(4, 20), + BND_ICON_OUTLINER_OB_LAMP = BND_ICONID!(5, 20), + BND_ICON_OUTLINER_OB_CAMERA = BND_ICONID!(6, 20), + BND_ICON_OUTLINER_OB_ARMATURE = BND_ICONID!(7, 20), + BND_ICON_OUTLINER_OB_FONT = BND_ICONID!(8, 20), + BND_ICON_OUTLINER_OB_SURFACE = BND_ICONID!(9, 20), + BND_ICON_OUTLINER_OB_SPEAKER = BND_ICONID!(10, 20), + BND_ICON_RESTRICT_VIEW_OFF = BND_ICONID!(19, 20), + BND_ICON_RESTRICT_VIEW_ON = BND_ICONID!(20, 20), + BND_ICON_RESTRICT_SELECT_OFF = BND_ICONID!(21, 20), + BND_ICON_RESTRICT_SELECT_ON = BND_ICONID!(22, 20), + BND_ICON_RESTRICT_RENDER_OFF = BND_ICONID!(23, 20), + BND_ICON_RESTRICT_RENDER_ON = BND_ICONID!(24, 20), +} +public enum /*BNDicon*/ { + BND_ICON_OUTLINER_DATA_EMPTY = BND_ICONID!(0, 19), + BND_ICON_OUTLINER_DATA_MESH = BND_ICONID!(1, 19), + BND_ICON_OUTLINER_DATA_CURVE = BND_ICONID!(2, 19), + BND_ICON_OUTLINER_DATA_LATTICE = BND_ICONID!(3, 19), + BND_ICON_OUTLINER_DATA_META = BND_ICONID!(4, 19), + BND_ICON_OUTLINER_DATA_LAMP = BND_ICONID!(5, 19), + BND_ICON_OUTLINER_DATA_CAMERA = BND_ICONID!(6, 19), + BND_ICON_OUTLINER_DATA_ARMATURE = BND_ICONID!(7, 19), + BND_ICON_OUTLINER_DATA_FONT = BND_ICONID!(8, 19), + BND_ICON_OUTLINER_DATA_SURFACE = BND_ICONID!(9, 19), + BND_ICON_OUTLINER_DATA_SPEAKER = BND_ICONID!(10, 19), + BND_ICON_OUTLINER_DATA_POSE = BND_ICONID!(11, 19), +} +public enum /*BNDicon*/ { + BND_ICON_MESH_PLANE = BND_ICONID!(0, 18), + BND_ICON_MESH_CUBE = BND_ICONID!(1, 18), + BND_ICON_MESH_CIRCLE = BND_ICONID!(2, 18), + BND_ICON_MESH_UVSPHERE = BND_ICONID!(3, 18), + BND_ICON_MESH_ICOSPHERE = BND_ICONID!(4, 18), + BND_ICON_MESH_GRID = BND_ICONID!(5, 18), + BND_ICON_MESH_MONKEY = BND_ICONID!(6, 18), + BND_ICON_MESH_CYLINDER = BND_ICONID!(7, 18), + BND_ICON_MESH_TORUS = BND_ICONID!(8, 18), + BND_ICON_MESH_CONE = BND_ICONID!(9, 18), + BND_ICON_LAMP_POINT = BND_ICONID!(12, 18), + BND_ICON_LAMP_SUN = BND_ICONID!(13, 18), + BND_ICON_LAMP_SPOT = BND_ICONID!(14, 18), + BND_ICON_LAMP_HEMI = BND_ICONID!(15, 18), + BND_ICON_LAMP_AREA = BND_ICONID!(16, 18), + BND_ICON_META_EMPTY = BND_ICONID!(19, 18), + BND_ICON_META_PLANE = BND_ICONID!(20, 18), + BND_ICON_META_CUBE = BND_ICONID!(21, 18), + BND_ICON_META_BALL = BND_ICONID!(22, 18), + BND_ICON_META_ELLIPSOID = BND_ICONID!(23, 18), + BND_ICON_META_CAPSULE = BND_ICONID!(24, 18), +} +public enum /*BNDicon*/ { + BND_ICON_SURFACE_NCURVE = BND_ICONID!(0, 17), + BND_ICON_SURFACE_NCIRCLE = BND_ICONID!(1, 17), + BND_ICON_SURFACE_NSURFACE = BND_ICONID!(2, 17), + BND_ICON_SURFACE_NCYLINDER = BND_ICONID!(3, 17), + BND_ICON_SURFACE_NSPHERE = BND_ICONID!(4, 17), + BND_ICON_SURFACE_NTORUS = BND_ICONID!(5, 17), + BND_ICON_CURVE_BEZCURVE = BND_ICONID!(9, 17), + BND_ICON_CURVE_BEZCIRCLE = BND_ICONID!(10, 17), + BND_ICON_CURVE_NCURVE = BND_ICONID!(11, 17), + BND_ICON_CURVE_NCIRCLE = BND_ICONID!(12, 17), + BND_ICON_CURVE_PATH = BND_ICONID!(13, 17), + BND_ICON_COLOR_RED = BND_ICONID!(19, 17), + BND_ICON_COLOR_GREEN = BND_ICONID!(20, 17), + BND_ICON_COLOR_BLUE = BND_ICONID!(21, 17), +} +public enum /*BNDicon*/ { + BND_ICON_FORCE_FORCE = BND_ICONID!(0, 16), + BND_ICON_FORCE_WIND = BND_ICONID!(1, 16), + BND_ICON_FORCE_VORTEX = BND_ICONID!(2, 16), + BND_ICON_FORCE_MAGNETIC = BND_ICONID!(3, 16), + BND_ICON_FORCE_HARMONIC = BND_ICONID!(4, 16), + BND_ICON_FORCE_CHARGE = BND_ICONID!(5, 16), + BND_ICON_FORCE_LENNARDJONES = BND_ICONID!(6, 16), + BND_ICON_FORCE_TEXTURE = BND_ICONID!(7, 16), + BND_ICON_FORCE_CURVE = BND_ICONID!(8, 16), + BND_ICON_FORCE_BOID = BND_ICONID!(9, 16), + BND_ICON_FORCE_TURBULENCE = BND_ICONID!(10, 16), + BND_ICON_FORCE_DRAG = BND_ICONID!(11, 16), + BND_ICON_FORCE_SMOKEFLOW = BND_ICONID!(12, 16), +} +public enum /*BNDicon*/ { + BND_ICON_MODIFIER = BND_ICONID!(0, 12), + BND_ICON_MOD_WAVE = BND_ICONID!(1, 12), + BND_ICON_MOD_BUILD = BND_ICONID!(2, 12), + BND_ICON_MOD_DECIM = BND_ICONID!(3, 12), + BND_ICON_MOD_MIRROR = BND_ICONID!(4, 12), + BND_ICON_MOD_SOFT = BND_ICONID!(5, 12), + BND_ICON_MOD_SUBSURF = BND_ICONID!(6, 12), + BND_ICON_HOOK = BND_ICONID!(7, 12), + BND_ICON_MOD_PHYSICS = BND_ICONID!(8, 12), + BND_ICON_MOD_PARTICLES = BND_ICONID!(9, 12), + BND_ICON_MOD_BOOLEAN = BND_ICONID!(10, 12), + BND_ICON_MOD_EDGESPLIT = BND_ICONID!(11, 12), + BND_ICON_MOD_ARRAY = BND_ICONID!(12, 12), + BND_ICON_MOD_UVPROJECT = BND_ICONID!(13, 12), + BND_ICON_MOD_DISPLACE = BND_ICONID!(14, 12), + BND_ICON_MOD_CURVE = BND_ICONID!(15, 12), + BND_ICON_MOD_LATTICE = BND_ICONID!(16, 12), + BND_ICON_CONSTRAINT_DATA = BND_ICONID!(17, 12), + BND_ICON_MOD_ARMATURE = BND_ICONID!(18, 12), + BND_ICON_MOD_SHRINKWRAP = BND_ICONID!(19, 12), + BND_ICON_MOD_CAST = BND_ICONID!(20, 12), + BND_ICON_MOD_MESHDEFORM = BND_ICONID!(21, 12), + BND_ICON_MOD_BEVEL = BND_ICONID!(22, 12), + BND_ICON_MOD_SMOOTH = BND_ICONID!(23, 12), + BND_ICON_MOD_SIMPLEDEFORM = BND_ICONID!(24, 12), + BND_ICON_MOD_MASK = BND_ICONID!(25, 12), +} +public enum /*BNDicon*/ { + BND_ICON_MOD_CLOTH = BND_ICONID!(0, 11), + BND_ICON_MOD_EXPLODE = BND_ICONID!(1, 11), + BND_ICON_MOD_FLUIDSIM = BND_ICONID!(2, 11), + BND_ICON_MOD_MULTIRES = BND_ICONID!(3, 11), + BND_ICON_MOD_SMOKE = BND_ICONID!(4, 11), + BND_ICON_MOD_SOLIDIFY = BND_ICONID!(5, 11), + BND_ICON_MOD_SCREW = BND_ICONID!(6, 11), + BND_ICON_MOD_VERTEX_WEIGHT = BND_ICONID!(7, 11), + BND_ICON_MOD_DYNAMICPAINT = BND_ICONID!(8, 11), + BND_ICON_MOD_REMESH = BND_ICONID!(9, 11), + BND_ICON_MOD_OCEAN = BND_ICONID!(10, 11), + BND_ICON_MOD_WARP = BND_ICONID!(11, 11), + BND_ICON_MOD_SKIN = BND_ICONID!(12, 11), + BND_ICON_MOD_TRIANGULATE = BND_ICONID!(13, 11), + BND_ICON_MOD_WIREFRAME = BND_ICONID!(14, 11), +} +public enum /*BNDicon*/ { + BND_ICON_REC = BND_ICONID!(0, 10), + BND_ICON_PLAY = BND_ICONID!(1, 10), + BND_ICON_FF = BND_ICONID!(2, 10), + BND_ICON_REW = BND_ICONID!(3, 10), + BND_ICON_PAUSE = BND_ICONID!(4, 10), + BND_ICON_PREV_KEYFRAME = BND_ICONID!(5, 10), + BND_ICON_NEXT_KEYFRAME = BND_ICONID!(6, 10), + BND_ICON_PLAY_AUDIO = BND_ICONID!(7, 10), + BND_ICON_PLAY_REVERSE = BND_ICONID!(8, 10), + BND_ICON_PREVIEW_RANGE = BND_ICONID!(9, 10), + BND_ICON_ACTION_TWEAK = BND_ICONID!(10, 10), + BND_ICON_PMARKER_ACT = BND_ICONID!(11, 10), + BND_ICON_PMARKER_SEL = BND_ICONID!(12, 10), + BND_ICON_PMARKER = BND_ICONID!(13, 10), + BND_ICON_MARKER_HLT = BND_ICONID!(14, 10), + BND_ICON_MARKER = BND_ICONID!(15, 10), + BND_ICON_SPACE2 = BND_ICONID!(16, 10), + BND_ICON_SPACE3 = BND_ICONID!(17, 10), + BND_ICON_KEYINGSET = BND_ICONID!(18, 10), + BND_ICON_KEY_DEHLT = BND_ICONID!(19, 10), + BND_ICON_KEY_HLT = BND_ICONID!(20, 10), + BND_ICON_MUTE_IPO_OFF = BND_ICONID!(21, 10), + BND_ICON_MUTE_IPO_ON = BND_ICONID!(22, 10), + BND_ICON_VISIBLE_IPO_OFF = BND_ICONID!(23, 10), + BND_ICON_VISIBLE_IPO_ON = BND_ICONID!(24, 10), + BND_ICON_DRIVER = BND_ICONID!(25, 10), +} +public enum /*BNDicon*/ { + BND_ICON_SOLO_OFF = BND_ICONID!(0, 9), + BND_ICON_SOLO_ON = BND_ICONID!(1, 9), + BND_ICON_FRAME_PREV = BND_ICONID!(2, 9), + BND_ICON_FRAME_NEXT = BND_ICONID!(3, 9), + BND_ICON_NLA_PUSHDOWN = BND_ICONID!(4, 9), + BND_ICON_IPO_CONSTANT = BND_ICONID!(5, 9), + BND_ICON_IPO_LINEAR = BND_ICONID!(6, 9), + BND_ICON_IPO_BEZIER = BND_ICONID!(7, 9), + BND_ICON_IPO_SINE = BND_ICONID!(8, 9), + BND_ICON_IPO_QUAD = BND_ICONID!(9, 9), + BND_ICON_IPO_CUBIC = BND_ICONID!(10, 9), + BND_ICON_IPO_QUART = BND_ICONID!(11, 9), + BND_ICON_IPO_QUINT = BND_ICONID!(12, 9), + BND_ICON_IPO_EXPO = BND_ICONID!(13, 9), + BND_ICON_IPO_CIRC = BND_ICONID!(14, 9), + BND_ICON_IPO_BOUNCE = BND_ICONID!(15, 9), + BND_ICON_IPO_ELASTIC = BND_ICONID!(16, 9), + BND_ICON_IPO_BACK = BND_ICONID!(17, 9), + BND_ICON_IPO_EASE_IN = BND_ICONID!(18, 9), + BND_ICON_IPO_EASE_OUT = BND_ICONID!(19, 9), + BND_ICON_IPO_EASE_IN_OUT = BND_ICONID!(20, 9), +} +public enum /*BNDicon*/ { + BND_ICON_VERTEXSEL = BND_ICONID!(0, 8), + BND_ICON_EDGESEL = BND_ICONID!(1, 8), + BND_ICON_FACESEL = BND_ICONID!(2, 8), + BND_ICON_LOOPSEL = BND_ICONID!(3, 8), + BND_ICON_ROTATE = BND_ICONID!(5, 8), + BND_ICON_CURSOR = BND_ICONID!(6, 8), + BND_ICON_ROTATECOLLECTION = BND_ICONID!(7, 8), + BND_ICON_ROTATECENTER = BND_ICONID!(8, 8), + BND_ICON_ROTACTIVE = BND_ICONID!(9, 8), + BND_ICON_ALIGN = BND_ICONID!(10, 8), + BND_ICON_SMOOTHCURVE = BND_ICONID!(12, 8), + BND_ICON_SPHERECURVE = BND_ICONID!(13, 8), + BND_ICON_ROOTCURVE = BND_ICONID!(14, 8), + BND_ICON_SHARPCURVE = BND_ICONID!(15, 8), + BND_ICON_LINCURVE = BND_ICONID!(16, 8), + BND_ICON_NOCURVE = BND_ICONID!(17, 8), + BND_ICON_RNDCURVE = BND_ICONID!(18, 8), + BND_ICON_PROP_OFF = BND_ICONID!(19, 8), + BND_ICON_PROP_ON = BND_ICONID!(20, 8), + BND_ICON_PROP_CON = BND_ICONID!(21, 8), + BND_ICON_SCULPT_DYNTOPO = BND_ICONID!(22, 8), + BND_ICON_PARTICLE_POINT = BND_ICONID!(23, 8), + BND_ICON_PARTICLE_TIP = BND_ICONID!(24, 8), + BND_ICON_PARTICLE_PATH = BND_ICONID!(25, 8), +} +public enum /*BNDicon*/ { + BND_ICON_MAN_TRANS = BND_ICONID!(0, 7), + BND_ICON_MAN_ROT = BND_ICONID!(1, 7), + BND_ICON_MAN_SCALE = BND_ICONID!(2, 7), + BND_ICON_MANIPUL = BND_ICONID!(3, 7), + BND_ICON_SNAP_OFF = BND_ICONID!(4, 7), + BND_ICON_SNAP_ON = BND_ICONID!(5, 7), + BND_ICON_SNAP_NORMAL = BND_ICONID!(6, 7), + BND_ICON_SNAP_INCREMENT = BND_ICONID!(7, 7), + BND_ICON_SNAP_VERTEX = BND_ICONID!(8, 7), + BND_ICON_SNAP_EDGE = BND_ICONID!(9, 7), + BND_ICON_SNAP_FACE = BND_ICONID!(10, 7), + BND_ICON_SNAP_VOLUME = BND_ICONID!(11, 7), + BND_ICON_STICKY_UVS_LOC = BND_ICONID!(13, 7), + BND_ICON_STICKY_UVS_DISABLE = BND_ICONID!(14, 7), + BND_ICON_STICKY_UVS_VERT = BND_ICONID!(15, 7), + BND_ICON_CLIPUV_DEHLT = BND_ICONID!(16, 7), + BND_ICON_CLIPUV_HLT = BND_ICONID!(17, 7), + BND_ICON_SNAP_PEEL_OBJECT = BND_ICONID!(18, 7), + BND_ICON_GRID = BND_ICONID!(19, 7), +} +public enum /*BNDicon*/ { + BND_ICON_PASTEDOWN = BND_ICONID!(0, 6), + BND_ICON_COPYDOWN = BND_ICONID!(1, 6), + BND_ICON_PASTEFLIPUP = BND_ICONID!(2, 6), + BND_ICON_PASTEFLIPDOWN = BND_ICONID!(3, 6), + BND_ICON_SNAP_SURFACE = BND_ICONID!(8, 6), + BND_ICON_AUTOMERGE_ON = BND_ICONID!(9, 6), + BND_ICON_AUTOMERGE_OFF = BND_ICONID!(10, 6), + BND_ICON_RETOPO = BND_ICONID!(11, 6), + BND_ICON_UV_VERTEXSEL = BND_ICONID!(12, 6), + BND_ICON_UV_EDGESEL = BND_ICONID!(13, 6), + BND_ICON_UV_FACESEL = BND_ICONID!(14, 6), + BND_ICON_UV_ISLANDSEL = BND_ICONID!(15, 6), + BND_ICON_UV_SYNC_SELECT = BND_ICONID!(16, 6), +} +public enum /*BNDicon*/ { + BND_ICON_BBOX = BND_ICONID!(0, 5), + BND_ICON_WIRE = BND_ICONID!(1, 5), + BND_ICON_SOLID = BND_ICONID!(2, 5), + BND_ICON_SMOOTH = BND_ICONID!(3, 5), + BND_ICON_POTATO = BND_ICONID!(4, 5), + BND_ICON_ORTHO = BND_ICONID!(6, 5), + BND_ICON_LOCKVIEW_OFF = BND_ICONID!(9, 5), + BND_ICON_LOCKVIEW_ON = BND_ICONID!(10, 5), + BND_ICON_AXIS_SIDE = BND_ICONID!(12, 5), + BND_ICON_AXIS_FRONT = BND_ICONID!(13, 5), + BND_ICON_AXIS_TOP = BND_ICONID!(14, 5), + BND_ICON_NDOF_DOM = BND_ICONID!(15, 5), + BND_ICON_NDOF_TURN = BND_ICONID!(16, 5), + BND_ICON_NDOF_FLY = BND_ICONID!(17, 5), + BND_ICON_NDOF_TRANS = BND_ICONID!(18, 5), + BND_ICON_LAYER_USED = BND_ICONID!(19, 5), + BND_ICON_LAYER_ACTIVE = BND_ICONID!(20, 5), +} +public enum /*BNDicon*/ { + BND_ICON_SORTALPHA = BND_ICONID!(0, 3), + BND_ICON_SORTBYEXT = BND_ICONID!(1, 3), + BND_ICON_SORTTIME = BND_ICONID!(2, 3), + BND_ICON_SORTSIZE = BND_ICONID!(3, 3), + BND_ICON_LONGDISPLAY = BND_ICONID!(4, 3), + BND_ICON_SHORTDISPLAY = BND_ICONID!(5, 3), + BND_ICON_GHOST = BND_ICONID!(6, 3), + BND_ICON_IMGDISPLAY = BND_ICONID!(7, 3), + BND_ICON_SAVE_AS = BND_ICONID!(8, 3), + BND_ICON_SAVE_COPY = BND_ICONID!(9, 3), + BND_ICON_BOOKMARKS = BND_ICONID!(10, 3), + BND_ICON_FONTPREVIEW = BND_ICONID!(11, 3), + BND_ICON_FILTER = BND_ICONID!(12, 3), + BND_ICON_NEWFOLDER = BND_ICONID!(13, 3), + BND_ICON_OPEN_RECENT = BND_ICONID!(14, 3), + BND_ICON_FILE_PARENT = BND_ICONID!(15, 3), + BND_ICON_FILE_REFRESH = BND_ICONID!(16, 3), + BND_ICON_FILE_FOLDER = BND_ICONID!(17, 3), + BND_ICON_FILE_BLANK = BND_ICONID!(18, 3), + BND_ICON_FILE_BLEND = BND_ICONID!(19, 3), + BND_ICON_FILE_IMAGE = BND_ICONID!(20, 3), + BND_ICON_FILE_MOVIE = BND_ICONID!(21, 3), + BND_ICON_FILE_SCRIPT = BND_ICONID!(22, 3), + BND_ICON_FILE_SOUND = BND_ICONID!(23, 3), + BND_ICON_FILE_FONT = BND_ICONID!(24, 3), + BND_ICON_FILE_TEXT = BND_ICONID!(25, 3), +} +public enum /*BNDicon*/ { + BND_ICON_RECOVER_AUTO = BND_ICONID!(0, 2), + BND_ICON_SAVE_PREFS = BND_ICONID!(1, 2), + BND_ICON_LINK_BLEND = BND_ICONID!(2, 2), + BND_ICON_APPEND_BLEND = BND_ICONID!(3, 2), + BND_ICON_IMPORT = BND_ICONID!(4, 2), + BND_ICON_EXPORT = BND_ICONID!(5, 2), + BND_ICON_EXTERNAL_DATA = BND_ICONID!(6, 2), + BND_ICON_LOAD_FACTORY = BND_ICONID!(7, 2), + BND_ICON_LOOP_BACK = BND_ICONID!(13, 2), + BND_ICON_LOOP_FORWARDS = BND_ICONID!(14, 2), + BND_ICON_BACK = BND_ICONID!(15, 2), + BND_ICON_FORWARD = BND_ICONID!(16, 2), + BND_ICON_FILE_BACKUP = BND_ICONID!(24, 2), + BND_ICON_DISK_DRIVE = BND_ICONID!(25, 2), +} +public enum /*BNDicon*/ { + BND_ICON_MATPLANE = BND_ICONID!(0, 1), + BND_ICON_MATSPHERE = BND_ICONID!(1, 1), + BND_ICON_MATCUBE = BND_ICONID!(2, 1), + BND_ICON_MONKEY = BND_ICONID!(3, 1), + BND_ICON_HAIR = BND_ICONID!(4, 1), + BND_ICON_ALIASED = BND_ICONID!(5, 1), + BND_ICON_ANTIALIASED = BND_ICONID!(6, 1), + BND_ICON_MAT_SPHERE_SKY = BND_ICONID!(7, 1), + BND_ICON_WORDWRAP_OFF = BND_ICONID!(12, 1), + BND_ICON_WORDWRAP_ON = BND_ICONID!(13, 1), + BND_ICON_SYNTAX_OFF = BND_ICONID!(14, 1), + BND_ICON_SYNTAX_ON = BND_ICONID!(15, 1), + BND_ICON_LINENUMBERS_OFF = BND_ICONID!(16, 1), + BND_ICON_LINENUMBERS_ON = BND_ICONID!(17, 1), + BND_ICON_SCRIPTPLUGINS = BND_ICONID!(18, 1), +} +public enum /*BNDicon*/ { + BND_ICON_SEQ_SEQUENCER = BND_ICONID!(0, 0), + BND_ICON_SEQ_PREVIEW = BND_ICONID!(1, 0), + BND_ICON_SEQ_LUMA_WAVEFORM = BND_ICONID!(2, 0), + BND_ICON_SEQ_CHROMA_SCOPE = BND_ICONID!(3, 0), + BND_ICON_SEQ_HISTOGRAM = BND_ICONID!(4, 0), + BND_ICON_SEQ_SPLITVIEW = BND_ICONID!(5, 0), + BND_ICON_IMAGE_RGB = BND_ICONID!(9, 0), + BND_ICON_IMAGE_RGB_ALPHA = BND_ICONID!(10, 0), + BND_ICON_IMAGE_ALPHA = BND_ICONID!(11, 0), + BND_ICON_IMAGE_ZDEPTH = BND_ICONID!(12, 0), + BND_ICON_IMAGEFILE = BND_ICONID!(13, 0), +} + + +//////////////////////////////////////////////////////////////////////////////// +public float bndMin(T) (in T a, in T b) if (__traits(isFloating, T)) { pragma(inline, true); import std.math : isNaN; return (isNaN(a) ? b : ( isNaN(b) ? a : (a < b ? a : b))); } +public float bndMax(T) (in T a, in T b) if (__traits(isFloating, T)) { pragma(inline, true); import std.math : isNaN; return (isNaN(a) ? b : ( isNaN(b) ? a : (a > b ? a : b))); } + + +//////////////////////////////////////////////////////////////////////////////// +/// default text size +public __gshared float BND_LABEL_FONT_SIZE = 13; + +/// default text padding in inner box +public __gshared int BND_PAD_LEFT = 8; +public __gshared int BND_PAD_RIGHT = 8; + +/// label: value separator string +public __gshared string BND_LABEL_SEPARATOR = ": "; + +/// alpha intensity of transparent items (0xa4) +public __gshared float BND_TRANSPARENT_ALPHA = 0.643; + +/// shade intensity of beveled panels +public __gshared int BND_BEVEL_SHADE = 30; +/// shade intensity of beveled insets +public __gshared int BND_INSET_BEVEL_SHADE = 30; +/// shade intensity of hovered inner boxes +public __gshared int BND_HOVER_SHADE = 15; +/// shade intensity of splitter bevels +public __gshared int BND_SPLITTER_SHADE = 100; + +/// width of icon sheet +public __gshared int BND_ICON_SHEET_WIDTH = 602; +/// height of icon sheet +public __gshared int BND_ICON_SHEET_HEIGHT = 640; +/// gridsize of icon sheet in both dimensions +public __gshared int BND_ICON_SHEET_GRID = 21; +/// offset of first icon tile relative to left border +public __gshared int BND_ICON_SHEET_OFFSET_X = 5; +/// offset of first icon tile relative to top border +public __gshared int BND_ICON_SHEET_OFFSET_Y = 10; +/// resolution of single icon +public __gshared int BND_ICON_SHEET_RES = 16; + +/// size of number field arrow +public __gshared float BND_NUMBER_ARROW_SIZE = 4; + +/// default text color +public enum BND_COLOR_TEXT = nvgRGBAf(0, 0, 0, 1); +/// default highlighted text color +public enum BND_COLOR_TEXT_SELECTED = nvgRGBAf(1, 1, 1, 1); +/// default color for active element +public enum BND_COLOR_ACTIVE = nvgRGBA(255, 127, 0, 255); + +/// radius of tool button +public __gshared float BND_TOOL_RADIUS = 4; + +/// radius of option button +public __gshared float BND_OPTION_RADIUS = 4; +/// width of option button checkbox +public __gshared float BND_OPTION_WIDTH = 14; +/// height of option button checkbox +public __gshared float BND_OPTION_HEIGHT = 15; + +/// radius of text field +public __gshared float BND_TEXT_RADIUS = 4; + +/// radius of number button +public __gshared float BND_NUMBER_RADIUS = 10; + +/// radius of menu popup +public __gshared float BND_MENU_RADIUS = 3; +/// feather of menu popup shadow +public __gshared float BND_SHADOW_FEATHER = 12; +/// alpha of menu popup shadow +public __gshared float BND_SHADOW_ALPHA = 0.5; + +/// radius of scrollbar +public __gshared float BND_SCROLLBAR_RADIUS = 7; +/// shade intensity of active scrollbar +public __gshared int BND_SCROLLBAR_ACTIVE_SHADE = 15; + +/// max glyphs for position testing +public enum BND_MAX_GLYPHS = 1024; + +/// max rows for position testing +public enum BND_MAX_ROWS = 32; + +/// text distance from bottom +public __gshared int BND_TEXT_PAD_DOWN = 7; + +/// stroke width of wire outline +public __gshared float BND_NODE_WIRE_OUTLINE_WIDTH = 4; +/// stroke width of wire +public __gshared float BND_NODE_WIRE_WIDTH = 2; +/// radius of node box +public __gshared float BND_NODE_RADIUS = 8; +/// feather of node title text +public __gshared float BND_NODE_TITLE_FEATHER = 1; +/// size of node title arrow +public __gshared float BND_NODE_ARROW_SIZE = 9; + + +//////////////////////////////////////////////////////////////////////////////// +public float bndClamp() (float v, float mn, float mx) { pragma(inline, true); return (v > mx ? mx : (v < mn ? mn : v)); } + + +//////////////////////////////////////////////////////////////////////////////// + +/// the initial theme +public __gshared BNDtheme bndTheme = BNDtheme( + "default theme", + // backgroundColor + nvgRGBA(113, 113, 113, 255), + // regularTheme + BNDwidgetTheme( + "regular", + nvgRGBA( 24, 24, 24, 255), // outlineColor + nvgRGBA( 24, 24, 24, 255), // itemColor + nvgRGBA(153, 153, 153, 255), // innerColor + nvgRGBA( 99, 99, 99, 255), // innerSelectedColor + BND_COLOR_TEXT, // textColor + BND_COLOR_TEXT_SELECTED, // textSelectedColor + 0, // shadeTop + 0, // shadeDown + ), + // toolTheme + BNDwidgetTheme( + "tool", + nvgRGBA( 24, 24, 24, 255), // outlineColor + nvgRGBA( 24, 24, 24, 255), // itemColor + nvgRGBA(153, 153, 153, 255), // innerColor + nvgRGBA( 99, 99, 99, 255), // innerSelectedColor + BND_COLOR_TEXT, // textColor + BND_COLOR_TEXT_SELECTED, // textSelectedColor + 15, // shadeTop + -15, // shadeDown + ), + // radioTheme + BNDwidgetTheme( + "radio", + nvgRGBA( 0, 0, 0, 255), // outlineColor + nvgRGBA(255, 255, 255, 255), // itemColor + nvgRGBA( 70, 70, 70, 255), // innerColor + BND_COLOR_ACTIVE, // innerSelectedColor + BND_COLOR_TEXT_SELECTED, // textColor + BND_COLOR_TEXT, // textSelectedColor + 15, // shadeTop + -15, // shadeDown + ), + // textFieldTheme + BNDwidgetTheme( + "text field", + nvgRGBA( 24, 24, 24, 255), // outlineColor + nvgRGBA( 60, 160, 160, 255), // itemColor + nvgRGBA(153, 153, 153, 255), // innerColor + nvgRGBA(213, 213, 213, 255), // innerSelectedColor + BND_COLOR_TEXT, // textColor + BND_COLOR_TEXT, // textSelectedColor + 0, // shadeTop + 25, // shadeDown + NVGColor.transparent, // textHoverColor + NVGColor.black, // textCaretColor + ), + // optionTheme + BNDwidgetTheme( + "option", + nvgRGBA( 0, 0, 0, 255), // outlineColor + nvgRGBA(255, 255, 255, 255), // itemColor + nvgRGBA( 70, 70, 70, 255), // innerColor + nvgRGBA( 70, 70, 70, 255), // innerSelectedColor + BND_COLOR_TEXT, // textColor + BND_COLOR_TEXT_SELECTED, // textSelectedColor + 15, // shadeTop + -15, // shadeDown + ), + // choiceTheme + BNDwidgetTheme( + "choice", + nvgRGBA( 0, 0, 0, 255), // outlineColor + nvgRGBA(255, 255, 255, 255), // itemColor + nvgRGBA( 70, 70, 70, 255), // innerColor + nvgRGBA( 70, 70, 70, 255), // innerSelectedColor + BND_COLOR_TEXT_SELECTED, // textColor + nvgRGBA(204, 204, 204, 255), // textSelectedColor + 15, // shadeTop + -15, // shadeDown + ), + // numberFieldTheme + BNDwidgetTheme( + "number field", + nvgRGBA( 24, 24, 24, 255), // outlineColor + nvgRGBA( 90, 90, 90, 255), // itemColor + nvgRGBA(180, 180, 180, 255), // innerColor + nvgRGBA(153, 153, 153, 255), // innerSelectedColor + BND_COLOR_TEXT, // textColor + BND_COLOR_TEXT_SELECTED, // textSelectedColor + -20, // shadeTop + 0, // shadeDown + ), + // sliderTheme + BNDwidgetTheme( + "slider", + nvgRGBA( 24, 24, 24, 255), // outlineColor + nvgRGBA(128, 128, 128, 255), // itemColor + nvgRGBA(180, 180, 180, 255), // innerColor + nvgRGBA(153, 153, 153, 255), // innerSelectedColor + BND_COLOR_TEXT, // textColor + BND_COLOR_TEXT_SELECTED, // textSelectedColor + -20, // shadeTop + 0, // shadeDown + ), + // scrollBarTheme + BNDwidgetTheme( + "scrollbar", + nvgRGBA( 49, 49, 49, 255), // outlineColor + nvgRGBA(128, 128, 128, 255), // itemColor + nvgRGBA( 80, 80, 80, 180), // innerColor + nvgRGBA( 99, 99, 99, 180), // innerSelectedColor + BND_COLOR_TEXT, // textColor + BND_COLOR_TEXT_SELECTED, // textSelectedColor + 5, // shadeTop + -5, // shadeDown + ), + // tooltipTheme + BNDwidgetTheme( + "tooltip", + nvgRGBA( 0, 0, 0, 255), // outlineColor + nvgRGBA( 99, 99, 99, 255), // itemColor + nvgRGBA( 24, 24, 24, 230), // innerColor + nvgRGBA( 44, 44, 44, 230), // innerSelectedColor + nvgRGBA(159, 159, 159, 255), // textColor + BND_COLOR_TEXT_SELECTED, // textSelectedColor + 0, // shadeTop + 0, // shadeDown + ), + // menuTheme + BNDwidgetTheme( + "menu", + nvgRGBA( 0, 0, 0, 255), // outlineColor + nvgRGBA( 99, 99, 99, 255), // itemColor + nvgRGBA( 24, 24, 24, 230), // innerColor + nvgRGBA( 44, 44, 44, 230), // innerSelectedColor + nvgRGBA(159, 159, 159, 255), // textColor + BND_COLOR_TEXT_SELECTED, // textSelectedColor + 0, // shadeTop + 0, // shadeDown + ), + // menuItemTheme + BNDwidgetTheme( + "menu item", + nvgRGBA( 0, 0, 0, 255), // outlineColor + nvgRGBA(172, 172, 172, 128), // itemColor + nvgRGBA( 0, 100, 180, 255), // innerColor + BND_COLOR_ACTIVE, // innerSelectedColor + BND_COLOR_TEXT_SELECTED, // textColor + BND_COLOR_TEXT, // textSelectedColor + 38, // shadeTop + 0, // shadeDown + nvgRGBA(255, 255, 255, 255), // textHoverColor + ), + // nodeTheme + BNDnodeTheme( + "node", + nvgRGBA(240, 87, 0, 255), // nodeSelectedColor + nvgRGBA( 0, 0, 0, 255), // wiresColor + nvgRGBA(126, 111, 111, 255), // textSelectedColor + nvgRGBA(255, 170, 64, 255), // activeNodeColor + nvgRGBA(255, 255, 255, 255), // wireSelectColor + nvgRGBA(155, 155, 155, 159), // nodeBackdropColor + 5, // noodleCurving + ), +); + +//////////////////////////////////////////////////////////////////////////////// + +/// set the current theme all widgets will be drawn with. the default Blender 2.6 theme is set by default. +public void bndSetTheme (in ref BNDtheme theme) { bndTheme = theme; } + +/// Returns the currently set theme +public BNDtheme* bndGetTheme () { return &bndTheme; } + +// the handle to the image containing the icon sheet +private __gshared int bndIconImage = -1; + +/** designates an image handle as returned by nvgCreateImage*() as the themes' + * icon sheet. The icon sheet format must be compatible to Blender 2.6's icon + * sheet; the order of icons does not matter. + * + * A valid icon sheet is e.g. shown at + * http://wiki.blender.org/index.php/Dev:2.5/Doc/How_to/Add_an_icon + */ +public void bndSetIconImage (int image) nothrow @trusted @nogc { pragma(inline, true); bndIconImage = image; } + +/// get icon sheet image. +public int bndGetIconImage () nothrow @trusted @nogc { pragma(inline, true); return bndIconImage; } + +// the handle to the UI font +private __gshared int bndFont = -1; + +/** designates an image handle as returned by nvgCreateFont*() as the themes' + * UI font. Blender's original UI font Droid Sans is perfectly suited and + * available here: + * https://svn.blender.org/svnroot/bf-blender/trunk/blender/release/datafiles/fonts/ + */ +public void bndSetFont (int font) nothrow @trusted @nogc { pragma(inline, true); bndFont = font; } + +/// get current font. +public int bndGetFont () nothrow @trusted @nogc { pragma(inline, true); return bndFont; } + +//////////////////////////////////////////////////////////////////////////////// +/// High Level Functions. Use these functions to draw themed widgets with your NVGcontext. + +/** Draw a label with its lower left origin at (x, y) and size of (w, h). + * + * if iconid >= 0, an icon will be added to the widget + * + * if label is not null, a label will be added to the widget + * + * widget looks best when height is BND_WIDGET_HEIGHT + */ +public void bndLabel(T=char) (NVGContext ctx, float x, float y, float w, float h, int iconid, const(T)[] label, int align_=BND_LEFT) +if (isAnyCharType!T) +{ + bndIconLabelValue(ctx, x, y, w, h, iconid, bndTheme.regularTheme.textColor, /*BND_LEFT*/align_, BND_LABEL_FONT_SIZE, label); +} + +/** Draw a tool button with its lower left origin at (x, y) and size of (w, h), + * where flags is one or multiple flags from BNDcornerFlags and state denotes + * the widgets current UI state. + * + * if iconid >= 0, an icon will be added to the widget + * + * if label is not null, a label will be added to the widget + * + * widget looks best when height is BND_WIDGET_HEIGHT + */ +public void bndToolButton(T=char) (NVGContext ctx, float x, float y, float w, float h, int flags, BNDwidgetState state, int iconid, const(T)[] label) +if (isAnyCharType!T) +{ + float[4] cr = void; + NVGColor shadeTop, shadeDown; + bndSelectCorners(cr[], BND_TOOL_RADIUS, flags); + bndBevelInset(ctx, x, y, w, h, cr[2], cr[3]); + bndInnerColors(&shadeTop, &shadeDown, &bndTheme.toolTheme, state, 1); + bndInnerBox(ctx, x, y, w, h, cr[0], cr[1], cr[2], cr[3], shadeTop, shadeDown); + bndOutlineBox(ctx, x, y, w, h, cr[0], cr[1], cr[2], cr[3], bndTransparent(bndTheme.toolTheme.outlineColor)); + bndIconLabelValue(ctx, x, y, w, h, iconid, bndTextColor(&bndTheme.toolTheme, state), BND_CENTER, BND_LABEL_FONT_SIZE, label); +} + +/** Draw a radio button with its lower left origin at (x, y) and size of (w, h), + * where flags is one or multiple flags from BNDcornerFlags and state denotes + * the widgets current UI state. + * + * if iconid >= 0, an icon will be added to the widget + * + * if label is not null, a label will be added to the widget + * + * widget looks best when height is BND_WIDGET_HEIGHT + */ +public void bndRadioButton(T=char) (NVGContext ctx, float x, float y, float w, float h, int flags, BNDwidgetState state, int iconid, const(T)[] label) +if (isAnyCharType!T) +{ + float[4] cr = void; + NVGColor shadeTop, shadeDown; + bndSelectCorners(cr[], BND_OPTION_RADIUS, flags); + bndBevelInset(ctx, x, y, w, h, cr[2], cr[3]); + bndInnerColors(&shadeTop, &shadeDown, &bndTheme.radioTheme, state, 1); + bndInnerBox(ctx, x, y, w, h, cr[0], cr[1], cr[2], cr[3], shadeTop, shadeDown); + bndOutlineBox(ctx, x, y, w, h, cr[0], cr[1], cr[2], cr[3], bndTransparent(bndTheme.radioTheme.outlineColor)); + bndIconLabelValue(ctx, x, y, w, h, iconid, bndTextColor(&bndTheme.radioTheme, state), BND_CENTER, BND_LABEL_FONT_SIZE, label); +} + +/** Draw a radio button with its lower left origin at (x, y) and size of (w, h), + * where flags is one or multiple flags from BNDcornerFlags and state denotes + * the widgets current UI state. + * + * if iconid >= 0, an icon will be added to the widget + * + * if label is not null, a label will be added to the widget + * + * widget looks best when height is BND_WIDGET_HEIGHT + */ +public void bndRadioButton2(T=char) (NVGContext ctx, float x, float y, float w, float h, int flags, BNDwidgetState state, int iconid, const(T)[] label) +if (isAnyCharType!T) +{ + float ox, oy; + NVGColor shadeTop, shadeDown; + ox = x; + oy = y+h-BND_OPTION_HEIGHT-3; + bndBevelInset(ctx, ox, oy, BND_OPTION_WIDTH, BND_OPTION_HEIGHT, BND_OPTION_RADIUS, BND_OPTION_RADIUS); + bndInnerColors(&shadeTop, &shadeDown, &bndTheme.optionTheme, state, 1); + bndInnerBox(ctx, ox, oy, BND_OPTION_WIDTH, BND_OPTION_HEIGHT, BND_OPTION_RADIUS, BND_OPTION_RADIUS, BND_OPTION_RADIUS, BND_OPTION_RADIUS, shadeTop, shadeDown); + bndOutlineBox(ctx, ox, oy, BND_OPTION_WIDTH, BND_OPTION_HEIGHT, BND_OPTION_RADIUS, BND_OPTION_RADIUS, BND_OPTION_RADIUS, BND_OPTION_RADIUS, bndTransparent(bndTheme.optionTheme.outlineColor)); + if (state == BND_ACTIVE) bndRadioCheck(ctx, ox, oy, bndTransparent(bndTheme.optionTheme.itemColor)); + bndIconLabelValue(ctx, x+12, y, w-12, h, -1, bndTextColor(&bndTheme.optionTheme, state), BND_LEFT, BND_LABEL_FONT_SIZE, label); +} + +/** Calculate the corresponding text position for given coordinates px/py + * in a text field. + * See bndTextField for more info. + */ +public int bndTextFieldTextPosition(T=char) (NVGContext ctx, float x, float y, float w, float h, int iconid, const(T)[] text, int px, int py) +if (isAnyCharType!T) +{ + return bndIconLabelTextPosition(ctx, x, y, w, h, iconid, BND_LABEL_FONT_SIZE, text, px, py); +} + +/** Draw a text field with its lower left origin at (x, y) and size of (w, h), + * where flags is one or multiple flags from BNDcornerFlags and state denotes + * the widgets current UI state. + * + * if iconid >= 0, an icon will be added to the widget + * + * if text is not null, text will be printed to the widget + * + * cbegin must be >= 0 and <= strlen(text) and denotes the beginning of the caret + * + * cend must be >= cbegin and <= strlen(text) and denotes the end of the caret + * + * if cend < cbegin, then no caret will be drawn + * + * widget looks best when height is BND_WIDGET_HEIGHT + */ +public void bndTextField(T=char) (NVGContext ctx, float x, float y, float w, float h, int flags, BNDwidgetState state, int iconid, const(T)[] text, int cbegin, int cend) +if (isAnyCharType!T) +{ + float[4] cr = void; + NVGColor shadeTop, shadeDown; + bndSelectCorners(cr[], BND_TEXT_RADIUS, flags); + bndBevelInset(ctx, x, y, w, h, cr[2], cr[3]); + bndInnerColors(&shadeTop, &shadeDown, &bndTheme.textFieldTheme, state, 0); + bndInnerBox(ctx, x, y, w, h, cr[0], cr[1], cr[2], cr[3], shadeTop, shadeDown); + bndOutlineBox(ctx, x, y, w, h, cr[0], cr[1], cr[2], cr[3], bndTransparent(bndTheme.textFieldTheme.outlineColor)); + if (state != BND_ACTIVE) cend = -1; + NVGColor cc = bndTheme.textFieldTheme.textCaretColor; + if (cc.isTransparent) cc = bndTheme.textFieldTheme.textColor; + bndIconLabelCaret(ctx, x, y, w, h, iconid, bndTextColor(&bndTheme.textFieldTheme, state), BND_LABEL_FONT_SIZE, text, bndTheme.textFieldTheme.itemColor, cbegin, cend, cc); +} + +/** Draw an option button with its lower left origin at (x, y) and size of (w, h), + * where flags is one or multiple flags from BNDcornerFlags and state denotes + * the widgets current UI state. + * + * if label is not null, a label will be added to the widget + * + * widget looks best when height is BND_WIDGET_HEIGHT + */ +public void bndOptionButton(T=char) (NVGContext ctx, float x, float y, float w, float h, BNDwidgetState state, const(T)[] label) +if (isAnyCharType!T) +{ + float ox, oy; + NVGColor shadeTop, shadeDown; + ox = x; + oy = y+h-BND_OPTION_HEIGHT-3; + bndBevelInset(ctx, ox, oy, BND_OPTION_WIDTH, BND_OPTION_HEIGHT, BND_OPTION_RADIUS, BND_OPTION_RADIUS); + bndInnerColors(&shadeTop, &shadeDown, &bndTheme.optionTheme, state, 1); + bndInnerBox(ctx, ox, oy, BND_OPTION_WIDTH, BND_OPTION_HEIGHT, BND_OPTION_RADIUS, BND_OPTION_RADIUS, BND_OPTION_RADIUS, BND_OPTION_RADIUS, shadeTop, shadeDown); + bndOutlineBox(ctx, ox, oy, BND_OPTION_WIDTH, BND_OPTION_HEIGHT, BND_OPTION_RADIUS, BND_OPTION_RADIUS, BND_OPTION_RADIUS, BND_OPTION_RADIUS, bndTransparent(bndTheme.optionTheme.outlineColor)); + if (state == BND_ACTIVE) bndCheck(ctx, ox, oy, bndTransparent(bndTheme.optionTheme.itemColor)); + bndIconLabelValue(ctx, x+12, y, w-12, h, -1, bndTextColor(&bndTheme.optionTheme, state), BND_LEFT, BND_LABEL_FONT_SIZE, label); +} + +/** Draw a choice button with its lower left origin at (x, y) and size of (w, h), + * where flags is one or multiple flags from BNDcornerFlags and state denotes + * the widgets current UI state. + * + * if iconid >= 0, an icon will be added to the widget + * + * if label is not null, a label will be added to the widget + * + * widget looks best when height is BND_WIDGET_HEIGHT + */ +public void bndChoiceButton(T=char) (NVGContext ctx, float x, float y, float w, float h, int flags, BNDwidgetState state, int iconid, const(T)[] label) +if (isAnyCharType!T) +{ + float[4] cr = void; + NVGColor shadeTop, shadeDown; + bndSelectCorners(cr[], BND_OPTION_RADIUS, flags); + bndBevelInset(ctx, x, y, w, h, cr[2], cr[3]); + bndInnerColors(&shadeTop, &shadeDown, &bndTheme.choiceTheme, state, 1); + bndInnerBox(ctx, x, y, w, h, cr[0], cr[1], cr[2], cr[3], shadeTop, shadeDown); + bndOutlineBox(ctx, x, y, w, h, cr[0], cr[1], cr[2], cr[3], bndTransparent(bndTheme.choiceTheme.outlineColor)); + bndIconLabelValue(ctx, x, y, w, h, iconid, bndTextColor(&bndTheme.choiceTheme, state), BND_LEFT, BND_LABEL_FONT_SIZE, label); + bndUpDownArrow(ctx, x+w-10, y+10, 5, bndTransparent(bndTheme.choiceTheme.itemColor)); +} + +/** Draw a color button with its lower left origin at (x, y) and size of (w, h), + * where flags is one or multiple flags from BNDcornerFlags and state denotes + * the widgets current UI state. + * + * widget looks best when height is BND_WIDGET_HEIGHT + */ +public void bndColorButton (NVGContext ctx, float x, float y, float w, float h, int flags, NVGColor color) { + float[4] cr = void; + bndSelectCorners(cr[], BND_TOOL_RADIUS, flags); + bndBevelInset(ctx, x, y, w, h, cr[2], cr[3]); + bndInnerBox(ctx, x, y, w, h, cr[0], cr[1], cr[2], cr[3], color, color); + bndOutlineBox(ctx, x, y, w, h, cr[0], cr[1], cr[2], cr[3], bndTransparent(bndTheme.toolTheme.outlineColor)); +} + +/** Draw a number field with its lower left origin at (x, y) and size of (w, h), + * where flags is one or multiple flags from BNDcornerFlags and state denotes + * the widgets current UI state. + * + * if label is not null, a label will be added to the widget + * + * if value is not null, a value will be added to the widget, along with a ":" separator + * + * widget looks best when height is BND_WIDGET_HEIGHT + */ +public void bndNumberField(T=char) (NVGContext ctx, float x, float y, float w, float h, int flags, BNDwidgetState state, const(T)[] label, const(char)[] value) +if (isAnyCharType!T) +{ + float[4] cr = void; + NVGColor shadeTop, shadeDown; + bndSelectCorners(cr[], BND_NUMBER_RADIUS, flags); + bndBevelInset(ctx, x, y, w, h, cr[2], cr[3]); + bndInnerColors(&shadeTop, &shadeDown, &bndTheme.numberFieldTheme, state, 0); + bndInnerBox(ctx, x, y, w, h, cr[0], cr[1], cr[2], cr[3], shadeTop, shadeDown); + bndOutlineBox(ctx, x, y, w, h, cr[0], cr[1], cr[2], cr[3], bndTransparent(bndTheme.numberFieldTheme.outlineColor)); + bndIconLabelValue(ctx, x, y, w, h, -1, bndTextColor(&bndTheme.numberFieldTheme, state), BND_CENTER, BND_LABEL_FONT_SIZE, label, value); + bndArrow(ctx, x+8, y+10, -BND_NUMBER_ARROW_SIZE, bndTransparent(bndTheme.numberFieldTheme.itemColor)); + bndArrow(ctx, x+w-8, y+10, BND_NUMBER_ARROW_SIZE, bndTransparent(bndTheme.numberFieldTheme.itemColor)); +} + +/** Draw slider control with its lower left origin at (x, y) and size of (w, h), + * where flags is one or multiple flags from BNDcornerFlags and state denotes + * the widgets current UI state. + * + * progress must be in the range 0..1 and controls the size of the slider bar + * + * if label is not null, a label will be added to the widget + * + * if value is not null, a value will be added to the widget, along with a ":" separator + * + * widget looks best when height is BND_WIDGET_HEIGHT + */ +public void bndSlider(T=char,TV=char) (NVGContext ctx, float x, float y, float w, float h, int flags, BNDwidgetState state, float progress, const(T)[] label, const(TV)[] value) +if (isAnyCharType!T && isAnyCharType!TV) +{ + float[4] cr = void; + NVGColor shadeTop, shadeDown; + + bndSelectCorners(cr[], BND_NUMBER_RADIUS, flags); + bndBevelInset(ctx, x, y, w, h, cr[2], cr[3]); + bndInnerColors(&shadeTop, &shadeDown, &bndTheme.sliderTheme, state, 0); + bndInnerBox(ctx, x, y, w, h, cr[0], cr[1], cr[2], cr[3], shadeTop, shadeDown); + + if (state == BND_ACTIVE) { + shadeTop = bndOffsetColor(bndTheme.sliderTheme.itemColor, bndTheme.sliderTheme.shadeTop); + shadeDown = bndOffsetColor(bndTheme.sliderTheme.itemColor, bndTheme.sliderTheme.shadeDown); + } else { + shadeTop = bndOffsetColor(bndTheme.sliderTheme.itemColor, bndTheme.sliderTheme.shadeDown); + shadeDown = bndOffsetColor(bndTheme.sliderTheme.itemColor, bndTheme.sliderTheme.shadeTop); + } + ctx.scissor(x, y, 8+(w-8)*bndClamp(progress, 0, 1), h); + bndInnerBox(ctx, x, y, w, h, cr[0], cr[1], cr[2], cr[3], shadeTop, shadeDown); + ctx.resetScissor(); + + bndOutlineBox(ctx, x, y, w, h, cr[0], cr[1], cr[2], cr[3], bndTransparent(bndTheme.sliderTheme.outlineColor)); + bndIconLabelValue(ctx, x, y, w, h, -1, bndTextColor(&bndTheme.sliderTheme, state), BND_CENTER, BND_LABEL_FONT_SIZE, label, value); +} + +/** Draw scrollbar with its lower left origin at (x, y) and size of (w, h), + * where state denotes the widgets current UI state. + * + * offset is in the range 0..1 and controls the position of the scroll handle + * + * size is in the range 0..1 and controls the size of the scroll handle + * + * horizontal widget looks best when height is BND_SCROLLBAR_HEIGHT, + * + * vertical looks best when width is BND_SCROLLBAR_WIDTH + */ +public void bndScrollBar (NVGContext ctx, float x, float y, float w, float h, BNDwidgetState state, float offset, float size) { + bndBevelInset(ctx, x, y, w, h, BND_SCROLLBAR_RADIUS, BND_SCROLLBAR_RADIUS); + bndInnerBox(ctx, x, y, w, h, + BND_SCROLLBAR_RADIUS, BND_SCROLLBAR_RADIUS, + BND_SCROLLBAR_RADIUS, BND_SCROLLBAR_RADIUS, + bndOffsetColor(bndTheme.scrollBarTheme.innerColor, 3*bndTheme.scrollBarTheme.shadeDown), + bndOffsetColor(bndTheme.scrollBarTheme.innerColor, 3*bndTheme.scrollBarTheme.shadeTop)); + bndOutlineBox(ctx, x, y, w, h, + BND_SCROLLBAR_RADIUS, BND_SCROLLBAR_RADIUS, + BND_SCROLLBAR_RADIUS, BND_SCROLLBAR_RADIUS, + bndTransparent(bndTheme.scrollBarTheme.outlineColor)); + + NVGColor itemColor = bndOffsetColor(bndTheme.scrollBarTheme.itemColor, (state == BND_ACTIVE ? BND_SCROLLBAR_ACTIVE_SHADE : 0)); + + bndScrollHandleRect(&x, &y, &w, &h, offset, size); + + bndInnerBox(ctx, x, y, w, h, + BND_SCROLLBAR_RADIUS, BND_SCROLLBAR_RADIUS, + BND_SCROLLBAR_RADIUS, BND_SCROLLBAR_RADIUS, + bndOffsetColor(itemColor, 3*bndTheme.scrollBarTheme.shadeTop), + bndOffsetColor(itemColor, 3*bndTheme.scrollBarTheme.shadeDown)); + bndOutlineBox(ctx, x, y, w, h, + BND_SCROLLBAR_RADIUS, BND_SCROLLBAR_RADIUS, + BND_SCROLLBAR_RADIUS, BND_SCROLLBAR_RADIUS, + bndTransparent(bndTheme.scrollBarTheme.outlineColor)); +} + +/** Draw scrollbar with its lower left origin at (x, y) and size of (w, h), + * where state denotes the widgets current UI state. + * + * offset is in the range 0..1 and controls the position of the scroll handle + * + * size is in the range 0..1 and controls the size of the scroll handle + * + * horizontal widget looks best when height is BND_SCROLLBAR_HEIGHT, + * + * vertical looks best when width is BND_SCROLLBAR_WIDTH + */ +public void bndScrollSlider (NVGContext ctx, float x, float y, float w, float h, BNDwidgetState state, float offset, float size) { + bndBevelInset(ctx, x, y, w, h, BND_SCROLLBAR_RADIUS, BND_SCROLLBAR_RADIUS); + bndInnerBox(ctx, x, y, w, h, + BND_SCROLLBAR_RADIUS, BND_SCROLLBAR_RADIUS, + BND_SCROLLBAR_RADIUS, BND_SCROLLBAR_RADIUS, + bndOffsetColor(bndTheme.scrollBarTheme.innerColor, 3*bndTheme.scrollBarTheme.shadeDown), + bndOffsetColor(bndTheme.scrollBarTheme.innerColor, 3*bndTheme.scrollBarTheme.shadeTop)); + bndOutlineBox(ctx, x, y, w, h, + BND_SCROLLBAR_RADIUS, BND_SCROLLBAR_RADIUS, + BND_SCROLLBAR_RADIUS, BND_SCROLLBAR_RADIUS, + bndTransparent(bndTheme.scrollBarTheme.outlineColor)); + + NVGColor itemColor = bndOffsetColor(bndTheme.scrollBarTheme.itemColor, (state == BND_ACTIVE ? BND_SCROLLBAR_ACTIVE_SHADE : 0)); + + bndScrollSliderRect(&w, &h, offset, size); + + bndInnerBox(ctx, x, y, w, h, + BND_SCROLLBAR_RADIUS, BND_SCROLLBAR_RADIUS, + BND_SCROLLBAR_RADIUS, BND_SCROLLBAR_RADIUS, + bndOffsetColor(itemColor, 3*bndTheme.scrollBarTheme.shadeTop), + bndOffsetColor(itemColor, 3*bndTheme.scrollBarTheme.shadeDown)); + bndOutlineBox(ctx, x, y, w, h, + BND_SCROLLBAR_RADIUS, BND_SCROLLBAR_RADIUS, + BND_SCROLLBAR_RADIUS, BND_SCROLLBAR_RADIUS, + bndTransparent(bndTheme.scrollBarTheme.outlineColor)); +} + +/** Draw a menu background with its lower left origin at (x, y) and size of (w, h), + * where flags is one or multiple flags from BNDcornerFlags. + */ +public void bndMenuBackground (NVGContext ctx, float x, float y, float w, float h, int flags) { + float[4] cr = void; + NVGColor shadeTop, shadeDown; + bndSelectCorners(cr[], BND_MENU_RADIUS, flags); + bndInnerColors(&shadeTop, &shadeDown, &bndTheme.menuTheme, BND_DEFAULT, 0); + bndInnerBox(ctx, x, y, w, h+1, cr[0], cr[1], cr[2], cr[3], shadeTop, shadeDown); + bndOutlineBox(ctx, x, y, w, h+1, cr[0], cr[1], cr[2], cr[3], bndTransparent(bndTheme.menuTheme.outlineColor)); + bndDropShadow(ctx, x, y, w, h, BND_MENU_RADIUS, BND_SHADOW_FEATHER, BND_SHADOW_ALPHA); +} + +/// Draw a tooltip background with its lower left origin at (x, y) and size of (w, h) +public void bndTooltipBackground (NVGContext ctx, float x, float y, float w, float h) { + NVGColor shadeTop, shadeDown; + bndInnerColors(&shadeTop, &shadeDown, &bndTheme.tooltipTheme, BND_DEFAULT, 0); + bndInnerBox(ctx, x, y, w, h+1, BND_MENU_RADIUS, BND_MENU_RADIUS, BND_MENU_RADIUS, BND_MENU_RADIUS, shadeTop, shadeDown); + bndOutlineBox(ctx, x, y, w, h+1, BND_MENU_RADIUS, BND_MENU_RADIUS, BND_MENU_RADIUS, BND_MENU_RADIUS, bndTransparent(bndTheme.tooltipTheme.outlineColor)); + bndDropShadow(ctx, x, y, w, h, BND_MENU_RADIUS, BND_SHADOW_FEATHER, BND_SHADOW_ALPHA); +} + +/** Draw a menu label with its lower left origin at (x, y) and size of (w, h). + * + * if iconid >= 0, an icon will be added to the widget + * + * if label is not null, a label will be added to the widget + * + * widget looks best when height is BND_WIDGET_HEIGHT + */ +public void bndMenuLabel(T=char) (NVGContext ctx, float x, float y, float w, float h, int iconid, const(T)[] label) +if (isAnyCharType!T) +{ + bndIconLabelValue(ctx, x, y, w, h, iconid, bndTheme.menuTheme.textColor, BND_LEFT, BND_LABEL_FONT_SIZE, label); +} + +/** Draw a menu item with its lower left origin at (x, y) and size of (w, h), + * where state denotes the widgets current UI state. + * + * if iconid >= 0, an icon will be added to the widget + * + * if label is not null, a label will be added to the widget + * + * widget looks best when height is BND_WIDGET_HEIGHT + */ +public void bndMenuItem(T=char) (NVGContext ctx, float x, float y, float w, float h, BNDwidgetState state, int iconid, const(T)[] label) +if (isAnyCharType!T) +{ + if (state != BND_DEFAULT) { + auto clr = (state == BND_HOVER ? bndOffsetColor(bndTheme.menuItemTheme.innerColor/*innerSelectedColor*/, BND_HOVER_SHADE) : bndTheme.menuItemTheme.innerSelectedColor); + bndInnerBox(ctx, x, y, w, h, 0, 0, 0, 0, + bndOffsetColor(clr, bndTheme.menuItemTheme.shadeTop), + bndOffsetColor(clr, bndTheme.menuItemTheme.shadeDown)); + //state = BND_ACTIVE; + } + bndIconLabelValue(ctx, x, y, w, h, iconid, + bndTextColor(&bndTheme.menuItemTheme, state), BND_LEFT, + BND_LABEL_FONT_SIZE, label); +} + +/// Draw a node port at the given position filled with the given color +public void bndNodePort (NVGContext ctx, float x, float y, BNDwidgetState state, NVGColor color) { + ctx.beginPath(); + ctx.circle(x, y, BND_NODE_PORT_RADIUS); + ctx.strokeColor(bndTheme.nodeTheme.wiresColor); + ctx.strokeWidth(1.0f); + ctx.stroke(); + ctx.fillColor((state != BND_DEFAULT ? bndOffsetColor(color, BND_HOVER_SHADE) : color)); + ctx.fill(); +} + +/// Draw a node wire originating at (x0, y0) and floating to (x1, y1), with a colored gradient based on the two colors color0 and color1 +public void bndColoredNodeWire (NVGContext ctx, float x0, float y0, float x1, float y1, NVGColor color0, NVGColor color1) { + import core.stdc.math : fabsf; + float length = bndMax(fabsf(x1-x0), fabsf(y1-y0)); + float delta = length*cast(float)bndTheme.nodeTheme.noodleCurving/10.0f; + + ctx.beginPath(); + ctx.moveTo(x0, y0); + ctx.bezierTo(x0+delta, y0, x1-delta, y1, x1, y1); + NVGColor colorw = bndTheme.nodeTheme.wiresColor; + colorw.a = (color0.a < color1.a ? color0.a : color1.a); + ctx.strokeColor(colorw); + ctx.strokeWidth(BND_NODE_WIRE_OUTLINE_WIDTH); + ctx.stroke(); + ctx.strokePaint(ctx.linearGradient(x0, y0, x1, y1, color0, color1)); + ctx.strokeWidth(BND_NODE_WIRE_WIDTH); + ctx.stroke(); +} + +/** Draw a node wire originating at (x0, y0) and floating to (x1, y1), with + * a colored gradient based on the states state0 and state1: + * + * BND_DEFAULT: default wire color + * + * BND_HOVER: selected wire color + * + * BND_ACTIVE: dragged wire color + */ +public void bndNodeWire (NVGContext ctx, float x0, float y0, float x1, float y1, BNDwidgetState state0, BNDwidgetState state1) { + bndColoredNodeWire(ctx, x0, y0, x1, y1, bndNodeWireColor(&bndTheme.nodeTheme, state0), bndNodeWireColor(&bndTheme.nodeTheme, state1)); +} + +/// Draw a node background with its upper left origin at (x, y) and size of (w, h) where titleColor provides the base color for the title bar +public void bndNodeBackground(T=char) (NVGContext ctx, float x, float y, float w, float h, BNDwidgetState state, int iconid, const(T)[] label, NVGColor titleColor) +if (isAnyCharType!T) +{ + bndInnerBox(ctx, x, y, w, BND_NODE_TITLE_HEIGHT+2, + BND_NODE_RADIUS, BND_NODE_RADIUS, 0, 0, + bndTransparent(bndOffsetColor(titleColor, BND_BEVEL_SHADE)), + bndTransparent(titleColor)); + bndInnerBox(ctx, x, y+BND_NODE_TITLE_HEIGHT-1, w, h+2-BND_NODE_TITLE_HEIGHT, + 0, 0, BND_NODE_RADIUS, BND_NODE_RADIUS, + bndTransparent(bndTheme.nodeTheme.nodeBackdropColor), + bndTransparent(bndTheme.nodeTheme.nodeBackdropColor)); + bndNodeIconLabel(ctx, + x+BND_NODE_ARROW_AREA_WIDTH, y, + w-BND_NODE_ARROW_AREA_WIDTH-BND_NODE_MARGIN_SIDE, BND_NODE_TITLE_HEIGHT, + iconid, bndTheme.regularTheme.textColor, + bndOffsetColor(titleColor, BND_BEVEL_SHADE), + BND_LEFT, BND_LABEL_FONT_SIZE, label); + NVGColor arrowColor; + NVGColor borderColor; + switch (state) { + default: + case BND_DEFAULT: + borderColor = nvgRGBf(0, 0, 0); + arrowColor = bndOffsetColor(titleColor, -BND_BEVEL_SHADE); + break; + case BND_HOVER: + borderColor = bndTheme.nodeTheme.nodeSelectedColor; + arrowColor = bndTheme.nodeTheme.nodeSelectedColor; + break; + case BND_ACTIVE: + borderColor = bndTheme.nodeTheme.activeNodeColor; + arrowColor = bndTheme.nodeTheme.nodeSelectedColor; + break; + } + bndOutlineBox(ctx, x, y, w, h+1, BND_NODE_RADIUS, BND_NODE_RADIUS, BND_NODE_RADIUS, BND_NODE_RADIUS, bndTransparent(borderColor)); + //bndNodeArrowDown(ctx, x+BND_NODE_MARGIN_SIDE, y+BND_NODE_TITLE_HEIGHT-4, BND_NODE_ARROW_SIZE, arrowColor); + bndDropShadow(ctx, x, y, w, h, BND_NODE_RADIUS, BND_SHADOW_FEATHER, BND_SHADOW_ALPHA); +} + +/// Draw a window with the upper right and lower left splitter widgets into the rectangle at origin (x, y) and size (w, h) +public void bndSplitterWidgets (NVGContext ctx, float x, float y, float w, float h) { + NVGColor insetLight = bndTransparent(bndOffsetColor(bndTheme.backgroundColor, BND_SPLITTER_SHADE)); + NVGColor insetDark = bndTransparent(bndOffsetColor(bndTheme.backgroundColor, -BND_SPLITTER_SHADE)); + NVGColor inset = bndTransparent(bndTheme.backgroundColor); + + float x2 = x+w; + float y2 = y+h; + + ctx.beginPath(); + ctx.moveTo(x, y2-13); + ctx.lineTo(x+13, y2); + ctx.moveTo(x, y2-9); + ctx.lineTo(x+9, y2); + ctx.moveTo(x, y2-5); + ctx.lineTo(x+5, y2); + + ctx.moveTo(x2-11, y); + ctx.lineTo(x2, y+11); + ctx.moveTo(x2-7, y); + ctx.lineTo(x2, y+7); + ctx.moveTo(x2-3, y); + ctx.lineTo(x2, y+3); + + ctx.strokeColor(insetDark); + ctx.stroke(); + + ctx.beginPath(); + ctx.moveTo(x, y2-11); + ctx.lineTo(x+11, y2); + ctx.moveTo(x, y2-7); + ctx.lineTo(x+7, y2); + ctx.moveTo(x, y2-3); + ctx.lineTo(x+3, y2); + + ctx.moveTo(x2-13, y); + ctx.lineTo(x2, y+13); + ctx.moveTo(x2-9, y); + ctx.lineTo(x2, y+9); + ctx.moveTo(x2-5, y); + ctx.lineTo(x2, y+5); + + ctx.strokeColor(insetLight); + ctx.stroke(); + + ctx.beginPath(); + ctx.moveTo(x, y2-12); + ctx.lineTo(x+12, y2); + ctx.moveTo(x, y2-8); + ctx.lineTo(x+8, y2); + ctx.moveTo(x, y2-4); + ctx.lineTo(x+4, y2); + + ctx.moveTo(x2-12, y); + ctx.lineTo(x2, y+12); + ctx.moveTo(x2-8, y); + ctx.lineTo(x2, y+8); + ctx.moveTo(x2-4, y); + ctx.lineTo(x2, y+4); + + ctx.strokeColor(inset); + ctx.stroke(); +} + +/** Draw the join area overlay stencil into the rectangle + * at origin (x, y) and size (w, h) + * + * vertical is `false` or `true` and designates the arrow orientation, mirror is `false` or `true` and flips the arrow side + */ +public void bndJoinAreaOverlay (NVGContext ctx, float x, float y, float w, float h, bool vertical, bool mirror) { + if (vertical) { + float u = w; + w = h; h = u; + } + + float s = (w < h ? w : h); + + float x0, y0, x1, y1; + if (mirror) { + x0 = w; + y0 = h; + x1 = 0; + y1 = 0; + s = -s; + } else { + x0 = 0; + y0 = 0; + x1 = w; + y1 = h; + } + + float yc = (y0+y1)*0.5f; + float s2 = s/2.0f; + float s4 = s/4.0f; + float s8 = s/8.0f; + float x4 = x0+s4; + + float[2][11] points = [ + [ x0, y0 ], + [ x1, y0 ], + [ x1, y1 ], + [ x0, y1 ], + [ x0, yc+s8 ], + [ x4, yc+s8 ], + [ x4, yc+s4 ], + [ x0+s2, yc ], + [ x4, yc-s4 ], + [ x4, yc-s8 ], + [ x0, yc-s8 ] + ]; + + ctx.beginPath(); + int count = cast(int)points.length; //sizeof(points)/(sizeof(float)*2); + ctx.moveTo(x+points[0][vertical&1], y+points[0][(vertical&1)^1]); + foreach (int i; 1..count) ctx.lineTo(x+points[i][vertical&1], y+points[i][(vertical&1)^1]); + + ctx.fillColor(nvgRGBAf(0, 0, 0, 0.3)); + ctx.fill(); +} + + +//////////////////////////////////////////////////////////////////////////////// +/// Estimator Functions +/// Use these functions to estimate sizes for widgets with your NVGcontext. + +/// 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; + if (iconid >= 0) w += BND_ICON_SHEET_RES; + if (label && bndFont >= 0) { + ctx.fontFaceId(bndFont); + ctx.fontSize(BND_LABEL_FONT_SIZE); + w += cast(int)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; +} + +/// 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; + width -= BND_TEXT_RADIUS*2; + if (iconid >= 0) width -= BND_ICON_SHEET_RES; + if (label && (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; + if (bh > h) h = bh; + } + return h; +} + + +//////////////////////////////////////////////////////////////////////////////// +/// Low Level Functions +/// these are part of the implementation detail and can be used to theme new kinds of controls in a similar fashion. + +/** Add a rounded box path at position (x, y) with size (w, h) and a separate + * radius for each corner listed in clockwise order, so that cr0 = top left, + * cr1 = top right, cr2 = bottom right, cr3 = bottom left; + * + * this is a low level drawing function: the path must be stroked or filled + * to become visible. + */ +public void bndRoundedBox (NVGContext ctx, float x, float y, float w, float h, float cr0, float cr1, float cr2, float cr3) { + float d; + w = bndMax(0, w); + h = bndMax(0, h); + d = bndMin(w, h); + ctx.moveTo(x, y+h*0.5f); + ctx.arcTo(x, y, x+w, y, bndMin(cr0, d/2)); + ctx.arcTo(x+w, y, x+w, y+h, bndMin(cr1, d/2)); + ctx.arcTo(x+w, y+h, x, y+h, bndMin(cr2, d/2)); + ctx.arcTo(x, y+h, x, y, bndMin(cr3, d/2)); + ctx.closePath(); +} + +/// make color transparent using the default alpha value +public NVGColor bndTransparent (NVGColor color) { + color.a *= BND_TRANSPARENT_ALPHA; + return color; +} + +/// offset a color by a given integer delta in the range -100 to 100 +public NVGColor bndOffsetColor (NVGColor color, int delta) { + float offset = cast(float)delta/255.0f; + return (delta ? nvgRGBAf(bndClamp(color.r+offset, 0, 1), bndClamp(color.g+offset, 0, 1), bndClamp(color.b+offset, 0, 1), color.a) : color); +} + +/// Draw a beveled border at position (x, y) with size (w, h) shaded with lighter and darker versions of backgroundColor +public void bndBevel (NVGContext ctx, float x, float y, float w, float h) { + ctx.strokeWidth(1); + + x += 0.5f; + y += 0.5f; + w -= 1; + h -= 1; + + ctx.beginPath(); + ctx.moveTo(x, y+h); + ctx.lineTo(x+w, y+h); + ctx.lineTo(x+w, y); + ctx.strokeColor(bndTransparent(bndOffsetColor(bndTheme.backgroundColor, -BND_BEVEL_SHADE))); + ctx.stroke(); + + ctx.beginPath(); + ctx.moveTo(x, y+h); + ctx.lineTo(x, y); + ctx.lineTo(x+w, y); + ctx.strokeColor(bndTransparent(bndOffsetColor(bndTheme.backgroundColor, BND_BEVEL_SHADE))); + ctx.stroke(); +} + +/** Draw a lower inset for a rounded box at position (x, y) with size (w, h) + * that gives the impression the surface has been pushed in. + * + * cr2 and cr3 contain the radiuses of the bottom right and bottom left + * corners of the rounded box. + */ +public void bndBevelInset (NVGContext ctx, float x, float y, float w, float h, float cr2, float cr3) { + float d; + + y -= 0.5f; + d = bndMin(w, h); + cr2 = bndMin(cr2, d/2); + cr3 = bndMin(cr3, d/2); + + ctx.beginPath(); + ctx.moveTo(x+w, y+h-cr2); + ctx.arcTo(x+w, y+h, x, y+h, cr2); + ctx.arcTo(x, y+h, x, y, cr3); + + NVGColor bevelColor = bndOffsetColor(bndTheme.backgroundColor, BND_INSET_BEVEL_SHADE); + + ctx.strokeWidth(1); + ctx.strokePaint(ctx.linearGradient(x, y+h-bndMax(cr2, cr3)-1, x, y+h-1, nvgRGBAf(bevelColor.r, bevelColor.g, bevelColor.b, 0), bevelColor)); + ctx.stroke(); +} + +/// Draw a flat panel without any decorations at position (x, y) with size (w, h) and fills it with backgroundColor +public void bndBackground (NVGContext ctx, float x, float y, float w, float h) { + ctx.beginPath(); + ctx.rect(x, y, w, h); + ctx.fillColor(bndTheme.backgroundColor); + ctx.fill(); +} + +/// Draw an icon with (x, y) as its upper left coordinate; the iconid selects the icon from the sheet; use the BND_ICONID macro to build icon IDs. +public void bndIcon (NVGContext ctx, float x, float y, int iconid) { + int ix, iy, u, v; + if (bndIconImage < 0) return; // no icons loaded + + ix = iconid&0xff; + iy = (iconid>>8)&0xff; + u = BND_ICON_SHEET_OFFSET_X+ix*BND_ICON_SHEET_GRID; + v = BND_ICON_SHEET_OFFSET_Y+iy*BND_ICON_SHEET_GRID; + + ctx.beginPath(); + ctx.rect(x, y, BND_ICON_SHEET_RES, BND_ICON_SHEET_RES); + ctx.fillPaint(ctx.imagePattern(x-u, y-v, BND_ICON_SHEET_WIDTH, BND_ICON_SHEET_HEIGHT, 0, bndIconImage, 1)); + ctx.fill(); +} + +/** Draw a drop shadow around the rounded box at (x, y) with size (w, h) and + * radius r, with feather as its maximum range in pixels. + * + * No shadow will be painted inside the rounded box. + */ +public void bndDropShadow (NVGContext ctx, float x, float y, float w, float h, float r, float feather, float alpha) { + ctx.beginPath(); + y += feather; + h -= feather; + + ctx.moveTo(x-feather, y-feather); + ctx.lineTo(x, y-feather); + ctx.lineTo(x, y+h-feather); + ctx.arcTo(x, y+h, x+r, y+h, r); + ctx.arcTo(x+w, y+h, x+w, y+h-r, r); + ctx.lineTo(x+w, y-feather); + ctx.lineTo(x+w+feather, y-feather); + ctx.lineTo(x+w+feather, y+h+feather); + ctx.lineTo(x-feather, y+h+feather); + ctx.closePath(); + + ctx.fillPaint(ctx.boxGradient(x-feather*0.5f, y-feather*0.5f, + w+feather, h+feather, + r+feather*0.5f, + feather, + nvgRGBAf(0, 0, 0, alpha*alpha), + nvgRGBAf(0, 0, 0, 0))); + ctx.fill(); +} + +/* Draw the inner part of a widget box, with a gradient from shadeTop to + * shadeDown. If h>w, the gradient will be horizontal instead of vertical. + */ +public void bndInnerBox (NVGContext ctx, float x, float y, float w, float h, float cr0, float cr1, float cr2, float cr3, NVGColor shadeTop, NVGColor shadeDown) { + ctx.beginPath(); + bndRoundedBox(ctx, x+1, y+1, w-2, h-3, bndMax(0, cr0-1), bndMax(0, cr1-1), bndMax(0, cr2-1), bndMax(0, cr3-1)); + ctx.fillPaint((h-2 > w ? ctx.linearGradient(x, y, x+w, y, shadeTop, shadeDown) : ctx.linearGradient(x, y, x, y+h, shadeTop, shadeDown))); + ctx.fill(); +} + +/// Draw the outline part of a widget box with the given color +public void bndOutlineBox (NVGContext ctx, float x, float y, float w, float h, float cr0, float cr1, float cr2, float cr3, NVGColor color) { + ctx.beginPath(); + bndRoundedBox(ctx, x+0.5f, y+0.5f, w-1, h-2, cr0, cr1, cr2, cr3); + ctx.strokeColor(color); + ctx.strokeWidth(1); + ctx.stroke(); +} + +/** assigns radius r to the four entries of array radiuses depending on whether + * the corner is marked as sharp or not; see BNDcornerFlags for possible + * flag values. + */ +public void bndSelectCorners (float[] radiuses, float r, int flags) { + if (radiuses.length > 0) radiuses.ptr[0] = (flags&BND_CORNER_TOP_LEFT ? 0 : r); + if (radiuses.length > 1) radiuses.ptr[1] = (flags&BND_CORNER_TOP_RIGHT ? 0 : r); + if (radiuses.length > 2) radiuses.ptr[2] = (flags&BND_CORNER_DOWN_RIGHT ? 0 : r); + if (radiuses.length > 3) radiuses.ptr[3] = (flags&BND_CORNER_DOWN_LEFT ? 0 : r); +} + +/** computes the upper and lower gradient colors for the inner box from a widget + * theme and the widgets state. If flipActive is set and the state is + * BND_ACTIVE, the upper and lower colors will be swapped. + */ +public void bndInnerColors (NVGColor* shadeTop, NVGColor* shadeDown, const(BNDwidgetTheme)* theme, BNDwidgetState state, int flipActive) { + switch (state) { + default: + case BND_DEFAULT: + if (shadeTop !is null) *shadeTop = bndOffsetColor(theme.innerColor, theme.shadeTop); + if (shadeDown !is null) *shadeDown = bndOffsetColor(theme.innerColor, theme.shadeDown); + break; + case BND_HOVER: + NVGColor color = bndOffsetColor(theme.innerColor, BND_HOVER_SHADE); + if (shadeTop !is null) *shadeTop = bndOffsetColor(color, theme.shadeTop); + if (shadeDown !is null) *shadeDown = bndOffsetColor(color, theme.shadeDown); + break; + case BND_ACTIVE: + if (shadeTop !is null) *shadeTop = bndOffsetColor(theme.innerSelectedColor, flipActive?theme.shadeDown:theme.shadeTop); + if (shadeDown !is null) *shadeDown = bndOffsetColor(theme.innerSelectedColor, flipActive?theme.shadeTop:theme.shadeDown); + break; + } +} + +/// computes the text color for a widget label from a widget theme and the widgets state. +public NVGColor bndTextColor (const(BNDwidgetTheme)* theme, BNDwidgetState state) nothrow @trusted @nogc { + pragma(inline, true); + return + state == BND_ACTIVE ? theme.textSelectedColor : + state == BND_HOVER ? (theme.textHoverColor.isTransparent ? theme.textColor : theme.textHoverColor) : + theme.textColor; +} + +/** Draw an optional icon specified by and an optional label with + * given alignment (BNDtextAlignment), fontsize and color within a widget box. + * + * if iconid is >= 0, an icon will be drawn and the labels remaining space will be adjusted. + * + * if label is not null, it will be drawn with the specified alignment, fontsize and color. + * + * if value is not null, label and value will be drawn with a ":" separator inbetween. + */ +public void bndIconLabelValue(T=char,TV=char) (NVGContext ctx, float x, float y, float w, float h, int iconid, NVGColor color, int align_, float fontsize, const(T)[] label, const(TV)[] value=null) +if (isAnyCharType!T && isAnyCharType!TV) +{ + float pleft = BND_PAD_LEFT; + if (label.length) { + if (iconid >= 0) { + bndIcon(ctx, x+4, y+2, iconid); + pleft += BND_ICON_SHEET_RES; + } + + if (bndFont < 0) return; + ctx.fontFaceId(bndFont); + ctx.fontSize(fontsize); + ctx.beginPath(); + ctx.fillColor(color); + if (value.length) { + float label_width = ctx.textBounds(1, 1, label, null); + float sep_width = ctx.textBounds(1, 1, BND_LABEL_SEPARATOR, null); + + ctx.textAlign(NVGTextAlign.H.Left, NVGTextAlign.V.Baseline); + x += pleft; + if (align_ == BND_CENTER) { + float width = label_width+sep_width+ctx.textBounds(1, 1, value, null); + x += ((w-BND_PAD_RIGHT-pleft)-width)*0.5f; + } else if (align_ == BND_RIGHT) { + float width = label_width+sep_width+ctx.textBounds(1, 1, value, null); + x += w-BND_PAD_RIGHT-width; + } + y += BND_WIDGET_HEIGHT-BND_TEXT_PAD_DOWN; + ctx.text(x, y, label); + x += label_width; + ctx.text(x, y, BND_LABEL_SEPARATOR); + x += sep_width; + ctx.text(x, y, value); + } else { + ctx.textAlign((align_ == BND_LEFT ? NVGTextAlign(NVGTextAlign.H.Left, NVGTextAlign.V.Baseline) : align_ == BND_CENTER ? NVGTextAlign(NVGTextAlign.H.Center, NVGTextAlign.V.Baseline) : NVGTextAlign(NVGTextAlign.H.Right, NVGTextAlign.V.Baseline))); + ctx.textBox(x+pleft, y+BND_WIDGET_HEIGHT-BND_TEXT_PAD_DOWN, w-BND_PAD_RIGHT-pleft, label); + //{ import core.stdc.stdio : printf; printf("l=%u\n", cast(uint)label.length); } + } + } else if (iconid >= 0) { + bndIcon(ctx, x+2, y+2, iconid); + } +} + +/** Draw an optional icon specified by and an optional label with + * given alignment (BNDtextAlignment), fontsize and color within a node title bar + * + * if iconid is >= 0, an icon will be drawn + * + * if label is not null, it will be drawn with the specified alignment, fontsize and color. + */ +public void bndNodeIconLabel(T=char) (NVGContext ctx, float x, float y, float w, float h, int iconid, NVGColor color, NVGColor shadowColor, int align_, float fontsize, const(T)[] label) +if (isAnyCharType!T) +{ + if (label.length && bndFont >= 0) { + ctx.fontFaceId(bndFont); + ctx.fontSize(fontsize); + ctx.beginPath(); + ctx.textAlign(NVGTextAlign.H.Left, NVGTextAlign.V.Baseline); + ctx.fillColor(shadowColor); + ctx.fontBlur(BND_NODE_TITLE_FEATHER); + ctx.textBox(x+1, y+h+3-BND_TEXT_PAD_DOWN, w, label); + ctx.fillColor(color); + ctx.fontBlur(0); + ctx.textBox(x, y+h+2-BND_TEXT_PAD_DOWN, w, label); + } + if (iconid >= 0) bndIcon(ctx, x+w-BND_ICON_SHEET_RES, y+3, iconid); +} + +/** Calculate the corresponding text position for given coordinates px/py in an iconLabel. + * See bndIconLabelCaret for more info. + */ +public int bndIconLabelTextPosition(T=char) (NVGContext ctx, float x, float y, float w, float h, int iconid, float fontsize, const(T)[] label, int px, int py) +if (isAnyCharType!T) +{ + float[4] bounds; + float pleft = BND_TEXT_RADIUS; + if (label.length == 0) return -1; + if (iconid >= 0) pleft += BND_ICON_SHEET_RES; + + if (bndFont < 0) return -1; + + x += pleft; + y += BND_WIDGET_HEIGHT-BND_TEXT_PAD_DOWN; + + ctx.fontFaceId(bndFont); + ctx.fontSize(fontsize); + ctx.textAlign(NVGTextAlign.H.Left, NVGTextAlign.V.Baseline); + + w -= BND_TEXT_RADIUS+pleft; + + float asc, desc, lh; + static NVGTextRow!T[BND_MAX_ROWS] rows; + auto rres = ctx.textBreakLines(label, w, rows[]); + //{ import core.stdc.stdio : printf; printf("rlen=%u\n", cast(uint)rres.length); } + if (rres.length == 0) return 0; + ctx.textBoxBounds(x, y, w, label, bounds[]); + ctx.textMetrics(&asc, &desc, &lh); + + // calculate vertical position + int row = cast(int)bndClamp(cast(int)(cast(float)(py-bounds[1])/lh), 0, cast(int)rres.length-1); + // search horizontal position + static NVGGlyphPosition[BND_MAX_GLYPHS] glyphs; + //int nglyphs = ctx.textGlyphPositions(x, y, rows[row].start, rows[row].end+1, glyphs.ptr, BND_MAX_GLYPHS); + auto rglyphs = ctx.textGlyphPositions(x, y, rows[row].row!T, glyphs[]); + int nglyphs = cast(int)rglyphs.length; + int col, p = 0; + for (col = 0; col < nglyphs && glyphs[col].x < px; ++col) p = cast(int)glyphs[col].strpos; + // see if we should move one character further + if (col > 0 && col < nglyphs && glyphs[col].x-px < px-glyphs[col-1].x) p = cast(int)glyphs[col].strpos; + return p; +} + +void bndCaretPosition(RT) (NVGContext ctx, float x, float y, float desc, float lineHeight, int caretpos, RT[] rows, int* cr, float* cx, float* cy) +if (is(RT : NVGTextRow!CT, CT)) +{ + static NVGGlyphPosition[BND_MAX_GLYPHS] glyphs; + usize r = 0; + //for (r = 0; r < nrows && rows[r].end < caret; ++r) {} + while (r < rows.length && rows[r].end < caretpos) ++r; + if (cr !is null) *cr = r; + if (cx !is null) *cx = x; + if (cy !is null) *cy = y-lineHeight-desc+r*lineHeight; + if (rows.length == 0) return; + if (cx !is null) *cx = rows[r].minx; + //auto rglyphs = (rows[r].isChar ? ctx.textGlyphPositions(x, y, rows[r].row!char, glyphs[]) : ctx.textGlyphPositions(x, y, rows[r].row!dchar, glyphs[])); + auto rglyphs = ctx.textGlyphPositions(x, y, rows[r].row, glyphs[]); + foreach (immutable i; 0..rglyphs.length) { + if (cx !is null) *cx = glyphs.ptr[i].x; + if (glyphs.ptr[i].strpos == caretpos) break; + } +} + +/** Draw an optional icon specified by , an optional label and + * a caret with given fontsize and color within a widget box. + * + * if iconid is >= 0, an icon will be drawn and the labels remaining space will be adjusted. + * + * if label is not null, it will be drawn with the specified alignment, fontsize and color. + * + * cbegin must be >= 0 and <= strlen(text) and denotes the beginning of the caret + * + * cend must be >= cbegin and <= strlen(text) and denotes the end of the caret if cend < cbegin, then no caret will be drawn + */ +public void bndIconLabelCaret(T=char) (NVGContext ctx, float x, float y, float w, float h, int iconid, NVGColor color, float fontsize, const(T)[] label, NVGColor caretcolor, int cbegin, int cend, NVGColor thinCaretColor=NVGColor.black) +if (isAnyCharType!T) +{ + float pleft = BND_TEXT_RADIUS; + if (label.length == 0) return; + if (iconid >= 0) { + bndIcon(ctx, x+4, y+2, iconid); + pleft += BND_ICON_SHEET_RES; + } + + if (bndFont < 0) return; + + x += pleft; + y += BND_WIDGET_HEIGHT-BND_TEXT_PAD_DOWN; + + ctx.fontFaceId(bndFont); + ctx.fontSize(fontsize); + ctx.textAlign(NVGTextAlign.H.Left, NVGTextAlign.V.Baseline); + + w -= BND_TEXT_RADIUS+pleft; + + if (cend >= cbegin) { + int c0r, c1r; + float c0x, c0y, c1x, c1y; + float desc, lh; + static NVGTextRow!T[BND_MAX_ROWS] rows; + auto rrows = ctx.textBreakLines(label[0..cend], w, rows[]); + ctx.textMetrics(null, &desc, &lh); + + bndCaretPosition(ctx, x, y, desc, lh, cbegin, rrows, &c0r, &c0x, &c0y); + bndCaretPosition(ctx, x, y, desc, lh, cend, rrows, &c1r, &c1x, &c1y); + + ctx.beginPath(); + if (cbegin == cend) { + //ctx.fillColor(nvgRGBf(0.337, 0.502, 0.761)); + ctx.fillColor(thinCaretColor); + //ctx.rect(c0x-1, c0y, 2, lh+1); + ctx.rect(c0x, c0y, 1, lh+1); + } else { + ctx.fillColor(caretcolor); + if (c0r == c1r) { + ctx.rect(c0x-1, c0y, c1x-c0x+1, lh+1); + } else { + int blk = c1r-c0r-1; + ctx.rect(c0x-1, c0y, x+w-c0x+1, lh+1); + ctx.rect(x, c1y, c1x-x+1, lh+1); + if (blk) ctx.rect(x, c0y+lh, w, blk*lh+1); + } + } + ctx.fill(); + } + + ctx.beginPath(); + ctx.fillColor(color); + ctx.textBox(x, y, w, label); +} + +/// Draw a checkmark for an option box with the given upper left coordinates (ox, oy) with the specified color. +public void bndCheck (NVGContext ctx, float ox, float oy, NVGColor color) { + ctx.beginPath(); + ctx.strokeWidth(2); + ctx.strokeColor(color); + ctx.lineCap(NVGLineCap.Butt); + ctx.lineJoin(NVGLineCap.Miter); + ctx.moveTo(ox+4, oy+5); + ctx.lineTo(ox+7, oy+8); + ctx.lineTo(ox+14, oy+1); + ctx.stroke(); +} + +/// Draw a checkmark for a radio with the given upper left coordinates (ox, oy) with the specified color. +public void bndRadioCheck (NVGContext ctx, float ox, float oy, NVGColor color) { + ctx.beginPath(); + ctx.fillColor(color); + ctx.circle(ox+7, oy+7, 3); + ctx.fill(); +} + +/// Draw a horizontal arrow for a number field with its center at (x, y) and size s; if s is negative, the arrow points to the left. +public void bndArrow (NVGContext ctx, float x, float y, float s, NVGColor color) { + ctx.beginPath(); + ctx.moveTo(x, y); + ctx.lineTo(x-s, y+s); + ctx.lineTo(x-s, y-s); + ctx.closePath(); + ctx.fillColor(color); + ctx.fill(); +} + +/// Draw an up/down arrow for a choice box with its center at (x, y) and size s +public void bndUpDownArrow (NVGContext ctx, float x, float y, float s, NVGColor color) { + float w; + ctx.beginPath(); + w = 1.1f*s; + ctx.moveTo(x, y-1); + ctx.lineTo(x+0.5*w, y-s-1); + ctx.lineTo(x+w, y-1); + ctx.closePath(); + ctx.moveTo(x, y+1); + ctx.lineTo(x+0.5*w, y+s+1); + ctx.lineTo(x+w, y+1); + ctx.closePath(); + ctx.fillColor(color); + ctx.fill(); +} + +/// Draw a node down-arrow with its tip at (x, y) and size s +public void bndNodeArrowDown (NVGContext ctx, float x, float y, float s, NVGColor color) { + float w; + ctx.beginPath(); + w = 1.0f*s; + ctx.moveTo(x, y); + ctx.lineTo(x+0.5*w, y-s); + ctx.lineTo(x-0.5*w, y-s); + ctx.closePath(); + ctx.fillColor(color); + ctx.fill(); +} + +/** computes the bounds of the scrollbar handle from the scrollbar size and the handles offset and size. + * + * offset is in the range 0..1 and defines the position of the scroll handle + * + * size is in the range 0..1 and defines the size of the scroll handle + */ +public void bndScrollHandleRect (float* x, float* y, float* w, float* h, float offset, float size) { + assert(w !is null); + assert(h !is null); + size = bndClamp(size, 0, 1); + offset = bndClamp(offset, 0, 1); + if (*h > *w) { + immutable float hs = bndMax(size*(*h), (*w)+1); + if (y !is null) *y = (*y)+((*h)-hs)*offset; + *h = hs; + } else { + immutable float ws = bndMax(size*(*w), (*h)-1); + if (x !is null) *x = (*x)+((*w)-ws)*offset; + *w = ws; + } +} + +/** computes the bounds of the scroll slider from the scrollbar size and the handles offset and size. + * + * offset is in the range 0..1 and defines the position of the scroll handle + * + * size is in the range 0..1 and defines the size of the scroll handle + */ +public void bndScrollSliderRect (float* w, float* h, float offset, float size) { + assert(w !is null); + assert(h !is null); + size = bndClamp(size, 0, 1); + offset = bndClamp(offset, 0, 1); + if (*h > *w) { + immutable float hs = bndMax(size*(*h), (*w)+1); + *h = ((*h)-hs)*offset+hs; + } else { + immutable float ws = bndMax(size*(*w), (*h)-1); + *w = ((*w)-ws)*offset+ws; + } +} + +/** return the color of a node wire based on state + * + * BND_HOVER indicates selected state, + * + * BND_ACTIVE indicates dragged state + */ +public NVGColor bndNodeWireColor (const(BNDnodeTheme)* theme, BNDwidgetState state) { + switch (state) { + default: + case BND_DEFAULT: return nvgRGBf(0.5f, 0.5f, 0.5f); + case BND_HOVER: return theme.wireSelectColor; + case BND_ACTIVE: return theme.activeNodeColor; + } +} diff --git a/nanovega.d b/nanovega.d new file mode 100644 index 0000000..8fa6765 --- /dev/null +++ b/nanovega.d @@ -0,0 +1,12380 @@ +// +// Copyright (c) 2013 Mikko Mononen memon@inside.org +// +// This software is provided 'as-is', without any express or implied +// warranty. In no event will the authors be held liable for any damages +// arising from the use of this software. +// Permission is granted to anyone to use this software for any purpose, +// including commercial applications, and to alter it and redistribute it +// freely, subject to the following restrictions: +// 1. The origin of this software must not be misrepresented; you must not +// claim that you wrote the original software. If you use this software +// in a product, an acknowledgment in the product documentation would be +// appreciated but is not required. +// 2. Altered source versions must be plainly marked as such, and must not be +// misrepresented as being the original software. +// 3. This notice may not be removed or altered from any source distribution. +// +// Fork developement, feature integration and new bugs: +// Ketmar // Invisible Vector +// Contains code from various contributors. +/** +The NanoVega API is modeled loosely on HTML5 canvas API. +If you know canvas, you're up to speed with NanoVega in no time. + + +Creating drawing context +======================== + +The drawing context is created using platform specific constructor function. + + --- + NVGContext vg = nvgCreateContext(); + --- + + +Drawing shapes with NanoVega +============================ + +Drawing a simple shape using NanoVega consists of four steps: +$(LIST + * begin a new shape, + * define the path to draw, + * set fill or stroke, + * and finally fill or stroke the path. +) + + --- + vg.beginPath(); + vg.rect(100, 100, 120, 30); + vg.fillColor(nvgRGBA(255, 192, 0, 255)); + vg.fill(); + --- + +Calling [beginPath] will clear any existing paths and start drawing from blank slate. +There are number of number of functions to define the path to draw, such as rectangle, +rounded rectangle and ellipse, or you can use the common moveTo, lineTo, bezierTo and +arcTo API to compose the paths step by step. + + +Understanding Composite Paths +============================= + +Because of the way the rendering backend is built in NanoVega, drawing a composite path, +that is path consisting from multiple paths defining holes and fills, is a bit more +involved. NanoVega uses non-zero filling rule and by default, and paths are wound in counter +clockwise order. Keep that in mind when drawing using the low level draw API. In order to +wind one of the predefined shapes as a hole, you should call `pathWinding(NVGSolidity.Hole)`, +or `pathWinding(NVGSolidity.Solid)` $(B after) defining the path. + + --- + vg.beginPath(); + vg.rect(100, 100, 120, 30); + vg.circle(120, 120, 5); + vg.pathWinding(NVGSolidity.Hole); // mark circle as a hole + vg.fillColor(nvgRGBA(255, 192, 0, 255)); + vg.fill(); + --- + + +Rendering is wrong, what to do? +=============================== + +$(LIST + * make sure you have created NanoVega context using [nvgCreateContext] call + * make sure you have initialised OpenGL with $(B stencil buffer) + * make sure you have cleared stencil buffer + * make sure all rendering calls happen between [beginFrame] and [endFrame] + * to enable more checks for OpenGL errors, add `NVGContextFlag.Debug` flag to [nvgCreateContext] +) + + +OpenGL state touched by the backend +=================================== + +The OpenGL back-end touches following states: + +When textures are uploaded or updated, the following pixel store is set to defaults: +`GL_UNPACK_ALIGNMENT`, `GL_UNPACK_ROW_LENGTH`, `GL_UNPACK_SKIP_PIXELS`, `GL_UNPACK_SKIP_ROWS`. +Texture binding is also affected. Texture updates can happen when the user loads images, +or when new font glyphs are added. Glyphs are added as needed between calls to [beginFrame] +and [endFrame]. + +The data for the whole frame is buffered and flushed in [endFrame]. +The following code illustrates the OpenGL state touched by the rendering code: + + --- + glUseProgram(prog); + glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); + glEnable(GL_CULL_FACE); + glCullFace(GL_BACK); + glFrontFace(GL_CCW); + glEnable(GL_BLEND); + glDisable(GL_DEPTH_TEST); + glDisable(GL_SCISSOR_TEST); + glColorMask(GL_TRUE, GL_TRUE, GL_TRUE, GL_TRUE); + glStencilMask(0xffffffff); + glStencilOp(GL_KEEP, GL_KEEP, GL_KEEP); + glStencilFunc(GL_ALWAYS, 0, 0xffffffff); + glActiveTexture(GL_TEXTURE0); + glBindBuffer(GL_UNIFORM_BUFFER, buf); + glBindVertexArray(arr); + glBindBuffer(GL_ARRAY_BUFFER, buf); + glBindTexture(GL_TEXTURE_2D, tex); + glUniformBlockBinding(... , GLNVG_FRAG_BINDING); + --- + + Symbol_groups: + + context_management = + ## Context Management + + Functions to create and destory NanoVega context. + + frame_management = + ## Frame Management + + To start drawing with NanoVega context, you have to "begin frame", and then + "end frame" to flush your rendering commands to GPU. + + composite_operation = + ## Composite Operation + + The composite operations in NanoVega are modeled after HTML Canvas API, and + the blend func is based on OpenGL (see corresponding manuals for more info). + The colors in the blending state have premultiplied alpha. + + color_utils = + ## Color Utils + + Colors in NanoVega are stored as ARGB. Zero alpha means "transparent color". + + matrices = + ## Matrices and Transformations + + The paths, gradients, patterns and scissor region are transformed by an transformation + matrix at the time when they are passed to the API. + The current transformation matrix is an affine matrix: + + ---------------------- + [sx kx tx] + [ky sy ty] + [ 0 0 1] + ---------------------- + + Where: (sx, sy) define scaling, (kx, ky) skewing, and (tx, ty) translation. + The last row is assumed to be (0, 0, 1) and is not stored. + + Apart from [resetTransform], each transformation function first creates + specific transformation matrix and pre-multiplies the current transformation by it. + + Current coordinate system (transformation) can be saved and restored using [save] and [restore]. + + The following functions can be used to make calculations on 2x3 transformation matrices. + A 2x3 matrix is represented as float[6]. + + state_handling = + ## State Handling + + NanoVega contains state which represents how paths will be rendered. + The state contains transform, fill and stroke styles, text and font styles, + and scissor clipping. + + render_styles = + ## Render Styles + + Fill and stroke render style can be either a solid color or a paint which is a gradient or a pattern. + Solid color is simply defined as a color value, different kinds of paints can be created + using [linearGradient], [boxGradient], [radialGradient] and [imagePattern]. + + Current render style can be saved and restored using [save] and [restore]. + + Note that if you want "almost perfect" pixel rendering, you should set aspect ratio to 1, + and use `integerCoord+0.5f` as pixel coordinates. + + render_transformations = + ## Render Transformations + + Transformation matrix management for the current rendering style. Transformations are applied in + backwards order. I.e. if you first translate, and then rotate, your path will be rotated around + it's origin, and then translated to the destination point. + + scissoring = + ## Scissoring + + Scissoring allows you to clip the rendering into a rectangle. This is useful for various + user interface cases like rendering a text edit or a timeline. + + images = + ## Images + + NanoVega allows you to load image files in various formats (if arsd loaders are in place) to be used for rendering. + In addition you can upload your own image. + The parameter imageFlags is combination of flags defined in NVGImageFlags. + + paints = + ## Paints + + NanoVega supports four types of paints: linear gradient, box gradient, radial gradient and image pattern. + These can be used as paints for strokes and fills. + + gpu_affine = + ## Render-Time Affine Transformations + + It is possible to set affine transformation matrix for GPU. That matrix will + be applied by the shader code. This can be used to quickly translate and rotate + saved paths. Call this $(B only) between [beginFrame] and [endFrame]. + + Note that [beginFrame] resets this matrix to identity one. + + WARNING! Don't use this for scaling or skewing, it will result in heavily distorted image! + + paths = + ## Paths + + Drawing a new shape starts with [beginPath], it clears all the currently defined paths. + Then you define one or more paths and sub-paths which describe the shape. The are functions + to draw common shapes like rectangles and circles, and lower level step-by-step functions, + which allow to define a path curve by curve. + + NanoVega uses even-odd fill rule to draw the shapes. Solid shapes should have counter clockwise + winding and holes should have counter clockwise order. To specify winding of a path you can + call [pathWinding]. This is useful especially for the common shapes, which are drawn CCW. + + Finally you can fill the path using current fill style by calling [fill], and stroke it + with current stroke style by calling [stroke]. + + The curve segments and sub-paths are transformed by the current transform. + + picking_api = + ## Picking API + + This is picking API that works directly on paths, without rasterizing them first. + + [beginFrame] resets picking state. Then you can create paths as usual, but + there is a possibility to perform hit checks $(B before) rasterizing a path. + Call either id assigning functions ([currFillHitId]/[currStrokeHitId]), or + immediate hit test functions ([hitTestCurrFill]/[hitTestCurrStroke]) + before rasterizing (i.e. calling [fill] or [stroke]) to perform hover + effects, for example. + + Also note that picking API is ignoring GPU affine transformation matrix. + You can "untransform" picking coordinates before checking with [gpuUntransformPoint]. + + text_api = + ## Text + + NanoVega allows you to load .ttf files and use the font to render text. + + The appearance of the text can be defined by setting the current text style + and by specifying the fill color. Common text and font settings such as + font size, letter spacing and text align are supported. Font blur allows you + to create simple text effects such as drop shadows. + + At render time the font face can be set based on the font handles or name. + + Font measure functions return values in local space, the calculations are + carried in the same resolution as the final rendering. This is done because + the text glyph positions are snapped to the nearest pixels sharp rendering. + + The local space means that values are not rotated or scale as per the current + transformation. For example if you set font size to 12, which would mean that + line height is 16, then regardless of the current scaling and rotation, the + returned line height is always 16. Some measures may vary because of the scaling + since aforementioned pixel snapping. + + While this may sound a little odd, the setup allows you to always render the + same way regardless of scaling. I.e. following works regardless of scaling: + + ---------------------- + string txt = "Text me up."; + vg.textBounds(x, y, txt, bounds); + vg.beginPath(); + vg.roundedRect(bounds[0], bounds[1], bounds[2]-bounds[0], bounds[3]-bounds[1]); + vg.fill(); + ---------------------- + + Note: currently only solid color fill is supported for text. + + path_recording = + ## Recording and Replaying Pathes + + $(B WARNING! This API is hightly experimental, and is subject to change. + While I will try to keep it compatible in future NanoVega + versions, no promises are made. Also note that NanoVega + rendering is quite fast, so you prolly don't need this + functionality. If you really want to render-once-and-copy, + consider rendering to FBO, and use imaging API to blit + FBO texture instead. Note that NanoVega supports alot of + blit/copy modes.) + + It is posible to record render commands and replay them later. This will allow + you to skip possible time-consuming tesselation stage. Potential uses of this + feature is, for example, rendering alot of similar complex paths, like game + tiles, or enemy sprites. + + Path replaying has some limitations, though: you cannot change stroke width, + fringe size, tesselation tolerance, or rescale path. But you can change fill + color/pattern, stroke color, translate and/or rotate saved paths. + + Note that text rendering commands are not saved, as technically text rendering + is not a path. + + To translate or rotate a record, use [affineGPU] API call. + + To record render commands, you must create new path set with [newPathSet] + function, then start recording with [startRecording]. You can cancel current + recording with [cancelRecording], or commit (save) recording with [stopRecording]. + + You can resume recording with [startRecording] after [stopRecording] call. + Calling [cancelRecording] will cancel only current recording session (i.e. it + will forget everything from the very latest [startRecording], not the whole + record). + + Finishing frame with [endFrame] will automatically commit current recording, and + calling [cancelFrame] will cancel recording by calling [cancelRecording]. + + Note that commit recording will clear current picking scene (but cancelling won't). + + Calling [startRecording] without commiting or cancelling recoriding will commit. + + $(B WARNING! Text output is not recorded now. Neither is scissor, so if you are using + scissoring or text in your pathes (UI, for example), things will not + work as you may expect.) + */ +module arsd.nanovega; +private: + +version(aliced) { + import iv.meta; + import iv.vfs; +} else { + private alias usize = size_t; + // i fear phobos! + private template Unqual(T) { + static if (is(T U == immutable U)) alias Unqual = U; + else static if (is(T U == shared inout const U)) alias Unqual = U; + else static if (is(T U == shared inout U)) alias Unqual = U; + else static if (is(T U == shared const U)) alias Unqual = U; + else static if (is(T U == shared U)) alias Unqual = U; + else static if (is(T U == inout const U)) alias Unqual = U; + else static if (is(T U == inout U)) alias Unqual = U; + else static if (is(T U == const U)) alias Unqual = U; + else alias Unqual = T; + } + private template isAnyCharType(T, bool unqual=false) { + static if (unqual) private alias UT = Unqual!T; else private alias UT = T; + enum isAnyCharType = is(UT == char) || is(UT == wchar) || is(UT == dchar); + } + private template isWideCharType(T, bool unqual=false) { + static if (unqual) private alias UT = Unqual!T; else private alias UT = T; + enum isWideCharType = is(UT == wchar) || is(UT == dchar); + } +} +version(nanovg_disable_vfs) { + enum NanoVegaHasIVVFS = false; +} else { + static if (is(typeof((){import iv.vfs;}))) { + enum NanoVegaHasIVVFS = true; + import iv.vfs; + } else { + enum NanoVegaHasIVVFS = false; + } +} + +// ////////////////////////////////////////////////////////////////////////// // +// engine +// ////////////////////////////////////////////////////////////////////////// // +import core.stdc.stdlib : malloc, realloc, free; +import core.stdc.string : memset, memcpy, strlen; +import std.math : PI; +//import arsd.fontstash; + +version(Posix) { + version = nanovg_use_freetype; +} else { + version = nanovg_disable_fontconfig; +} +version(aliced) { + version = nanovg_default_no_font_aa; + version = nanovg_builtin_fontconfig_bindings; + version = nanovg_builtin_freetype_bindings; + version = nanovg_builtin_opengl_bindings; // use `arsd.simpledisplay` to get basic bindings +} else { + version = nanovg_builtin_fontconfig_bindings; + version = nanovg_builtin_freetype_bindings; + version = nanovg_builtin_opengl_bindings; // use `arsd.simpledisplay` to get basic bindings +} + +version(nanovg_disable_fontconfig) { + public enum NanoVegaHasFontConfig = false; +} else { + public enum NanoVegaHasFontConfig = true; + version(nanovg_builtin_fontconfig_bindings) {} else import iv.fontconfig; +} + +//version = nanovg_bench_flatten; + +public: +alias NVG_PI = PI; + +enum NanoVegaHasArsdColor = (is(typeof((){ import arsd.color; }))); +enum NanoVegaHasArsdImage = (is(typeof((){ import arsd.color; import arsd.image; }))); + +static if (NanoVegaHasArsdColor) private import arsd.color; +static if (NanoVegaHasArsdImage) { + private import arsd.image; +} else { + void stbi_set_unpremultiply_on_load (int flag_true_if_should_unpremultiply) {} + void stbi_convert_iphone_png_to_rgb (int flag_true_if_should_convert) {} + ubyte* stbi_load (const(char)* filename, int* x, int* y, int* comp, int req_comp) { return null; } + ubyte* stbi_load_from_memory (const(void)* buffer, int len, int* x, int* y, int* comp, int req_comp) { return null; } + void stbi_image_free (void* retval_from_stbi_load) {} +} + +version(nanovg_default_no_font_aa) { + __gshared bool NVG_INVERT_FONT_AA = false; +} else { + __gshared bool NVG_INVERT_FONT_AA = true; +} + + +/// this is branchless for ints on x86, and even for longs on x86_64 +public ubyte nvgClampToByte(T) (T n) pure nothrow @safe @nogc if (__traits(isIntegral, T)) { + static if (__VERSION__ > 2067) pragma(inline, true); + static if (T.sizeof == 2 || T.sizeof == 4) { + static if (__traits(isUnsigned, T)) { + return cast(ubyte)(n&0xff|(255-((-cast(int)(n < 256))>>24))); + } else { + n &= -cast(int)(n >= 0); + return cast(ubyte)(n|((255-cast(int)n)>>31)); + } + } else static if (T.sizeof == 1) { + static assert(__traits(isUnsigned, T), "clampToByte: signed byte? no, really?"); + return cast(ubyte)n; + } else static if (T.sizeof == 8) { + static if (__traits(isUnsigned, T)) { + return cast(ubyte)(n&0xff|(255-((-cast(long)(n < 256))>>56))); + } else { + n &= -cast(long)(n >= 0); + return cast(ubyte)(n|((255-cast(long)n)>>63)); + } + } else { + static assert(false, "clampToByte: integer too big"); + } +} + + +/// NanoVega RGBA color +/// Group: color_utils +public align(1) struct NVGColor { +align(1): +public: + float[4] rgba = 0; /// default color is transparent (a=1 is opaque) + +public: + @property string toString () const @safe { import std.string : format; return "NVGColor(%s,%s,%s,%s)".format(r, g, b, a); } + +public: + enum transparent = NVGColor(0.0f, 0.0f, 0.0f, 0.0f); + enum k8orange = NVGColor(1.0f, 0.5f, 0.0f, 1.0f); + + enum aliceblue = NVGColor(240, 248, 255); + enum antiquewhite = NVGColor(250, 235, 215); + enum aqua = NVGColor(0, 255, 255); + enum aquamarine = NVGColor(127, 255, 212); + enum azure = NVGColor(240, 255, 255); + enum beige = NVGColor(245, 245, 220); + enum bisque = NVGColor(255, 228, 196); + enum black = NVGColor(0, 0, 0); // basic color + enum blanchedalmond = NVGColor(255, 235, 205); + enum blue = NVGColor(0, 0, 255); // basic color + enum blueviolet = NVGColor(138, 43, 226); + enum brown = NVGColor(165, 42, 42); + enum burlywood = NVGColor(222, 184, 135); + enum cadetblue = NVGColor(95, 158, 160); + enum chartreuse = NVGColor(127, 255, 0); + enum chocolate = NVGColor(210, 105, 30); + enum coral = NVGColor(255, 127, 80); + enum cornflowerblue = NVGColor(100, 149, 237); + enum cornsilk = NVGColor(255, 248, 220); + enum crimson = NVGColor(220, 20, 60); + enum cyan = NVGColor(0, 255, 255); // basic color + enum darkblue = NVGColor(0, 0, 139); + enum darkcyan = NVGColor(0, 139, 139); + enum darkgoldenrod = NVGColor(184, 134, 11); + enum darkgray = NVGColor(169, 169, 169); + enum darkgreen = NVGColor(0, 100, 0); + enum darkgrey = NVGColor(169, 169, 169); + enum darkkhaki = NVGColor(189, 183, 107); + enum darkmagenta = NVGColor(139, 0, 139); + enum darkolivegreen = NVGColor(85, 107, 47); + enum darkorange = NVGColor(255, 140, 0); + enum darkorchid = NVGColor(153, 50, 204); + enum darkred = NVGColor(139, 0, 0); + enum darksalmon = NVGColor(233, 150, 122); + enum darkseagreen = NVGColor(143, 188, 143); + enum darkslateblue = NVGColor(72, 61, 139); + enum darkslategray = NVGColor(47, 79, 79); + enum darkslategrey = NVGColor(47, 79, 79); + enum darkturquoise = NVGColor(0, 206, 209); + enum darkviolet = NVGColor(148, 0, 211); + enum deeppink = NVGColor(255, 20, 147); + enum deepskyblue = NVGColor(0, 191, 255); + enum dimgray = NVGColor(105, 105, 105); + enum dimgrey = NVGColor(105, 105, 105); + enum dodgerblue = NVGColor(30, 144, 255); + enum firebrick = NVGColor(178, 34, 34); + enum floralwhite = NVGColor(255, 250, 240); + enum forestgreen = NVGColor(34, 139, 34); + enum fuchsia = NVGColor(255, 0, 255); + enum gainsboro = NVGColor(220, 220, 220); + enum ghostwhite = NVGColor(248, 248, 255); + enum gold = NVGColor(255, 215, 0); + enum goldenrod = NVGColor(218, 165, 32); + enum gray = NVGColor(128, 128, 128); // basic color + enum green = NVGColor(0, 128, 0); // basic color + enum greenyellow = NVGColor(173, 255, 47); + enum grey = NVGColor(128, 128, 128); // basic color + enum honeydew = NVGColor(240, 255, 240); + enum hotpink = NVGColor(255, 105, 180); + enum indianred = NVGColor(205, 92, 92); + enum indigo = NVGColor(75, 0, 130); + enum ivory = NVGColor(255, 255, 240); + enum khaki = NVGColor(240, 230, 140); + enum lavender = NVGColor(230, 230, 250); + enum lavenderblush = NVGColor(255, 240, 245); + enum lawngreen = NVGColor(124, 252, 0); + enum lemonchiffon = NVGColor(255, 250, 205); + enum lightblue = NVGColor(173, 216, 230); + enum lightcoral = NVGColor(240, 128, 128); + enum lightcyan = NVGColor(224, 255, 255); + enum lightgoldenrodyellow = NVGColor(250, 250, 210); + enum lightgray = NVGColor(211, 211, 211); + enum lightgreen = NVGColor(144, 238, 144); + enum lightgrey = NVGColor(211, 211, 211); + enum lightpink = NVGColor(255, 182, 193); + enum lightsalmon = NVGColor(255, 160, 122); + enum lightseagreen = NVGColor(32, 178, 170); + enum lightskyblue = NVGColor(135, 206, 250); + enum lightslategray = NVGColor(119, 136, 153); + enum lightslategrey = NVGColor(119, 136, 153); + enum lightsteelblue = NVGColor(176, 196, 222); + enum lightyellow = NVGColor(255, 255, 224); + enum lime = NVGColor(0, 255, 0); + enum limegreen = NVGColor(50, 205, 50); + enum linen = NVGColor(250, 240, 230); + enum magenta = NVGColor(255, 0, 255); // basic color + enum maroon = NVGColor(128, 0, 0); + enum mediumaquamarine = NVGColor(102, 205, 170); + enum mediumblue = NVGColor(0, 0, 205); + enum mediumorchid = NVGColor(186, 85, 211); + enum mediumpurple = NVGColor(147, 112, 219); + enum mediumseagreen = NVGColor(60, 179, 113); + enum mediumslateblue = NVGColor(123, 104, 238); + enum mediumspringgreen = NVGColor(0, 250, 154); + enum mediumturquoise = NVGColor(72, 209, 204); + enum mediumvioletred = NVGColor(199, 21, 133); + enum midnightblue = NVGColor(25, 25, 112); + enum mintcream = NVGColor(245, 255, 250); + enum mistyrose = NVGColor(255, 228, 225); + enum moccasin = NVGColor(255, 228, 181); + enum navajowhite = NVGColor(255, 222, 173); + enum navy = NVGColor(0, 0, 128); + enum oldlace = NVGColor(253, 245, 230); + enum olive = NVGColor(128, 128, 0); + enum olivedrab = NVGColor(107, 142, 35); + enum orange = NVGColor(255, 165, 0); + enum orangered = NVGColor(255, 69, 0); + enum orchid = NVGColor(218, 112, 214); + enum palegoldenrod = NVGColor(238, 232, 170); + enum palegreen = NVGColor(152, 251, 152); + enum paleturquoise = NVGColor(175, 238, 238); + enum palevioletred = NVGColor(219, 112, 147); + enum papayawhip = NVGColor(255, 239, 213); + enum peachpuff = NVGColor(255, 218, 185); + enum peru = NVGColor(205, 133, 63); + enum pink = NVGColor(255, 192, 203); + enum plum = NVGColor(221, 160, 221); + enum powderblue = NVGColor(176, 224, 230); + enum purple = NVGColor(128, 0, 128); + enum red = NVGColor(255, 0, 0); // basic color + enum rosybrown = NVGColor(188, 143, 143); + enum royalblue = NVGColor(65, 105, 225); + enum saddlebrown = NVGColor(139, 69, 19); + enum salmon = NVGColor(250, 128, 114); + enum sandybrown = NVGColor(244, 164, 96); + enum seagreen = NVGColor(46, 139, 87); + enum seashell = NVGColor(255, 245, 238); + enum sienna = NVGColor(160, 82, 45); + enum silver = NVGColor(192, 192, 192); + enum skyblue = NVGColor(135, 206, 235); + enum slateblue = NVGColor(106, 90, 205); + enum slategray = NVGColor(112, 128, 144); + enum slategrey = NVGColor(112, 128, 144); + enum snow = NVGColor(255, 250, 250); + enum springgreen = NVGColor(0, 255, 127); + enum steelblue = NVGColor(70, 130, 180); + enum tan = NVGColor(210, 180, 140); + enum teal = NVGColor(0, 128, 128); + enum thistle = NVGColor(216, 191, 216); + enum tomato = NVGColor(255, 99, 71); + enum turquoise = NVGColor(64, 224, 208); + enum violet = NVGColor(238, 130, 238); + enum wheat = NVGColor(245, 222, 179); + enum white = NVGColor(255, 255, 255); // basic color + enum whitesmoke = NVGColor(245, 245, 245); + enum yellow = NVGColor(255, 255, 0); // basic color + enum yellowgreen = NVGColor(154, 205, 50); + +nothrow @safe @nogc: +public: + /// + this (int ar, int ag, int ab, int aa=255) pure { + pragma(inline, true); + r = nvgClampToByte(ar)/255.0f; + g = nvgClampToByte(ag)/255.0f; + b = nvgClampToByte(ab)/255.0f; + a = nvgClampToByte(aa)/255.0f; + } + + /// + this (float ar, float ag, float ab, float aa=1.0f) pure { + pragma(inline, true); + r = ar; + g = ag; + b = ab; + a = aa; + } + + /// AABBGGRR (same format as little-endian RGBA image, coincidentally, the same as arsd.color) + this (uint c) pure { + pragma(inline, true); + r = (c&0xff)/255.0f; + g = ((c>>8)&0xff)/255.0f; + b = ((c>>16)&0xff)/255.0f; + a = ((c>>24)&0xff)/255.0f; + } + + /// Supports: "#rgb", "#rrggbb", "#argb", "#aarrggbb" + this (const(char)[] srgb) { + static int c2d (char ch) pure nothrow @safe @nogc { + pragma(inline, true); + return + ch >= '0' && ch <= '9' ? ch-'0' : + ch >= 'A' && ch <= 'F' ? ch-'A'+10 : + ch >= 'a' && ch <= 'f' ? ch-'a'+10 : + -1; + } + int[8] digs; + int dc = -1; + foreach (immutable char ch; srgb) { + if (ch <= ' ') continue; + if (ch == '#') { + if (dc != -1) { dc = -1; break; } + dc = 0; + } else { + if (dc >= digs.length) { dc = -1; break; } + if ((digs[dc++] = c2d(ch)) < 0) { dc = -1; break; } + } + } + switch (dc) { + case 3: // rgb + a = 1.0f; + r = digs[0]/15.0f; + g = digs[1]/15.0f; + b = digs[2]/15.0f; + break; + case 4: // argb + a = digs[0]/15.0f; + r = digs[1]/15.0f; + g = digs[2]/15.0f; + b = digs[3]/15.0f; + break; + case 6: // rrggbb + a = 1.0f; + r = (digs[0]*16+digs[1])/255.0f; + g = (digs[2]*16+digs[3])/255.0f; + b = (digs[4]*16+digs[5])/255.0f; + break; + case 8: // aarrggbb + a = (digs[0]*16+digs[1])/255.0f; + r = (digs[2]*16+digs[3])/255.0f; + g = (digs[4]*16+digs[5])/255.0f; + b = (digs[6]*16+digs[7])/255.0f; + break; + default: + break; + } + } + + /// Is this color completely opaque? + @property bool isOpaque () const pure nothrow @trusted @nogc { pragma(inline, true); return (rgba.ptr[3] >= 1.0f); } + /// Is this color completely transparent? + @property bool isTransparent () const pure nothrow @trusted @nogc { pragma(inline, true); return (rgba.ptr[3] <= 0.0f); } + + /// AABBGGRR (same format as little-endian RGBA image, coincidentally, the same as arsd.color) + @property uint asUint () const pure { + pragma(inline, true); + return + cast(uint)(r*255)| + (cast(uint)(g*255)<<8)| + (cast(uint)(b*255)<<16)| + (cast(uint)(a*255)<<24); + } + + alias asUintABGR = asUint; /// Ditto. + + /// AABBGGRR (same format as little-endian RGBA image, coincidentally, the same as arsd.color) + static NVGColor fromUint (uint c) pure { pragma(inline, true); return NVGColor(c); } + + alias fromUintABGR = fromUint; /// Ditto. + + /// AARRGGBB + @property uint asUintARGB () const pure { + pragma(inline, true); + return + cast(uint)(b*255)| + (cast(uint)(g*255)<<8)| + (cast(uint)(r*255)<<16)| + (cast(uint)(a*255)<<24); + } + + /// AARRGGBB + static NVGColor fromUintARGB (uint c) pure { pragma(inline, true); return NVGColor((c>>16)&0xff, (c>>8)&0xff, c&0xff, (c>>24)&0xff); } + + @property ref inout(float) r () inout pure @trusted { pragma(inline, true); return rgba.ptr[0]; } /// + @property ref inout(float) g () inout pure @trusted { pragma(inline, true); return rgba.ptr[1]; } /// + @property ref inout(float) b () inout pure @trusted { pragma(inline, true); return rgba.ptr[2]; } /// + @property ref inout(float) a () inout pure @trusted { pragma(inline, true); return rgba.ptr[3]; } /// + + ref NVGColor applyTint() (in auto ref NVGColor tint) nothrow @trusted @nogc { + if (tint.a == 0) return this; + foreach (immutable idx, ref float v; rgba[0..4]) { + v = nvg__clamp(v*tint.rgba.ptr[idx], 0.0f, 1.0f); + } + return this; + } + + NVGHSL asHSL() (bool useWeightedLightness=false) const { pragma(inline, true); return NVGHSL.fromColor(this, useWeightedLightness); } /// + static fromHSL() (in auto ref NVGHSL hsl) { pragma(inline, true); return hsl.asColor; } /// + + static if (NanoVegaHasArsdColor) { + Color toArsd () const { pragma(inline, true); return Color(cast(int)(r*255), cast(int)(g*255), cast(int)(b*255), cast(int)(a*255)); } /// + static NVGColor fromArsd (in Color c) { pragma(inline, true); return NVGColor(c.r, c.g, c.b, c.a); } /// + /// + this (in Color c) { + pragma(inline, true); + r = c.r/255.0f; + g = c.g/255.0f; + b = c.b/255.0f; + a = c.a/255.0f; + } + } +} + + +/// NanoVega A-HSL color +/// Group: color_utils +public align(1) struct NVGHSL { +align(1): + float h=0, s=0, l=1, a=1; /// + + string toString () const { import std.format : format; return (a != 1 ? "HSL(%s,%s,%s,%d)".format(h, s, l, a) : "HSL(%s,%s,%s)".format(h, s, l)); } + +nothrow @safe @nogc: +public: + /// + this (float ah, float as, float al, float aa=1) pure { pragma(inline, true); h = ah; s = as; l = al; a = aa; } + + NVGColor asColor () const { pragma(inline, true); return nvgHSLA(h, s, l, a); } /// + + // taken from Adam's arsd.color + /** Converts an RGB color into an HSL triplet. + * [useWeightedLightness] will try to get a better value for luminosity for the human eye, + * which is more sensitive to green than red and more to red than blue. + * If it is false, it just does average of the rgb. */ + static NVGHSL fromColor() (in auto ref NVGColor c, bool useWeightedLightness=false) pure { + NVGHSL res; + res.a = c.a; + float r1 = c.r; + float g1 = c.g; + float b1 = c.b; + + float maxColor = r1; + if (g1 > maxColor) maxColor = g1; + if (b1 > maxColor) maxColor = b1; + float minColor = r1; + if (g1 < minColor) minColor = g1; + if (b1 < minColor) minColor = b1; + + res.l = (maxColor+minColor)/2; + if (useWeightedLightness) { + // the colors don't affect the eye equally + // this is a little more accurate than plain HSL numbers + res.l = 0.2126*r1+0.7152*g1+0.0722*b1; + } + if (maxColor != minColor) { + if (res.l < 0.5) { + res.s = (maxColor-minColor)/(maxColor+minColor); + } else { + res.s = (maxColor-minColor)/(2.0-maxColor-minColor); + } + if (r1 == maxColor) { + res.h = (g1-b1)/(maxColor-minColor); + } else if(g1 == maxColor) { + res.h = 2.0+(b1-r1)/(maxColor-minColor); + } else { + res.h = 4.0+(r1-g1)/(maxColor-minColor); + } + } + + res.h = res.h*60; + if (res.h < 0) res.h += 360; + res.h /= 360; + + return res; + } +} + + +/// Paint parameters for various fills. Don't change anything here! +/// Group: render_styles +public struct NVGPaint { + NVGMatrix xform; + float[2] extent; + float radius; + float feather; + NVGColor innerColor; /// this can be used to modulate images (fill/font) + NVGColor outerColor; + int image; +} + +/// Path winding. +/// Group: paths +public enum NVGWinding { + CCW = 1, /// Winding for solid shapes + CW = 2, /// Winding for holes +} + +/// Path solidity. +/// Group: paths +public enum NVGSolidity { + Solid = 1, /// Solid shape (CCW winding). + Hole = 2, /// Hole (CW winding). +} + +/// Line cap style. +/// Group: render_styles +public enum NVGLineCap { + Butt, /// + Round, /// + Square, /// + Bevel, /// + Miter, /// +} + +/// Text align. +/// Group: text_api +public align(1) struct NVGTextAlign { +align(1): + /// Horizontal align. + enum H : ubyte { + Left = 0, /// Default, align text horizontally to left. + Center = 1, /// Align text horizontally to center. + Right = 2, /// Align text horizontally to right. + } + + /// Vertical align. + enum V : ubyte { + Baseline = 0, /// Default, align text vertically to baseline. + Top = 1, /// Align text vertically to top. + Middle = 2, /// Align text vertically to middle. + Bottom = 3, /// Align text vertically to bottom. + } + +pure nothrow @safe @nogc: +public: + this (H h) { pragma(inline, true); value = h; } /// + this (V v) { pragma(inline, true); value = cast(ubyte)(v<<4); } /// + this (H h, V v) { pragma(inline, true); value = cast(ubyte)(h|(v<<4)); } /// + this (V v, H h) { pragma(inline, true); value = cast(ubyte)(h|(v<<4)); } /// + void reset () { pragma(inline, true); value = 0; } /// + void reset (H h, V v) { pragma(inline, true); value = cast(ubyte)(h|(v<<4)); } /// + void reset (V v, H h) { pragma(inline, true); value = cast(ubyte)(h|(v<<4)); } /// +@property: + bool left () const { pragma(inline, true); return ((value&0x0f) == H.Left); } /// + void left (bool v) { pragma(inline, true); value = cast(ubyte)((value&0xf0)|(v ? H.Left : 0)); } /// + bool center () const { pragma(inline, true); return ((value&0x0f) == H.Center); } /// + void center (bool v) { pragma(inline, true); value = cast(ubyte)((value&0xf0)|(v ? H.Center : 0)); } /// + bool right () const { pragma(inline, true); return ((value&0x0f) == H.Right); } /// + void right (bool v) { pragma(inline, true); value = cast(ubyte)((value&0xf0)|(v ? H.Right : 0)); } /// + // + bool baseline () const { pragma(inline, true); return (((value>>4)&0x0f) == V.Baseline); } /// + void baseline (bool v) { pragma(inline, true); value = cast(ubyte)((value&0x0f)|(v ? V.Baseline<<4 : 0)); } /// + bool top () const { pragma(inline, true); return (((value>>4)&0x0f) == V.Top); } /// + void top (bool v) { pragma(inline, true); value = cast(ubyte)((value&0x0f)|(v ? V.Top<<4 : 0)); } /// + bool middle () const { pragma(inline, true); return (((value>>4)&0x0f) == V.Middle); } /// + void middle (bool v) { pragma(inline, true); value = cast(ubyte)((value&0x0f)|(v ? V.Middle<<4 : 0)); } /// + bool bottom () const { pragma(inline, true); return (((value>>4)&0x0f) == V.Bottom); } /// + void bottom (bool v) { pragma(inline, true); value = cast(ubyte)((value&0x0f)|(v ? V.Bottom<<4 : 0)); } /// + // + H horizontal () const { pragma(inline, true); return cast(H)(value&0x0f); } /// + void horizontal (H v) { pragma(inline, true); value = (value&0xf0)|v; } /// + // + V vertical () const { pragma(inline, true); return cast(V)((value>>4)&0x0f); } /// + void vertical (V v) { pragma(inline, true); value = (value&0x0f)|cast(ubyte)(v<<4); } /// + // +private: + ubyte value = 0; // low nibble: horizontal; high nibble: vertical +} + +/// Blending type. +/// Group: composite_operation +public enum NVGBlendFactor { + Zero = 1<<0, /// + One = 1<<1, /// + SrcColor = 1<<2, /// + OneMinusSrcColor = 1<<3, /// + DstColor = 1<<4, /// + OneMinusDstColor = 1<<5, /// + SrcAlpha = 1<<6, /// + OneMinusSrcAlpha = 1<<7, /// + DstAlpha = 1<<8, /// + OneMinusDstAlpha = 1<<9, /// + SrcAlphaSaturate = 1<<10, /// +} + +/// Composite operation (HTML5-alike). +/// Group: composite_operation +public enum NVGCompositeOperation { + SourceOver, /// + SourceIn, /// + SourceOut, /// + SourceAtop, /// + DestinationOver, /// + DestinationIn, /// + DestinationOut, /// + DestinationAtop, /// + Lighter, /// + Copy, /// + Xor, /// +} + +/// Composite operation state. +/// Group: composite_operation +public struct NVGCompositeOperationState { + NVGBlendFactor srcRGB; /// + NVGBlendFactor dstRGB; /// + NVGBlendFactor srcAlpha; /// + NVGBlendFactor dstAlpha; /// +} + +/// Glyph position info. +/// Group: text_api +public struct NVGGlyphPosition { + usize strpos; /// Position of the glyph in the input string. + float x; /// The x-coordinate of the logical glyph position. + float minx, maxx; /// The bounds of the glyph shape. +} + +/// Text row storage. +/// Group: text_api +public struct NVGTextRow(CT) if (isAnyCharType!CT) { + alias CharType = CT; + const(CT)[] s; + int start; /// Index in the input text where the row starts. + int end; /// Index in the input text where the row ends (one past the last character). + float width; /// Logical width of the row. + float minx, maxx; /// Actual bounds of the row. Logical with and bounds can differ because of kerning and some parts over extending. + /// Get rest of the string. + @property const(CT)[] rest () const pure nothrow @trusted @nogc { pragma(inline, true); return (end <= s.length ? s[end..$] : null); } + /// Get current row. + @property const(CT)[] row () const pure nothrow @trusted @nogc { pragma(inline, true); return s[start..end]; } + @property const(CT)[] string () const pure nothrow @trusted @nogc { pragma(inline, true); return s; } + @property void string(CT) (const(CT)[] v) pure nothrow @trusted @nogc { pragma(inline, true); s = v; } +} + +/// Image creation flags. +/// Group: images +public enum NVGImageFlags : uint { + None = 0, /// Nothing special. + GenerateMipmaps = 1<<0, /// Generate mipmaps during creation of the image. + RepeatX = 1<<1, /// Repeat image in X direction. + RepeatY = 1<<2, /// Repeat image in Y direction. + FlipY = 1<<3, /// Flips (inverses) image in Y direction when rendered. + Premultiplied = 1<<4, /// Image data has premultiplied alpha. + NoFiltering = 1<<8, /// use GL_NEAREST instead of GL_LINEAR + Nearest = NoFiltering, /// compatibility with original NanoVG +} + + +// ////////////////////////////////////////////////////////////////////////// // +package/*(arsd)*/: + +static T* xdup(T) (const(T)* ptr, int count) nothrow @trusted @nogc { + import core.stdc.stdlib : malloc; + import core.stdc.string : memcpy; + if (count == 0) return null; + T* res = cast(T*)malloc(T.sizeof*count); + if (res is null) assert(0, "NanoVega: out of memory"); + memcpy(res, ptr, T.sizeof*count); + return res; +} + +// Internal Render API +enum NVGtexture { + Alpha = 0x01, + RGBA = 0x02, +} + +struct NVGscissor { + NVGMatrix xform; + float[2] extent; +} + +struct NVGvertex { + float x, y, u, v; +} + +struct NVGpath { + int first; + int count; + bool closed; + int nbevel; + NVGvertex* fill; + int nfill; + NVGvertex* stroke; + int nstroke; + NVGWinding winding; + int convex; + bool cloned; + + @disable this (this); // no copies + + void clear () nothrow @trusted @nogc { + import core.stdc.stdlib : free; + if (cloned) { + if (stroke !is null && stroke !is fill) free(stroke); + if (fill !is null) free(fill); + } + this = this.init; + } + + // won't clear current path + void copyFrom (const NVGpath* src) nothrow @trusted @nogc { + import core.stdc.string : memcpy; + assert(src !is null); + memcpy(&this, src, NVGpath.sizeof); + this.fill = xdup(src.fill, src.nfill); + if (src.stroke is src.fill) { + this.stroke = this.fill; + } else { + this.stroke = xdup(src.stroke, src.nstroke); + } + this.cloned = true; + } +} + + +struct NVGparams { + void* userPtr; + bool edgeAntiAlias; + bool fontAA; + bool function (void* uptr) nothrow @trusted @nogc renderCreate; + int function (void* uptr, NVGtexture type, int w, int h, int imageFlags, const(ubyte)* data) nothrow @trusted @nogc renderCreateTexture; + bool function (void* uptr, int image) nothrow @trusted @nogc renderTextureIncRef; + bool function (void* uptr, int image) nothrow @trusted @nogc renderDeleteTexture; + bool function (void* uptr, int image, int x, int y, int w, int h, const(ubyte)* data) nothrow @trusted @nogc renderUpdateTexture; + bool function (void* uptr, int image, int* w, int* h) nothrow @trusted @nogc renderGetTextureSize; + void function (void* uptr, int width, int height) nothrow @trusted @nogc renderViewport; + void function (void* uptr) nothrow @trusted @nogc renderCancel; + void function (void* uptr, NVGCompositeOperationState compositeOperation) nothrow @trusted @nogc renderFlush; + void function (void* uptr, NVGPaint* paint, NVGscissor* scissor, float fringe, const(float)* bounds, const(NVGpath)* paths, int npaths, bool evenOdd) nothrow @trusted @nogc renderFill; + void function (void* uptr, NVGPaint* paint, NVGscissor* scissor, float fringe, float strokeWidth, const(NVGpath)* paths, int npaths) nothrow @trusted @nogc renderStroke; + void function (void* uptr, NVGPaint* paint, NVGscissor* scissor, const(NVGvertex)* verts, int nverts) nothrow @trusted @nogc renderTriangles; + void function (void* uptr, const(float)[] mat...) nothrow @trusted @nogc renderSetAffine; + void function (void* uptr) nothrow @trusted @nogc renderDelete; +} + +// ////////////////////////////////////////////////////////////////////////// // +private: + +enum NVG_INIT_FONTIMAGE_SIZE = 512; +enum NVG_MAX_FONTIMAGE_SIZE = 2048; +enum NVG_MAX_FONTIMAGES = 4; + +enum NVG_INIT_COMMANDS_SIZE = 256; +enum NVG_INIT_POINTS_SIZE = 128; +enum NVG_INIT_PATHS_SIZE = 16; +enum NVG_INIT_VERTS_SIZE = 256; +enum NVG_MAX_STATES = 32; + +enum NVG_KAPPA90 = 0.5522847493f; // Length proportional to radius of a cubic bezier handle for 90deg arcs. +enum NVG_MIN_FEATHER = 0.001f; // it should be greater than zero, 'cause it is used in shader for divisions + +enum Command { + MoveTo = 0, + LineTo = 1, + BezierTo = 2, + Close = 3, + Winding = 4, +} + +enum PointFlag : int { + Corner = 0x01, + Left = 0x02, + Bevel = 0x04, + InnerBevelPR = 0x08, +} + +struct NVGstate { + NVGCompositeOperationState compositeOperation; + bool shapeAntiAlias; + NVGPaint fill; + NVGPaint stroke; + float strokeWidth; + float miterLimit; + NVGLineCap lineJoin; + NVGLineCap lineCap; + float alpha; + NVGMatrix xform; + NVGscissor scissor; + float fontSize; + float letterSpacing; + float lineHeight; + float fontBlur; + NVGTextAlign textAlign; + int fontId; + bool evenOddMode; // use even-odd filling rule (required for some svgs); otherwise use non-zero fill +} + +struct NVGpoint { + float x, y; + float dx, dy; + float len; + float dmx, dmy; + ubyte flags; +} + +struct NVGpathCache { + NVGpoint* points; + int npoints; + int cpoints; + NVGpath* paths; + int npaths; + int cpaths; + NVGvertex* verts; + int nverts; + int cverts; + float[4] bounds; + // this is required for saved paths + bool strokeReady; + bool fillReady; + float strokeAlphaMul; + float strokeWidth; + float fringeWidth; + bool evenOddMode; + // non-saved path will not have this + float* commands; + int ncommands; + + @disable this (this); // no copies + + // won't clear current path + void copyFrom (const NVGpathCache* src) nothrow @trusted @nogc { + import core.stdc.stdlib : malloc; + import core.stdc.string : memcpy, memset; + assert(src !is null); + memcpy(&this, src, NVGpathCache.sizeof); + this.points = xdup(src.points, src.npoints); + this.cpoints = src.npoints; + this.verts = xdup(src.verts, src.nverts); + this.cverts = src.nverts; + this.commands = xdup(src.commands, src.ncommands); + if (src.npaths > 0) { + this.paths = cast(NVGpath*)malloc(src.npaths*NVGpath.sizeof); + memset(this.paths, 0, npaths*NVGpath.sizeof); + foreach (immutable pidx; 0..npaths) this.paths[pidx].copyFrom(&src.paths[pidx]); + } else { + this.npaths = this.cpaths = 0; + } + } + + void clear () nothrow @trusted @nogc { + import core.stdc.stdlib : free; + if (paths !is null) { + foreach (ref p; paths[0..npaths]) p.clear(); + free(paths); + } + if (points !is null) free(points); + if (verts !is null) free(verts); + if (commands !is null) free(commands); + this = this.init; + } +} + +/// Pointer to opaque NanoVega context structure. +/// Group: context_management +public alias NVGContext = NVGcontextinternal*; + +// Returns FontStash context of the given NanoVega context. +public FONScontext* fonsContext (NVGContext ctx) { return (ctx !is null ? ctx.fs : null); } + +/** De Casteljau Bezier tesselator is faster for small number of bezier curves. + * But if your path has alot of degenerate curves (i.e. straight lines), then + * AFD tesselator is MUCH faster. */ +public enum NVGTesselation { + Combined, /// default: this will use AFD tesselator for degenerate Beziers, and DeCasteljau for normal beziers + DeCasteljau, /// standard well-known tesselation algorithm + AFD, /// adaptive forward differencing +} + +/// Default tesselator for Bezier curves. +public __gshared NVGTesselation NVG_DEFAULT_TESSELATOR = NVGTesselation.Combined; + + +// some public info + +/// valid only inside [beginFrame]/[endFrame] +/// Group: context_management +public int width (NVGContext ctx) pure nothrow @trusted @nogc { pragma(inline, true); return (ctx !is null ? ctx.mWidth : 0); } + +/// valid only inside [beginFrame]/[endFrame] +/// Group: context_management +public int height (NVGContext ctx) pure nothrow @trusted @nogc { pragma(inline, true); return (ctx !is null ? ctx.mHeight : 0); } + +/// 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); } + +// path autoregistration + +/// [pickid] to stop autoregistration. +/// Group: context_management +public enum NVGNoPick = -1; + +/// >=0: this pickid will be assigned to all filled/stroked paths +/// Group: context_management +public int pickid (NVGContext ctx) pure nothrow @trusted @nogc { pragma(inline, true); return (ctx !is null ? ctx.pathPickId : NVGNoPick); } + +/// >=0: this pickid will be assigned to all filled/stroked paths +/// Group: context_management +public void pickid (NVGContext ctx, int v) nothrow @trusted @nogc { pragma(inline, true); if (ctx !is null) ctx.pathPickId = v; } + +/// pick autoregistration mode; see [NVGPickKind] +/// Group: context_management +public uint pickmode (NVGContext ctx) pure nothrow @trusted @nogc { pragma(inline, true); return (ctx !is null ? ctx.pathPickRegistered&NVGPickKind.All : 0); } + +/// pick autoregistration mode; see [NVGPickKind] +/// Group: context_management +public void pickmode (NVGContext ctx, uint v) nothrow @trusted @nogc { pragma(inline, true); if (ctx !is null) ctx.pathPickRegistered = (ctx.pathPickRegistered&0xffff_0000u)|(v&NVGPickKind.All); } + +// tesselator options + +/// +/// Group: context_management +public NVGTesselation tesselation (NVGContext ctx) pure nothrow @trusted @nogc { pragma(inline, true); return (ctx !is null ? ctx.tesselatortype : NVGTesselation.Combined); } + +/// +/// Group: context_management +public void tesselation (NVGContext ctx, NVGTesselation v) nothrow @trusted @nogc { pragma(inline, true); if (ctx !is null) ctx.tesselatortype = v; } + + +private struct NVGcontextinternal { +private: + NVGparams params; + float* commands; + int ccommands; + int ncommands; + float commandx, commandy; + NVGstate[NVG_MAX_STATES] states; + int nstates; + NVGpathCache* cache; + float tessTol; + float distTol; + float fringeWidth; + float devicePxRatio; + FONScontext* fs; // this is public, so i can use it in text layouter, for example; WARNING: DON'T MODIFY! + int[NVG_MAX_FONTIMAGES] fontImages; + int fontImageIdx; + int drawCallCount; + int fillTriCount; + int strokeTriCount; + int textTriCount; + NVGTesselation tesselatortype; + // picking API + NVGpickScene* pickScene; + int pathPickId; // >=0: register all paths for picking using this id + uint pathPickRegistered; // if [pathPickId] >= 0, this is used to avoid double-registration (see [NVGPickKind]); hi 16 bit is check flags, lo 16 bit is mode + // path recording + NVGPathSet recset; + int recstart; // used to cancel recording + bool recblockdraw; + // internals + NVGMatrix gpuAffine; + int mWidth, mHeight; + float mDeviceRatio; + void delegate (NVGContext ctx) nothrow @trusted @nogc cleanup; + + @disable this (this); // no copies +} + +public import core.stdc.math : + nvg__sqrtf = sqrtf, + nvg__modf = fmodf, + nvg__sinf = sinf, + nvg__cosf = cosf, + nvg__tanf = tanf, + nvg__atan2f = atan2f, + nvg__acosf = acosf, + nvg__ceilf = ceilf; + +version(Windows) { + public int nvg__lrintf (float f) nothrow @trusted @nogc { pragma(inline, true); return cast(int)(f+0.5); } +} else { + public import core.stdc.math : nvg__lrintf = lrintf; +} + +public auto nvg__min(T) (T a, T b) { pragma(inline, true); return (a < b ? a : b); } +public auto nvg__max(T) (T a, T b) { pragma(inline, true); return (a > b ? a : b); } +public auto nvg__clamp(T) (T a, T mn, T mx) { pragma(inline, true); return (a < mn ? mn : (a > mx ? mx : a)); } +//float nvg__absf() (float a) { pragma(inline, true); return (a >= 0.0f ? a : -a); } +public auto nvg__sign(T) (T a) { pragma(inline, true); return (a >= cast(T)0 ? cast(T)1 : cast(T)(-1)); } +public float nvg__cross() (float dx0, float dy0, float dx1, float dy1) { pragma(inline, true); return (dx1*dy0-dx0*dy1); } + +public import core.stdc.math : nvg__absf = fabsf; + + +float nvg__normalize (float* x, float* y) nothrow @safe @nogc { + float d = nvg__sqrtf((*x)*(*x)+(*y)*(*y)); + if (d > 1e-6f) { + immutable float id = 1.0f/d; + *x *= id; + *y *= id; + } + return d; +} + +void nvg__deletePathCache (ref NVGpathCache* c) nothrow @trusted @nogc { + if (c !is null) { + c.clear(); + free(c); + } +} + +NVGpathCache* nvg__allocPathCache () nothrow @trusted @nogc { + NVGpathCache* c = cast(NVGpathCache*)malloc(NVGpathCache.sizeof); + if (c is null) goto error; + memset(c, 0, NVGpathCache.sizeof); + + c.points = cast(NVGpoint*)malloc(NVGpoint.sizeof*NVG_INIT_POINTS_SIZE); + if (c.points is null) goto error; + assert(c.npoints == 0); + c.cpoints = NVG_INIT_POINTS_SIZE; + + c.paths = cast(NVGpath*)malloc(NVGpath.sizeof*NVG_INIT_PATHS_SIZE); + if (c.paths is null) goto error; + assert(c.npaths == 0); + c.cpaths = NVG_INIT_PATHS_SIZE; + + c.verts = cast(NVGvertex*)malloc(NVGvertex.sizeof*NVG_INIT_VERTS_SIZE); + if (c.verts is null) goto error; + assert(c.nverts == 0); + c.cverts = NVG_INIT_VERTS_SIZE; + + return c; + +error: + nvg__deletePathCache(c); + return null; +} + +void nvg__setDevicePixelRatio (NVGContext ctx, float ratio) pure nothrow @safe @nogc { + ctx.tessTol = 0.25f/ratio; + ctx.distTol = 0.01f/ratio; + ctx.fringeWidth = 1.0f/ratio; + ctx.devicePxRatio = ratio; +} + +NVGCompositeOperationState nvg__compositeOperationState (NVGCompositeOperation op) pure nothrow @safe @nogc { + NVGBlendFactor sfactor, dfactor; + if (op == NVGCompositeOperation.SourceOver) { sfactor = NVGBlendFactor.One; dfactor = NVGBlendFactor.OneMinusSrcAlpha;} + else if (op == NVGCompositeOperation.SourceIn) { sfactor = NVGBlendFactor.DstAlpha; dfactor = NVGBlendFactor.Zero; } + else if (op == NVGCompositeOperation.SourceOut) { sfactor = NVGBlendFactor.OneMinusDstAlpha; dfactor = NVGBlendFactor.Zero; } + else if (op == NVGCompositeOperation.SourceAtop) { sfactor = NVGBlendFactor.DstAlpha; dfactor = NVGBlendFactor.OneMinusSrcAlpha; } + else if (op == NVGCompositeOperation.DestinationOver) { sfactor = NVGBlendFactor.OneMinusDstAlpha; dfactor = NVGBlendFactor.One; } + else if (op == NVGCompositeOperation.DestinationIn) { sfactor = NVGBlendFactor.Zero; dfactor = NVGBlendFactor.SrcAlpha; } + else if (op == NVGCompositeOperation.DestinationOut) { sfactor = NVGBlendFactor.Zero; dfactor = NVGBlendFactor.OneMinusSrcAlpha; } + else if (op == NVGCompositeOperation.DestinationAtop) { sfactor = NVGBlendFactor.OneMinusDstAlpha; dfactor = NVGBlendFactor.SrcAlpha; } + else if (op == NVGCompositeOperation.Lighter) { sfactor = NVGBlendFactor.One; dfactor = NVGBlendFactor.One; } + else if (op == NVGCompositeOperation.Copy) { sfactor = NVGBlendFactor.One; dfactor = NVGBlendFactor.Zero; } + else if (op == NVGCompositeOperation.Xor) { sfactor = NVGBlendFactor.OneMinusDstAlpha; dfactor = NVGBlendFactor.OneMinusSrcAlpha; } + else { sfactor = NVGBlendFactor.One; dfactor = NVGBlendFactor.OneMinusSrcAlpha;} // default value for invalid op: SourceOver + + NVGCompositeOperationState state; + state.srcRGB = sfactor; + state.dstRGB = dfactor; + state.srcAlpha = sfactor; + state.dstAlpha = dfactor; + return state; +} + +NVGstate* nvg__getState (NVGContext ctx) pure nothrow @trusted @nogc { + pragma(inline, true); + return &ctx.states.ptr[ctx.nstates-1]; +} + +// Constructor called by the render back-end. +package/*(arsd)*/ NVGContext createInternal (NVGparams* params) nothrow @trusted @nogc { + FONSparams fontParams = void; + NVGContext ctx = cast(NVGContext)malloc(NVGcontextinternal.sizeof); + if (ctx is null) goto error; + memset(ctx, 0, NVGcontextinternal.sizeof); + + ctx.params = *params; + ctx.fontImages[0..NVG_MAX_FONTIMAGES] = 0; + + ctx.commands = cast(float*)malloc(float.sizeof*NVG_INIT_COMMANDS_SIZE); + if (ctx.commands is null) goto error; + ctx.ncommands = 0; + ctx.ccommands = NVG_INIT_COMMANDS_SIZE; + + ctx.cache = nvg__allocPathCache(); + if (ctx.cache is null) goto error; + + ctx.save(); + ctx.reset(); + + nvg__setDevicePixelRatio(ctx, 1.0f); + + if (!ctx.params.renderCreate(ctx.params.userPtr)) goto error; + + // init font rendering + memset(&fontParams, 0, fontParams.sizeof); + fontParams.width = NVG_INIT_FONTIMAGE_SIZE; + fontParams.height = NVG_INIT_FONTIMAGE_SIZE; + fontParams.flags = FONS_ZERO_TOPLEFT; + fontParams.renderCreate = null; + fontParams.renderUpdate = null; + fontParams.renderDraw = null; + fontParams.renderDelete = null; + fontParams.userPtr = null; + ctx.fs = fonsCreateInternal(&fontParams); + if (ctx.fs is null) goto error; + + // create font texture + ctx.fontImages[0] = ctx.params.renderCreateTexture(ctx.params.userPtr, NVGtexture.Alpha, fontParams.width, fontParams.height, (ctx.params.fontAA ? 0 : NVGImageFlags.NoFiltering), null); + if (ctx.fontImages[0] == 0) goto error; + ctx.fontImageIdx = 0; + + ctx.pathPickId = -1; + ctx.tesselatortype = NVG_DEFAULT_TESSELATOR; + + return ctx; + +error: + ctx.deleteInternal(); + return null; +} + +// Called by render backend. +package/*(arsd)*/ NVGparams* internalParams (NVGContext ctx) nothrow @trusted @nogc { + return &ctx.params; +} + +// Destructor called by the render back-end. +package/*(arsd)*/ void deleteInternal (ref NVGContext ctx) nothrow @trusted @nogc { + if (ctx is null) return; + + if (ctx.commands !is null) free(ctx.commands); + nvg__deletePathCache(ctx.cache); + + if (ctx.fs) fonsDeleteInternal(ctx.fs); + + foreach (uint i; 0..NVG_MAX_FONTIMAGES) { + if (ctx.fontImages[i] != 0) { + ctx.deleteImage(ctx.fontImages[i]); + ctx.fontImages[i] = 0; + } + } + + if (ctx.params.renderDelete !is null) ctx.params.renderDelete(ctx.params.userPtr); + + if (ctx.pickScene !is null) nvg__deletePickScene(ctx.pickScene); + + if (ctx.cleanup !is null) ctx.cleanup(ctx); + + free(ctx); +} + +/// Delete NanoVega context. +/// Group: context_management +public void kill (ref NVGContext ctx) nothrow @trusted @nogc { + if (ctx !is null) { + ctx.deleteInternal(); + ctx = null; + } +} + +// ////////////////////////////////////////////////////////////////////////// // +// Frame Management + +/** Begin drawing a new frame. + * + * Calls to NanoVega drawing API should be wrapped in [beginFrame] and [endFrame] + * + * [beginFrame] defines the size of the window to render to in relation currently + * set viewport (i.e. glViewport on GL backends). Device pixel ration allows to + * control the rendering on Hi-DPI devices. + * + * For example, GLFW returns two dimension for an opened window: window size and + * frame buffer size. In that case you would set windowWidth/windowHeight to the window size, + * devicePixelRatio to: `windowWidth/windowHeight`. + * + * Default ratio is `1`. + * + * Note that fractional ratio can (and will) distort your fonts and images. + * + * This call also resets pick marks (see picking API for non-rasterized paths), + * path recording, and GPU affine transformatin matrix. + * + * see also [glNVGClearFlags], which returns necessary flags for [glClear]. + * + * Group: frame_management + */ +public void beginFrame (NVGContext ctx, int windowWidth, int windowHeight, float devicePixelRatio=1.0f) nothrow @trusted @nogc { + import std.math : isNaN; + /* + printf("Tris: draws:%d fill:%d stroke:%d text:%d TOT:%d\n", + ctx.drawCallCount, ctx.fillTriCount, ctx.strokeTriCount, ctx.textTriCount, + ctx.fillTriCount+ctx.strokeTriCount+ctx.textTriCount); + */ + + if (isNaN(devicePixelRatio)) devicePixelRatio = (windowHeight > 0 ? cast(float)windowWidth/cast(float)windowHeight : 1024.0/768.0); + + ctx.nstates = 0; + ctx.save(); + ctx.reset(); + + nvg__setDevicePixelRatio(ctx, devicePixelRatio); + + ctx.params.renderViewport(ctx.params.userPtr, windowWidth, windowHeight); + ctx.mWidth = windowWidth; + ctx.mHeight = windowHeight; + ctx.mDeviceRatio = devicePixelRatio; + + ctx.recset = null; + ctx.recstart = -1; + + ctx.drawCallCount = 0; + ctx.fillTriCount = 0; + ctx.strokeTriCount = 0; + ctx.textTriCount = 0; + + ctx.gpuAffine = NVGMatrix.Identity; + + nvg__pickBeginFrame(ctx, windowWidth, windowHeight); +} + +/// Cancels drawing the current frame. Cancels path recording. +/// Group: frame_management +public void cancelFrame (NVGContext ctx) nothrow @trusted @nogc { + ctx.cancelRecording(); + ctx.mWidth = 0; + ctx.mHeight = 0; + ctx.mDeviceRatio = 0; + // cancel render queue + ctx.params.renderCancel(ctx.params.userPtr); +} + +/// Ends drawing the current frame (flushing remaining render state). Commits recorded paths. +/// Group: frame_management +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; + // flush render queue + NVGstate* state = nvg__getState(ctx); + ctx.params.renderFlush(ctx.params.userPtr, state.compositeOperation); + if (ctx.fontImageIdx != 0) { + int fontImage = ctx.fontImages[ctx.fontImageIdx]; + int j = 0, iw, ih; + // delete images that smaller than current one + if (fontImage == 0) return; + ctx.imageSize(fontImage, iw, ih); + foreach (int i; 0..ctx.fontImageIdx) { + if (ctx.fontImages[i] != 0) { + int nw, nh; + ctx.imageSize(ctx.fontImages[i], nw, nh); + if (nw < iw || nh < ih) { + ctx.deleteImage(ctx.fontImages[i]); + } else { + ctx.fontImages[j++] = ctx.fontImages[i]; + } + } + } + // make current font image to first + ctx.fontImages[j++] = ctx.fontImages[0]; + ctx.fontImages[0] = fontImage; + ctx.fontImageIdx = 0; + // clear all images after j + ctx.fontImages[j..NVG_MAX_FONTIMAGES] = 0; + } +} + + +// ////////////////////////////////////////////////////////////////////////// // +// Recording and Replaying Pathes + +/// Saved path set. +/// Group: path_recording +public alias NVGPathSet = NVGPathSetS*; + + +//TODO: save scissor info? +struct NVGPathSetS { +private: + // either path cache, or text item + static struct Node { + NVGPaint paint; + NVGpathCache* path; + } + +private: + Node* nodes; + int nnodes, cnodes; + NVGpickScene* pickscene; + //int npickscenes, cpickscenes; + NVGContext svctx; // used to do some sanity checks, and to free resources + +private: + Node* allocNode () nothrow @trusted @nogc { + import core.stdc.string : memset; + // grow buffer if necessary + if (nnodes+1 > cnodes) { + import core.stdc.stdlib : realloc; + int newsz = (cnodes == 0 ? 8 : cnodes <= 1024 ? cnodes*2 : cnodes+1024); + nodes = cast(Node*)realloc(nodes, newsz*Node.sizeof); + if (nodes is null) assert(0, "NanoVega: out of memory"); + //memset(svp.caches+svp.ccaches, 0, (newsz-svp.ccaches)*NVGpathCache.sizeof); + cnodes = newsz; + } + assert(nnodes < cnodes); + memset(nodes+nnodes, 0, Node.sizeof); + return &nodes[nnodes++]; + } + + Node* allocPathNode () nothrow @trusted @nogc { + import core.stdc.stdlib : malloc; + import core.stdc.string : memset; + auto node = allocNode(); + // allocate path cache + auto pc = cast(NVGpathCache*)malloc(NVGpathCache.sizeof); + if (pc is null) assert(0, "NanoVega: out of memory"); + node.path = pc; + return node; + } + + void clearNode (int idx) nothrow @trusted @nogc { + if (idx < 0 || idx >= nnodes) return; + Node* node = &nodes[idx]; + if (svctx !is null && node.paint.image > 0) { + svctx.params.renderDeleteTexture(svctx.params.userPtr, node.paint.image); + } + if (node.path !is null) node.path.clear(); + } + +private: + void takeCurrentPickScene (NVGContext ctx) nothrow @trusted @nogc { + NVGpickScene* ps = ctx.pickScene; + if (ps is null) return; // nothing to do + if (ps.npaths == 0) return; // pick scene is empty + ctx.pickScene = null; + pickscene = ps; + } + + void replay (NVGContext ctx, in ref NVGColor fillTint, in ref NVGColor strokeTint) nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + foreach (ref node; nodes[0..nnodes]) { + if (auto cc = node.path) { + if (cc.npaths <= 0) continue; + + if (cc.fillReady) { + NVGPaint fillPaint = node.paint; + + // apply global alpha + fillPaint.innerColor.a *= state.alpha; + fillPaint.outerColor.a *= state.alpha; + + fillPaint.innerColor.applyTint(fillTint); + fillPaint.outerColor.applyTint(fillTint); + + ctx.params.renderFill(ctx.params.userPtr, &fillPaint, &state.scissor, cc.fringeWidth, cc.bounds.ptr, cc.paths, cc.npaths, cc.evenOddMode); + + // count triangles + foreach (int i; 0..cc.npaths) { + NVGpath* path = &cc.paths[i]; + ctx.fillTriCount += path.nfill-2; + ctx.fillTriCount += path.nstroke-2; + ctx.drawCallCount += 2; + } + } + + if (cc.strokeReady) { + NVGPaint strokePaint = node.paint; + + strokePaint.innerColor.a *= cc.strokeAlphaMul; + strokePaint.outerColor.a *= cc.strokeAlphaMul; + + // apply global alpha + strokePaint.innerColor.a *= state.alpha; + strokePaint.outerColor.a *= state.alpha; + + strokePaint.innerColor.applyTint(strokeTint); + strokePaint.outerColor.applyTint(strokeTint); + + ctx.params.renderStroke(ctx.params.userPtr, &strokePaint, &state.scissor, cc.fringeWidth, cc.strokeWidth, cc.paths, cc.npaths); + + // count triangles + foreach (int i; 0..cc.npaths) { + NVGpath* path = &cc.paths[i]; + ctx.strokeTriCount += path.nstroke-2; + ++ctx.drawCallCount; + } + } + } + } + } + +public: + @disable this (this); // no copies + + // pick test + // Call delegate [dg] for each path under the specified position (in no particular order). + // Returns the id of the path for which delegate [dg] returned true or -1. + // dg is: `bool delegate (int id, int order)` -- [order] is path ordering (ascending). + int hitTestDG(DG) (in float x, in float y, uint kind, scope DG dg) if (IsGoodHitTestDG!DG || IsGoodHitTestInternalDG!DG) { + if (pickscene is null) return -1; + + NVGpickScene* ps = pickscene; + int levelwidth = 1<<(ps.nlevels-1); + int cellx = nvg__clamp(cast(int)(x/ps.xdim), 0, levelwidth); + int celly = nvg__clamp(cast(int)(y/ps.ydim), 0, levelwidth); + int npicked = 0; + + for (int lvl = ps.nlevels-1; lvl >= 0; --lvl) { + NVGpickPath* pp = ps.levels[lvl][celly*levelwidth+cellx]; + while (pp !is null) { + if (nvg__pickPathTestBounds(svctx, ps, pp, x, y)) { + int hit = 0; + if ((kind&NVGPickKind.Stroke) && (pp.flags&NVGPathFlags.Stroke)) hit = nvg__pickPathStroke(ps, pp, x, y); + if (!hit && (kind&NVGPickKind.Fill) && (pp.flags&NVGPathFlags.Fill)) hit = nvg__pickPath(ps, pp, x, y); + if (hit) { + static if (IsGoodHitTestDG!DG) { + static if (__traits(compiles, (){ DG dg; bool res = dg(cast(int)42, cast(int)666); })) { + if (dg(pp.id, cast(int)pp.order)) return pp.id; + } else { + dg(pp.id, cast(int)pp.order); + } + } else { + static if (__traits(compiles, (){ DG dg; NVGpickPath* pp; bool res = dg(pp); })) { + if (dg(pp)) return pp.id; + } else { + dg(pp); + } + } + } + } + pp = pp.next; + } + cellx >>= 1; + celly >>= 1; + levelwidth >>= 1; + } + + return -1; + } + + // Fills ids with a list of the top most hit ids under the specified position. + // Returns the slice of [ids]. + int[] hitTestAll (in float x, in float y, uint kind, int[] ids) nothrow @trusted @nogc { + if (pickscene is null || ids.length == 0) return ids[0..0]; + + int npicked = 0; + NVGpickScene* ps = pickscene; + + hitTestDG(x, y, kind, delegate (NVGpickPath* pp) nothrow @trusted @nogc { + if (npicked == ps.cpicked) { + int cpicked = ps.cpicked+ps.cpicked; + NVGpickPath** picked = cast(NVGpickPath**)realloc(ps.picked, (NVGpickPath*).sizeof*ps.cpicked); + if (picked is null) return true; // abort + ps.cpicked = cpicked; + ps.picked = picked; + } + ps.picked[npicked] = pp; + ++npicked; + return false; // go on + }); + + qsort(ps.picked, npicked, (NVGpickPath*).sizeof, &nvg__comparePaths); + + assert(npicked >= 0); + if (npicked > ids.length) npicked = cast(int)ids.length; + foreach (immutable nidx, ref int did; ids[0..npicked]) did = ps.picked[nidx].id; + + return ids[0..npicked]; + } + + // Returns the id of the pickable shape containing x,y or -1 if no shape was found. + int hitTest (in float x, in float y, uint kind) nothrow @trusted @nogc { + if (pickscene is null) return -1; + + int bestOrder = -1; + int bestID = -1; + + hitTestDG(x, y, kind, delegate (NVGpickPath* pp) nothrow @trusted @nogc { + if (pp.order > bestOrder) { + bestOrder = pp.order; + bestID = pp.id; + } + }); + + return bestID; + } +} + +// Append current path to existing path set. Is is safe to call this with `null` [svp]. +void appendCurrentPathToCache (NVGContext ctx, NVGPathSet svp, in ref NVGPaint paint) nothrow @trusted @nogc { + if (ctx is null || svp is null) return; + if (ctx !is svp.svctx) assert(0, "NanoVega: cannot save paths from different contexts"); + if (ctx.ncommands == 0) { + assert(ctx.cache.npaths == 0); + return; + } + if (!ctx.cache.fillReady && !ctx.cache.strokeReady) return; + + // tesselate current path + //if (!ctx.cache.fillReady) nvg__prepareFill(ctx); + //if (!ctx.cache.strokeReady) nvg__prepareStroke(ctx); + + auto node = svp.allocPathNode(); + NVGpathCache* cc = node.path; + cc.copyFrom(ctx.cache); + node.paint = paint; + ctx.params.renderTextureIncRef(ctx.params.userPtr, node.paint.image); + // copy path commands (we may need 'em for picking) + version(all) { + cc.ncommands = ctx.ncommands; + if (cc.ncommands) { + import core.stdc.stdlib : malloc; + import core.stdc.string : memcpy; + cc.commands = cast(float*)malloc(ctx.ncommands*float.sizeof); + if (cc.commands is null) assert(0, "NanoVega: out of memory"); + memcpy(cc.commands, ctx.commands, ctx.ncommands*float.sizeof); + } else { + cc.commands = null; + } + } +} + +/// Create new empty path set. +/// Group: path_recording +public NVGPathSet newPathSet (NVGContext ctx) nothrow @trusted @nogc { + import core.stdc.stdlib : malloc; + import core.stdc.string : memset; + if (ctx is null) return null; + NVGPathSet res = cast(NVGPathSet)malloc(NVGPathSetS.sizeof); + if (res is null) assert(0, "NanoVega: out of memory"); + memset(res, 0, NVGPathSetS.sizeof); + res.svctx = ctx; + return res; +} + +/// Is the given path set empty? Empty path set can be `null`. +/// Group: path_recording +public bool empty (NVGPathSet svp) pure nothrow @safe @nogc { pragma(inline, true); return (svp is null || svp.nnodes == 0); } + +/// Clear path set contents. Will release $(B some) allocated memory (this function is meant to clear something that will be reused). +/// Group: path_recording +public void clear (NVGPathSet svp) nothrow @trusted @nogc { + if (svp !is null) { + import core.stdc.stdlib : free; + foreach (immutable idx; 0.. svp.nnodes) svp.clearNode(idx); + svp.nnodes = 0; + } +} + +/// Destroy path set (frees all allocated memory). +/// Group: path_recording +public void kill (ref NVGPathSet svp) nothrow @trusted @nogc { + if (svp !is null) { + import core.stdc.stdlib : free; + svp.clear(); + if (svp.nodes !is null) free(svp.nodes); + free(svp); + if (svp.pickscene !is null) nvg__deletePickScene(svp.pickscene); + svp = null; + } +} + +/// Start path recording. [svp] should be alive until recording is cancelled or stopped. +/// Group: path_recording +public void startRecording (NVGContext ctx, NVGPathSet svp) nothrow @trusted @nogc { + if (svp !is null && svp.svctx !is ctx) assert(0, "NanoVega: cannot share path set between contexts"); + ctx.stopRecording(); + ctx.recset = svp; + ctx.recstart = (svp !is null ? svp.nnodes : -1); + ctx.recblockdraw = false; +} + +/** Start path recording. [svp] should be alive until recording is cancelled or stopped. + * + * This will block all rendering, so you can call your rendering functions to record paths without actual drawing. + * Commiting or cancelling will re-enable rendering. + * You can call this with `null` svp to block rendering without recording any paths. + * + * Group: path_recording + */ +public void startBlockingRecording (NVGContext ctx, NVGPathSet svp) nothrow @trusted @nogc { + if (svp !is null && svp.svctx !is ctx) assert(0, "NanoVega: cannot share path set between contexts"); + ctx.stopRecording(); + ctx.recset = svp; + ctx.recstart = (svp !is null ? svp.nnodes : -1); + ctx.recblockdraw = true; +} + +/// Commit recorded paths. It is safe to call this when recording is not started. +/// Group: path_recording +public void stopRecording (NVGContext ctx) nothrow @trusted @nogc { + if (ctx.recset !is null && ctx.recset.svctx !is ctx) assert(0, "NanoVega: cannot share path set between contexts"); + if (ctx.recset !is null) ctx.recset.takeCurrentPickScene(ctx); + ctx.recset = null; + ctx.recstart = -1; + ctx.recblockdraw = false; +} + +/// Cancel path recording. +/// Group: path_recording +public void cancelRecording (NVGContext ctx) nothrow @trusted @nogc { + if (ctx.recset !is null) { + if (ctx.recset.svctx !is ctx) assert(0, "NanoVega: cannot share path set between contexts"); + assert(ctx.recstart >= 0 && ctx.recstart <= ctx.recset.nnodes); + foreach (immutable idx; ctx.recstart..ctx.recset.nnodes) ctx.recset.clearNode(idx); + ctx.recset.nnodes = ctx.recstart; + ctx.recset = null; + ctx.recstart = -1; + } + ctx.recblockdraw = false; +} + +/** Replay saved path set. + * + * Replaying record while you're recording another one is undefined behavior. + * + * Group: path_recording + */ +public void replayRecording() (NVGContext ctx, NVGPathSet svp, in auto ref NVGColor fillTint, in auto ref NVGColor strokeTint) nothrow @trusted @nogc { + if (svp !is null && svp.svctx !is ctx) assert(0, "NanoVega: cannot share path set between contexts"); + svp.replay(ctx, fillTint, strokeTint); +} + +/// Ditto. +public void replayRecording() (NVGContext ctx, NVGPathSet svp, in auto ref NVGColor fillTint) nothrow @trusted @nogc { ctx.replayRecording(svp, fillTint, NVGColor.transparent); } + +/// Ditto. +public void replayRecording (NVGContext ctx, NVGPathSet svp) nothrow @trusted @nogc { ctx.replayRecording(svp, NVGColor.transparent, NVGColor.transparent); } + + +// ////////////////////////////////////////////////////////////////////////// // +// Composite operation + +/// Sets the composite operation. +/// Group: composite_operation +public void globalCompositeOperation (NVGContext ctx, NVGCompositeOperation op) nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + state.compositeOperation = nvg__compositeOperationState(op); +} + +/// Sets the composite operation with custom pixel arithmetic. +/// Group: composite_operation +public void globalCompositeBlendFunc (NVGContext ctx, NVGBlendFactor sfactor, NVGBlendFactor dfactor) nothrow @trusted @nogc { + ctx.globalCompositeBlendFuncSeparate(sfactor, dfactor, sfactor, dfactor); +} + +/// Sets the composite operation with custom pixel arithmetic for RGB and alpha components separately. +/// Group: composite_operation +public void globalCompositeBlendFuncSeparate (NVGContext ctx, NVGBlendFactor srcRGB, NVGBlendFactor dstRGB, NVGBlendFactor srcAlpha, NVGBlendFactor dstAlpha) nothrow @trusted @nogc { + NVGCompositeOperationState op; + op.srcRGB = srcRGB; + op.dstRGB = dstRGB; + op.srcAlpha = srcAlpha; + op.dstAlpha = dstAlpha; + NVGstate* state = nvg__getState(ctx); + state.compositeOperation = op; +} + + +// ////////////////////////////////////////////////////////////////////////// // +// Color utils + +/// Returns a color value from string form. +/// Supports: "#rgb", "#rrggbb", "#argb", "#aarrggbb" +/// Group: color_utils +public NVGColor nvgRGB (const(char)[] srgb) nothrow @trusted @nogc { pragma(inline, true); return NVGColor(srgb); } + +/// Ditto. +public NVGColor nvgRGBA (const(char)[] srgb) nothrow @trusted @nogc { pragma(inline, true); return NVGColor(srgb); } + +/// Returns a color value from red, green, blue values. Alpha will be set to 255 (1.0f). +/// Group: color_utils +public NVGColor nvgRGB (int r, int g, int b) nothrow @trusted @nogc { pragma(inline, true); return NVGColor(r, g, b, 255); } + +/// Returns a color value from red, green, blue values. Alpha will be set to 1.0f. +/// Group: color_utils +public NVGColor nvgRGBf (float r, float g, float b) nothrow @trusted @nogc { pragma(inline, true); return NVGColor(r, g, b, 1.0f); } + +/// Returns a color value from red, green, blue and alpha values. +/// Group: color_utils +public NVGColor nvgRGBA (int r, int g, int b, int a=255) nothrow @trusted @nogc { pragma(inline, true); return NVGColor(r, g, b, a); } + +/// Returns a color value from red, green, blue and alpha values. +/// Group: color_utils +public NVGColor nvgRGBAf (float r, float g, float b, float a=1.0f) nothrow @trusted @nogc { pragma(inline, true); return NVGColor(r, g, b, a); } + +/// Returns new color with transparency (alpha) set to [a]. +/// Group: color_utils +public NVGColor nvgTransRGBA (NVGColor c, ubyte a) nothrow @trusted @nogc { + pragma(inline, true); + c.a = a/255.0f; + return c; +} + +/// Ditto. +public NVGColor nvgTransRGBAf (NVGColor c, float a) nothrow @trusted @nogc { + pragma(inline, true); + c.a = a; + return c; +} + +/// Linearly interpolates from color c0 to c1, and returns resulting color value. +/// Group: color_utils +public NVGColor nvgLerpRGBA() (in auto ref NVGColor c0, in auto ref NVGColor c1, float u) nothrow @trusted @nogc { + NVGColor cint = void; + u = nvg__clamp(u, 0.0f, 1.0f); + float oneminu = 1.0f-u; + foreach (uint i; 0..4) cint.rgba.ptr[i] = c0.rgba.ptr[i]*oneminu+c1.rgba.ptr[i]*u; + return cint; +} + +/* see below +public NVGColor nvgHSL() (float h, float s, float l) { + //pragma(inline, true); // alas + return nvgHSLA(h, s, l, 255); +} +*/ + +float nvg__hue (float h, float m1, float m2) pure nothrow @safe @nogc { + if (h < 0) h += 1; + if (h > 1) h -= 1; + if (h < 1.0f/6.0f) return m1+(m2-m1)*h*6.0f; + if (h < 3.0f/6.0f) return m2; + if (h < 4.0f/6.0f) return m1+(m2-m1)*(2.0f/3.0f-h)*6.0f; + return m1; +} + +/// Returns color value specified by hue, saturation and lightness. +/// HSL values are all in range [0..1], alpha will be set to 255. +/// Group: color_utils +public alias nvgHSL = nvgHSLA; // trick to allow inlining + +/// Returns color value specified by hue, saturation and lightness and alpha. +/// HSL values are all in range [0..1], alpha in range [0..255]. +/// Group: color_utils +public NVGColor nvgHSLA (float h, float s, float l, ubyte a=255) nothrow @trusted @nogc { + pragma(inline, true); + NVGColor col = void; + h = nvg__modf(h, 1.0f); + if (h < 0.0f) h += 1.0f; + s = nvg__clamp(s, 0.0f, 1.0f); + l = nvg__clamp(l, 0.0f, 1.0f); + immutable float m2 = (l <= 0.5f ? l*(1+s) : l+s-l*s); + immutable float m1 = 2*l-m2; + col.r = nvg__clamp(nvg__hue(h+1.0f/3.0f, m1, m2), 0.0f, 1.0f); + col.g = nvg__clamp(nvg__hue(h, m1, m2), 0.0f, 1.0f); + col.b = nvg__clamp(nvg__hue(h-1.0f/3.0f, m1, m2), 0.0f, 1.0f); + col.a = a/255.0f; + return col; +} + +/// Returns color value specified by hue, saturation and lightness and alpha. +/// HSL values and alpha are all in range [0..1]. +/// Group: color_utils +public NVGColor nvgHSLA (float h, float s, float l, float a) nothrow @trusted @nogc { + // sorry for copypasta, it is for inliner + static if (__VERSION__ >= 2072) pragma(inline, true); + NVGColor col = void; + h = nvg__modf(h, 1.0f); + if (h < 0.0f) h += 1.0f; + s = nvg__clamp(s, 0.0f, 1.0f); + l = nvg__clamp(l, 0.0f, 1.0f); + immutable m2 = (l <= 0.5f ? l*(1+s) : l+s-l*s); + immutable m1 = 2*l-m2; + col.r = nvg__clamp(nvg__hue(h+1.0f/3.0f, m1, m2), 0.0f, 1.0f); + col.g = nvg__clamp(nvg__hue(h, m1, m2), 0.0f, 1.0f); + col.b = nvg__clamp(nvg__hue(h-1.0f/3.0f, m1, m2), 0.0f, 1.0f); + col.a = a; + return col; +} + + +// ////////////////////////////////////////////////////////////////////////// // +// Matrices and Transformations + +/** Matrix class. + * + * Group: matrices + */ +public align(1) struct NVGMatrix { +align(1): +private: + static immutable float[6] IdentityMat = [ + 1.0f, 0.0f, + 0.0f, 1.0f, + 0.0f, 0.0f, + ]; + +public: + /// Matrix values. Initial value is identity matrix. + float[6] mat = [ + 1.0f, 0.0f, + 0.0f, 1.0f, + 0.0f, 0.0f, + ]; + +public nothrow @trusted @nogc: + /// Create Matrix with the given values. + this (const(float)[] amat...) { + pragma(inline, true); + if (amat.length >= 6) { + mat.ptr[0..6] = amat.ptr[0..6]; + } else { + mat.ptr[0..6] = 0; + mat.ptr[0..amat.length] = amat[]; + } + } + + /// Can be used to check validity of [inverted] result + @property bool valid () const { import core.stdc.math : isfinite; return (isfinite(mat.ptr[0]) != 0); } + + /// Returns `true` if this matrix is identity matrix. + @property bool isIdentity () const { pragma(inline, true); return (mat[] == IdentityMat[]); } + + /// Returns new inverse matrix. + /// If inverted matrix cannot be calculated, `res.valid` fill be `false`. + NVGMatrix inverted () const { + NVGMatrix res = this; + res.invert; + return res; + } + + /// Inverts this matrix. + /// If inverted matrix cannot be calculated, `this.valid` fill be `false`. + ref NVGMatrix invert () { + float[6] inv = void; + immutable double det = cast(double)mat.ptr[0]*mat.ptr[3]-cast(double)mat.ptr[2]*mat.ptr[1]; + if (det > -1e-6 && det < 1e-6) { + inv[] = float.nan; + } else { + immutable double invdet = 1.0/det; + inv.ptr[0] = cast(float)(mat.ptr[3]*invdet); + inv.ptr[2] = cast(float)(-mat.ptr[2]*invdet); + inv.ptr[4] = cast(float)((cast(double)mat.ptr[2]*mat.ptr[5]-cast(double)mat.ptr[3]*mat.ptr[4])*invdet); + inv.ptr[1] = cast(float)(-mat.ptr[1]*invdet); + inv.ptr[3] = cast(float)(mat.ptr[0]*invdet); + inv.ptr[5] = cast(float)((cast(double)mat.ptr[1]*mat.ptr[4]-cast(double)mat.ptr[0]*mat.ptr[5])*invdet); + } + mat.ptr[0..6] = inv.ptr[0..6]; + return this; + } + + /// Sets this matrix to identity matrix. + ref NVGMatrix identity () { version(aliced) pragma(inline, true); mat[] = IdentityMat[]; return this; } + + /// Translate this matrix. + ref NVGMatrix translate (in float tx, in float ty) { + version(aliced) pragma(inline, true); + return this.mul(Translated(tx, ty)); + } + + /// Scale this matrix. + ref NVGMatrix scale (in float sx, in float sy) { + version(aliced) pragma(inline, true); + return this.mul(Scaled(sx, sy)); + } + + /// Rotate this matrix. + ref NVGMatrix rotate (in float a) { + version(aliced) pragma(inline, true); + return this.mul(Rotated(a)); + } + + /// Skew this matrix by X axis. + ref NVGMatrix skewX (in float a) { + version(aliced) pragma(inline, true); + return this.mul(SkewedX(a)); + } + + /// Skew this matrix by Y axis. + ref NVGMatrix skewY (in float a) { + version(aliced) pragma(inline, true); + return this.mul(SkewedY(a)); + } + + /// Skew this matrix by both axes. + ref NVGMatrix skewY (in float ax, in float ay) { + version(aliced) pragma(inline, true); + return this.mul(SkewedXY(ax, ay)); + } + + /// Transform point with this matrix. `null` destinations are allowed. + /// [sx] and [sy] is the source point. [dx] and [dy] may point to the same variables. + void point (float* dx, float* dy, float sx, float sy) nothrow @trusted @nogc { + version(aliced) pragma(inline, true); + if (dx !is null) *dx = sx*mat.ptr[0]+sy*mat.ptr[2]+mat.ptr[4]; + if (dy !is null) *dy = sx*mat.ptr[1]+sy*mat.ptr[3]+mat.ptr[5]; + } + + /// Transform point with this matrix. + void point (ref float x, ref float y) nothrow @trusted @nogc { + version(aliced) pragma(inline, true); + immutable float nx = x*mat.ptr[0]+y*mat.ptr[2]+mat.ptr[4]; + immutable float ny = x*mat.ptr[1]+y*mat.ptr[3]+mat.ptr[5]; + x = nx; + y = ny; + } + + /// Sets this matrix to the result of multiplication of `this` and [s] (this * S). + ref NVGMatrix mul() (in auto ref NVGMatrix s) { + immutable float t0 = mat.ptr[0]*s.mat.ptr[0]+mat.ptr[1]*s.mat.ptr[2]; + immutable float t2 = mat.ptr[2]*s.mat.ptr[0]+mat.ptr[3]*s.mat.ptr[2]; + immutable float t4 = mat.ptr[4]*s.mat.ptr[0]+mat.ptr[5]*s.mat.ptr[2]+s.mat.ptr[4]; + mat.ptr[1] = mat.ptr[0]*s.mat.ptr[1]+mat.ptr[1]*s.mat.ptr[3]; + mat.ptr[3] = mat.ptr[2]*s.mat.ptr[1]+mat.ptr[3]*s.mat.ptr[3]; + mat.ptr[5] = mat.ptr[4]*s.mat.ptr[1]+mat.ptr[5]*s.mat.ptr[3]+s.mat.ptr[5]; + mat.ptr[0] = t0; + mat.ptr[2] = t2; + mat.ptr[4] = t4; + return this; + } + + /// Sets this matrix to the result of multiplication of [s] and `this` (S * this). + /// Sets the transform to the result of multiplication of two transforms, of A = B*A. + /// Group: matrices + ref NVGMatrix premul() (in auto ref NVGMatrix s) { + NVGMatrix s2 = s; + s2.mul(this); + mat[] = s2.mat[]; + return this; + } + + /// Multiply this matrix by [s], return result as new matrix. + /// Performs operations in this left-to-right order. + NVGMatrix opBinary(string op="*") (in auto ref NVGMatrix s) const { + version(aliced) pragma(inline, true); + NVGMatrix res = this; + res.mul(s); + return res; + } + + /// Multiply this matrix by [s]. + /// Performs operations in this left-to-right order. + ref NVGMatrix opOpAssign(string op="*") (in auto ref NVGMatrix s) { + version(aliced) pragma(inline, true); + return this.mul(s); + } + + float scaleX () const { pragma(inline, true); return nvg__sqrtf(mat.ptr[0]*mat.ptr[0]+mat.ptr[2]*mat.ptr[2]); } /// Returns x scaling of this matrix. + float scaleY () const { pragma(inline, true); return nvg__sqrtf(mat.ptr[1]*mat.ptr[1]+mat.ptr[3]*mat.ptr[3]); } /// Returns y scaling of this matrix. + float rotation () const { pragma(inline, true); return nvg__atan2f(mat.ptr[1], mat.ptr[0]); } /// Returns rotation of this matrix. + float tx () const { pragma(inline, true); return mat.ptr[4]; } /// Returns x translation of this matrix. + float ty () const { pragma(inline, true); return mat.ptr[5]; } /// Returns y translation of this matrix. + + ref NVGMatrix scaleX (in float v) { pragma(inline, true); return scaleRotateTransform(v, scaleY, rotation, tx, ty); } /// Sets x scaling of this matrix. + ref NVGMatrix scaleY (in float v) { pragma(inline, true); return scaleRotateTransform(scaleX, v, rotation, tx, ty); } /// Sets y scaling of this matrix. + ref NVGMatrix rotation (in float v) { pragma(inline, true); return scaleRotateTransform(scaleX, scaleY, v, tx, ty); } /// Sets rotation of this matrix. + ref NVGMatrix tx (in float v) { pragma(inline, true); mat.ptr[4] = v; return this; } /// Sets x translation of this matrix. + ref NVGMatrix ty (in float v) { pragma(inline, true); mat.ptr[5] = v; return this; } /// Sets y translation of this matrix. + + /// Utility function to be used in `setXXX()`. + /// This is the same as doing: `mat.identity.rotate(a).scale(xs, ys).translate(tx, ty)`, only faster + ref NVGMatrix scaleRotateTransform (in float xscale, in float yscale, in float a, in float tx, in float ty) { + immutable float cs = nvg__cosf(a), sn = nvg__sinf(a); + mat.ptr[0] = xscale*cs; mat.ptr[1] = yscale*sn; + mat.ptr[2] = xscale*-sn; mat.ptr[3] = yscale*cs; + mat.ptr[4] = tx; mat.ptr[5] = ty; + return this; + } + + /// This is the same as doing: `mat.identity.rotate(a).translate(tx, ty)`, only faster + ref NVGMatrix rotateTransform (in float a, in float tx, in float ty) { + immutable float cs = nvg__cosf(a), sn = nvg__sinf(a); + mat.ptr[0] = cs; mat.ptr[1] = sn; + mat.ptr[2] = -sn; mat.ptr[3] = cs; + mat.ptr[4] = tx; mat.ptr[5] = ty; + return this; + } + + /// Returns new identity matrix. + static NVGMatrix Identity () { pragma(inline, true); return NVGMatrix.init; } + + /// Returns new translation matrix. + static NVGMatrix Translated (in float tx, in float ty) { + version(aliced) pragma(inline, true); + NVGMatrix res = void; + res.mat.ptr[0] = 1.0f; res.mat.ptr[1] = 0.0f; + res.mat.ptr[2] = 0.0f; res.mat.ptr[3] = 1.0f; + res.mat.ptr[4] = tx; res.mat.ptr[5] = ty; + return res; + } + + /// Returns new scaling matrix. + static NVGMatrix Scaled (in float sx, in float sy) { + version(aliced) pragma(inline, true); + NVGMatrix res = void; + res.mat.ptr[0] = sx; res.mat.ptr[1] = 0.0f; + res.mat.ptr[2] = 0.0f; res.mat.ptr[3] = sy; + res.mat.ptr[4] = 0.0f; res.mat.ptr[5] = 0.0f; + return res; + } + + /// Returns new rotation matrix. Angle is specified in radians. + static NVGMatrix Rotated (in float a) { + version(aliced) pragma(inline, true); + immutable float cs = nvg__cosf(a), sn = nvg__sinf(a); + NVGMatrix res = void; + res.mat.ptr[0] = cs; res.mat.ptr[1] = sn; + res.mat.ptr[2] = -sn; res.mat.ptr[3] = cs; + res.mat.ptr[4] = 0.0f; res.mat.ptr[5] = 0.0f; + return res; + } + + /// Returns new x-skewing matrix. Angle is specified in radians. + static NVGMatrix SkewedX (in float a) { + version(aliced) pragma(inline, true); + NVGMatrix res = void; + res.mat.ptr[0] = 1.0f; res.mat.ptr[1] = 0.0f; + res.mat.ptr[2] = nvg__tanf(a); res.mat.ptr[3] = 1.0f; + res.mat.ptr[4] = 0.0f; res.mat.ptr[5] = 0.0f; + return res; + } + + /// Returns new y-skewing matrix. Angle is specified in radians. + static NVGMatrix SkewedY (in float a) { + version(aliced) pragma(inline, true); + NVGMatrix res = void; + res.mat.ptr[0] = 1.0f; res.mat.ptr[1] = nvg__tanf(a); + res.mat.ptr[2] = 0.0f; res.mat.ptr[3] = 1.0f; + res.mat.ptr[4] = 0.0f; res.mat.ptr[5] = 0.0f; + return res; + } + + /// Returns new xy-skewing matrix. Angles are specified in radians. + static NVGMatrix SkewedXY (in float ax, in float ay) { + version(aliced) pragma(inline, true); + NVGMatrix res = void; + res.mat.ptr[0] = 1.0f; res.mat.ptr[1] = nvg__tanf(ay); + res.mat.ptr[2] = nvg__tanf(ax); res.mat.ptr[3] = 1.0f; + res.mat.ptr[4] = 0.0f; res.mat.ptr[5] = 0.0f; + return res; + } + + /// Utility function to be used in `setXXX()`. + /// This is the same as doing: `NVGMatrix.Identity.rotate(a).scale(xs, ys).translate(tx, ty)`, only faster + static NVGMatrix ScaledRotatedTransformed (in float xscale, in float yscale, in float a, in float tx, in float ty) { + NVGMatrix res = void; + res.scaleRotateTransform(xscale, yscale, a, tx, ty); + return res; + } + + /// This is the same as doing: `NVGMatrix.Identity.rotate(a).translate(tx, ty)`, only faster + static NVGMatrix RotatedTransformed (in float a, in float tx, in float ty) { + NVGMatrix res = void; + res.rotateTransform(a, tx, ty); + return res; + } +} + + +/// Converts degrees to radians. +/// Group: matrices +public float nvgDegToRad() (in float deg) pure nothrow @safe @nogc { pragma(inline, true); return deg/180.0f*NVG_PI; } + +/// Converts radians to degrees. +/// Group: matrices +public float nvgRadToDeg() (in float rad) pure nothrow @safe @nogc { pragma(inline, true); return rad/NVG_PI*180.0f; } + +public alias nvgDegrees = nvgDegToRad; /// Use this like `42.nvgDegrees` +public float nvgRadians() (in float rad) pure nothrow @safe @nogc { pragma(inline, true); return rad; } /// Use this like `0.1.nvgRadians` + + +// ////////////////////////////////////////////////////////////////////////// // +void nvg__setPaintColor (ref NVGPaint p, NVGColor color) nothrow @trusted @nogc { + //pragma(inline, true); + memset(&p, 0, p.sizeof); + p.xform.identity; + p.radius = 0.0f; + p.feather = 1.0f; + p.innerColor = color; + p.outerColor = color; +} + + +// ////////////////////////////////////////////////////////////////////////// // +// State handling + +/** Pushes and saves the current render state into a state stack. + * A matching [restore] must be used to restore the state. + * Returns `false` if state stack overflowed. + * + * Group: state_handling + */ +public bool save (NVGContext ctx) nothrow @trusted @nogc { + if (ctx.nstates >= NVG_MAX_STATES) return false; + if (ctx.nstates > 0) memcpy(&ctx.states[ctx.nstates], &ctx.states[ctx.nstates-1], NVGstate.sizeof); + ++ctx.nstates; + return true; +} + +/// Pops and restores current render state. +/// Group: state_handling +public bool restore (NVGContext ctx) nothrow @trusted @nogc { + if (ctx.nstates <= 1) return false; + --ctx.nstates; + return true; +} + +/// Resets current render state to default values. Does not affect the render state stack. +/// Group: state_handling +public void reset (NVGContext ctx) nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + memset(state, 0, (*state).sizeof); + + nvg__setPaintColor(state.fill, nvgRGBA(255, 255, 255, 255)); + nvg__setPaintColor(state.stroke, nvgRGBA(0, 0, 0, 255)); + state.compositeOperation = nvg__compositeOperationState(NVGCompositeOperation.SourceOver); + state.shapeAntiAlias = true; + state.strokeWidth = 1.0f; + state.miterLimit = 10.0f; + state.lineCap = NVGLineCap.Butt; + state.lineJoin = NVGLineCap.Miter; + state.alpha = 1.0f; + state.xform.identity; + + state.scissor.extent.ptr[0] = -1.0f; + state.scissor.extent.ptr[1] = -1.0f; + + state.fontSize = 16.0f; + state.letterSpacing = 0.0f; + state.lineHeight = 1.0f; + state.fontBlur = 0.0f; + state.textAlign.reset; + state.fontId = 0; + state.evenOddMode = false; +} + + +// ////////////////////////////////////////////////////////////////////////// // +// Render styles + +/// Sets filling mode to "even-odd". +/// Group: render_styles +public void evenOddFill (NVGContext ctx) nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + state.evenOddMode = true; +} + +/// Sets filling mode to "non-zero" (this is default mode). +/// Group: render_styles +public void nonZeroFill (NVGContext ctx) nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + state.evenOddMode = false; +} + +/// Sets whether to draw antialias for [stroke] and [fill]. It's enabled by default. +/// Group: render_styles +public void shapeAntiAlias (NVGContext ctx, bool enabled) { + NVGstate* state = nvg__getState(ctx); + state.shapeAntiAlias = enabled; +} + +/// Sets the stroke width of the stroke style. +/// Group: render_styles +public void strokeWidth (NVGContext ctx, float width) nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + state.strokeWidth = width; +} + +/// Sets the miter limit of the stroke style. Miter limit controls when a sharp corner is beveled. +/// Group: render_styles +public void miterLimit (NVGContext ctx, float limit) nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + state.miterLimit = limit; +} + +/// Sets how the end of the line (cap) is drawn, +/// Can be one of: NVGLineCap.Butt (default), NVGLineCap.Round, NVGLineCap.Square. +/// Group: render_styles +public void lineCap (NVGContext ctx, NVGLineCap cap) nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + state.lineCap = cap; +} + +/// Sets how sharp path corners are drawn. +/// Can be one of NVGLineCap.Miter (default), NVGLineCap.Round, NVGLineCap.Bevel. +/// Group: render_styles +public void lineJoin (NVGContext ctx, NVGLineCap join) nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + state.lineJoin = join; +} + +/// Sets the transparency applied to all rendered shapes. +/// Already transparent paths will get proportionally more transparent as well. +/// Group: render_styles +public void globalAlpha (NVGContext ctx, float alpha) nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + state.alpha = alpha; +} + +/// Sets current stroke style to a solid color. +/// Group: render_styles +public void strokeColor (NVGContext ctx, NVGColor color) nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + nvg__setPaintColor(state.stroke, color); +} + +/// Sets current stroke style to a paint, which can be a one of the gradients or a pattern. +/// Group: render_styles +public void strokePaint (NVGContext ctx, NVGPaint paint) nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + state.stroke = paint; + //nvgTransformMultiply(state.stroke.xform[], state.xform[]); + state.stroke.xform.mul(state.xform); +} + +/// Sets current fill style to a solid color. +/// Group: render_styles +public void fillColor (NVGContext ctx, NVGColor color) nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + nvg__setPaintColor(state.fill, color); +} + +/// Sets current fill style to a paint, which can be a one of the gradients or a pattern. +/// Group: render_styles +public void fillPaint (NVGContext ctx, NVGPaint paint) nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + state.fill = paint; + //nvgTransformMultiply(state.fill.xform[], state.xform[]); + state.fill.xform.mul(state.xform); +} + +/// Returns current transformation matrix. +/// Group: render_transformations +public NVGMatrix currTransform (NVGContext ctx) pure nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + return state.xform; +} + +/// Sets current transformation matrix. +/// Group: render_transformations +public void currTransform() (NVGContext ctx, in auto ref NVGMatrix m) nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + state.xform = m; +} + +/// Resets current transform to an identity matrix. +/// Group: render_transformations +public void resetTransform (NVGContext ctx) nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + state.xform.identity; +} + +/// Premultiplies current coordinate system by specified matrix. +/// Group: render_transformations +public void transform() (NVGContext ctx, in auto ref NVGMatrix mt) nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + //nvgTransformPremultiply(state.xform[], t[]); + state.xform *= mt; +} + +/// Translates current coordinate system. +/// Group: render_transformations +public void translate (NVGContext ctx, in float x, in float y) nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + //NVGMatrix t = void; + //nvgTransformTranslate(t[], x, y); + //nvgTransformPremultiply(state.xform[], t[]); + state.xform.premul(NVGMatrix.Translated(x, y)); +} + +/// Rotates current coordinate system. Angle is specified in radians. +/// Group: render_transformations +public void rotate (NVGContext ctx, in float angle) nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + //NVGMatrix t = void; + //nvgTransformRotate(t[], angle); + //nvgTransformPremultiply(state.xform[], t[]); + state.xform.premul(NVGMatrix.Rotated(angle)); +} + +/// Skews the current coordinate system along X axis. Angle is specified in radians. +/// Group: render_transformations +public void skewX (NVGContext ctx, in float angle) nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + //NVGMatrix t = void; + //nvgTransformSkewX(t[], angle); + //nvgTransformPremultiply(state.xform[], t[]); + state.xform.premul(NVGMatrix.SkewedX(angle)); +} + +/// Skews the current coordinate system along Y axis. Angle is specified in radians. +/// Group: render_transformations +public void skewY (NVGContext ctx, in float angle) nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + //NVGMatrix t = void; + //nvgTransformSkewY(t[], angle); + //nvgTransformPremultiply(state.xform[], t[]); + state.xform.premul(NVGMatrix.SkewedY(angle)); +} + +/// Scales the current coordinate system. +/// Group: render_transformations +public void scale (NVGContext ctx, in float x, in float y) nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + //NVGMatrix t = void; + //nvgTransformScale(t[], x, y); + //nvgTransformPremultiply(state.xform[], t[]); + state.xform.premul(NVGMatrix.Scaled(x, y)); +} + + +// ////////////////////////////////////////////////////////////////////////// // +// Images + +static if (NanoVegaHasArsdImage) { + // do we have new arsd API to load images? + static if (!is(typeof(MemoryImage.fromImageFile))) { + static assert(0, "Sorry, your ARSD is too old. Please, update it."); + } else { + alias ArsdImage = MemoryImage.fromImageFile; + } +} + +/// Creates image by loading it from the disk from specified file name. +/// Returns handle to the image or 0 on error. +/// Group: images +public int createImage (NVGContext ctx, const(char)[] filename, int imageFlags=NVGImageFlags.None) { + static if (NanoVegaHasArsdImage) { + try { + auto oimg = ArsdImage(filename); + if (auto img = cast(TrueColorImage)oimg) { + scope(exit) { oimg.destroy; } + return ctx.createImageRGBA(img.width, img.height, img.imageData.bytes[], imageFlags); + } else { + TrueColorImage img = oimg.getAsTrueColorImage; + oimg.destroy; + scope(exit) { img.destroy; } + return ctx.createImageRGBA(img.width, img.height, img.imageData.bytes[], imageFlags); + } + } catch (Exception) {} + return 0; + } else { + import std.internal.cstring; + ubyte* img; + int w, h, n, image; + stbi_set_unpremultiply_on_load(1); + stbi_convert_iphone_png_to_rgb(1); + img = stbi_load(filename.tempCString, &w, &h, &n, 4); + if (img is null) { + //printf("Failed to load %s - %s\n", filename, stbi_failure_reason()); + return 0; + } + image = ctx.createImageRGBA(w, h, imageFlags, img[0..w*h*4]); + stbi_image_free(img); + return image; + } +} + +static if (NanoVegaHasArsdImage) { + /// Creates image by loading it from the specified chunk of memory. + /// Returns handle to the image or 0 on error. + /// Group: images + public int createImageFromMemoryImage (NVGContext ctx, MemoryImage img, int imageFlags=NVGImageFlags.None) { + if (img is null) return 0; + if (auto tc = cast(TrueColorImage)img) { + return ctx.createImageRGBA(tc.width, tc.height, tc.imageData.bytes[], imageFlags); + } else { + auto tc = img.getAsTrueColorImage; + scope(exit) { tc.destroy; } + return ctx.createImageRGBA(tc.width, tc.height, tc.imageData.bytes[], imageFlags); + } + } +} else { + /// Creates image by loading it from the specified chunk of memory. + /// Returns handle to the image or 0 on error. + /// Group: images + public int createImageMem (NVGContext ctx, const(ubyte)* data, int ndata, int imageFlags=NVGImageFlags.None) { + int w, h, n, image; + ubyte* img = stbi_load_from_memory(data, ndata, &w, &h, &n, 4); + if (img is null) { + //printf("Failed to load %s - %s\n", filename, stbi_failure_reason()); + return 0; + } + image = ctx.createImageRGBA(w, h, img[0..w*h*4], imageFlags); + stbi_image_free(img); + return image; + } +} + +/// Creates image from specified image data. +/// Returns handle to the image or 0 on error. +/// Group: images +public int createImageRGBA (NVGContext ctx, int w, int h, const(void)[] data, int imageFlags=NVGImageFlags.None) nothrow @trusted @nogc { + if (w < 1 || h < 1 || data.length < w*h*4) return 0; + return ctx.params.renderCreateTexture(ctx.params.userPtr, NVGtexture.RGBA, w, h, imageFlags, cast(const(ubyte)*)data.ptr); +} + +/// Updates image data specified by image handle. +/// Group: images +public void updateImage (NVGContext ctx, int image, const(void)[] data) nothrow @trusted @nogc { + if (image > 0) { + int w, h; + ctx.params.renderGetTextureSize(ctx.params.userPtr, image, &w, &h); + ctx.params.renderUpdateTexture(ctx.params.userPtr, image, 0, 0, w, h, cast(const(ubyte)*)data.ptr); + } +} + +/// Returns the dimensions of a created image. +/// Group: images +public void imageSize (NVGContext ctx, int image, out int w, out int h) nothrow @trusted @nogc { + if (image > 0) ctx.params.renderGetTextureSize(ctx.params.userPtr, image, &w, &h); +} + +/// Deletes created image. +/// Group: images +public void deleteImage (NVGContext ctx, int image) nothrow @trusted @nogc { + if (ctx is null || image <= 0) return; + ctx.params.renderDeleteTexture(ctx.params.userPtr, image); +} + + +// ////////////////////////////////////////////////////////////////////////// // +// Paints + +/** Creates and returns a linear gradient. Parameters `(sx, sy) (ex, ey)` specify the start and end coordinates + * of the linear gradient, icol specifies the start color and ocol the end color. + * The gradient is transformed by the current transform when it is passed to [fillPaint] or [strokePaint]. + * + * Group: paints + */ +public NVGPaint linearGradient (NVGContext ctx, float sx, float sy, float ex, float ey, NVGColor icol, NVGColor ocol) nothrow @trusted @nogc { + enum large = 1e5f; + + NVGPaint p = void; + memset(&p, 0, p.sizeof); + + // Calculate transform aligned to the line + float dx = ex-sx; + float dy = ey-sy; + immutable float d = nvg__sqrtf(dx*dx+dy*dy); + if (d > 0.0001f) { + dx /= d; + dy /= d; + } else { + dx = 0; + dy = 1; + } + + p.xform.mat.ptr[0] = dy; p.xform.mat.ptr[1] = -dx; + p.xform.mat.ptr[2] = dx; p.xform.mat.ptr[3] = dy; + p.xform.mat.ptr[4] = sx-dx*large; p.xform.mat.ptr[5] = sy-dy*large; + + p.extent.ptr[0] = large; + p.extent.ptr[1] = large+d*0.5f; + + p.radius = 0.0f; + + p.feather = nvg__max(NVG_MIN_FEATHER, d); + + p.innerColor = icol; + p.outerColor = ocol; + + return p; +} + +/** Creates and returns a radial gradient. Parameters (cx, cy) specify the center, inr and outr specify + * the inner and outer radius of the gradient, icol specifies the start color and ocol the end color. + * The gradient is transformed by the current transform when it is passed to [fillPaint] or [strokePaint]. + * + * Group: paints + */ +public NVGPaint radialGradient (NVGContext ctx, float cx, float cy, float inr, float outr, NVGColor icol, NVGColor ocol) nothrow @trusted @nogc { + immutable float r = (inr+outr)*0.5f; + immutable float f = (outr-inr); + + NVGPaint p = void; + memset(&p, 0, p.sizeof); + + p.xform.identity; + p.xform.mat.ptr[4] = cx; + p.xform.mat.ptr[5] = cy; + + p.extent.ptr[0] = r; + p.extent.ptr[1] = r; + + p.radius = r; + + p.feather = nvg__max(NVG_MIN_FEATHER, f); + + p.innerColor = icol; + p.outerColor = ocol; + + return p; +} + +/** Creates and returns a box gradient. Box gradient is a feathered rounded rectangle, it is useful for rendering + * drop shadows or highlights for boxes. Parameters (x, y) define the top-left corner of the rectangle, + * (w, h) define the size of the rectangle, r defines the corner radius, and f feather. Feather defines how blurry + * the border of the rectangle is. Parameter icol specifies the inner color and ocol the outer color of the gradient. + * The gradient is transformed by the current transform when it is passed to [fillPaint] or [strokePaint]. + * + * Group: paints + */ +public NVGPaint boxGradient (NVGContext ctx, float x, float y, float w, float h, float r, float f, NVGColor icol, NVGColor ocol) nothrow @trusted @nogc { + NVGPaint p = void; + memset(&p, 0, p.sizeof); + + p.xform.identity; + p.xform.mat.ptr[4] = x+w*0.5f; + p.xform.mat.ptr[5] = y+h*0.5f; + + p.extent.ptr[0] = w*0.5f; + p.extent.ptr[1] = h*0.5f; + + p.radius = r; + + p.feather = nvg__max(NVG_MIN_FEATHER, f); + + p.innerColor = icol; + p.outerColor = ocol; + + return p; +} + +/** Creates and returns an image pattern. Parameters `(cx, cy)` specify the left-top location of the image pattern, + * `(w, h)` the size of one image, [angle] rotation around the top-left corner, [image] is handle to the image to render. + * The gradient is transformed by the current transform when it is passed to [fillPaint] or [strokePaint]. + * + * Group: paints + */ +public NVGPaint imagePattern (NVGContext ctx, float cx, float cy, float w, float h, float angle, int image, float alpha=1) nothrow @trusted @nogc { + NVGPaint p = void; + memset(&p, 0, p.sizeof); + + p.xform.identity.rotate(angle); + p.xform.mat.ptr[4] = cx; + p.xform.mat.ptr[5] = cy; + + p.extent.ptr[0] = w; + p.extent.ptr[1] = h; + + p.image = image; + + p.innerColor = p.outerColor = nvgRGBAf(1, 1, 1, alpha); + + return p; +} + +/// Linear gradient with multiple stops. +/// WARNING: THIS IS EXPERIMENTAL API AND MAY BE CHANGED/BROKEN IN NEXT RELEASES! +/// Group: paints +public alias NVGLGS = NVGLGSdata*; + +private struct NVGLGSdata { + int imgid; // 0: invalid + // [imagePattern] arguments + float cx, cy, w, h, angle; + + @disable this (this); // no copies + public @property bool valid () const pure nothrow @safe @nogc { pragma(inline, true); return (imgid > 0); } /// +} + +/// Destroy linear gradient with stops +/// WARNING: THIS IS EXPERIMENTAL API AND MAY BE CHANGED/BROKEN IN NEXT RELEASES! +/// Group: paints +public void kill (NVGContext ctx, ref NVGLGS lgs) nothrow @trusted @nogc { + if (lgs is null) return; + if (lgs.imgid > 0) { ctx.deleteImage(lgs.imgid); lgs.imgid = 0; } + free(lgs); + lgs = null; +} + +/** Sets linear gradient with stops, created with [createLinearGradientWithStops]. + * The gradient is transformed by the current transform when it is passed to [fillPaint] or [strokePaint]. + * + * WARNING: THIS IS EXPERIMENTAL API AND MAY BE CHANGED/BROKEN IN NEXT RELEASES! + * Group: paints + */ +public NVGPaint linearGradient (NVGContext ctx, NVGLGS lgs) nothrow @trusted @nogc { + if (lgs is null || !lgs.valid) { + NVGPaint p = void; + memset(&p, 0, p.sizeof); + nvg__setPaintColor(p, NVGColor.red); + return p; + } else { + return ctx.imagePattern(lgs.cx, lgs.cy, lgs.w, lgs.h, lgs.angle, lgs.imgid); + } +} + +/// Gradient Stop Point. +/// WARNING: THIS IS EXPERIMENTAL API AND MAY BE CHANGED/BROKEN IN NEXT RELEASES! +/// Group: paints +public struct NVGGradientStop { + float offset; /// [0..1] + NVGColor color; /// +} + +/// Create linear gradient data suitable to use with `linearGradient(res)`. +/// Don't forget to destroy the result when you don't need it anymore with `ctx.kill(res);`. +/// WARNING: THIS IS EXPERIMENTAL API AND MAY BE CHANGED/BROKEN IN NEXT RELEASES! +/// Group: paints +public NVGLGS createLinearGradientWithStops (NVGContext ctx, float sx, float sy, float ex, float ey, const(NVGGradientStop)[] stops) nothrow @trusted @nogc { + // based on the code by Jorge Acereda + enum NVG_GRADIENT_SAMPLES = 1024; + static void gradientSpan (uint* dst, const(NVGGradientStop)* s0, const(NVGGradientStop)* s1) nothrow @trusted @nogc { + float s0o = nvg__clamp(s0.offset, 0.0f, 1.0f); + float s1o = nvg__clamp(s1.offset, 0.0f, 1.0f); + uint s = cast(uint)(s0o*NVG_GRADIENT_SAMPLES); + uint e = cast(uint)(s1o*NVG_GRADIENT_SAMPLES); + uint sc = 0xffffffffU; + uint sh = 24; + uint r = cast(uint)(s0.color.rgba[0]*sc); + uint g = cast(uint)(s0.color.rgba[1]*sc); + uint b = cast(uint)(s0.color.rgba[2]*sc); + uint a = cast(uint)(s0.color.rgba[3]*sc); + uint dr = cast(uint)((s1.color.rgba[0]*sc-r)/(e-s)); + uint dg = cast(uint)((s1.color.rgba[1]*sc-g)/(e-s)); + uint db = cast(uint)((s1.color.rgba[2]*sc-b)/(e-s)); + uint da = cast(uint)((s1.color.rgba[3]*sc-a)/(e-s)); + for (uint i = s; i < e; ++i) { + version(BigEndian) { + dst[i] = ((r>>sh)<<24)+((g>>sh)<<16)+((b>>sh)<<8)+((a>>sh)<<0); + } else { + dst[i] = ((a>>sh)<<24)+((b>>sh)<<16)+((g>>sh)<<8)+((r>>sh)<<0); + } + r += dr; + g += dg; + b += db; + a += da; + } + } + + uint[NVG_GRADIENT_SAMPLES] data = void; + float w = ex-sx; + float h = ey-sy; + float len = nvg__sqrtf(w*w + h*h); + auto s0 = NVGGradientStop(0, nvgRGBAf(0, 0, 0, 1)); + auto s1 = NVGGradientStop(1, nvgRGBAf(1, 1, 1, 1)); + int img; + if (stops.length > 64) stops = stops[0..64]; + if (stops.length) { + s0.color = stops[0].color; + s1.color = stops[$-1].color; + } + gradientSpan(data.ptr, &s0, (stops.length ? stops.ptr : &s1)); + if (stops.length) { + foreach (immutable i; 0..stops.length-1) gradientSpan(data.ptr, stops.ptr+i, stops.ptr+i+1); + } + gradientSpan(data.ptr, (stops.length ? stops.ptr+stops.length-1 : &s0), &s1); + img = ctx.createImageRGBA(NVG_GRADIENT_SAMPLES, 1, data[], NVGImageFlags.RepeatX|NVGImageFlags.RepeatY); + if (img <= 0) return null; + // allocate data + NVGLGS res = cast(NVGLGS)malloc((*NVGLGS).sizeof); + if (res is null) { ctx.deleteImage(img); return null; } + // fill result + res.imgid = img; + res.cx = sx; + res.cy = sy; + res.w = len; + res.h = len; + res.angle = nvg__atan2f(ey-sy, ex-sx); + return res; +} + + +// ////////////////////////////////////////////////////////////////////////// // +// Scissoring + +/// Sets the current scissor rectangle. The scissor rectangle is transformed by the current transform. +/// Arguments: [x, y, w, h]* +/// Group: scissoring +public void scissor (NVGContext ctx, in float[] args...) nothrow @trusted @nogc { + enum ArgC = 4; + if (args.length%ArgC != 0) assert(0, "NanoVega: invalid [scissor] call"); + if (args.length < ArgC) return; + NVGstate* state = nvg__getState(ctx); + const(float)* aptr = args.ptr; + foreach (immutable idx; 0..args.length/ArgC) { + immutable x = *aptr++; + immutable y = *aptr++; + immutable w = nvg__max(0.0f, *aptr++); + immutable h = nvg__max(0.0f, *aptr++); + + state.scissor.xform.identity; + state.scissor.xform.mat.ptr[4] = x+w*0.5f; + state.scissor.xform.mat.ptr[5] = y+h*0.5f; + //nvgTransformMultiply(state.scissor.xform[], state.xform[]); + state.scissor.xform.mul(state.xform); + + state.scissor.extent.ptr[0] = w*0.5f; + state.scissor.extent.ptr[1] = h*0.5f; + } +} + +void nvg__isectRects (float* dst, float ax, float ay, float aw, float ah, float bx, float by, float bw, float bh) nothrow @trusted @nogc { + immutable float minx = nvg__max(ax, bx); + immutable float miny = nvg__max(ay, by); + immutable float maxx = nvg__min(ax+aw, bx+bw); + immutable float maxy = nvg__min(ay+ah, by+bh); + dst[0] = minx; + dst[1] = miny; + dst[2] = nvg__max(0.0f, maxx-minx); + dst[3] = nvg__max(0.0f, maxy-miny); +} + +/** Intersects current scissor rectangle with the specified rectangle. + * The scissor rectangle is transformed by the current transform. + * Note: in case the rotation of previous scissor rect differs from + * the current one, the intersection will be done between the specified + * rectangle and the previous scissor rectangle transformed in the current + * transform space. The resulting shape is always rectangle. + * + * Arguments: [x, y, w, h]* + * + * Group: scissoring + */ +public void intersectScissor (NVGContext ctx, in float[] args...) nothrow @trusted @nogc { + enum ArgC = 4; + if (args.length%ArgC != 0) assert(0, "NanoVega: invalid [intersectScissor] call"); + if (args.length < ArgC) return; + NVGstate* state = nvg__getState(ctx); + const(float)* aptr = args.ptr; + foreach (immutable idx; 0..args.length/ArgC) { + immutable x = *aptr++; + immutable y = *aptr++; + immutable w = *aptr++; + immutable h = *aptr++; + + // If no previous scissor has been set, set the scissor as current scissor. + if (state.scissor.extent.ptr[0] < 0) { + ctx.scissor(x, y, w, h); + continue; + } + + NVGMatrix pxform = void; + NVGMatrix invxorm = void; + float[4] rect = void; + + // Transform the current scissor rect into current transform space. + // If there is difference in rotation, this will be approximation. + //memcpy(pxform.mat.ptr, state.scissor.xform.ptr, float.sizeof*6); + pxform = state.scissor.xform; + immutable float ex = state.scissor.extent.ptr[0]; + immutable float ey = state.scissor.extent.ptr[1]; + //nvgTransformInverse(invxorm[], state.xform[]); + invxorm = state.xform.inverted; + //nvgTransformMultiply(pxform[], invxorm[]); + pxform.mul(invxorm); + immutable float tex = ex*nvg__absf(pxform.mat.ptr[0])+ey*nvg__absf(pxform.mat.ptr[2]); + immutable float tey = ex*nvg__absf(pxform.mat.ptr[1])+ey*nvg__absf(pxform.mat.ptr[3]); + + // Intersect rects. + nvg__isectRects(rect.ptr, pxform.mat.ptr[4]-tex, pxform.mat.ptr[5]-tey, tex*2, tey*2, x, y, w, h); + + //ctx.scissor(rect.ptr[0], rect.ptr[1], rect.ptr[2], rect.ptr[3]); + ctx.scissor(rect.ptr[0..4]); + } +} + +/// Reset and disables scissoring. +/// Group: scissoring +public void resetScissor (NVGContext ctx) nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + state.scissor.xform.mat[] = 0.0f; + state.scissor.extent[] = -1.0f; +} + + +// ////////////////////////////////////////////////////////////////////////// // +// Render-Time Affine Transformations + +/// Sets GPU affine transformatin matrix. Don't do scaling or skewing here. +/// This matrix won't be saved/restored with context state save/restore operations, as it is not a part of that state. +/// Group: gpu_affine +public void affineGPU() (NVGContext ctx, in auto ref NVGMatrix mat) nothrow @trusted @nogc { + ctx.gpuAffine = mat; + ctx.params.renderSetAffine(ctx.params.userPtr, ctx.gpuAffine.mat[]); +} + +/// Get current GPU affine transformatin matrix. +/// Group: gpu_affine +public NVGMatrix affineGPU (NVGContext ctx) nothrow @safe @nogc { + pragma(inline, true); + return ctx.gpuAffine; +} + +/// "Untransform" point using current GPU affine matrix. +/// Group: gpu_affine +public void gpuUntransformPoint (NVGContext ctx, float *dx, float *dy, in float x, in float y) nothrow @safe @nogc { + if (ctx.gpuAffine.isIdentity) { + if (dx !is null) *dx = x; + if (dy !is null) *dy = y; + } else { + // inverse GPU transformation + NVGMatrix igpu = ctx.gpuAffine.inverted; + igpu.point(dx, dy, x, y); + } +} + + +// ////////////////////////////////////////////////////////////////////////// // +// rasterization (tesselation) code + +int nvg__ptEquals (float x1, float y1, float x2, float y2, float tol) pure nothrow @safe @nogc { + //pragma(inline, true); + immutable float dx = x2-x1; + immutable float dy = y2-y1; + return dx*dx+dy*dy < tol*tol; +} + +float nvg__distPtSeg (float x, float y, float px, float py, float qx, float qy) pure nothrow @safe @nogc { + immutable float pqx = qx-px; + immutable float pqy = qy-py; + float dx = x-px; + float dy = y-py; + immutable float d = pqx*pqx+pqy*pqy; + float t = pqx*dx+pqy*dy; + if (d > 0) t /= d; + if (t < 0) t = 0; else if (t > 1) t = 1; + dx = px+t*pqx-x; + dy = py+t*pqy-y; + return dx*dx+dy*dy; +} + +void nvg__appendCommands(bool useCommand=true) (NVGContext ctx, Command acmd, const(float)[] vals...) nothrow @trusted @nogc { + int nvals = cast(int)vals.length; + static if (useCommand) { + enum addon = 1; + } else { + enum addon = 0; + if (nvals == 0) return; // nothing to do + } + + NVGstate* state = nvg__getState(ctx); + + if (ctx.ncommands+nvals+addon > ctx.ccommands) { + //int ccommands = ctx.ncommands+nvals+ctx.ccommands/2; + int ccommands = ((ctx.ncommands+(nvals+addon))|0xfff)+1; + float* commands = cast(float*)realloc(ctx.commands, float.sizeof*ccommands); + if (commands is null) assert(0, "NanoVega: out of memory"); + ctx.commands = commands; + ctx.ccommands = ccommands; + assert(ctx.ncommands+(nvals+addon) <= ctx.ccommands); + } + + static if (!useCommand) acmd = cast(Command)vals.ptr[0]; + + if (acmd != Command.Close && acmd != Command.Winding) { + //assert(nvals+addon >= 3); + ctx.commandx = vals.ptr[nvals-2]; + ctx.commandy = vals.ptr[nvals-1]; + } + + // copy commands + float* vp = ctx.commands+ctx.ncommands; + static if (useCommand) { + vp[0] = cast(float)acmd; + if (nvals > 0) memcpy(vp+1, vals.ptr, nvals*float.sizeof); + } else { + memcpy(vp, vals.ptr, nvals*float.sizeof); + } + ctx.ncommands += nvals+addon; + + // transform commands + int i = nvals+addon; + while (i > 0) { + int nlen = 1; + final switch (cast(Command)(*vp)) { + case Command.MoveTo: + case Command.LineTo: + assert(i >= 3); + state.xform.point(vp+1, vp+2, vp[1], vp[2]); + nlen = 3; + break; + case Command.BezierTo: + assert(i >= 7); + state.xform.point(vp+1, vp+2, vp[1], vp[2]); + state.xform.point(vp+3, vp+4, vp[3], vp[4]); + state.xform.point(vp+5, vp+6, vp[5], vp[6]); + nlen = 7; + break; + case Command.Close: + nlen = 1; + break; + case Command.Winding: + nlen = 2; + break; + } + assert(nlen > 0 && nlen <= i); + i -= nlen; + vp += nlen; + } +} + +void nvg__clearPathCache (NVGContext ctx) nothrow @trusted @nogc { + // no need to clear paths, as data is not copied there + //foreach (ref p; ctx.cache.paths[0..ctx.cache.npaths]) p.clear(); + ctx.cache.npoints = 0; + ctx.cache.npaths = 0; + ctx.cache.fillReady = ctx.cache.strokeReady = false; +} + +NVGpath* nvg__lastPath (NVGContext ctx) nothrow @trusted @nogc { + return (ctx.cache.npaths > 0 ? &ctx.cache.paths[ctx.cache.npaths-1] : null); +} + +void nvg__addPath (NVGContext ctx) nothrow @trusted @nogc { + import core.stdc.stdlib : realloc; + import core.stdc.string : memset; + + if (ctx.cache.npaths+1 > ctx.cache.cpaths) { + int cpaths = ctx.cache.npaths+1+ctx.cache.cpaths/2; + NVGpath* paths = cast(NVGpath*)realloc(ctx.cache.paths, NVGpath.sizeof*cpaths); + if (paths is null) assert(0, "NanoVega: out of memory"); + ctx.cache.paths = paths; + ctx.cache.cpaths = cpaths; + } + + NVGpath* path = &ctx.cache.paths[ctx.cache.npaths++]; + memset(path, 0, NVGpath.sizeof); + path.first = ctx.cache.npoints; + path.winding = NVGWinding.CCW; +} + +NVGpoint* nvg__lastPoint (NVGContext ctx) nothrow @trusted @nogc { + return (ctx.cache.npoints > 0 ? &ctx.cache.points[ctx.cache.npoints-1] : null); +} + +void nvg__addPoint (NVGContext ctx, float x, float y, int flags) nothrow @trusted @nogc { + NVGpath* path = nvg__lastPath(ctx); + if (path is null) return; + + if (path.count > 0 && ctx.cache.npoints > 0) { + NVGpoint* pt = nvg__lastPoint(ctx); + if (nvg__ptEquals(pt.x, pt.y, x, y, ctx.distTol)) { + pt.flags |= flags; + return; + } + } + + if (ctx.cache.npoints+1 > ctx.cache.cpoints) { + int cpoints = ctx.cache.npoints+1+ctx.cache.cpoints/2; + NVGpoint* points = cast(NVGpoint*)realloc(ctx.cache.points, NVGpoint.sizeof*cpoints); + if (points is null) return; + ctx.cache.points = points; + ctx.cache.cpoints = cpoints; + } + + NVGpoint* pt = &ctx.cache.points[ctx.cache.npoints]; + memset(pt, 0, (*pt).sizeof); + pt.x = x; + pt.y = y; + pt.flags = cast(ubyte)flags; + + ++ctx.cache.npoints; + ++path.count; +} + +void nvg__closePath (NVGContext ctx) nothrow @trusted @nogc { + NVGpath* path = nvg__lastPath(ctx); + if (path is null) return; + path.closed = true; +} + +void nvg__pathWinding (NVGContext ctx, NVGWinding winding) nothrow @trusted @nogc { + NVGpath* path = nvg__lastPath(ctx); + if (path is null) return; + path.winding = winding; +} + +float nvg__getAverageScale() (in auto ref NVGMatrix t) nothrow @trusted @nogc { + immutable float sx = nvg__sqrtf(t.mat.ptr[0]*t.mat.ptr[0]+t.mat.ptr[2]*t.mat.ptr[2]); + immutable float sy = nvg__sqrtf(t.mat.ptr[1]*t.mat.ptr[1]+t.mat.ptr[3]*t.mat.ptr[3]); + return (sx+sy)*0.5f; +} + +NVGvertex* nvg__allocTempVerts (NVGContext ctx, int nverts) nothrow @trusted @nogc { + if (nverts > ctx.cache.cverts) { + int cverts = (nverts+0xff)&~0xff; // Round up to prevent allocations when things change just slightly. + NVGvertex* verts = cast(NVGvertex*)realloc(ctx.cache.verts, NVGvertex.sizeof*cverts); + if (verts is null) return null; + ctx.cache.verts = verts; + ctx.cache.cverts = cverts; + } + + return ctx.cache.verts; +} + +float nvg__triarea2 (float ax, float ay, float bx, float by, float cx, float cy) pure nothrow @safe @nogc { + immutable float abx = bx-ax; + immutable float aby = by-ay; + immutable float acx = cx-ax; + immutable float acy = cy-ay; + return acx*aby-abx*acy; +} + +float nvg__polyArea (NVGpoint* pts, int npts) nothrow @trusted @nogc { + float area = 0; + foreach (int i; 2..npts) { + NVGpoint* a = &pts[0]; + NVGpoint* b = &pts[i-1]; + NVGpoint* c = &pts[i]; + area += nvg__triarea2(a.x, a.y, b.x, b.y, c.x, c.y); + } + return area*0.5f; +} + +void nvg__polyReverse (NVGpoint* pts, int npts) nothrow @trusted @nogc { + NVGpoint tmp = void; + int i = 0, j = npts-1; + while (i < j) { + tmp = pts[i]; + pts[i] = pts[j]; + pts[j] = tmp; + ++i; + --j; + } +} + +void nvg__vset (NVGvertex* vtx, float x, float y, float u, float v) nothrow @trusted @nogc { + vtx.x = x; + vtx.y = y; + vtx.u = u; + vtx.v = v; +} + +void nvg__tesselateBezier (NVGContext ctx, in float x1, in float y1, in float x2, in float y2, in float x3, in float y3, in float x4, in float y4, in int level, in int type) nothrow @trusted @nogc { + if (level > 10) return; + + // check for collinear points, and use AFD tesselator on such curves (it is WAY faster for this case) + if (level == 0 && ctx.tesselatortype == NVGTesselation.Combined) { + static bool collinear (in float v0x, in float v0y, in float v1x, in float v1y, in float v2x, in float v2y) nothrow @trusted @nogc { + immutable float cz = (v1x-v0x)*(v2y-v0y)-(v2x-v0x)*(v1y-v0y); + return (nvg__absf(cz*cz) <= 0.01f); // arbitrary number, seems to work ok with NanoSVG output + } + if (collinear(x1, y1, x2, y2, x3, y3) && collinear(x2, y2, x3, y3, x3, y4)) { + //{ import core.stdc.stdio; printf("AFD fallback!\n"); } + ctx.nvg__tesselateBezierAFD(x1, y1, x2, y2, x3, y3, x4, y4, type); + return; + } + } + + immutable float x12 = (x1+x2)*0.5f; + immutable float y12 = (y1+y2)*0.5f; + immutable float x23 = (x2+x3)*0.5f; + immutable float y23 = (y2+y3)*0.5f; + immutable float x34 = (x3+x4)*0.5f; + immutable float y34 = (y3+y4)*0.5f; + immutable float x123 = (x12+x23)*0.5f; + immutable float y123 = (y12+y23)*0.5f; + + immutable float dx = x4-x1; + immutable float dy = y4-y1; + immutable float d2 = nvg__absf(((x2-x4)*dy-(y2-y4)*dx)); + immutable float d3 = nvg__absf(((x3-x4)*dy-(y3-y4)*dx)); + + if ((d2+d3)*(d2+d3) < ctx.tessTol*(dx*dx+dy*dy)) { + nvg__addPoint(ctx, x4, y4, type); + return; + } + + /* + if (nvg__absf(x1+x3-x2-x2)+nvg__absf(y1+y3-y2-y2)+nvg__absf(x2+x4-x3-x3)+nvg__absf(y2+y4-y3-y3) < ctx.tessTol) { + nvg__addPoint(ctx, x4, y4, type); + return; + } + */ + + immutable float x234 = (x23+x34)*0.5f; + immutable float y234 = (y23+y34)*0.5f; + immutable float x1234 = (x123+x234)*0.5f; + immutable float y1234 = (y123+y234)*0.5f; + + nvg__tesselateBezier(ctx, x1, y1, x12, y12, x123, y123, x1234, y1234, level+1, 0); + nvg__tesselateBezier(ctx, x1234, y1234, x234, y234, x34, y34, x4, y4, level+1, type); +} + +// Adaptive forward differencing for bezier tesselation. +// See Lien, Sheue-Ling, Michael Shantz, and Vaughan Pratt. +// "Adaptive forward differencing for rendering curves and surfaces." +// ACM SIGGRAPH Computer Graphics. Vol. 21. No. 4. ACM, 1987. +// original code by Taylor Holliday +void nvg__tesselateBezierAFD (NVGContext ctx, in float x1, in float y1, in float x2, in float y2, in float x3, in float y3, in float x4, in float y4, in int type) nothrow @trusted @nogc { + enum AFD_ONE = (1<<10); + + // power basis + immutable float ax = -x1+3*x2-3*x3+x4; + immutable float ay = -y1+3*y2-3*y3+y4; + immutable float bx = 3*x1-6*x2+3*x3; + immutable float by = 3*y1-6*y2+3*y3; + immutable float cx = -3*x1+3*x2; + immutable float cy = -3*y1+3*y2; + + // Transform to forward difference basis (stepsize 1) + float px = x1; + float py = y1; + float dx = ax+bx+cx; + float dy = ay+by+cy; + float ddx = 6*ax+2*bx; + float ddy = 6*ay+2*by; + float dddx = 6*ax; + float dddy = 6*ay; + + //printf("dx: %f, dy: %f\n", dx, dy); + //printf("ddx: %f, ddy: %f\n", ddx, ddy); + //printf("dddx: %f, dddy: %f\n", dddx, dddy); + + int t = 0; + int dt = AFD_ONE; + + immutable float tol = ctx.tessTol*4; + + while (t < AFD_ONE) { + // Flatness measure. + float d = ddx*ddx+ddy*ddy+dddx*dddx+dddy*dddy; + + // printf("d: %f, th: %f\n", d, th); + + // Go to higher resolution if we're moving a lot or overshooting the end. + while ((d > tol && dt > 1) || (t+dt > AFD_ONE)) { + // printf("up\n"); + + // Apply L to the curve. Increase curve resolution. + dx = 0.5f*dx-(1.0f/8.0f)*ddx+(1.0f/16.0f)*dddx; + dy = 0.5f*dy-(1.0f/8.0f)*ddy+(1.0f/16.0f)*dddy; + ddx = (1.0f/4.0f)*ddx-(1.0f/8.0f)*dddx; + ddy = (1.0f/4.0f)*ddy-(1.0f/8.0f)*dddy; + dddx = (1.0f/8.0f)*dddx; + dddy = (1.0f/8.0f)*dddy; + + // Half the stepsize. + dt >>= 1; + + // Recompute d + d = ddx*ddx+ddy*ddy+dddx*dddx+dddy*dddy; + } + + // Go to lower resolution if we're really flat + // and we aren't going to overshoot the end. + // XXX: tol/32 is just a guess for when we are too flat. + while ((d > 0 && d < tol/32.0f && dt < AFD_ONE) && (t+2*dt <= AFD_ONE)) { + // printf("down\n"); + + // Apply L^(-1) to the curve. Decrease curve resolution. + dx = 2*dx+ddx; + dy = 2*dy+ddy; + ddx = 4*ddx+4*dddx; + ddy = 4*ddy+4*dddy; + dddx = 8*dddx; + dddy = 8*dddy; + + // Double the stepsize. + dt <<= 1; + + // Recompute d + d = ddx*ddx+ddy*ddy+dddx*dddx+dddy*dddy; + } + + // Forward differencing. + px += dx; + py += dy; + dx += ddx; + dy += ddy; + ddx += dddx; + ddy += dddy; + + // Output a point. + nvg__addPoint(ctx, px, py, (t > 0 ? type : 0)); + + // Advance along the curve. + t += dt; + + // Ensure we don't overshoot. + assert(t <= AFD_ONE); + } +} + +version(nanovg_bench_flatten) import iv.timer : Timer; + +void nvg__flattenPaths (NVGContext ctx) nothrow @trusted @nogc { + version(nanovg_bench_flatten) { + Timer timer; + char[128] tmbuf; + int bzcount; + } + NVGpathCache* cache = ctx.cache; + + if (cache.npaths > 0) return; + + // flatten + version(nanovg_bench_flatten) timer.restart(); + int i = 0; + while (i < ctx.ncommands) { + final switch (cast(Command)ctx.commands[i]) { + case Command.MoveTo: + //assert(i+3 <= ctx.ncommands); + nvg__addPath(ctx); + const p = &ctx.commands[i+1]; + nvg__addPoint(ctx, p[0], p[1], PointFlag.Corner); + i += 3; + break; + case Command.LineTo: + //assert(i+3 <= ctx.ncommands); + const p = &ctx.commands[i+1]; + nvg__addPoint(ctx, p[0], p[1], PointFlag.Corner); + i += 3; + break; + case Command.BezierTo: + //assert(i+7 <= ctx.ncommands); + const last = nvg__lastPoint(ctx); + if (last !is null) { + const cp1 = &ctx.commands[i+1]; + const cp2 = &ctx.commands[i+3]; + const p = &ctx.commands[i+5]; + if (ctx.tesselatortype != NVGTesselation.AFD) { + nvg__tesselateBezier(ctx, last.x, last.y, cp1[0], cp1[1], cp2[0], cp2[1], p[0], p[1], 0, PointFlag.Corner); + } else { + nvg__tesselateBezierAFD(ctx, last.x, last.y, cp1[0], cp1[1], cp2[0], cp2[1], p[0], p[1], PointFlag.Corner); + } + version(nanovg_bench_flatten) ++bzcount; + } + i += 7; + break; + case Command.Close: + //assert(i+1 <= ctx.ncommands); + nvg__closePath(ctx); + i += 1; + break; + case Command.Winding: + //assert(i+2 <= ctx.ncommands); + nvg__pathWinding(ctx, cast(NVGWinding)ctx.commands[i+1]); + i += 2; + break; + } + } + version(nanovg_bench_flatten) {{ + timer.stop(); + auto xb = timer.toBuffer(tmbuf[]); + import core.stdc.stdio : printf; + printf("flattening time: [%.*s] (%d beziers)\n", cast(uint)xb.length, xb.ptr, bzcount); + }} + + cache.bounds.ptr[0] = cache.bounds.ptr[1] = 1e6f; + cache.bounds.ptr[2] = cache.bounds.ptr[3] = -1e6f; + + // calculate the direction and length of line segments + version(nanovg_bench_flatten) timer.restart(); + foreach (int j; 0..cache.npaths) { + NVGpath* path = &cache.paths[j]; + NVGpoint* pts = &cache.points[path.first]; + + // if the first and last points are the same, remove the last, mark as closed path + NVGpoint* p0 = &pts[path.count-1]; + NVGpoint* p1 = &pts[0]; + if (nvg__ptEquals(p0.x, p0.y, p1.x, p1.y, ctx.distTol)) { + --path.count; + p0 = &pts[path.count-1]; + path.closed = true; + } + + // enforce winding + if (path.count > 2) { + immutable float area = nvg__polyArea(pts, path.count); + if (path.winding == NVGWinding.CCW && area < 0.0f) nvg__polyReverse(pts, path.count); + if (path.winding == NVGWinding.CW && area > 0.0f) nvg__polyReverse(pts, path.count); + } + + foreach (immutable _; 0..path.count) { + // calculate segment direction and length + p0.dx = p1.x-p0.x; + p0.dy = p1.y-p0.y; + p0.len = nvg__normalize(&p0.dx, &p0.dy); + // update bounds + cache.bounds.ptr[0] = nvg__min(cache.bounds.ptr[0], p0.x); + cache.bounds.ptr[1] = nvg__min(cache.bounds.ptr[1], p0.y); + cache.bounds.ptr[2] = nvg__max(cache.bounds.ptr[2], p0.x); + cache.bounds.ptr[3] = nvg__max(cache.bounds.ptr[3], p0.y); + // advance + p0 = p1++; + } + } + version(nanovg_bench_flatten) {{ + timer.stop(); + auto xb = timer.toBuffer(tmbuf[]); + import core.stdc.stdio : printf; + printf("segment calculation time: [%.*s]\n", cast(uint)xb.length, xb.ptr); + }} +} + +int nvg__curveDivs (float r, float arc, float tol) nothrow @trusted @nogc { + immutable float da = nvg__acosf(r/(r+tol))*2.0f; + return nvg__max(2, cast(int)nvg__ceilf(arc/da)); +} + +void nvg__chooseBevel (int bevel, NVGpoint* p0, NVGpoint* p1, float w, float* x0, float* y0, float* x1, float* y1) nothrow @trusted @nogc { + if (bevel) { + *x0 = p1.x+p0.dy*w; + *y0 = p1.y-p0.dx*w; + *x1 = p1.x+p1.dy*w; + *y1 = p1.y-p1.dx*w; + } else { + *x0 = p1.x+p1.dmx*w; + *y0 = p1.y+p1.dmy*w; + *x1 = p1.x+p1.dmx*w; + *y1 = p1.y+p1.dmy*w; + } +} + +NVGvertex* nvg__roundJoin (NVGvertex* dst, NVGpoint* p0, NVGpoint* p1, float lw, float rw, float lu, float ru, int ncap, float fringe) nothrow @trusted @nogc { + float dlx0 = p0.dy; + float dly0 = -p0.dx; + float dlx1 = p1.dy; + float dly1 = -p1.dx; + //NVG_NOTUSED(fringe); + + if (p1.flags&PointFlag.Left) { + float lx0 = void, ly0 = void, lx1 = void, ly1 = void; + nvg__chooseBevel(p1.flags&PointFlag.InnerBevelPR, p0, p1, lw, &lx0, &ly0, &lx1, &ly1); + immutable float a0 = nvg__atan2f(-dly0, -dlx0); + float a1 = nvg__atan2f(-dly1, -dlx1); + if (a1 > a0) a1 -= NVG_PI*2; + + nvg__vset(dst, lx0, ly0, lu, 1); ++dst; + nvg__vset(dst, p1.x-dlx0*rw, p1.y-dly0*rw, ru, 1); ++dst; + + int n = nvg__clamp(cast(int)nvg__ceilf(((a0-a1)/NVG_PI)*ncap), 2, ncap); + for (int i = 0; i < n; ++i) { + float u = i/cast(float)(n-1); + float a = a0+u*(a1-a0); + float rx = p1.x+nvg__cosf(a)*rw; + float ry = p1.y+nvg__sinf(a)*rw; + nvg__vset(dst, p1.x, p1.y, 0.5f, 1); ++dst; + nvg__vset(dst, rx, ry, ru, 1); ++dst; + } + + nvg__vset(dst, lx1, ly1, lu, 1); ++dst; + nvg__vset(dst, p1.x-dlx1*rw, p1.y-dly1*rw, ru, 1); ++dst; + + } else { + float rx0 = void, ry0 = void, rx1 = void, ry1 = void; + nvg__chooseBevel(p1.flags&PointFlag.InnerBevelPR, p0, p1, -rw, &rx0, &ry0, &rx1, &ry1); + immutable float a0 = nvg__atan2f(dly0, dlx0); + float a1 = nvg__atan2f(dly1, dlx1); + if (a1 < a0) a1 += NVG_PI*2; + + nvg__vset(dst, p1.x+dlx0*rw, p1.y+dly0*rw, lu, 1); ++dst; + nvg__vset(dst, rx0, ry0, ru, 1); ++dst; + + int n = nvg__clamp(cast(int)nvg__ceilf(((a1-a0)/NVG_PI)*ncap), 2, ncap); + for (int i = 0; i < n; i++) { + float u = i/cast(float)(n-1); + float a = a0+u*(a1-a0); + float lx = p1.x+nvg__cosf(a)*lw; + float ly = p1.y+nvg__sinf(a)*lw; + nvg__vset(dst, lx, ly, lu, 1); ++dst; + nvg__vset(dst, p1.x, p1.y, 0.5f, 1); ++dst; + } + + nvg__vset(dst, p1.x+dlx1*rw, p1.y+dly1*rw, lu, 1); ++dst; + nvg__vset(dst, rx1, ry1, ru, 1); ++dst; + + } + return dst; +} + +NVGvertex* nvg__bevelJoin (NVGvertex* dst, NVGpoint* p0, NVGpoint* p1, float lw, float rw, float lu, float ru, float fringe) nothrow @trusted @nogc { + float rx0, ry0, rx1, ry1; + float lx0, ly0, lx1, ly1; + float dlx0 = p0.dy; + float dly0 = -p0.dx; + float dlx1 = p1.dy; + float dly1 = -p1.dx; + //NVG_NOTUSED(fringe); + + if (p1.flags&PointFlag.Left) { + nvg__chooseBevel(p1.flags&PointFlag.InnerBevelPR, p0, p1, lw, &lx0, &ly0, &lx1, &ly1); + + nvg__vset(dst, lx0, ly0, lu, 1); ++dst; + nvg__vset(dst, p1.x-dlx0*rw, p1.y-dly0*rw, ru, 1); ++dst; + + if (p1.flags&PointFlag.Bevel) { + nvg__vset(dst, lx0, ly0, lu, 1); ++dst; + nvg__vset(dst, p1.x-dlx0*rw, p1.y-dly0*rw, ru, 1); ++dst; + + nvg__vset(dst, lx1, ly1, lu, 1); ++dst; + nvg__vset(dst, p1.x-dlx1*rw, p1.y-dly1*rw, ru, 1); ++dst; + } else { + rx0 = p1.x-p1.dmx*rw; + ry0 = p1.y-p1.dmy*rw; + + nvg__vset(dst, p1.x, p1.y, 0.5f, 1); ++dst; + nvg__vset(dst, p1.x-dlx0*rw, p1.y-dly0*rw, ru, 1); ++dst; + + nvg__vset(dst, rx0, ry0, ru, 1); ++dst; + nvg__vset(dst, rx0, ry0, ru, 1); ++dst; + + nvg__vset(dst, p1.x, p1.y, 0.5f, 1); ++dst; + nvg__vset(dst, p1.x-dlx1*rw, p1.y-dly1*rw, ru, 1); ++dst; + } + + nvg__vset(dst, lx1, ly1, lu, 1); ++dst; + nvg__vset(dst, p1.x-dlx1*rw, p1.y-dly1*rw, ru, 1); ++dst; + + } else { + nvg__chooseBevel(p1.flags&PointFlag.InnerBevelPR, p0, p1, -rw, &rx0, &ry0, &rx1, &ry1); + + nvg__vset(dst, p1.x+dlx0*lw, p1.y+dly0*lw, lu, 1); ++dst; + nvg__vset(dst, rx0, ry0, ru, 1); ++dst; + + if (p1.flags&PointFlag.Bevel) { + nvg__vset(dst, p1.x+dlx0*lw, p1.y+dly0*lw, lu, 1); ++dst; + nvg__vset(dst, rx0, ry0, ru, 1); ++dst; + + nvg__vset(dst, p1.x+dlx1*lw, p1.y+dly1*lw, lu, 1); ++dst; + nvg__vset(dst, rx1, ry1, ru, 1); ++dst; + } else { + lx0 = p1.x+p1.dmx*lw; + ly0 = p1.y+p1.dmy*lw; + + nvg__vset(dst, p1.x+dlx0*lw, p1.y+dly0*lw, lu, 1); ++dst; + nvg__vset(dst, p1.x, p1.y, 0.5f, 1); ++dst; + + nvg__vset(dst, lx0, ly0, lu, 1); ++dst; + nvg__vset(dst, lx0, ly0, lu, 1); ++dst; + + nvg__vset(dst, p1.x+dlx1*lw, p1.y+dly1*lw, lu, 1); ++dst; + nvg__vset(dst, p1.x, p1.y, 0.5f, 1); ++dst; + } + + nvg__vset(dst, p1.x+dlx1*lw, p1.y+dly1*lw, lu, 1); ++dst; + nvg__vset(dst, rx1, ry1, ru, 1); ++dst; + } + + return dst; +} + +NVGvertex* nvg__buttCapStart (NVGvertex* dst, NVGpoint* p, float dx, float dy, float w, float d, float aa) nothrow @trusted @nogc { + immutable float px = p.x-dx*d; + immutable float py = p.y-dy*d; + immutable float dlx = dy; + immutable float dly = -dx; + nvg__vset(dst, px+dlx*w-dx*aa, py+dly*w-dy*aa, 0, 0); ++dst; + nvg__vset(dst, px-dlx*w-dx*aa, py-dly*w-dy*aa, 1, 0); ++dst; + nvg__vset(dst, px+dlx*w, py+dly*w, 0, 1); ++dst; + nvg__vset(dst, px-dlx*w, py-dly*w, 1, 1); ++dst; + return dst; +} + +NVGvertex* nvg__buttCapEnd (NVGvertex* dst, NVGpoint* p, float dx, float dy, float w, float d, float aa) nothrow @trusted @nogc { + immutable float px = p.x+dx*d; + immutable float py = p.y+dy*d; + immutable float dlx = dy; + immutable float dly = -dx; + nvg__vset(dst, px+dlx*w, py+dly*w, 0, 1); ++dst; + nvg__vset(dst, px-dlx*w, py-dly*w, 1, 1); ++dst; + nvg__vset(dst, px+dlx*w+dx*aa, py+dly*w+dy*aa, 0, 0); ++dst; + nvg__vset(dst, px-dlx*w+dx*aa, py-dly*w+dy*aa, 1, 0); ++dst; + return dst; +} + +NVGvertex* nvg__roundCapStart (NVGvertex* dst, NVGpoint* p, float dx, float dy, float w, int ncap, float aa) nothrow @trusted @nogc { + immutable float px = p.x; + immutable float py = p.y; + immutable float dlx = dy; + immutable float dly = -dx; + //NVG_NOTUSED(aa); + immutable float ncpf = cast(float)(ncap-1); + foreach (int i; 0..ncap) { + float a = i/*/cast(float)(ncap-1)*//ncpf*NVG_PI; + float ax = nvg__cosf(a)*w, ay = nvg__sinf(a)*w; + nvg__vset(dst, px-dlx*ax-dx*ay, py-dly*ax-dy*ay, 0, 1); ++dst; + nvg__vset(dst, px, py, 0.5f, 1); ++dst; + } + nvg__vset(dst, px+dlx*w, py+dly*w, 0, 1); ++dst; + nvg__vset(dst, px-dlx*w, py-dly*w, 1, 1); ++dst; + return dst; +} + +NVGvertex* nvg__roundCapEnd (NVGvertex* dst, NVGpoint* p, float dx, float dy, float w, int ncap, float aa) nothrow @trusted @nogc { + immutable float px = p.x; + immutable float py = p.y; + immutable float dlx = dy; + immutable float dly = -dx; + //NVG_NOTUSED(aa); + nvg__vset(dst, px+dlx*w, py+dly*w, 0, 1); ++dst; + nvg__vset(dst, px-dlx*w, py-dly*w, 1, 1); ++dst; + immutable float ncpf = cast(float)(ncap-1); + foreach (int i; 0..ncap) { + float a = i/*cast(float)(ncap-1)*//ncpf*NVG_PI; + float ax = nvg__cosf(a)*w, ay = nvg__sinf(a)*w; + nvg__vset(dst, px, py, 0.5f, 1); ++dst; + nvg__vset(dst, px-dlx*ax+dx*ay, py-dly*ax+dy*ay, 0, 1); ++dst; + } + return dst; +} + +void nvg__calculateJoins (NVGContext ctx, float w, int lineJoin, float miterLimit) nothrow @trusted @nogc { + NVGpathCache* cache = ctx.cache; + float iw = 0.0f; + + if (w > 0.0f) iw = 1.0f/w; + + // Calculate which joins needs extra vertices to append, and gather vertex count. + foreach (int i; 0..cache.npaths) { + NVGpath* path = &cache.paths[i]; + NVGpoint* pts = &cache.points[path.first]; + NVGpoint* p0 = &pts[path.count-1]; + NVGpoint* p1 = &pts[0]; + int nleft = 0; + + path.nbevel = 0; + + foreach (int j; 0..path.count) { + //float dlx0, dly0, dlx1, dly1, dmr2, cross, limit; + immutable float dlx0 = p0.dy; + immutable float dly0 = -p0.dx; + immutable float dlx1 = p1.dy; + immutable float dly1 = -p1.dx; + // Calculate extrusions + p1.dmx = (dlx0+dlx1)*0.5f; + p1.dmy = (dly0+dly1)*0.5f; + immutable float dmr2 = p1.dmx*p1.dmx+p1.dmy*p1.dmy; + if (dmr2 > 0.000001f) { + float scale = 1.0f/dmr2; + if (scale > 600.0f) scale = 600.0f; + p1.dmx *= scale; + p1.dmy *= scale; + } + + // Clear flags, but keep the corner. + p1.flags = (p1.flags&PointFlag.Corner) ? PointFlag.Corner : 0; + + // Keep track of left turns. + immutable float cross = p1.dx*p0.dy-p0.dx*p1.dy; + if (cross > 0.0f) { + nleft++; + p1.flags |= PointFlag.Left; + } + + // Calculate if we should use bevel or miter for inner join. + immutable float limit = nvg__max(1.01f, nvg__min(p0.len, p1.len)*iw); + if ((dmr2*limit*limit) < 1.0f) p1.flags |= PointFlag.InnerBevelPR; + + // Check to see if the corner needs to be beveled. + if (p1.flags&PointFlag.Corner) { + if ((dmr2*miterLimit*miterLimit) < 1.0f || lineJoin == NVGLineCap.Bevel || lineJoin == NVGLineCap.Round) { + p1.flags |= PointFlag.Bevel; + } + } + + if ((p1.flags&(PointFlag.Bevel|PointFlag.InnerBevelPR)) != 0) path.nbevel++; + + p0 = p1++; + } + + path.convex = (nleft == path.count) ? 1 : 0; + } +} + +void nvg__expandStroke (NVGContext ctx, float w, int lineCap, int lineJoin, float miterLimit) nothrow @trusted @nogc { + NVGpathCache* cache = ctx.cache; + immutable float aa = ctx.fringeWidth; + int ncap = nvg__curveDivs(w, NVG_PI, ctx.tessTol); // Calculate divisions per half circle. + + nvg__calculateJoins(ctx, w, lineJoin, miterLimit); + + // Calculate max vertex usage. + int cverts = 0; + foreach (int i; 0..cache.npaths) { + NVGpath* path = &cache.paths[i]; + immutable bool loop = path.closed; + if (lineJoin == NVGLineCap.Round) { + cverts += (path.count+path.nbevel*(ncap+2)+1)*2; // plus one for loop + } else { + cverts += (path.count+path.nbevel*5+1)*2; // plus one for loop + } + if (!loop) { + // space for caps + if (lineCap == NVGLineCap.Round) { + cverts += (ncap*2+2)*2; + } else { + cverts += (3+3)*2; + } + } + } + + NVGvertex* verts = nvg__allocTempVerts(ctx, cverts); + if (verts is null) return; + + foreach (int i; 0..cache.npaths) { + NVGpath* path = &cache.paths[i]; + NVGpoint* pts = &cache.points[path.first]; + NVGpoint* p0; + NVGpoint* p1; + int s, e; + + path.fill = null; + path.nfill = 0; + + // Calculate fringe or stroke + immutable bool loop = path.closed; + NVGvertex* dst = verts; + path.stroke = dst; + + if (loop) { + // Looping + p0 = &pts[path.count-1]; + p1 = &pts[0]; + s = 0; + e = path.count; + } else { + // Add cap + p0 = &pts[0]; + p1 = &pts[1]; + s = 1; + e = path.count-1; + } + + if (!loop) { + // Add cap + float dx = p1.x-p0.x; + float dy = p1.y-p0.y; + nvg__normalize(&dx, &dy); + if (lineCap == NVGLineCap.Butt) dst = nvg__buttCapStart(dst, p0, dx, dy, w, -aa*0.5f, aa); + else if (lineCap == NVGLineCap.Butt || lineCap == NVGLineCap.Square) dst = nvg__buttCapStart(dst, p0, dx, dy, w, w-aa, aa); + else if (lineCap == NVGLineCap.Round) dst = nvg__roundCapStart(dst, p0, dx, dy, w, ncap, aa); + } + + foreach (int j; s..e) { + if ((p1.flags&(PointFlag.Bevel|PointFlag.InnerBevelPR)) != 0) { + if (lineJoin == NVGLineCap.Round) { + dst = nvg__roundJoin(dst, p0, p1, w, w, 0, 1, ncap, aa); + } else { + dst = nvg__bevelJoin(dst, p0, p1, w, w, 0, 1, aa); + } + } else { + nvg__vset(dst, p1.x+(p1.dmx*w), p1.y+(p1.dmy*w), 0, 1); ++dst; + nvg__vset(dst, p1.x-(p1.dmx*w), p1.y-(p1.dmy*w), 1, 1); ++dst; + } + p0 = p1++; + } + + if (loop) { + // Loop it + nvg__vset(dst, verts[0].x, verts[0].y, 0, 1); ++dst; + nvg__vset(dst, verts[1].x, verts[1].y, 1, 1); ++dst; + } else { + // Add cap + float dx = p1.x-p0.x; + float dy = p1.y-p0.y; + nvg__normalize(&dx, &dy); + if (lineCap == NVGLineCap.Butt) dst = nvg__buttCapEnd(dst, p1, dx, dy, w, -aa*0.5f, aa); + else if (lineCap == NVGLineCap.Butt || lineCap == NVGLineCap.Square) dst = nvg__buttCapEnd(dst, p1, dx, dy, w, w-aa, aa); + else if (lineCap == NVGLineCap.Round) dst = nvg__roundCapEnd(dst, p1, dx, dy, w, ncap, aa); + } + + path.nstroke = cast(int)(dst-verts); + + verts = dst; + } +} + +void nvg__expandFill (NVGContext ctx, float w, int lineJoin, float miterLimit) nothrow @trusted @nogc { + NVGpathCache* cache = ctx.cache; + immutable float aa = ctx.fringeWidth; + bool fringe = (w > 0.0f); + + nvg__calculateJoins(ctx, w, lineJoin, miterLimit); + + // Calculate max vertex usage. + int cverts = 0; + foreach (int i; 0..cache.npaths) { + NVGpath* path = &cache.paths[i]; + cverts += path.count+path.nbevel+1; + if (fringe) cverts += (path.count+path.nbevel*5+1)*2; // plus one for loop + } + + NVGvertex* verts = nvg__allocTempVerts(ctx, cverts); + if (verts is null) return; + + bool convex = (cache.npaths == 1 && cache.paths[0].convex); + + foreach (int i; 0..cache.npaths) { + NVGpath* path = &cache.paths[i]; + NVGpoint* pts = &cache.points[path.first]; + + // Calculate shape vertices. + immutable float woff = 0.5f*aa; + NVGvertex* dst = verts; + path.fill = dst; + + if (fringe) { + // Looping + NVGpoint* p0 = &pts[path.count-1]; + NVGpoint* p1 = &pts[0]; + foreach (int j; 0..path.count) { + if (p1.flags&PointFlag.Bevel) { + immutable float dlx0 = p0.dy; + immutable float dly0 = -p0.dx; + immutable float dlx1 = p1.dy; + immutable float dly1 = -p1.dx; + if (p1.flags&PointFlag.Left) { + immutable float lx = p1.x+p1.dmx*woff; + immutable float ly = p1.y+p1.dmy*woff; + nvg__vset(dst, lx, ly, 0.5f, 1); ++dst; + } else { + immutable float lx0 = p1.x+dlx0*woff; + immutable float ly0 = p1.y+dly0*woff; + immutable float lx1 = p1.x+dlx1*woff; + immutable float ly1 = p1.y+dly1*woff; + nvg__vset(dst, lx0, ly0, 0.5f, 1); ++dst; + nvg__vset(dst, lx1, ly1, 0.5f, 1); ++dst; + } + } else { + nvg__vset(dst, p1.x+(p1.dmx*woff), p1.y+(p1.dmy*woff), 0.5f, 1); ++dst; + } + p0 = p1++; + } + } else { + foreach (int j; 0..path.count) { + nvg__vset(dst, pts[j].x, pts[j].y, 0.5f, 1); + ++dst; + } + } + + path.nfill = cast(int)(dst-verts); + verts = dst; + + // Calculate fringe + if (fringe) { + float lw = w+woff; + immutable float rw = w-woff; + float lu = 0; + immutable float ru = 1; + dst = verts; + path.stroke = dst; + + // Create only half a fringe for convex shapes so that + // the shape can be rendered without stenciling. + if (convex) { + lw = woff; // This should generate the same vertex as fill inset above. + lu = 0.5f; // Set outline fade at middle. + } + + // Looping + NVGpoint* p0 = &pts[path.count-1]; + NVGpoint* p1 = &pts[0]; + + foreach (int j; 0..path.count) { + if ((p1.flags&(PointFlag.Bevel|PointFlag.InnerBevelPR)) != 0) { + dst = nvg__bevelJoin(dst, p0, p1, lw, rw, lu, ru, ctx.fringeWidth); + } else { + nvg__vset(dst, p1.x+(p1.dmx*lw), p1.y+(p1.dmy*lw), lu, 1); ++dst; + nvg__vset(dst, p1.x-(p1.dmx*rw), p1.y-(p1.dmy*rw), ru, 1); ++dst; + } + p0 = p1++; + } + + // Loop it + nvg__vset(dst, verts[0].x, verts[0].y, lu, 1); ++dst; + nvg__vset(dst, verts[1].x, verts[1].y, ru, 1); ++dst; + + path.nstroke = cast(int)(dst-verts); + verts = dst; + } else { + path.stroke = null; + path.nstroke = 0; + } + } +} + + +// ////////////////////////////////////////////////////////////////////////// // +// Paths + +/// Clears the current path and sub-paths. +/// Will call [nvgOnBeginPath] callback if current path is not empty. +/// Group: paths +public void beginPath (NVGContext ctx) nothrow @trusted @nogc { + ctx.ncommands = 0; + ctx.pathPickRegistered &= NVGPickKind.All; // reset "registered" flags + nvg__clearPathCache(ctx); +} + +public alias newPath = beginPath; /// Ditto. + +/// Starts new sub-path with specified point as first point. +/// Arguments: [x, y]* +/// Group: paths +public void moveTo (NVGContext ctx, in float[] args...) nothrow @trusted @nogc { + enum ArgC = 2; + if (args.length%ArgC != 0) assert(0, "NanoVega: invalid [moveTo] call"); + if (args.length < ArgC) return; + nvg__appendCommands(ctx, Command.MoveTo, args[$-2..$]); +} + +/// Adds line segment from the last point in the path to the specified point. +/// Arguments: [x, y]* +/// Group: paths +public void lineTo (NVGContext ctx, in float[] args...) nothrow @trusted @nogc { + enum ArgC = 2; + if (args.length%ArgC != 0) assert(0, "NanoVega: invalid [lineTo] call"); + if (args.length < ArgC) return; + foreach (immutable idx; 0..args.length/ArgC) { + nvg__appendCommands(ctx, Command.LineTo, args.ptr[idx*ArgC..idx*ArgC+ArgC]); + } +} + +/// Adds cubic bezier segment from last point in the path via two control points to the specified point. +/// Arguments: [c1x, c1y, c2x, c2y, x, y]* +/// Group: paths +public void bezierTo (NVGContext ctx, in float[] args...) nothrow @trusted @nogc { + enum ArgC = 6; + if (args.length%ArgC != 0) assert(0, "NanoVega: invalid [bezierTo] call"); + if (args.length < ArgC) return; + foreach (immutable idx; 0..args.length/ArgC) { + nvg__appendCommands(ctx, Command.BezierTo, args.ptr[idx*ArgC..idx*ArgC+ArgC]); + } +} + +/// Adds quadratic bezier segment from last point in the path via a control point to the specified point. +/// Arguments: [cx, cy, x, y]* +/// Group: paths +public void quadTo (NVGContext ctx, in float[] args...) nothrow @trusted @nogc { + enum ArgC = 4; + if (args.length%ArgC != 0) assert(0, "NanoVega: invalid [quadTo] call"); + if (args.length < ArgC) return; + const(float)* aptr = args.ptr; + foreach (immutable idx; 0..args.length/ArgC) { + immutable float x0 = ctx.commandx; + immutable float y0 = ctx.commandy; + immutable float cx = *aptr++; + immutable float cy = *aptr++; + immutable float x = *aptr++; + immutable float y = *aptr++; + nvg__appendCommands(ctx, + Command.BezierTo, + x0+2.0f/3.0f*(cx-x0), y0+2.0f/3.0f*(cy-y0), + x+2.0f/3.0f*(cx-x), y+2.0f/3.0f*(cy-y), + x, y, + ); + } +} + +/// Adds an arc segment at the corner defined by the last path point, and two specified points. +/// Arguments: [x1, y1, x2, y2, radius]* +/// Group: paths +public void arcTo (NVGContext ctx, in float[] args...) nothrow @trusted @nogc { + enum ArgC = 5; + if (args.length%ArgC != 0) assert(0, "NanoVega: invalid [arcTo] call"); + if (args.length < ArgC) return; + + if (ctx.ncommands == 0) return; + + const(float)* aptr = args.ptr; + foreach (immutable idx; 0..args.length/ArgC) { + immutable float x0 = ctx.commandx; + immutable float y0 = ctx.commandy; + immutable float x1 = *aptr++; + immutable float y1 = *aptr++; + immutable float x2 = *aptr++; + immutable float y2 = *aptr++; + immutable float radius = *aptr++; + + // handle degenerate cases + if (nvg__ptEquals(x0, y0, x1, y1, ctx.distTol) || + nvg__ptEquals(x1, y1, x2, y2, ctx.distTol) || + nvg__distPtSeg(x1, y1, x0, y0, x2, y2) < ctx.distTol*ctx.distTol || + radius < ctx.distTol) + { + ctx.lineTo(x1, y1); + continue; + } + + // calculate tangential circle to lines (x0, y0)-(x1, y1) and (x1, y1)-(x2, y2) + float dx0 = x0-x1; + float dy0 = y0-y1; + float dx1 = x2-x1; + float dy1 = y2-y1; + nvg__normalize(&dx0, &dy0); + nvg__normalize(&dx1, &dy1); + immutable float a = nvg__acosf(dx0*dx1+dy0*dy1); + immutable float d = radius/nvg__tanf(a/2.0f); + + //printf("a=%f° d=%f\n", a/NVG_PI*180.0f, d); + + if (d > 10000.0f) { + ctx.lineTo(x1, y1); + continue; + } + + float cx = void, cy = void, a0 = void, a1 = void; + NVGWinding dir; + if (nvg__cross(dx0, dy0, dx1, dy1) > 0.0f) { + cx = x1+dx0*d+dy0*radius; + cy = y1+dy0*d+-dx0*radius; + a0 = nvg__atan2f(dx0, -dy0); + a1 = nvg__atan2f(-dx1, dy1); + dir = NVGWinding.CW; + //printf("CW c=(%f, %f) a0=%f° a1=%f°\n", cx, cy, a0/NVG_PI*180.0f, a1/NVG_PI*180.0f); + } else { + cx = x1+dx0*d+-dy0*radius; + cy = y1+dy0*d+dx0*radius; + a0 = nvg__atan2f(-dx0, dy0); + a1 = nvg__atan2f(dx1, -dy1); + dir = NVGWinding.CCW; + //printf("CCW c=(%f, %f) a0=%f° a1=%f°\n", cx, cy, a0/NVG_PI*180.0f, a1/NVG_PI*180.0f); + } + + ctx.arc(dir, cx, cy, radius, a0, a1); // first is line + } +} + +/// Closes current sub-path with a line segment. +/// Group: paths +public void closePath (NVGContext ctx) nothrow @trusted @nogc { + nvg__appendCommands(ctx, Command.Close); +} + +/// Sets the current sub-path winding, see NVGWinding and NVGSolidity. +/// Group: paths +public void pathWinding (NVGContext ctx, NVGWinding dir) nothrow @trusted @nogc { + nvg__appendCommands(ctx, Command.Winding, cast(float)dir); +} + +/// Ditto. +public void pathWinding (NVGContext ctx, NVGSolidity dir) nothrow @trusted @nogc { + nvg__appendCommands(ctx, Command.Winding, cast(float)dir); +} + +/** Creates new circle arc shaped sub-path. The arc center is at (cx, cy), the arc radius is r, + * and the arc is drawn from angle a0 to a1, and swept in direction dir (NVGWinding.CCW, or NVGWinding.CW). + * Angles are specified in radians. + * + * Arguments: [cx, cy, r, a0, a1]* + * + * [mode] is: "original", "move", "line" -- first command will be like original NanoVega, MoveTo, or LineTo + * + * Group: paths + */ +public void arc(string mode="original") (NVGContext ctx, NVGWinding dir, in float[] args...) nothrow @trusted @nogc { + static assert(mode == "original" || mode == "move" || mode == "line"); + enum ArgC = 5; + if (args.length%ArgC != 0) assert(0, "NanoVega: invalid [arc] call"); + if (args.length < ArgC) return; + const(float)* aptr = args.ptr; + foreach (immutable idx; 0..args.length/ArgC) { + immutable cx = *aptr++; + immutable cy = *aptr++; + immutable r = *aptr++; + immutable a0 = *aptr++; + immutable a1 = *aptr++; + + float[3+5*7+100] vals = void; + //int move = (ctx.ncommands > 0 ? Command.LineTo : Command.MoveTo); + static if (mode == "original") { + immutable int move = (ctx.ncommands > 0 ? Command.LineTo : Command.MoveTo); + } else static if (mode == "move") { + enum move = Command.MoveTo; + } else static if (mode == "line") { + enum move = Command.LineTo; + } else { + static assert(0, "wtf?!"); + } + + // Clamp angles + float da = a1-a0; + if (dir == NVGWinding.CW) { + if (nvg__absf(da) >= NVG_PI*2) { + da = NVG_PI*2; + } else { + while (da < 0.0f) da += NVG_PI*2; + } + } else { + if (nvg__absf(da) >= NVG_PI*2) { + da = -NVG_PI*2; + } else { + while (da > 0.0f) da -= NVG_PI*2; + } + } + + // Split arc into max 90 degree segments. + immutable int ndivs = nvg__max(1, nvg__min(cast(int)(nvg__absf(da)/(NVG_PI*0.5f)+0.5f), 5)); + immutable float hda = (da/cast(float)ndivs)/2.0f; + float kappa = nvg__absf(4.0f/3.0f*(1.0f-nvg__cosf(hda))/nvg__sinf(hda)); + + if (dir == NVGWinding.CCW) kappa = -kappa; + + int nvals = 0; + float px = 0, py = 0, ptanx = 0, ptany = 0; + foreach (int i; 0..ndivs+1) { + immutable float a = a0+da*(i/cast(float)ndivs); + immutable float dx = nvg__cosf(a); + immutable float dy = nvg__sinf(a); + immutable float x = cx+dx*r; + immutable float y = cy+dy*r; + immutable float tanx = -dy*r*kappa; + immutable float tany = dx*r*kappa; + + if (i == 0) { + if (vals.length-nvals < 3) { + // flush + nvg__appendCommands!false(ctx, Command.MoveTo, vals.ptr[0..nvals]); // ignore command + nvals = 0; + } + vals.ptr[nvals++] = cast(float)move; + vals.ptr[nvals++] = x; + vals.ptr[nvals++] = y; + } else { + if (vals.length-nvals < 7) { + // flush + nvg__appendCommands!false(ctx, Command.MoveTo, vals.ptr[0..nvals]); // ignore command + nvals = 0; + } + vals.ptr[nvals++] = Command.BezierTo; + vals.ptr[nvals++] = px+ptanx; + vals.ptr[nvals++] = py+ptany; + vals.ptr[nvals++] = x-tanx; + vals.ptr[nvals++] = y-tany; + vals.ptr[nvals++] = x; + vals.ptr[nvals++] = y; + } + px = x; + py = y; + ptanx = tanx; + ptany = tany; + } + + nvg__appendCommands!false(ctx, Command.MoveTo, vals.ptr[0..nvals]); // ignore command + } +} + +/// Creates new rectangle shaped sub-path. +/// Arguments: [x, y, w, h]* +/// Group: paths +public void rect (NVGContext ctx, in float[] args...) nothrow @trusted @nogc { + enum ArgC = 4; + if (args.length%ArgC != 0) assert(0, "NanoVega: invalid [rect] call"); + if (args.length < ArgC) return; + const(float)* aptr = args.ptr; + foreach (immutable idx; 0..args.length/ArgC) { + immutable x = *aptr++; + immutable y = *aptr++; + immutable w = *aptr++; + immutable h = *aptr++; + nvg__appendCommands!false(ctx, Command.MoveTo, // ignore command + Command.MoveTo, x, y, + Command.LineTo, x, y+h, + Command.LineTo, x+w, y+h, + Command.LineTo, x+w, y, + Command.Close, + ); + } +} + +/// Creates new rounded rectangle shaped sub-path. +/// Arguments: [x, y, w, h, radius]* +/// Group: paths +public void roundedRect (NVGContext ctx, in float[] args...) nothrow @trusted @nogc { + enum ArgC = 5; + if (args.length%ArgC != 0) assert(0, "NanoVega: invalid [roundedRect] call"); + if (args.length < ArgC) return; + const(float)* aptr = args.ptr; + foreach (immutable idx; 0..args.length/ArgC) { + immutable x = *aptr++; + immutable y = *aptr++; + immutable w = *aptr++; + immutable h = *aptr++; + immutable r = *aptr++; + ctx.roundedRectVarying(x, y, w, h, r, r, r, r); + } +} + +/// Creates new rounded rectangle shaped sub-path. Specify ellipse width and height to round corners according to it. +/// Arguments: [x, y, w, h, rw, rh]* +/// Group: paths +public void roundedRectEllipse (NVGContext ctx, in float[] args...) nothrow @trusted @nogc { + enum ArgC = 6; + if (args.length%ArgC != 0) assert(0, "NanoVega: invalid [roundedRectEllipse] call"); + if (args.length < ArgC) return; + const(float)* aptr = args.ptr; + foreach (immutable idx; 0..args.length/ArgC) { + immutable x = *aptr++; + immutable y = *aptr++; + immutable w = *aptr++; + immutable h = *aptr++; + immutable rw = *aptr++; + immutable rh = *aptr++; + if (rw < 0.1f || rh < 0.1f) { + rect(ctx, x, y, w, h); + } else { + nvg__appendCommands!false(ctx, Command.MoveTo, // ignore command + Command.MoveTo, x+rw, y, + Command.LineTo, x+w-rw, y, + Command.BezierTo, x+w-rw*(1-NVG_KAPPA90), y, x+w, y+rh*(1-NVG_KAPPA90), x+w, y+rh, + Command.LineTo, x+w, y+h-rh, + Command.BezierTo, x+w, y+h-rh*(1-NVG_KAPPA90), x+w-rw*(1-NVG_KAPPA90), y+h, x+w-rw, y+h, + Command.LineTo, x+rw, y+h, + Command.BezierTo, x+rw*(1-NVG_KAPPA90), y+h, x, y+h-rh*(1-NVG_KAPPA90), x, y+h-rh, + Command.LineTo, x, y+rh, + Command.BezierTo, x, y+rh*(1-NVG_KAPPA90), x+rw*(1-NVG_KAPPA90), y, x+rw, y, + Command.Close, + ); + } + } +} + +/// Creates new rounded rectangle shaped sub-path. This one allows you to specify different rounding radii for each corner. +/// Arguments: [x, y, w, h, radTopLeft, radTopRight, radBottomRight, radBottomLeft]* +/// Group: paths +public void roundedRectVarying (NVGContext ctx, in float[] args...) nothrow @trusted @nogc { + enum ArgC = 8; + if (args.length%ArgC != 0) assert(0, "NanoVega: invalid [roundedRectVarying] call"); + if (args.length < ArgC) return; + const(float)* aptr = args.ptr; + foreach (immutable idx; 0..args.length/ArgC) { + immutable x = *aptr++; + immutable y = *aptr++; + immutable w = *aptr++; + immutable h = *aptr++; + immutable radTopLeft = *aptr++; + immutable radTopRight = *aptr++; + immutable radBottomRight = *aptr++; + immutable radBottomLeft = *aptr++; + if (radTopLeft < 0.1f && radTopRight < 0.1f && radBottomRight < 0.1f && radBottomLeft < 0.1f) { + ctx.rect(x, y, w, h); + } else { + immutable float halfw = nvg__absf(w)*0.5f; + immutable float halfh = nvg__absf(h)*0.5f; + immutable float rxBL = nvg__min(radBottomLeft, halfw)*nvg__sign(w), ryBL = nvg__min(radBottomLeft, halfh)*nvg__sign(h); + immutable float rxBR = nvg__min(radBottomRight, halfw)*nvg__sign(w), ryBR = nvg__min(radBottomRight, halfh)*nvg__sign(h); + immutable float rxTR = nvg__min(radTopRight, halfw)*nvg__sign(w), ryTR = nvg__min(radTopRight, halfh)*nvg__sign(h); + immutable float rxTL = nvg__min(radTopLeft, halfw)*nvg__sign(w), ryTL = nvg__min(radTopLeft, halfh)*nvg__sign(h); + nvg__appendCommands!false(ctx, Command.MoveTo, // ignore command + Command.MoveTo, x, y+ryTL, + Command.LineTo, x, y+h-ryBL, + Command.BezierTo, x, y+h-ryBL*(1-NVG_KAPPA90), x+rxBL*(1-NVG_KAPPA90), y+h, x+rxBL, y+h, + Command.LineTo, x+w-rxBR, y+h, + Command.BezierTo, x+w-rxBR*(1-NVG_KAPPA90), y+h, x+w, y+h-ryBR*(1-NVG_KAPPA90), x+w, y+h-ryBR, + Command.LineTo, x+w, y+ryTR, + Command.BezierTo, x+w, y+ryTR*(1-NVG_KAPPA90), x+w-rxTR*(1-NVG_KAPPA90), y, x+w-rxTR, y, + Command.LineTo, x+rxTL, y, + Command.BezierTo, x+rxTL*(1-NVG_KAPPA90), y, x, y+ryTL*(1-NVG_KAPPA90), x, y+ryTL, + Command.Close, + ); + } + } +} + +/// Creates new ellipse shaped sub-path. +/// Arguments: [cx, cy, rx, ry]* +/// Group: paths +public void ellipse (NVGContext ctx, in float[] args...) nothrow @trusted @nogc { + enum ArgC = 4; + if (args.length%ArgC != 0) assert(0, "NanoVega: invalid [ellipse] call"); + if (args.length < ArgC) return; + const(float)* aptr = args.ptr; + foreach (immutable idx; 0..args.length/ArgC) { + immutable cx = *aptr++; + immutable cy = *aptr++; + immutable rx = *aptr++; + immutable ry = *aptr++; + nvg__appendCommands!false(ctx, Command.MoveTo, // ignore command + Command.MoveTo, cx-rx, cy, + Command.BezierTo, cx-rx, cy+ry*NVG_KAPPA90, cx-rx*NVG_KAPPA90, cy+ry, cx, cy+ry, + Command.BezierTo, cx+rx*NVG_KAPPA90, cy+ry, cx+rx, cy+ry*NVG_KAPPA90, cx+rx, cy, + Command.BezierTo, cx+rx, cy-ry*NVG_KAPPA90, cx+rx*NVG_KAPPA90, cy-ry, cx, cy-ry, + Command.BezierTo, cx-rx*NVG_KAPPA90, cy-ry, cx-rx, cy-ry*NVG_KAPPA90, cx-rx, cy, + Command.Close, + ); + } +} + +/// Creates new circle shaped sub-path. +/// Arguments: [cx, cy, r]* +/// Group: paths +public void circle (NVGContext ctx, in float[] args...) nothrow @trusted @nogc { + enum ArgC = 3; + if (args.length%ArgC != 0) assert(0, "NanoVega: invalid [circle] call"); + if (args.length < ArgC) return; + const(float)* aptr = args.ptr; + foreach (immutable idx; 0..args.length/ArgC) { + immutable cx = *aptr++; + immutable cy = *aptr++; + immutable r = *aptr++; + ctx.ellipse(cx, cy, r, r); + } +} + +// Debug function to dump cached path data. +debug public void debugDumpPathCache (NVGContext ctx) nothrow @trusted @nogc { + import core.stdc.stdio : printf; + const(NVGpath)* path; + printf("Dumping %d cached paths\n", ctx.cache.npaths); + for (int i = 0; i < ctx.cache.npaths; ++i) { + path = &ctx.cache.paths[i]; + printf("-Path %d\n", i); + if (path.nfill) { + printf("-fill: %d\n", path.nfill); + for (int j = 0; j < path.nfill; ++j) printf("%f\t%f\n", path.fill[j].x, path.fill[j].y); + } + if (path.nstroke) { + printf("-stroke: %d\n", path.nstroke); + for (int j = 0; j < path.nstroke; ++j) printf("%f\t%f\n", path.stroke[j].x, path.stroke[j].y); + } + } +} + +// Flatten path, prepare it for fill operation. +void nvg__prepareFill (NVGContext ctx) nothrow @trusted @nogc { + NVGpathCache* cache = ctx.cache; + NVGstate* state = nvg__getState(ctx); + + nvg__flattenPaths(ctx); + + if (ctx.params.edgeAntiAlias && state.shapeAntiAlias) { + nvg__expandFill(ctx, ctx.fringeWidth, NVGLineCap.Miter, 2.4f); + } else { + nvg__expandFill(ctx, 0.0f, NVGLineCap.Miter, 2.4f); + } + + cache.evenOddMode = state.evenOddMode; + cache.fringeWidth = ctx.fringeWidth; + cache.fillReady = true; + cache.strokeReady = false; +} + +// Flatten path, prepare it for stroke operation. +void nvg__prepareStroke (NVGContext ctx) nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + NVGpathCache* cache = ctx.cache; + + nvg__flattenPaths(ctx); + + immutable float scale = nvg__getAverageScale(state.xform); + float strokeWidth = nvg__clamp(state.strokeWidth*scale, 0.0f, 200.0f); + + if (strokeWidth < ctx.fringeWidth) { + // If the stroke width is less than pixel size, use alpha to emulate coverage. + // Since coverage is area, scale by alpha*alpha. + immutable float alpha = nvg__clamp(strokeWidth/ctx.fringeWidth, 0.0f, 1.0f); + cache.strokeAlphaMul = alpha*alpha; + strokeWidth = ctx.fringeWidth; + } else { + cache.strokeAlphaMul = 1.0f; + } + cache.strokeWidth = strokeWidth; + + if (ctx.params.edgeAntiAlias && state.shapeAntiAlias) { + nvg__expandStroke(ctx, strokeWidth*0.5f+ctx.fringeWidth*0.5f, state.lineCap, state.lineJoin, state.miterLimit); + } else { + nvg__expandStroke(ctx, strokeWidth*0.5f, state.lineCap, state.lineJoin, state.miterLimit); + } + + cache.fringeWidth = ctx.fringeWidth; + cache.fillReady = false; + cache.strokeReady = true; +} + +/// Fills the current path with current fill style. +/// Group: paths +public void fill (NVGContext ctx) nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + + if (ctx.pathPickId >= 0 && (ctx.pathPickRegistered&(NVGPickKind.Fill|(NVGPickKind.Fill<<16))) == NVGPickKind.Fill) { + ctx.pathPickRegistered |= NVGPickKind.Fill<<16; + ctx.currFillHitId = ctx.pathPickId; + } + + nvg__prepareFill(ctx); + + // apply global alpha + NVGPaint fillPaint = state.fill; + fillPaint.innerColor.a *= state.alpha; + fillPaint.outerColor.a *= state.alpha; + + ctx.appendCurrentPathToCache(ctx.recset, state.fill); + + if (ctx.recblockdraw) return; + + ctx.params.renderFill(ctx.params.userPtr, &fillPaint, &state.scissor, ctx.fringeWidth, ctx.cache.bounds.ptr, ctx.cache.paths, ctx.cache.npaths, state.evenOddMode); + + // count triangles + foreach (int i; 0..ctx.cache.npaths) { + NVGpath* path = &ctx.cache.paths[i]; + ctx.fillTriCount += path.nfill-2; + ctx.fillTriCount += path.nstroke-2; + ctx.drawCallCount += 2; + } +} + +/// Fills the current path with current stroke style. +/// Group: paths +public void stroke (NVGContext ctx) nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + + if (ctx.pathPickId >= 0 && (ctx.pathPickRegistered&(NVGPickKind.Stroke|(NVGPickKind.Stroke<<16))) == NVGPickKind.Stroke) { + ctx.pathPickRegistered |= NVGPickKind.Stroke<<16; + ctx.currStrokeHitId = ctx.pathPickId; + } + + nvg__prepareStroke(ctx); + + NVGpathCache* cache = ctx.cache; + + NVGPaint strokePaint = state.stroke; + strokePaint.innerColor.a *= cache.strokeAlphaMul; + strokePaint.outerColor.a *= cache.strokeAlphaMul; + + // apply global alpha + strokePaint.innerColor.a *= state.alpha; + strokePaint.outerColor.a *= state.alpha; + + ctx.appendCurrentPathToCache(ctx.recset, state.stroke); + + if (ctx.recblockdraw) return; + + ctx.params.renderStroke(ctx.params.userPtr, &strokePaint, &state.scissor, ctx.fringeWidth, cache.strokeWidth, ctx.cache.paths, ctx.cache.npaths); + + // count triangles + foreach (int i; 0..ctx.cache.npaths) { + NVGpath* path = &ctx.cache.paths[i]; + ctx.strokeTriCount += path.nstroke-2; + ++ctx.drawCallCount; + } +} + + +// ////////////////////////////////////////////////////////////////////////// // +// Picking API + +// most of the code is by Michael Wynne +// https://github.com/memononen/nanovg/pull/230 +// https://github.com/MikeWW/nanovg + +/// Pick type query. Used in [hitTest] and [hitTestAll]. +/// Group: picking_api +public enum NVGPickKind : ubyte { + Fill = 0x01, /// + Stroke = 0x02, /// + All = 0x03, /// +} + +/// Marks the fill of the current path as pickable with the specified id. +/// Note that you can create and mark path without rasterizing it. +/// Group: picking_api +public void currFillHitId (NVGContext ctx, int id) nothrow @trusted @nogc { + NVGpickScene* ps = nvg__pickSceneGet(ctx); + NVGpickPath* pp = nvg__pickPathCreate(ctx, ctx.commands[0..ctx.ncommands], id, /*forStroke:*/false); + nvg__pickSceneInsert(ps, pp); +} + +/// Marks the stroke of the current path as pickable with the specified id. +/// Note that you can create and mark path without rasterizing it. +/// Group: picking_api +public void currStrokeHitId (NVGContext ctx, int id) nothrow @trusted @nogc { + NVGpickScene* ps = nvg__pickSceneGet(ctx); + NVGpickPath* pp = nvg__pickPathCreate(ctx, ctx.commands[0..ctx.ncommands], id, /*forStroke:*/true); + nvg__pickSceneInsert(ps, pp); +} + +// Marks the saved path set (fill) as pickable with the specified id. +// WARNING: this doesn't work right yet (it is using current context transformation and other settings instead of record settings)! +// Group: picking_api +/+ +public void pathSetFillHitId (NVGContext ctx, NVGPathSet svp, int id) nothrow @trusted @nogc { + if (svp is null) return; + if (svp.svctx !is ctx) assert(0, "NanoVega: cannot register path set from different context"); + foreach (ref cp; svp.caches[0..svp.ncaches]) { + NVGpickScene* ps = nvg__pickSceneGet(ctx); + NVGpickPath* pp = nvg__pickPathCreate(ctx, cp.commands[0..cp.ncommands], id, /*forStroke:*/false); + nvg__pickSceneInsert(ps, pp); + } +} ++/ + +// Marks the saved path set (stroke) as pickable with the specified id. +// WARNING: this doesn't work right yet (it is using current context transformation and other settings instead of record settings)! +// Group: picking_api +/+ +public void pathSetStrokeHitId (NVGContext ctx, NVGPathSet svp, int id) nothrow @trusted @nogc { + if (svp is null) return; + if (svp.svctx !is ctx) assert(0, "NanoVega: cannot register path set from different context"); + foreach (ref cp; svp.caches[0..svp.ncaches]) { + NVGpickScene* ps = nvg__pickSceneGet(ctx); + NVGpickPath* pp = nvg__pickPathCreate(ctx, cp.commands[0..cp.ncommands], id, /*forStroke:*/true); + nvg__pickSceneInsert(ps, pp); + } +} ++/ + +private template IsGoodHitTestDG(DG) { + enum IsGoodHitTestDG = + __traits(compiles, (){ DG dg; bool res = dg(cast(int)42, cast(int)666); }) || + __traits(compiles, (){ DG dg; dg(cast(int)42, cast(int)666); }); +} + +private template IsGoodHitTestInternalDG(DG) { + enum IsGoodHitTestInternalDG = + __traits(compiles, (){ DG dg; NVGpickPath* pp; bool res = dg(pp); }) || + __traits(compiles, (){ DG dg; NVGpickPath* pp; dg(pp); }); +} + +/// Call delegate [dg] for each path under the specified position (in no particular order). +/// Returns the id of the path for which delegate [dg] returned true or -1. +/// dg is: `bool delegate (int id, int order)` -- [order] is path ordering (ascending). +/// Group: picking_api +public int hitTestDG(DG) (NVGContext ctx, in float x, in float y, uint kind, scope DG dg) if (IsGoodHitTestDG!DG || IsGoodHitTestInternalDG!DG) { + if (ctx.pickScene is null || ctx.pickScene.npaths == 0) return -1; + + NVGpickScene* ps = ctx.pickScene; + int levelwidth = 1<<(ps.nlevels-1); + int cellx = nvg__clamp(cast(int)(x/ps.xdim), 0, levelwidth); + int celly = nvg__clamp(cast(int)(y/ps.ydim), 0, levelwidth); + int npicked = 0; + + //{ import core.stdc.stdio; printf("npaths=%d\n", ps.npaths); } + for (int lvl = ps.nlevels-1; lvl >= 0; --lvl) { + NVGpickPath* pp = ps.levels[lvl][celly*levelwidth+cellx]; + while (pp !is null) { + //{ import core.stdc.stdio; printf("... pos=(%g,%g); bounds=(%g,%g)-(%g,%g); flags=0x%02x; kind=0x%02x\n", x, y, pp.bounds[0], pp.bounds[1], pp.bounds[2], pp.bounds[3], pp.flags, kind); } + if (nvg__pickPathTestBounds(ctx, ps, pp, x, y)) { + //{ import core.stdc.stdio; printf("in bounds!\n"); } + int hit = 0; + if ((kind&NVGPickKind.Stroke) && (pp.flags&NVGPathFlags.Stroke)) hit = nvg__pickPathStroke(ps, pp, x, y); + if (!hit && (kind&NVGPickKind.Fill) && (pp.flags&NVGPathFlags.Fill)) hit = nvg__pickPath(ps, pp, x, y); + if (hit) { + //{ import core.stdc.stdio; printf(" HIT!\n"); } + static if (IsGoodHitTestDG!DG) { + static if (__traits(compiles, (){ DG dg; bool res = dg(cast(int)42, cast(int)666); })) { + if (dg(pp.id, cast(int)pp.order)) return pp.id; + } else { + dg(pp.id, cast(int)pp.order); + } + } else { + static if (__traits(compiles, (){ DG dg; NVGpickPath* pp; bool res = dg(pp); })) { + if (dg(pp)) return pp.id; + } else { + dg(pp); + } + } + } + } + pp = pp.next; + } + cellx >>= 1; + celly >>= 1; + levelwidth >>= 1; + } + + return -1; +} + +/// Fills ids with a list of the top most hit ids under the specified position. +/// Returns the slice of [ids]. +/// Group: picking_api +public int[] hitTestAll (NVGContext ctx, in float x, in float y, uint kind, int[] ids) nothrow @trusted @nogc { + if (ctx.pickScene is null || ids.length == 0) return ids[0..0]; + + int npicked = 0; + NVGpickScene* ps = ctx.pickScene; + + ctx.hitTestDG(x, y, kind, delegate (NVGpickPath* pp) nothrow @trusted @nogc { + if (npicked == ps.cpicked) { + int cpicked = ps.cpicked+ps.cpicked; + NVGpickPath** picked = cast(NVGpickPath**)realloc(ps.picked, (NVGpickPath*).sizeof*ps.cpicked); + if (picked is null) return true; // abort + ps.cpicked = cpicked; + ps.picked = picked; + } + ps.picked[npicked] = pp; + ++npicked; + return false; // go on + }); + + qsort(ps.picked, npicked, (NVGpickPath*).sizeof, &nvg__comparePaths); + + assert(npicked >= 0); + if (npicked > ids.length) npicked = cast(int)ids.length; + foreach (immutable nidx, ref int did; ids[0..npicked]) did = ps.picked[nidx].id; + + return ids[0..npicked]; +} + +/// Returns the id of the pickable shape containing x,y or -1 if no shape was found. +/// Group: picking_api +public int hitTest (NVGContext ctx, in float x, in float y, uint kind) nothrow @trusted @nogc { + if (ctx.pickScene is null) return -1; + + int bestOrder = -1; + int bestID = -1; + + ctx.hitTestDG(x, y, kind, delegate (NVGpickPath* pp) { + if (pp.order > bestOrder) { + bestOrder = pp.order; + bestID = pp.id; + } + }); + + return bestID; +} + +/// Returns `true` if the given point is within the fill of the currently defined path. +/// This operation can be done before rasterizing the current path. +/// Group: picking_api +public bool hitTestCurrFill (NVGContext ctx, in float x, in float y) nothrow @trusted @nogc { + NVGpickScene* ps = nvg__pickSceneGet(ctx); + int oldnpoints = ps.npoints; + int oldnsegments = ps.nsegments; + NVGpickPath* pp = nvg__pickPathCreate(ctx, ctx.commands[0..ctx.ncommands], 1, /*forStroke:*/false); + if (pp is null) return false; // oops + scope(exit) { + nvg__freePickPath(ps, pp); + ps.npoints = oldnpoints; + ps.nsegments = oldnsegments; + } + return (nvg__pointInBounds(x, y, pp.bounds) ? nvg__pickPath(ps, pp, x, y) : false); +} + +/// Returns `true` if the given point is within the stroke of the currently defined path. +/// This operation can be done before rasterizing the current path. +/// Group: picking_api +public bool hitTestCurrStroke (NVGContext ctx, in float x, in float y) nothrow @trusted @nogc { + NVGpickScene* ps = nvg__pickSceneGet(ctx); + int oldnpoints = ps.npoints; + int oldnsegments = ps.nsegments; + NVGpickPath* pp = nvg__pickPathCreate(ctx, ctx.commands[0..ctx.ncommands], 1, /*forStroke:*/true); + if (pp is null) return false; // oops + scope(exit) { + nvg__freePickPath(ps, pp); + ps.npoints = oldnpoints; + ps.nsegments = oldnsegments; + } + return (nvg__pointInBounds(x, y, pp.bounds) ? nvg__pickPathStroke(ps, pp, x, y) : false); +} + + +nothrow @trusted @nogc { +extern(C) { + private alias _compare_fp_t = int function (const void*, const void*) nothrow @nogc; + private extern(C) void qsort (scope void* base, size_t nmemb, size_t size, _compare_fp_t compar) nothrow @nogc; + + extern(C) int nvg__comparePaths (const void* a, const void* b) { + return (*cast(const(NVGpickPath)**)b).order-(*cast(const(NVGpickPath)**)a).order; + } +} + +enum NVGPickEPS = 0.0001f; + +// Segment flags +enum NVGSegmentFlags { + Corner = 1, + Bevel = 2, + InnerBevel = 4, + Cap = 8, + Endcap = 16, +} + +// Path flags +enum NVGPathFlags { + Scissor = 1, + Stroke = 2, + Fill = 4, +} + +struct NVGsegment { + int firstPoint; // Index into NVGpickScene.points + short type; // NVG_LINETO or NVG_BEZIERTO + short flags; // Flags relate to the corner between the prev segment and this one. + float[4] bounds; + float[2] startDir; // Direction at t == 0 + float[2] endDir; // Direction at t == 1 + float[2] miterDir; // Direction of miter of corner between the prev segment and this one. +} + +struct NVGpickSubPath { + short winding; // TODO: Merge to flag field + bool closed; // TODO: Merge to flag field + + int firstSegment; // Index into NVGpickScene.segments + int nsegments; + + float[4] bounds; + + NVGpickSubPath* next; +} + +struct NVGpickPath { + int id; + short flags; + short order; + float strokeWidth; + float miterLimit; + short lineCap; + short lineJoin; + bool evenOddMode; + + float[4] bounds; + int scissor; // Indexes into ps->points and defines scissor rect as XVec, YVec and Center + + NVGpickSubPath* subPaths; + NVGpickPath* next; + NVGpickPath* cellnext; +} + +struct NVGpickScene { + int npaths; + + NVGpickPath* paths; // Linked list of paths + NVGpickPath* lastPath; // The last path in the paths linked list (the first path added) + NVGpickPath* freePaths; // Linked list of free paths + + NVGpickSubPath* freeSubPaths; // Linked list of free sub paths + + int width; + int height; + + // Points for all path sub paths. + float* points; + int npoints; + int cpoints; + + // Segments for all path sub paths + NVGsegment* segments; + int nsegments; + int csegments; + + // Implicit quadtree + float xdim; // Width / (1 << nlevels) + float ydim; // Height / (1 << nlevels) + int ncells; // Total number of cells in all levels + int nlevels; + NVGpickPath*** levels; // Index: [Level][LevelY * LevelW + LevelX] Value: Linked list of paths + + // Temp storage for picking + int cpicked; + NVGpickPath** picked; +} + + +// bounds utilities +void nvg__initBounds (ref float[4] bounds) { + bounds.ptr[0] = bounds.ptr[1] = 1e6f; + bounds.ptr[2] = bounds.ptr[3] = -1e6f; +} + +void nvg__expandBounds (ref float[4] bounds, const(float)* points, int npoints) { + npoints *= 2; + for (int i = 0; i < npoints; i += 2) { + bounds.ptr[0] = nvg__min(bounds.ptr[0], points[i]); + bounds.ptr[1] = nvg__min(bounds.ptr[1], points[i+1]); + bounds.ptr[2] = nvg__max(bounds.ptr[2], points[i]); + bounds.ptr[3] = nvg__max(bounds.ptr[3], points[i+1]); + } +} + +void nvg__unionBounds (ref float[4] bounds, in ref float[4] boundsB) { + bounds.ptr[0] = nvg__min(bounds.ptr[0], boundsB.ptr[0]); + bounds.ptr[1] = nvg__min(bounds.ptr[1], boundsB.ptr[1]); + bounds.ptr[2] = nvg__max(bounds.ptr[2], boundsB.ptr[2]); + bounds.ptr[3] = nvg__max(bounds.ptr[3], boundsB.ptr[3]); +} + +void nvg__intersectBounds (ref float[4] bounds, in ref float[4] boundsB) { + bounds.ptr[0] = nvg__max(boundsB.ptr[0], bounds.ptr[0]); + bounds.ptr[1] = nvg__max(boundsB.ptr[1], bounds.ptr[1]); + bounds.ptr[2] = nvg__min(boundsB.ptr[2], bounds.ptr[2]); + bounds.ptr[3] = nvg__min(boundsB.ptr[3], bounds.ptr[3]); + + bounds.ptr[2] = nvg__max(bounds.ptr[0], bounds.ptr[2]); + bounds.ptr[3] = nvg__max(bounds.ptr[1], bounds.ptr[3]); +} + +bool nvg__pointInBounds (in float x, in float y, in ref float[4] bounds) { + pragma(inline, true); + return (x >= bounds.ptr[0] && x <= bounds.ptr[2] && y >= bounds.ptr[1] && y <= bounds.ptr[3]); +} + +// building paths & sub paths +int nvg__pickSceneAddPoints (NVGpickScene* ps, const(float)* xy, int n) { + import core.stdc.string : memcpy; + if (ps.npoints+n > ps.cpoints) { + import core.stdc.stdlib : realloc; + int cpoints = ps.npoints+n+(ps.cpoints<<1); + float* points = cast(float*)realloc(ps.points, float.sizeof*2*cpoints); + if (points is null) assert(0, "NanoVega: out of memory"); + ps.points = points; + ps.cpoints = cpoints; + } + int i = ps.npoints; + if (xy !is null) memcpy(&ps.points[i*2], xy, float.sizeof*2*n); + ps.npoints += n; + return i; +} + +void nvg__pickSubPathAddSegment (NVGpickScene* ps, NVGpickSubPath* psp, int firstPoint, int type, short flags) { + NVGsegment* seg = null; + if (ps.nsegments == ps.csegments) { + int csegments = 1+ps.csegments+(ps.csegments<<1); + NVGsegment* segments = cast(NVGsegment*)realloc(ps.segments, NVGsegment.sizeof*csegments); + if (segments is null) assert(0, "NanoVega: out of memory"); + ps.segments = segments; + ps.csegments = csegments; + } + + if (psp.firstSegment == -1) psp.firstSegment = ps.nsegments; + + seg = &ps.segments[ps.nsegments]; + ++ps.nsegments; + seg.firstPoint = firstPoint; + seg.type = cast(short)type; + seg.flags = flags; + ++psp.nsegments; + + nvg__segmentDir(ps, psp, seg, 0, seg.startDir); + nvg__segmentDir(ps, psp, seg, 1, seg.endDir); +} + +void nvg__segmentDir (NVGpickScene* ps, NVGpickSubPath* psp, NVGsegment* seg, float t, ref float[2] d) { + const(float)* points = &ps.points[seg.firstPoint*2]; + immutable float x0 = points[0*2+0], x1 = points[1*2+0]; + immutable float y0 = points[0*2+1], y1 = points[1*2+1]; + switch (seg.type) { + case Command.LineTo: + d.ptr[0] = x1-x0; + d.ptr[1] = y1-y0; + nvg__normalize(&d.ptr[0], &d.ptr[1]); + break; + case Command.BezierTo: + immutable float x2 = points[2*2+0]; + immutable float y2 = points[2*2+1]; + immutable float x3 = points[3*2+0]; + immutable float y3 = points[3*2+1]; + + immutable float omt = 1.0f-t; + immutable float omt2 = omt*omt; + immutable float t2 = t*t; + + d.ptr[0] = + 3.0f*omt2*(x1-x0)+ + 6.0f*omt*t*(x2-x1)+ + 3.0f*t2*(x3-x2); + + d.ptr[1] = + 3.0f*omt2*(y1-y0)+ + 6.0f*omt*t*(y2-y1)+ + 3.0f*t2*(y3-y2); + + nvg__normalize(&d.ptr[0], &d.ptr[1]); + break; + default: + break; + } +} + +void nvg__pickSubPathAddFillSupports (NVGpickScene* ps, NVGpickSubPath* psp) { + NVGsegment* segments = &ps.segments[psp.firstSegment]; + for (int s = 0; s < psp.nsegments; ++s) { + NVGsegment* seg = &segments[s]; + const(float)* points = &ps.points[seg.firstPoint*2]; + if (seg.type == Command.LineTo) { + nvg__initBounds(seg.bounds); + nvg__expandBounds(seg.bounds, points, 2); + } else { + nvg__bezierBounds(points, seg.bounds); + } + } +} + +void nvg__pickSubPathAddStrokeSupports (NVGpickScene* ps, NVGpickSubPath* psp, float strokeWidth, int lineCap, int lineJoin, float miterLimit) { + immutable bool closed = psp.closed; + const(float)* points = ps.points; + NVGsegment* seg = null; + NVGsegment* segments = &ps.segments[psp.firstSegment]; + int nsegments = psp.nsegments; + NVGsegment* prevseg = (closed ? &segments[psp.nsegments-1] : null); + + int ns = 0; // nsupports + float[32] supportingPoints; + int firstPoint, lastPoint; + + if (!closed) { + segments[0].flags |= NVGSegmentFlags.Cap; + segments[nsegments-1].flags |= NVGSegmentFlags.Endcap; + } + + for (int s = 0; s < nsegments; ++s) { + seg = &segments[s]; + nvg__initBounds(seg.bounds); + + firstPoint = seg.firstPoint*2; + lastPoint = firstPoint+(seg.type == Command.LineTo ? 2 : 6); + + ns = 0; + + // First two supporting points are either side of the start point + supportingPoints.ptr[ns++] = points[firstPoint]-seg.startDir.ptr[1]*strokeWidth; + supportingPoints.ptr[ns++] = points[firstPoint+1]+seg.startDir.ptr[0]*strokeWidth; + + supportingPoints.ptr[ns++] = points[firstPoint]+seg.startDir.ptr[1]*strokeWidth; + supportingPoints.ptr[ns++] = points[firstPoint+1]-seg.startDir.ptr[0]*strokeWidth; + + // Second two supporting points are either side of the end point + supportingPoints.ptr[ns++] = points[lastPoint]-seg.endDir.ptr[1]*strokeWidth; + supportingPoints.ptr[ns++] = points[lastPoint+1]+seg.endDir.ptr[0]*strokeWidth; + + supportingPoints.ptr[ns++] = points[lastPoint]+seg.endDir.ptr[1]*strokeWidth; + supportingPoints.ptr[ns++] = points[lastPoint+1]-seg.endDir.ptr[0]*strokeWidth; + + if (seg.flags&NVGSegmentFlags.Corner && prevseg !is null) { + seg.miterDir.ptr[0] = 0.5f*(-prevseg.endDir.ptr[1]-seg.startDir.ptr[1]); + seg.miterDir.ptr[1] = 0.5f*(prevseg.endDir.ptr[0]+seg.startDir.ptr[0]); + + immutable float M2 = seg.miterDir.ptr[0]*seg.miterDir.ptr[0]+seg.miterDir.ptr[1]*seg.miterDir.ptr[1]; + + if (M2 > 0.000001f) { + float scale = 1.0f/M2; + if (scale > 600.0f) scale = 600.0f; + seg.miterDir.ptr[0] *= scale; + seg.miterDir.ptr[1] *= scale; + } + + //NVG_PICK_DEBUG_VECTOR_SCALE(&points[firstPoint], seg.miterDir, 10); + + // Add an additional support at the corner on the other line + supportingPoints.ptr[ns++] = points[firstPoint]-prevseg.endDir.ptr[1]*strokeWidth; + supportingPoints.ptr[ns++] = points[firstPoint+1]+prevseg.endDir.ptr[0]*strokeWidth; + + if (lineJoin == NVGLineCap.Miter || lineJoin == NVGLineCap.Bevel) { + // Set a corner as beveled if the join type is bevel or mitered and + // miterLimit is hit. + if (lineJoin == NVGLineCap.Bevel || (M2*miterLimit*miterLimit) < 1.0f) { + seg.flags |= NVGSegmentFlags.Bevel; + } else { + // Corner is mitered - add miter point as a support + supportingPoints.ptr[ns++] = points[firstPoint]+seg.miterDir.ptr[0]*strokeWidth; + supportingPoints.ptr[ns++] = points[firstPoint+1]+seg.miterDir.ptr[1]*strokeWidth; + } + } else if (lineJoin == NVGLineCap.Round) { + // ... and at the midpoint of the corner arc + float[2] vertexN = [ -seg.startDir.ptr[0]+prevseg.endDir.ptr[0], -seg.startDir.ptr[1]+prevseg.endDir.ptr[1] ]; + nvg__normalize(&vertexN[0], &vertexN[1]); + + supportingPoints.ptr[ns++] = points[firstPoint]+vertexN[0]*strokeWidth; + supportingPoints.ptr[ns++] = points[firstPoint+1]+vertexN[1]*strokeWidth; + } + } + + if (seg.flags&NVGSegmentFlags.Cap) { + switch (lineCap) { + case NVGLineCap.Butt: + // supports for butt already added + break; + case NVGLineCap.Square: + // square cap supports are just the original two supports moved out along the direction + supportingPoints.ptr[ns++] = supportingPoints.ptr[0]-seg.startDir.ptr[0]*strokeWidth; + supportingPoints.ptr[ns++] = supportingPoints.ptr[1]-seg.startDir.ptr[1]*strokeWidth; + supportingPoints.ptr[ns++] = supportingPoints.ptr[2]-seg.startDir.ptr[0]*strokeWidth; + supportingPoints.ptr[ns++] = supportingPoints.ptr[3]-seg.startDir.ptr[1]*strokeWidth; + break; + case NVGLineCap.Round: + // add one additional support for the round cap along the dir + supportingPoints.ptr[ns++] = points[firstPoint]-seg.startDir.ptr[0]*strokeWidth; + supportingPoints.ptr[ns++] = points[firstPoint+1]-seg.startDir.ptr[1]*strokeWidth; + break; + default: + break; + } + } + + if (seg.flags&NVGSegmentFlags.Endcap) { + // end supporting points, either side of line + int end = 4; + switch(lineCap) { + case NVGLineCap.Butt: + // supports for butt already added + break; + case NVGLineCap.Square: + // square cap supports are just the original two supports moved out along the direction + supportingPoints.ptr[ns++] = supportingPoints.ptr[end+0]+seg.endDir.ptr[0]*strokeWidth; + supportingPoints.ptr[ns++] = supportingPoints.ptr[end+1]+seg.endDir.ptr[1]*strokeWidth; + supportingPoints.ptr[ns++] = supportingPoints.ptr[end+2]+seg.endDir.ptr[0]*strokeWidth; + supportingPoints.ptr[ns++] = supportingPoints.ptr[end+3]+seg.endDir.ptr[1]*strokeWidth; + break; + case NVGLineCap.Round: + // add one additional support for the round cap along the dir + supportingPoints.ptr[ns++] = points[lastPoint]+seg.endDir.ptr[0]*strokeWidth; + supportingPoints.ptr[ns++] = points[lastPoint+1]+seg.endDir.ptr[1]*strokeWidth; + break; + default: + break; + } + } + + nvg__expandBounds(seg.bounds, supportingPoints.ptr, ns/2); + + prevseg = seg; + } +} + +NVGpickPath* nvg__pickPathCreate (NVGContext context, const(float)[] acommands, int id, bool forStroke) { + NVGpickScene* ps = nvg__pickSceneGet(context); + if (ps is null) return null; + + int i = 0; + + int ncommands = cast(int)acommands.length; + const(float)* commands = acommands.ptr; + + NVGpickPath* pp = null; + NVGpickSubPath* psp = null; + float[2] start = void; + int firstPoint; + + int hasHoles = 0; + NVGpickSubPath* prev = null; + + float[8] points = void; + float[2] inflections = void; + int ninflections = 0; + + NVGstate* state = nvg__getState(context); + float[4] totalBounds = void; + NVGsegment* segments = null; + const(NVGsegment)* seg = null; + NVGpickSubPath *curpsp; + + pp = nvg__allocPickPath(ps); + if (pp is null) return null; + + pp.id = id; + + while (i < ncommands) { + int cmd = cast(int)commands[i++]; + switch (cmd) { + case Command.MoveTo: // one coordinate pair + const(float)* tfxy = commands+i; + i += 2; + + // new starting point + start.ptr[0..2] = tfxy[0..2]; + + // start a new path for each sub path to handle sub paths that intersect other sub paths + prev = psp; + psp = nvg__allocPickSubPath(ps); + if (psp is null) { psp = prev; break; } + psp.firstSegment = -1; + psp.winding = NVGSolidity.Solid; + psp.next = prev; + + nvg__pickSceneAddPoints(ps, tfxy, 1); + break; + case Command.LineTo: // one coordinate pair + const(float)* tfxy = commands+i; + i += 2; + firstPoint = nvg__pickSceneAddPoints(ps, tfxy, 1); + nvg__pickSubPathAddSegment(ps, psp, firstPoint-1, cmd, NVGSegmentFlags.Corner); + break; + case Command.BezierTo: // three coordinate pairs + const(float)* tfxy = commands+i; + i += 3*2; + + // Split the curve at it's dx==0 or dy==0 inflection points. + // Thus: + // A horizontal line only ever interects the curves once. + // and + // Finding the closest point on any curve converges more reliably. + + // NOTE: We could just split on dy==0 here. + + memcpy(&points.ptr[0], &ps.points[(ps.npoints-1)*2], float.sizeof*2); + memcpy(&points.ptr[2], tfxy, float.sizeof*2*3); + + ninflections = 0; + nvg__bezierInflections(points.ptr, 1, &ninflections, inflections.ptr); + nvg__bezierInflections(points.ptr, 0, &ninflections, inflections.ptr); + + if (ninflections) { + float previnfl = 0; + float[8] pointsA = void, pointsB = void; + + nvg__smallsort(inflections.ptr, ninflections); + + for (int infl = 0; infl < ninflections; ++infl) { + if (nvg__absf(inflections.ptr[infl]-previnfl) < NVGPickEPS) continue; + + immutable float t = (inflections.ptr[infl]-previnfl)*(1.0f/(1.0f-previnfl)); + + previnfl = inflections.ptr[infl]; + + nvg__splitBezier(points.ptr, t, pointsA.ptr, pointsB.ptr); + + firstPoint = nvg__pickSceneAddPoints(ps, &pointsA.ptr[2], 3); + nvg__pickSubPathAddSegment(ps, psp, firstPoint-1, cmd, (infl == 0) ? NVGSegmentFlags.Corner : 0); + + memcpy(points.ptr, pointsB.ptr, float.sizeof*8); + } + + firstPoint = nvg__pickSceneAddPoints(ps, &pointsB.ptr[2], 3); + nvg__pickSubPathAddSegment(ps, psp, firstPoint-1, cmd, 0); + } else { + firstPoint = nvg__pickSceneAddPoints(ps, tfxy, 3); + nvg__pickSubPathAddSegment(ps, psp, firstPoint-1, cmd, NVGSegmentFlags.Corner); + } + break; + case Command.Close: + if (ps.points[(ps.npoints-1)*2] != start.ptr[0] || ps.points[(ps.npoints-1)*2+1] != start.ptr[1]) { + firstPoint = nvg__pickSceneAddPoints(ps, start.ptr, 1); + nvg__pickSubPathAddSegment(ps, psp, firstPoint-1, Command.LineTo, NVGSegmentFlags.Corner); + } + psp.closed = true; + break; + case Command.Winding: + psp.winding = cast(short)cast(int)commands[i]; + if (psp.winding == NVGSolidity.Hole) hasHoles = 1; + i += 1; + break; + default: + break; + } + } + + pp.flags = (forStroke ? NVGPathFlags.Stroke : NVGPathFlags.Fill); + pp.subPaths = psp; + pp.strokeWidth = state.strokeWidth*0.5f; + pp.miterLimit = state.miterLimit; + pp.lineCap = cast(short)state.lineCap; + pp.lineJoin = cast(short)state.lineJoin; + pp.evenOddMode = nvg__getState(context).evenOddMode; + + nvg__initBounds(totalBounds); + + for (curpsp = psp; curpsp; curpsp = curpsp.next) { + if (forStroke) { + nvg__pickSubPathAddStrokeSupports(ps, curpsp, pp.strokeWidth, pp.lineCap, pp.lineJoin, pp.miterLimit); + } else { + nvg__pickSubPathAddFillSupports(ps, curpsp); + } + + segments = &ps.segments[curpsp.firstSegment]; + nvg__initBounds(curpsp.bounds); + for (int s = 0; s < curpsp.nsegments; ++s) { + seg = &segments[s]; + //NVG_PICK_DEBUG_BOUNDS(seg.bounds); + nvg__unionBounds(curpsp.bounds, seg.bounds); + } + + nvg__unionBounds(totalBounds, curpsp.bounds); + } + + // Store the scissor rect if present. + if (state.scissor.extent.ptr[0] != -1.0f) { + // Use points storage to store the scissor data + pp.scissor = nvg__pickSceneAddPoints(ps, null, 4); + float* scissor = &ps.points[pp.scissor*2]; + + //memcpy(scissor, state.scissor.xform.ptr, 6*float.sizeof); + scissor[0..6] = state.scissor.xform.mat[]; + memcpy(scissor+6, state.scissor.extent.ptr, 2*float.sizeof); + + pp.flags |= NVGPathFlags.Scissor; + } + + memcpy(pp.bounds.ptr, totalBounds.ptr, float.sizeof*4); + + return pp; +} + + +// Struct management +NVGpickPath* nvg__allocPickPath (NVGpickScene* ps) { + NVGpickPath* pp = ps.freePaths; + if (pp !is null) { + ps.freePaths = pp.next; + } else { + pp = cast(NVGpickPath*)malloc(NVGpickPath.sizeof); + } + memset(pp, 0, NVGpickPath.sizeof); + return pp; +} + +// Put a pick path and any sub paths (back) to the free lists. +void nvg__freePickPath (NVGpickScene* ps, NVGpickPath* pp) { + // Add all sub paths to the sub path free list. + // Finds the end of the path sub paths, links that to the current + // sub path free list head and replaces the head ptr with the + // head path sub path entry. + NVGpickSubPath* psp = null; + for (psp = pp.subPaths; psp !is null && psp.next !is null; psp = psp.next) {} + + if (psp) { + psp.next = ps.freeSubPaths; + ps.freeSubPaths = pp.subPaths; + } + pp.subPaths = null; + + // Add the path to the path freelist + pp.next = ps.freePaths; + ps.freePaths = pp; + if (pp.next is null) ps.lastPath = pp; +} + +NVGpickSubPath* nvg__allocPickSubPath (NVGpickScene* ps) { + NVGpickSubPath* psp = ps.freeSubPaths; + if (psp !is null) { + ps.freeSubPaths = psp.next; + } else { + psp = cast(NVGpickSubPath*)malloc(NVGpickSubPath.sizeof); + if (psp is null) return null; + } + memset(psp, 0, NVGpickSubPath.sizeof); + return psp; +} + +void nvg__returnPickSubPath (NVGpickScene* ps, NVGpickSubPath* psp) { + psp.next = ps.freeSubPaths; + ps.freeSubPaths = psp; +} + +NVGpickScene* nvg__allocPickScene () { + NVGpickScene* ps = cast(NVGpickScene*)malloc(NVGpickScene.sizeof); + if (ps is null) return null; + memset(ps, 0, NVGpickScene.sizeof); + ps.nlevels = 5; + return ps; +} + +void nvg__deletePickScene (NVGpickScene* ps) { + NVGpickPath* pp; + NVGpickSubPath* psp; + + // Add all paths (and thus sub paths) to the free list(s). + while (ps.paths !is null) { + pp = ps.paths.next; + nvg__freePickPath(ps, ps.paths); + ps.paths = pp; + } + + // Delete all paths + while (ps.freePaths !is null) { + pp = ps.freePaths; + ps.freePaths = pp.next; + while (pp.subPaths !is null) { + psp = pp.subPaths; + pp.subPaths = psp.next; + free(psp); + } + free(pp); + } + + // Delete all sub paths + while (ps.freeSubPaths !is null) { + psp = ps.freeSubPaths.next; + free(ps.freeSubPaths); + ps.freeSubPaths = psp; + } + + ps.npoints = 0; + ps.nsegments = 0; + + if (ps.levels !is null) { + free(ps.levels[0]); + free(ps.levels); + } + + if (ps.picked !is null) free(ps.picked); + if (ps.points !is null) free(ps.points); + if (ps.segments !is null) free(ps.segments); + + free(ps); +} + +NVGpickScene* nvg__pickSceneGet (NVGContext ctx) { + if (ctx.pickScene is null) ctx.pickScene = nvg__allocPickScene(); + return ctx.pickScene; +} + + +// Applies Casteljau's algorithm to a cubic bezier for a given parameter t +// points is 4 points (8 floats) +// lvl1 is 3 points (6 floats) +// lvl2 is 2 points (4 floats) +// lvl3 is 1 point (2 floats) +void nvg__casteljau (const(float)* points, float t, float* lvl1, float* lvl2, float* lvl3) { + enum x0 = 0*2+0; enum x1 = 1*2+0; enum x2 = 2*2+0; enum x3 = 3*2+0; + enum y0 = 0*2+1; enum y1 = 1*2+1; enum y2 = 2*2+1; enum y3 = 3*2+1; + + // Level 1 + lvl1[x0] = (points[x1]-points[x0])*t+points[x0]; + lvl1[y0] = (points[y1]-points[y0])*t+points[y0]; + + lvl1[x1] = (points[x2]-points[x1])*t+points[x1]; + lvl1[y1] = (points[y2]-points[y1])*t+points[y1]; + + lvl1[x2] = (points[x3]-points[x2])*t+points[x2]; + lvl1[y2] = (points[y3]-points[y2])*t+points[y2]; + + // Level 2 + lvl2[x0] = (lvl1[x1]-lvl1[x0])*t+lvl1[x0]; + lvl2[y0] = (lvl1[y1]-lvl1[y0])*t+lvl1[y0]; + + lvl2[x1] = (lvl1[x2]-lvl1[x1])*t+lvl1[x1]; + lvl2[y1] = (lvl1[y2]-lvl1[y1])*t+lvl1[y1]; + + // Level 3 + lvl3[x0] = (lvl2[x1]-lvl2[x0])*t+lvl2[x0]; + lvl3[y0] = (lvl2[y1]-lvl2[y0])*t+lvl2[y0]; +} + +// Calculates a point on a bezier at point t. +void nvg__bezierEval (const(float)* points, float t, ref float[2] tpoint) { + immutable float omt = 1-t; + immutable float omt3 = omt*omt*omt; + immutable float omt2 = omt*omt; + immutable float t3 = t*t*t; + immutable float t2 = t*t; + + tpoint.ptr[0] = + points[0]*omt3+ + points[2]*3.0f*omt2*t+ + points[4]*3.0f*omt*t2+ + points[6]*t3; + + tpoint.ptr[1] = + points[1]*omt3+ + points[3]*3.0f*omt2*t+ + points[5]*3.0f*omt*t2+ + points[7]*t3; +} + +// Splits a cubic bezier curve into two parts at point t. +void nvg__splitBezier (const(float)* points, float t, float* pointsA, float* pointsB) { + enum x0 = 0*2+0; enum x1 = 1*2+0; enum x2 = 2*2+0; enum x3 = 3*2+0; + enum y0 = 0*2+1; enum y1 = 1*2+1; enum y2 = 2*2+1; enum y3 = 3*2+1; + + float[6] lvl1 = void; + float[4] lvl2 = void; + float[2] lvl3 = void; + + nvg__casteljau(points, t, lvl1.ptr, lvl2.ptr, lvl3.ptr); + + // First half + pointsA[x0] = points[x0]; + pointsA[y0] = points[y0]; + + pointsA[x1] = lvl1.ptr[x0]; + pointsA[y1] = lvl1.ptr[y0]; + + pointsA[x2] = lvl2.ptr[x0]; + pointsA[y2] = lvl2.ptr[y0]; + + pointsA[x3] = lvl3.ptr[x0]; + pointsA[y3] = lvl3.ptr[y0]; + + // Second half + pointsB[x0] = lvl3.ptr[x0]; + pointsB[y0] = lvl3.ptr[y0]; + + pointsB[x1] = lvl2.ptr[x1]; + pointsB[y1] = lvl2.ptr[y1]; + + pointsB[x2] = lvl1.ptr[x2]; + pointsB[y2] = lvl1.ptr[y2]; + + pointsB[x3] = points[x3]; + pointsB[y3] = points[y3]; +} + +// Calculates the inflection points in coordinate coord (X = 0, Y = 1) of a cubic bezier. +// Appends any found inflection points to the array inflections and increments *ninflections. +// So finds the parameters where dx/dt or dy/dt is 0 +void nvg__bezierInflections (const(float)* points, int coord, int* ninflections, float* inflections) { + immutable float v0 = points[0*2+coord], v1 = points[1*2+coord], v2 = points[2*2+coord], v3 = points[3*2+coord]; + float[2] t = void; + int nvalid = *ninflections; + + immutable float a = 3.0f*( -v0+3.0f*v1-3.0f*v2+v3 ); + immutable float b = 6.0f*( v0-2.0f*v1+v2 ); + immutable float c = 3.0f*( v1-v0 ); + + float d = b*b-4.0f*a*c; + if (nvg__absf(d-0.0f) < NVGPickEPS) { + // Zero or one root + t.ptr[0] = -b/2.0f*a; + if (t.ptr[0] > NVGPickEPS && t.ptr[0] < (1.0f-NVGPickEPS)) { + inflections[nvalid] = t.ptr[0]; + ++nvalid; + } + } else if (d > NVGPickEPS) { + // zero, one or two roots + d = nvg__sqrtf(d); + + t.ptr[0] = (-b+d)/(2.0f*a); + t.ptr[1] = (-b-d)/(2.0f*a); + + for (int i = 0; i < 2; ++i) { + if (t.ptr[i] > NVGPickEPS && t.ptr[i] < (1.0f-NVGPickEPS)) { + inflections[nvalid] = t.ptr[i]; + ++nvalid; + } + } + } else { + // zero roots + } + + *ninflections = nvalid; +} + +// Sort a small number of floats in ascending order (0 < n < 6) +void nvg__smallsort (float* values, int n) { + bool bSwapped = true; + for (int j = 0; j < n-1 && bSwapped; ++j) { + bSwapped = false; + for (int i = 0; i < n-1; ++i) { + if (values[i] > values[i+1]) { + auto tmp = values[i]; + values[i] = values[i+1]; + values[i+1] = tmp; + } + } + } +} + +// Calculates the bounding rect of a given cubic bezier curve. +void nvg__bezierBounds (const(float)* points, ref float[4] bounds) { + float[4] inflections = void; + int ninflections = 0; + float[2] tpoint = void; + + nvg__initBounds(bounds); + + // Include start and end points in bounds + nvg__expandBounds(bounds, &points[0], 1); + nvg__expandBounds(bounds, &points[6], 1); + + // Calculate dx==0 and dy==0 inflection points and add then to the bounds + + nvg__bezierInflections(points, 0, &ninflections, inflections.ptr); + nvg__bezierInflections(points, 1, &ninflections, inflections.ptr); + + for (int i = 0; i < ninflections; ++i) { + nvg__bezierEval(points, inflections[i], tpoint); + nvg__expandBounds(bounds, tpoint.ptr, 1); + } +} + +// Checks to see if a line originating from x,y along the +ve x axis +// intersects the given line (points[0],points[1]) -> (points[2], points[3]). +// Returns `true` on intersection. +// Horizontal lines are never hit. +bool nvg__intersectLine (const(float)* points, float x, float y) { + immutable float x1 = points[0]; + immutable float y1 = points[1]; + immutable float x2 = points[2]; + immutable float y2 = points[3]; + immutable float d = y2-y1; + if (d > NVGPickEPS || d < -NVGPickEPS) { + immutable float s = (x2-x1)/d; + immutable float lineX = x1+(y-y1)*s; + return (lineX > x); + } else { + return false; + } +} + +// Checks to see if a line originating from x,y along the +ve x axis intersects the given bezier. +// It is assumed that the line originates from within the bounding box of +// the bezier and that the curve has no dy=0 inflection points. +// Returns the number of intersections found (which is either 1 or 0). +int nvg__intersectBezier (const(float)* points, float x, float y) { + immutable float x0 = points[0*2+0], x1 = points[1*2+0], x2 = points[2*2+0], x3 = points[3*2+0]; + immutable float y0 = points[0*2+1], y1 = points[1*2+1], y2 = points[2*2+1], y3 = points[3*2+1]; + + if (y0 == y1 && y1 == y2 && y2 == y3) return 0; + + // Initial t guess + float t = void; + if (y3 != y0) t = (y-y0)/(y3-y0); + else if (x3 != x0) t = (x-x0)/(x3-x0); + else t = 0.5f; + + // A few Newton iterations + for (int iter = 0; iter < 6; ++iter) { + immutable float omt = 1-t; + immutable float omt2 = omt*omt; + immutable float t2 = t*t; + immutable float omt3 = omt2*omt; + immutable float t3 = t2*t; + + immutable float ty = y0*omt3 + + y1*3.0f*omt2*t + + y2*3.0f*omt*t2 + + y3*t3; + + // Newton iteration + immutable float dty = 3.0f*omt2*(y1-y0) + + 6.0f*omt*t*(y2-y1) + + 3.0f*t2*(y3-y2); + + // dty will never == 0 since: + // Either omt, omt2 are zero OR t2 is zero + // y0 != y1 != y2 != y3 (checked above) + t = t-(ty-y)/dty; + } + + { + immutable float omt = 1-t; + immutable float omt2 = omt*omt; + immutable float t2 = t*t; + immutable float omt3 = omt2*omt; + immutable float t3 = t2*t; + + immutable float tx = + x0*omt3+ + x1*3.0f*omt2*t+ + x2*3.0f*omt*t2+ + x3*t3; + + return (tx > x ? 1 : 0); + } +} + +// Finds the closest point on a line to a given point +void nvg__closestLine (const(float)* points, float x, float y, float* closest, float* ot) { + immutable float x1 = points[0]; + immutable float y1 = points[1]; + immutable float x2 = points[2]; + immutable float y2 = points[3]; + immutable float pqx = x2-x1; + immutable float pqz = y2-y1; + immutable float dx = x-x1; + immutable float dz = y-y1; + immutable float d = pqx*pqx+pqz*pqz; + float t = pqx*dx+pqz*dz; + if (d > 0) t /= d; + if (t < 0) t = 0; else if (t > 1) t = 1; + closest[0] = x1+t*pqx; + closest[1] = y1+t*pqz; + *ot = t; +} + +// Finds the closest point on a curve for a given point (x,y). +// Assumes that the curve has no dx==0 or dy==0 inflection points. +void nvg__closestBezier (const(float)* points, float x, float y, float* closest, float *ot) { + immutable float x0 = points[0*2+0], x1 = points[1*2+0], x2 = points[2*2+0], x3 = points[3*2+0]; + immutable float y0 = points[0*2+1], y1 = points[1*2+1], y2 = points[2*2+1], y3 = points[3*2+1]; + + // This assumes that the curve has no dy=0 inflection points. + + // Initial t guess + float t = 0.5f; + + // A few Newton iterations + for (int iter = 0; iter < 6; ++iter) { + immutable float omt = 1-t; + immutable float omt2 = omt*omt; + immutable float t2 = t*t; + immutable float omt3 = omt2*omt; + immutable float t3 = t2*t; + + immutable float ty = + y0*omt3+ + y1*3.0f*omt2*t+ + y2*3.0f*omt*t2+ + y3*t3; + + immutable float tx = + x0*omt3+ + x1*3.0f*omt2*t+ + x2*3.0f*omt*t2+ + x3*t3; + + // Newton iteration + immutable float dty = + 3.0f*omt2*(y1-y0)+ + 6.0f*omt*t*(y2-y1)+ + 3.0f*t2*(y3-y2); + + immutable float ddty = + 6.0f*omt*(y2-2.0f*y1+y0)+ + 6.0f*t*(y3-2.0f*y2+y1); + + immutable float dtx = + 3.0f*omt2*(x1-x0)+ + 6.0f*omt*t*(x2-x1)+ + 3.0f*t2*(x3-x2); + + immutable float ddtx = + 6.0f*omt*(x2-2.0f*x1+x0)+ + 6.0f*t*(x3-2.0f*x2+x1); + + immutable float errorx = tx-x; + immutable float errory = ty-y; + + immutable float n = errorx*dtx+errory*dty; + if (n == 0) break; + + immutable float d = dtx*dtx+dty*dty+errorx*ddtx+errory*ddty; + if (d != 0) t = t-n/d; else break; + } + + t = nvg__max(0, nvg__min(1.0, t)); + *ot = t; + { + immutable float omt = 1-t; + immutable float omt2 = omt*omt; + immutable float t2 = t*t; + immutable float omt3 = omt2*omt; + immutable float t3 = t2*t; + + immutable float ty = + y0*omt3+ + y1*3.0f*omt2*t+ + y2*3.0f*omt*t2+ + y3*t3; + + immutable float tx = + x0*omt3+ + x1*3.0f*omt2*t+ + x2*3.0f*omt*t2+ + x3*t3; + + closest[0] = tx; + closest[1] = ty; + } +} + +// Returns: +// 1 If (x,y) is contained by the stroke of the path +// 0 If (x,y) is not contained by the path. +int nvg__pickSubPathStroke (const NVGpickScene* ps, const NVGpickSubPath* psp, float x, float y, float strokeWidth, int lineCap, int lineJoin) { + if (!nvg__pointInBounds(x, y, psp.bounds)) return 0; + + float[2] closest = void; + float[2] d = void; + float t = void; + + // trace a line from x,y out along the positive x axis and count the number of intersections + int nsegments = psp.nsegments; + const(NVGsegment)* seg = ps.segments+psp.firstSegment; + const(NVGsegment)* prevseg = (psp.closed ? &ps.segments[psp.firstSegment+nsegments-1] : null); + immutable float strokeWidthSqd = strokeWidth*strokeWidth; + + for (int s = 0; s < nsegments; ++s, prevseg = seg, ++seg) { + if (nvg__pointInBounds(x, y, seg.bounds)) { + // Line potentially hits stroke. + switch (seg.type) { + case Command.LineTo: + nvg__closestLine(&ps.points[seg.firstPoint*2], x, y, closest.ptr, &t); + break; + case Command.BezierTo: + nvg__closestBezier(&ps.points[seg.firstPoint*2], x, y, closest.ptr, &t); + break; + default: + continue; + } + + d.ptr[0] = x-closest.ptr[0]; + d.ptr[1] = y-closest.ptr[1]; + + if ((t >= NVGPickEPS && t <= 1.0f-NVGPickEPS) || + (seg.flags&(NVGSegmentFlags.Corner|NVGSegmentFlags.Cap|NVGSegmentFlags.Endcap)) == 0 || + (lineJoin == NVGLineCap.Round)) + { + // Closest point is in the middle of the line/curve, at a rounded join/cap + // or at a smooth join + immutable float distSqd = d.ptr[0]*d.ptr[0]+d.ptr[1]*d.ptr[1]; + if (distSqd < strokeWidthSqd) return 1; + } else if ((t > 1.0f-NVGPickEPS && (seg.flags&NVGSegmentFlags.Endcap)) || + (t < NVGPickEPS && (seg.flags&NVGSegmentFlags.Cap))) { + switch (lineCap) { + case NVGLineCap.Butt: + immutable float distSqd = d.ptr[0]*d.ptr[0]+d.ptr[1]*d.ptr[1]; + immutable float dirD = (t < NVGPickEPS ? + -(d.ptr[0]*seg.startDir.ptr[0]+d.ptr[1]*seg.startDir.ptr[1]) : + d.ptr[0]*seg.endDir.ptr[0]+d.ptr[1]*seg.endDir.ptr[1]); + if (dirD < -NVGPickEPS && distSqd < strokeWidthSqd) return 1; + break; + case NVGLineCap.Square: + if (nvg__absf(d.ptr[0]) < strokeWidth && nvg__absf(d.ptr[1]) < strokeWidth) return 1; + break; + case NVGLineCap.Round: + immutable float distSqd = d.ptr[0]*d.ptr[0]+d.ptr[1]*d.ptr[1]; + if (distSqd < strokeWidthSqd) return 1; + break; + default: + break; + } + } else if (seg.flags&NVGSegmentFlags.Corner) { + // Closest point is at a corner + const(NVGsegment)* seg0, seg1; + + if (t < NVGPickEPS) { + seg0 = prevseg; + seg1 = seg; + } else { + seg0 = seg; + seg1 = (s == nsegments-1 ? &ps.segments[psp.firstSegment] : seg+1); + } + + if (!(seg1.flags&NVGSegmentFlags.Bevel)) { + immutable float prevNDist = -seg0.endDir.ptr[1]*d.ptr[0]+seg0.endDir.ptr[0]*d.ptr[1]; + immutable float curNDist = seg1.startDir.ptr[1]*d.ptr[0]-seg1.startDir.ptr[0]*d.ptr[1]; + if (nvg__absf(prevNDist) < strokeWidth && nvg__absf(curNDist) < strokeWidth) return 1; + } else { + d.ptr[0] -= -seg1.startDir.ptr[1]*strokeWidth; + d.ptr[1] -= +seg1.startDir.ptr[0]*strokeWidth; + if (seg1.miterDir.ptr[0]*d.ptr[0]+seg1.miterDir.ptr[1]*d.ptr[1] < 0) return 1; + } + } + } + } + + return 0; +} + +// Returns: +// 1 If (x,y) is contained by the path and the path is solid. +// -1 If (x,y) is contained by the path and the path is a hole. +// 0 If (x,y) is not contained by the path. +int nvg__pickSubPath (const NVGpickScene* ps, const NVGpickSubPath* psp, float x, float y, bool evenOddMode) { + if (!nvg__pointInBounds(x, y, psp.bounds)) return 0; + + const(NVGsegment)* seg = &ps.segments[psp.firstSegment]; + int nsegments = psp.nsegments; + int nintersections = 0; + + // trace a line from x,y out along the positive x axis and count the number of intersections + for (int s = 0; s < nsegments; ++s, ++seg) { + if ((seg.bounds.ptr[1]-NVGPickEPS) < y && + (seg.bounds.ptr[3]-NVGPickEPS) > y && + seg.bounds.ptr[2] > x) + { + // Line hits the box. + switch (seg.type) { + case Command.LineTo: + if (seg.bounds.ptr[0] > x) { + // line originates outside the box + ++nintersections; + } else { + // line originates inside the box + nintersections += nvg__intersectLine(&ps.points[seg.firstPoint*2], x, y); + } + break; + case Command.BezierTo: + if (seg.bounds.ptr[0] > x) { + // line originates outside the box + ++nintersections; + } else { + // line originates inside the box + nintersections += nvg__intersectBezier(&ps.points[seg.firstPoint*2], x, y); + } + break; + default: + break; + } + } + } + + if (evenOddMode) { + return nintersections; + } else { + return (nintersections&1 ? (psp.winding == NVGSolidity.Solid ? 1 : -1) : 0); + } +} + +bool nvg__pickPath (const(NVGpickScene)* ps, const(NVGpickPath)* pp, float x, float y) { + int pickCount = 0; + const(NVGpickSubPath)* psp = pp.subPaths; + while (psp !is null) { + pickCount += nvg__pickSubPath(ps, psp, x, y, pp.evenOddMode); + psp = psp.next; + } + return ((pp.evenOddMode ? pickCount&1 : pickCount) != 0); +} + +bool nvg__pickPathStroke (const(NVGpickScene)* ps, const(NVGpickPath)* pp, float x, float y) { + const(NVGpickSubPath)* psp = pp.subPaths; + while (psp !is null) { + if (nvg__pickSubPathStroke(ps, psp, x, y, pp.strokeWidth, pp.lineCap, pp.lineJoin)) return true; + psp = psp.next; + } + return false; +} + +bool nvg__pickPathTestBounds (NVGContext ctx, const NVGpickScene* ps, const NVGpickPath* pp, float x, float y) { + if (nvg__pointInBounds(x, y, pp.bounds)) { + //{ import core.stdc.stdio; printf(" (0): in bounds!\n"); } + if (pp.flags&NVGPathFlags.Scissor) { + const(float)* scissor = &ps.points[pp.scissor*2]; + // untransform scissor translation + float stx = void, sty = void; + ctx.gpuUntransformPoint(&stx, &sty, scissor[4], scissor[5]); + immutable float rx = x-stx; + immutable float ry = y-sty; + //{ import core.stdc.stdio; printf(" (1): rxy=(%g,%g); scissor=[%g,%g,%g,%g,%g] [%g,%g]!\n", rx, ry, scissor[0], scissor[1], scissor[2], scissor[3], scissor[4], scissor[5], scissor[6], scissor[7]); } + if (nvg__absf((scissor[0]*rx)+(scissor[1]*ry)) > scissor[6] || + nvg__absf((scissor[2]*rx)+(scissor[3]*ry)) > scissor[7]) + { + //{ import core.stdc.stdio; printf(" (1): scissor reject!\n"); } + return false; + } + } + return true; + } + return false; +} + +int nvg__countBitsUsed (int v) { + pragma(inline, true); + import core.bitop : popcnt; + return (v != 0 ? popcnt(cast(uint)v) : 0); +} + +void nvg__pickSceneInsert (NVGpickScene* ps, NVGpickPath* pp) { + if (ps is null || pp is null) return; + + int[4] cellbounds; + int base = ps.nlevels-1; + int level; + int levelwidth; + int levelshift; + int levelx; + int levely; + NVGpickPath** cell = null; + + // Bit tricks for inserting into an implicit quadtree. + + // Calc bounds of path in cells at the lowest level + cellbounds.ptr[0] = cast(int)(pp.bounds.ptr[0]/ps.xdim); + cellbounds.ptr[1] = cast(int)(pp.bounds.ptr[1]/ps.ydim); + cellbounds.ptr[2] = cast(int)(pp.bounds.ptr[2]/ps.xdim); + cellbounds.ptr[3] = cast(int)(pp.bounds.ptr[3]/ps.ydim); + + // Find which bits differ between the min/max x/y coords + cellbounds.ptr[0] ^= cellbounds.ptr[2]; + cellbounds.ptr[1] ^= cellbounds.ptr[3]; + + // Use the number of bits used (countBitsUsed(x) == sizeof(int) * 8 - clz(x); + // to calculate the level to insert at (the level at which the bounds fit in a single cell) + level = nvg__min(base-nvg__countBitsUsed(cellbounds.ptr[0]), base-nvg__countBitsUsed(cellbounds.ptr[1])); + if (level < 0) level = 0; + + // Find the correct cell in the chosen level, clamping to the edges. + levelwidth = 1<>levelshift, 0, levelwidth-1); + levely = nvg__clamp(cellbounds.ptr[3]>>levelshift, 0, levelwidth-1); + + // Insert the path into the linked list at that cell. + cell = &ps.levels[level][levely*levelwidth+levelx]; + + pp.cellnext = *cell; + *cell = pp; + + if (ps.paths is null) ps.lastPath = pp; + pp.next = ps.paths; + ps.paths = pp; + + // Store the order (depth) of the path for picking ops. + pp.order = cast(short)ps.npaths; + ++ps.npaths; +} + +void nvg__pickBeginFrame (NVGContext ctx, int width, int height) { + NVGpickScene* ps = nvg__pickSceneGet(ctx); + + //NVG_PICK_DEBUG_NEWFRAME(); + + // Return all paths & sub paths from last frame to the free list + while (ps.paths !is null) { + NVGpickPath* pp = ps.paths.next; + nvg__freePickPath(ps, ps.paths); + ps.paths = pp; + } + + ps.paths = null; + ps.npaths = 0; + + // Store the screen metrics for the quadtree + ps.width = width; + ps.height = height; + + immutable float lowestSubDiv = cast(float)(1<<(ps.nlevels-1)); + ps.xdim = cast(float)width/lowestSubDiv; + ps.ydim = cast(float)height/lowestSubDiv; + + // Allocate the quadtree if required. + if (ps.levels is null) { + int ncells = 1; + + ps.levels = cast(NVGpickPath***)malloc((NVGpickPath**).sizeof*ps.nlevels); + for (int l = 0; l < ps.nlevels; ++l) { + int leveldim = 1<= 0 && code <= 3 && args.length >= 2); } /// + + /// perform NanoVega command with stored data. + void perform (NVGContext ctx) const nothrow @trusted @nogc { + if (ctx is null) return; + switch (code) { + case Kind.MoveTo: if (args.length > 1) ctx.moveTo(args.ptr[0..2]); break; + case Kind.LineTo: if (args.length > 1) ctx.lineTo(args.ptr[0..2]); break; + case Kind.QuadTo: if (args.length > 3) ctx.quadTo(args.ptr[0..4]); break; + case Kind.BezierTo: if (args.length > 5) ctx.bezierTo(args.ptr[0..6]); break; + default: break; + } + } + + /// perform NanoVega command with stored data, transforming points with [xform] transformation matrix. + void perform() (NVGContext ctx, in auto ref NVGMatrix xform) const nothrow @trusted @nogc { + if (ctx is null || !valid) return; + float[6] pts = void; + pts[0..args.length] = args[]; + foreach (immutable pidx; 0..args.length/2) xform.point(pts.ptr[pidx*2+0], pts.ptr[pidx*2+1]); + switch (code) { + case Kind.MoveTo: if (args.length > 1) ctx.moveTo(pts.ptr[0..2]); break; + case Kind.LineTo: if (args.length > 1) ctx.lineTo(pts.ptr[0..2]); break; + case Kind.QuadTo: if (args.length > 3) ctx.quadTo(pts.ptr[0..4]); break; + case Kind.BezierTo: if (args.length > 5) ctx.bezierTo(pts.ptr[0..6]); break; + default: break; + } + } + } + + @disable this (this); // no copies + +private: + ubyte* data; + uint used; + uint size; + uint ccount; // number of commands + +private: + void clear () nothrow @trusted @nogc { + import core.stdc.stdlib : free; + if (data !is null) { free(data); data = null; } + used = size = ccount = 0; + bounds[] = 0; + } + +public: + float[4] bounds = 0; + + @property int length () const pure { pragma(inline, true); return ccount; } + +public: + /// Returns forward range with all glyph commands. + /// WARNING! returned rande should not outlive parent struct! + auto commands () nothrow @trusted @nogc { + static struct Range { + private nothrow @trusted @nogc: + const(ubyte)* data; + uint cleft; // number of commands left + public: + @property bool empty () const pure { pragma(inline, true); return (cleft == 0); } + @property int length () const pure { pragma(inline, true); return cleft; } + @property Range save () const pure { pragma(inline, true); Range res = this; return res; } + @property Command front () const { + Command res = void; + if (cleft > 0) { + res.code = cast(Command.Kind)data[0]; + switch (res.code) { + case Command.Kind.MoveTo: + case Command.Kind.LineTo: + res.args = (cast(const(float*))(data+1))[0..1*2]; + break; + case Command.Kind.QuadTo: + res.args = (cast(const(float*))(data+1))[0..2*2]; + break; + case Command.Kind.BezierTo: + res.args = (cast(const(float*))(data+1))[0..3*2]; + break; + default: + res.code = cast(Command.Kind)255; + res.args = null; + break; + } + } else { + res.code = cast(Command.Kind)255; + res.args = null; + } + return res; + } + void popFront () { + if (cleft == 0) return; + if (--cleft == 0) return; // don't waste time skipping last command + switch (data[0]) { + case Command.Kind.MoveTo: + case Command.Kind.LineTo: + data += 1+1*2*cast(uint)float.sizeof; + break; + case Command.Kind.QuadTo: + data += 1+2*2*cast(uint)float.sizeof; + break; + case Command.Kind.BezierTo: + data += 1+3*2*cast(uint)float.sizeof; + break; + default: + cleft = 0; + break; + } + } + } + return Range(data, ccount); + } +} + +/// Destroy glyph outiline and free allocated memory. +/// Group: text_api +public void kill (ref NVGGlyphOutline* ol) nothrow @trusted @nogc { + if (ol !is null) { + import core.stdc.stdlib : free; + ol.clear(); + free(ol); + ol = null; + } +} + +static if (is(typeof(&fons__nvg__toOutline))) { + public enum NanoVegaHasCharOutline = true; /// +} else { + public enum NanoVegaHasCharOutline = false; /// +} + +/// Returns glyph outlines as array of commands. Vertical 0 is baseline. +/// The glyph is not scaled in any way, so you have to use NanoVega transformations instead. +/// Returns `null` if there is no such glyph, or current font is not scalable. +/// Group: text_api +public NVGGlyphOutline* charOutline (NVGContext ctx, dchar dch) nothrow @trusted @nogc { + import core.stdc.stdlib : malloc; + import core.stdc.string : memcpy; + NVGstate* state = nvg__getState(ctx); + fonsSetFont(ctx.fs, state.fontId); + NVGGlyphOutline oline; + if (!fonsToOutline(ctx.fs, dch, &oline)) { oline.clear(); return null; } + auto res = cast(NVGGlyphOutline*)malloc(NVGGlyphOutline.sizeof); + if (res is null) { oline.clear(); return null; } + memcpy(res, &oline, oline.sizeof); + return res; +} + + +float nvg__quantize (float a, float d) pure nothrow @safe @nogc { + pragma(inline, true); + return (cast(int)(a/d+0.5f))*d; +} + +float nvg__getFontScale (NVGstate* state) nothrow @safe @nogc { + pragma(inline, true); + return nvg__min(nvg__quantize(nvg__getAverageScale(state.xform), 0.01f), 4.0f); +} + +void nvg__flushTextTexture (NVGContext ctx) nothrow @trusted @nogc { + int[4] dirty = void; + if (fonsValidateTexture(ctx.fs, dirty.ptr)) { + int fontImage = ctx.fontImages[ctx.fontImageIdx]; + // Update texture + if (fontImage != 0) { + int iw, ih; + const(ubyte)* data = fonsGetTextureData(ctx.fs, &iw, &ih); + int x = dirty[0]; + int y = dirty[1]; + int w = dirty[2]-dirty[0]; + int h = dirty[3]-dirty[1]; + ctx.params.renderUpdateTexture(ctx.params.userPtr, fontImage, x, y, w, h, data); + } + } +} + +bool nvg__allocTextAtlas (NVGContext ctx) nothrow @trusted @nogc { + int iw, ih; + nvg__flushTextTexture(ctx); + if (ctx.fontImageIdx >= NVG_MAX_FONTIMAGES-1) return false; + // if next fontImage already have a texture + if (ctx.fontImages[ctx.fontImageIdx+1] != 0) { + ctx.imageSize(ctx.fontImages[ctx.fontImageIdx+1], iw, ih); + } else { + // calculate the new font image size and create it + ctx.imageSize(ctx.fontImages[ctx.fontImageIdx], iw, ih); + if (iw > ih) ih *= 2; else iw *= 2; + if (iw > NVG_MAX_FONTIMAGE_SIZE || ih > NVG_MAX_FONTIMAGE_SIZE) iw = ih = NVG_MAX_FONTIMAGE_SIZE; + ctx.fontImages[ctx.fontImageIdx+1] = ctx.params.renderCreateTexture(ctx.params.userPtr, NVGtexture.Alpha, iw, ih, (ctx.params.fontAA ? 0 : NVGImageFlags.NoFiltering), null); + } + ++ctx.fontImageIdx; + fonsResetAtlas(ctx.fs, iw, ih); + return true; +} + +void nvg__renderText (NVGContext ctx, NVGvertex* verts, int nverts) nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + NVGPaint paint = state.fill; + + // Render triangles. + paint.image = ctx.fontImages[ctx.fontImageIdx]; + + // Apply global alpha + paint.innerColor.a *= state.alpha; + paint.outerColor.a *= state.alpha; + + ctx.params.renderTriangles(ctx.params.userPtr, &paint, &state.scissor, verts, nverts); + + ++ctx.drawCallCount; + ctx.textTriCount += nverts/3; +} + +/// Draws text string at specified location. Returns next x position. +/// Group: text_api +public float text(T) (NVGContext ctx, float x, float y, const(T)[] str) nothrow @trusted @nogc if (isAnyCharType!T) { + NVGstate* state = nvg__getState(ctx); + FONStextIter!T iter, prevIter; + FONSquad q; + NVGvertex* verts; + float scale = nvg__getFontScale(state)*ctx.devicePxRatio; + float invscale = 1.0f/scale; + int cverts = 0; + int nverts = 0; + + if (state.fontId == FONS_INVALID) return x; + if (str.length == 0) return x; + + fonsSetSize(ctx.fs, state.fontSize*scale); + fonsSetSpacing(ctx.fs, state.letterSpacing*scale); + fonsSetBlur(ctx.fs, state.fontBlur*scale); + fonsSetAlign(ctx.fs, state.textAlign); + fonsSetFont(ctx.fs, state.fontId); + + cverts = nvg__max(2, cast(int)(str.length))*6; // conservative estimate + verts = nvg__allocTempVerts(ctx, cverts); + if (verts is null) return x; + + fonsTextIterInit(ctx.fs, &iter, x*scale, y*scale, str, FONS_GLYPH_BITMAP_REQUIRED); + prevIter = iter; + while (fonsTextIterNext(ctx.fs, &iter, &q)) { + float[4*2] c = void; + if (iter.prevGlyphIndex < 0) { // can not retrieve glyph? + if (nverts != 0) { + // TODO: add back-end bit to do this just once per frame + nvg__flushTextTexture(ctx); + nvg__renderText(ctx, verts, nverts); + nverts = 0; + } + if (!nvg__allocTextAtlas(ctx)) break; // no memory :( + iter = prevIter; + fonsTextIterNext(ctx.fs, &iter, &q); // try again + if (iter.prevGlyphIndex < 0) { + // still can not find glyph, try replacement + iter = prevIter; + if (!fonsTextIterGetDummyChar(ctx.fs, &iter, &q)) break; + } + } + prevIter = iter; + // Transform corners. + state.xform.point(&c[0], &c[1], q.x0*invscale, q.y0*invscale); + state.xform.point(&c[2], &c[3], q.x1*invscale, q.y0*invscale); + state.xform.point(&c[4], &c[5], q.x1*invscale, q.y1*invscale); + state.xform.point(&c[6], &c[7], q.x0*invscale, q.y1*invscale); + // Create triangles + if (nverts+6 <= cverts) { + nvg__vset(&verts[nverts], c[0], c[1], q.s0, q.t0); ++nverts; + nvg__vset(&verts[nverts], c[4], c[5], q.s1, q.t1); ++nverts; + nvg__vset(&verts[nverts], c[2], c[3], q.s1, q.t0); ++nverts; + nvg__vset(&verts[nverts], c[0], c[1], q.s0, q.t0); ++nverts; + nvg__vset(&verts[nverts], c[6], c[7], q.s0, q.t1); ++nverts; + nvg__vset(&verts[nverts], c[4], c[5], q.s1, q.t1); ++nverts; + } + } + + // TODO: add back-end bit to do this just once per frame + if (nverts > 0) { + nvg__flushTextTexture(ctx); + nvg__renderText(ctx, verts, nverts); + } + + return iter.nextx/scale; +} + +/** Draws multi-line text string at specified location wrapped at the specified width. + * White space is stripped at the beginning of the rows, the text is split at word boundaries or when new-line characters are encountered. + * Words longer than the max width are slit at nearest character (i.e. no hyphenation). + * + * Group: text_api + */ +public void textBox(T) (NVGContext ctx, float x, float y, float breakRowWidth, const(T)[] str) nothrow @trusted @nogc if (isAnyCharType!T) { + NVGstate* state = nvg__getState(ctx); + if (state.fontId == FONS_INVALID) return; + + NVGTextRow!T[2] rows; + auto oldAlign = state.textAlign; + scope(exit) state.textAlign = oldAlign; + auto halign = state.textAlign.horizontal; + float lineh = 0; + + ctx.textMetrics(null, null, &lineh); + state.textAlign.horizontal = NVGTextAlign.H.Left; + for (;;) { + auto rres = ctx.textBreakLines(str, breakRowWidth, rows[]); + //{ import core.stdc.stdio : printf; printf("slen=%u; rlen=%u; bw=%f\n", cast(uint)str.length, cast(uint)rres.length, cast(double)breakRowWidth); } + if (rres.length == 0) break; + foreach (ref row; rres) { + final switch (halign) { + case NVGTextAlign.H.Left: ctx.text(x, y, row.row); break; + case NVGTextAlign.H.Center: ctx.text(x+breakRowWidth*0.5f-row.width*0.5f, y, row.row); break; + case NVGTextAlign.H.Right: ctx.text(x+breakRowWidth-row.width, y, row.row); break; + } + y += lineh*state.lineHeight; + } + str = rres[$-1].rest; + } +} + +private template isGoodPositionDelegate(DG) { + private DG dg; + static if (is(typeof({ NVGGlyphPosition pos; bool res = dg(pos); })) || + is(typeof({ NVGGlyphPosition pos; dg(pos); }))) + enum isGoodPositionDelegate = true; + else + enum isGoodPositionDelegate = false; +} + +/** Calculates the glyph x positions of the specified text. + * Measured values are returned in local coordinate space. + * + * Group: text_api + */ +public NVGGlyphPosition[] textGlyphPositions(T) (NVGContext ctx, float x, float y, const(T)[] str, NVGGlyphPosition[] positions) nothrow @trusted @nogc +if (isAnyCharType!T) +{ + if (str.length == 0 || positions.length == 0) return positions[0..0]; + usize posnum; + auto len = ctx.textGlyphPositions(x, y, str, (in ref NVGGlyphPosition pos) { + positions.ptr[posnum++] = pos; + return (posnum < positions.length); + }); + return positions[0..len]; +} + +/// Ditto. +public int textGlyphPositions(T, DG) (NVGContext ctx, float x, float y, const(T)[] str, scope DG dg) +if (isAnyCharType!T && isGoodPositionDelegate!DG) +{ + import std.traits : ReturnType; + static if (is(ReturnType!dg == void)) enum RetBool = false; else enum RetBool = true; + + NVGstate* state = nvg__getState(ctx); + float scale = nvg__getFontScale(state)*ctx.devicePxRatio; + float invscale = 1.0f/scale; + FONStextIter!T iter, prevIter; + FONSquad q; + int npos = 0; + + if (str.length == 0) return 0; + + fonsSetSize(ctx.fs, state.fontSize*scale); + fonsSetSpacing(ctx.fs, state.letterSpacing*scale); + fonsSetBlur(ctx.fs, state.fontBlur*scale); + fonsSetAlign(ctx.fs, state.textAlign); + fonsSetFont(ctx.fs, state.fontId); + + fonsTextIterInit(ctx.fs, &iter, x*scale, y*scale, str, FONS_GLYPH_BITMAP_OPTIONAL); + prevIter = iter; + while (fonsTextIterNext(ctx.fs, &iter, &q)) { + if (iter.prevGlyphIndex < 0) { // can not retrieve glyph? + if (!nvg__allocTextAtlas(ctx)) break; // no memory + iter = prevIter; + fonsTextIterNext(ctx.fs, &iter, &q); // try again + if (iter.prevGlyphIndex < 0) { + // still can not find glyph, try replacement + iter = prevIter; + if (!fonsTextIterGetDummyChar(ctx.fs, &iter, &q)) break; + } + } + prevIter = iter; + NVGGlyphPosition position = void; //WARNING! + position.strpos = cast(usize)(iter.string-str.ptr); + position.x = iter.x*invscale; + position.minx = nvg__min(iter.x, q.x0)*invscale; + position.maxx = nvg__max(iter.nextx, q.x1)*invscale; + ++npos; + static if (RetBool) { if (!dg(position)) return npos; } else dg(position); + } + + return npos; +} + +private template isGoodRowDelegate(CT, DG) { + private DG dg; + static if (is(typeof({ NVGTextRow!CT row; bool res = dg(row); })) || + is(typeof({ NVGTextRow!CT row; dg(row); }))) + enum isGoodRowDelegate = true; + else + enum isGoodRowDelegate = false; +} + +/** Breaks the specified text into lines. + * White space is stripped at the beginning of the rows, the text is split at word boundaries or when new-line characters are encountered. + * Words longer than the max width are slit at nearest character (i.e. no hyphenation). + * + * Group: text_api + */ +public NVGTextRow!T[] textBreakLines(T) (NVGContext ctx, const(T)[] str, float breakRowWidth, NVGTextRow!T[] rows) nothrow @trusted @nogc +if (isAnyCharType!T) +{ + if (rows.length == 0) return rows; + if (rows.length > int.max-1) rows = rows[0..int.max-1]; + int nrow = 0; + auto count = ctx.textBreakLines(str, breakRowWidth, (in ref NVGTextRow!T row) { + rows[nrow++] = row; + return (nrow < rows.length); + }); + return rows[0..count]; +} + +/// Ditto. +public int textBreakLines(T, DG) (NVGContext ctx, const(T)[] str, float breakRowWidth, scope DG dg) +if (isAnyCharType!T && isGoodRowDelegate!(T, DG)) +{ + import std.traits : ReturnType; + static if (is(ReturnType!dg == void)) enum RetBool = false; else enum RetBool = true; + + enum NVGcodepointType : int { + Space, + NewLine, + Char, + } + + NVGstate* state = nvg__getState(ctx); + float scale = nvg__getFontScale(state)*ctx.devicePxRatio; + float invscale = 1.0f/scale; + FONStextIter!T iter, prevIter; + FONSquad q; + int nrows = 0; + float rowStartX = 0; + float rowWidth = 0; + float rowMinX = 0; + float rowMaxX = 0; + int rowStart = 0; + int rowEnd = 0; + int wordStart = 0; + float wordStartX = 0; + float wordMinX = 0; + int breakEnd = 0; + float breakWidth = 0; + float breakMaxX = 0; + int type = NVGcodepointType.Space, ptype = NVGcodepointType.Space; + uint pcodepoint = 0; + + if (state.fontId == FONS_INVALID) return 0; + if (str.length == 0 || dg is null) return 0; + + fonsSetSize(ctx.fs, state.fontSize*scale); + fonsSetSpacing(ctx.fs, state.letterSpacing*scale); + fonsSetBlur(ctx.fs, state.fontBlur*scale); + fonsSetAlign(ctx.fs, state.textAlign); + fonsSetFont(ctx.fs, state.fontId); + + breakRowWidth *= scale; + + enum Phase { + Normal, // searching for breaking point + SkipBlanks, // skip leading blanks + } + Phase phase = Phase.SkipBlanks; // don't skip blanks on first line + + fonsTextIterInit(ctx.fs, &iter, 0, 0, str, FONS_GLYPH_BITMAP_OPTIONAL); + prevIter = iter; + while (fonsTextIterNext(ctx.fs, &iter, &q)) { + if (iter.prevGlyphIndex < 0) { // can not retrieve glyph? + if (!nvg__allocTextAtlas(ctx)) break; // no memory + iter = prevIter; + fonsTextIterNext(ctx.fs, &iter, &q); // try again + if (iter.prevGlyphIndex < 0) { + // still can not find glyph, try replacement + iter = prevIter; + if (!fonsTextIterGetDummyChar(ctx.fs, &iter, &q)) break; + } + } + prevIter = iter; + switch (iter.codepoint) { + case 9: // \t + case 11: // \v + case 12: // \f + case 32: // space + case 0x00a0: // NBSP + type = NVGcodepointType.Space; + break; + case 10: // \n + type = (pcodepoint == 13 ? NVGcodepointType.Space : NVGcodepointType.NewLine); + break; + case 13: // \r + type = (pcodepoint == 10 ? NVGcodepointType.Space : NVGcodepointType.NewLine); + break; + case 0x0085: // NEL + case 0x2028: // Line Separator + case 0x2029: // Paragraph Separator + type = NVGcodepointType.NewLine; + break; + default: + type = NVGcodepointType.Char; + break; + } + if (phase == Phase.SkipBlanks) { + // fix row start + rowStart = cast(int)(iter.string-str.ptr); + rowEnd = rowStart; + rowStartX = iter.x; + rowWidth = iter.nextx-rowStartX; // q.x1-rowStartX; + rowMinX = q.x0-rowStartX; + rowMaxX = q.x1-rowStartX; + wordStart = rowStart; + wordStartX = iter.x; + wordMinX = q.x0-rowStartX; + breakEnd = rowStart; + breakWidth = 0.0; + breakMaxX = 0.0; + if (type == NVGcodepointType.Space) continue; + phase = Phase.Normal; + } + + if (type == NVGcodepointType.NewLine) { + // always handle new lines + NVGTextRow!T row; + row.string = str; + row.start = rowStart; + row.end = rowEnd; + row.width = rowWidth*invscale; + row.minx = rowMinX*invscale; + row.maxx = rowMaxX*invscale; + ++nrows; + static if (RetBool) { if (!dg(row)) return nrows; } else dg(row); + phase = Phase.SkipBlanks; + } else { + float nextWidth = iter.nextx-rowStartX; + // track last non-white space character + if (type == NVGcodepointType.Char) { + rowEnd = cast(int)(iter.nextp-str.ptr); + rowWidth = iter.nextx-rowStartX; + rowMaxX = q.x1-rowStartX; + } + // track last end of a word + if (ptype == NVGcodepointType.Char && type == NVGcodepointType.Space) { + breakEnd = cast(int)(iter.string-str.ptr); + breakWidth = rowWidth; + breakMaxX = rowMaxX; + } + // track last beginning of a word + if (ptype == NVGcodepointType.Space && type == NVGcodepointType.Char) { + wordStart = cast(int)(iter.string-str.ptr); + wordStartX = iter.x; + wordMinX = q.x0-rowStartX; + } + // break to new line when a character is beyond break width + if (type == NVGcodepointType.Char && nextWidth > breakRowWidth) { + // the run length is too long, need to break to new line + NVGTextRow!T row; + row.string = str; + if (breakEnd == rowStart) { + // the current word is longer than the row length, just break it from here + row.start = rowStart; + row.end = cast(int)(iter.string-str.ptr); + row.width = rowWidth*invscale; + row.minx = rowMinX*invscale; + row.maxx = rowMaxX*invscale; + ++nrows; + static if (RetBool) { if (!dg(row)) return nrows; } else dg(row); + rowStartX = iter.x; + rowStart = cast(int)(iter.string-str.ptr); + rowEnd = cast(int)(iter.nextp-str.ptr); + rowWidth = iter.nextx-rowStartX; + rowMinX = q.x0-rowStartX; + rowMaxX = q.x1-rowStartX; + wordStart = rowStart; + wordStartX = iter.x; + wordMinX = q.x0-rowStartX; + } else { + // break the line from the end of the last word, and start new line from the beginning of the new + //{ import core.stdc.stdio : printf; printf("rowStart=%u; rowEnd=%u; breakEnd=%u; len=%u\n", rowStart, rowEnd, breakEnd, cast(uint)str.length); } + row.start = rowStart; + row.end = breakEnd; + row.width = breakWidth*invscale; + row.minx = rowMinX*invscale; + row.maxx = breakMaxX*invscale; + ++nrows; + static if (RetBool) { if (!dg(row)) return nrows; } else dg(row); + rowStartX = wordStartX; + rowStart = wordStart; + rowEnd = cast(int)(iter.nextp-str.ptr); + rowWidth = iter.nextx-rowStartX; + rowMinX = wordMinX; + rowMaxX = q.x1-rowStartX; + // no change to the word start + } + // set null break point + breakEnd = rowStart; + breakWidth = 0.0; + breakMaxX = 0.0; + } + } + + pcodepoint = iter.codepoint; + ptype = type; + } + + // break the line from the end of the last word, and start new line from the beginning of the new + if (phase != Phase.SkipBlanks && rowStart < str.length) { + //{ import core.stdc.stdio : printf; printf(" rowStart=%u; len=%u\n", rowStart, cast(uint)str.length); } + NVGTextRow!T row; + row.string = str; + row.start = rowStart; + row.end = cast(int)str.length; + row.width = rowWidth*invscale; + row.minx = rowMinX*invscale; + row.maxx = rowMaxX*invscale; + ++nrows; + static if (RetBool) { if (!dg(row)) return nrows; } else dg(row); + } + + return nrows; +} + +/** Returns iterator which you can use to calculate text bounds and advancement. + * This is usable when you need to do some text layouting with wrapping, to avoid + * guesswork ("will advancement for this space stay the same?"), and Schlemiel's + * algorithm. Note that you can copy the returned struct to save iterator state. + * + * You can check if iterator is valid with [valid] property, put new chars with + * [put] method, get current advance with [advance] property, and current + * bounds with `getBounds(ref float[4] bounds)` method. + * + * WARNING! Don't change font parameters while iterating! Or use [restoreFont] + * method. + * + * Group: text_api + */ +public struct TextBoundsIterator { +private: + NVGContext ctx; + FonsTextBoundsIterator fsiter; // fontstash iterator + float scale, invscale, xscaled, yscaled; + // font settings + float fsSize, fsSpacing, fsBlur; + int fsFontId; + NVGTextAlign fsAlign; + +public: + this (NVGContext actx, float ax, float ay) nothrow @trusted @nogc { reset(actx, ax, ay); } + + void reset (NVGContext actx, float ax, float ay) nothrow @trusted @nogc { + fsiter = fsiter.init; + this = this.init; + if (actx is null) return; + NVGstate* state = nvg__getState(actx); + if (state is null) return; + if (state.fontId == FONS_INVALID) { ctx = null; return; } + + ctx = actx; + scale = nvg__getFontScale(state)*ctx.devicePxRatio; + invscale = 1.0f/scale; + + fsSize = state.fontSize*scale; + fsSpacing = state.letterSpacing*scale; + fsBlur = state.fontBlur*scale; + fsAlign = state.textAlign; + fsFontId = state.fontId; + restoreFont(); + + xscaled = ax*scale; + yscaled = ay*scale; + fsiter.reset(ctx.fs, xscaled, yscaled); + } + + /// Restart iteration. Will not restore font. + void restart () nothrow @trusted @nogc { + if (ctx !is null) fsiter.reset(ctx.fs, xscaled, yscaled); + } + + /// Restore font settings for the context. + void restoreFont () nothrow @trusted @nogc { + if (ctx !is null) { + fonsSetSize(ctx.fs, fsSize); + fonsSetSpacing(ctx.fs, fsSpacing); + fonsSetBlur(ctx.fs, fsBlur); + fonsSetAlign(ctx.fs, fsAlign); + fonsSetFont(ctx.fs, fsFontId); + } + } + + /// Is this iterator valid? + @property bool valid () const pure nothrow @safe @nogc { pragma(inline, true); return (ctx !is null); } + + /// Add chars. + void put(T) (const(T)[] str...) nothrow @trusted @nogc if (isAnyCharType!T) { pragma(inline, true); if (ctx !is null) fsiter.put(str[]); } + + /// Returns current advance + @property float advance () const pure nothrow @safe @nogc { pragma(inline, true); return (ctx !is null ? fsiter.advance*invscale : 0); } + + /// Returns current text bounds. + void getBounds (ref float[4] bounds) nothrow @trusted @nogc { + if (ctx !is null) { + fsiter.getBounds(bounds); + fonsLineBounds(ctx.fs, yscaled, &bounds[1], &bounds[3]); + bounds[0] *= invscale; + bounds[1] *= invscale; + bounds[2] *= invscale; + bounds[3] *= invscale; + } else { + bounds[] = 0; + } + } + + /// Returns current horizontal text bounds. + void getHBounds (out float xmin, out float xmax) nothrow @trusted @nogc { + if (ctx !is null) { + fsiter.getHBounds(xmin, xmax); + xmin *= invscale; + xmax *= invscale; + } + } + + /// Returns current vertical text bounds. + void getVBounds (out float ymin, out float ymax) nothrow @trusted @nogc { + if (ctx !is null) { + //fsiter.getVBounds(ymin, ymax); + fonsLineBounds(ctx.fs, yscaled, &ymin, &ymax); + ymin *= invscale; + ymax *= invscale; + } + } +} + +/// Returns font line height (without line spacing), measured in local coordinate space. +/// Group: text_api +public float textFontHeight (NVGContext ctx) nothrow @trusted @nogc { + float res = void; + ctx.textMetrics(null, null, &res); + return res; +} + +/// Returns font ascender, measured in local coordinate space. +/// Group: text_api +public float textFontAscender (NVGContext ctx) nothrow @trusted @nogc { + float res = void; + ctx.textMetrics(&res, null, null); + return res; +} + +/// Returns font descender, measured in local coordinate space. +/// Group: text_api +public float textFontDescender (NVGContext ctx) nothrow @trusted @nogc { + float res = void; + ctx.textMetrics(null, &res, null); + return res; +} + +/** Measures the specified text string. Returns horizontal and vertical sizes of the measured text. + * Measured values are returned in local coordinate space. + * + * Group: text_api + */ +public void textExtents(T) (NVGContext ctx, const(T)[] str, float *w, float *h) nothrow @trusted @nogc if (isAnyCharType!T) { + float[4] bnd = void; + ctx.textBounds(0, 0, str, bnd[]); + if (!fonsGetFontAA(ctx.fs, nvg__getState(ctx).fontId)) { + if (w !is null) *w = nvg__lrintf(bnd.ptr[2]-bnd.ptr[0]); + if (h !is null) *h = nvg__lrintf(bnd.ptr[3]-bnd.ptr[1]); + } else { + if (w !is null) *w = bnd.ptr[2]-bnd.ptr[0]; + if (h !is null) *h = bnd.ptr[3]-bnd.ptr[1]; + } +} + +/** Measures the specified text string. Returns horizontal size of the measured text. + * Measured values are returned in local coordinate space. + * + * Group: text_api + */ +public float textWidth(T) (NVGContext ctx, const(T)[] str) nothrow @trusted @nogc if (isAnyCharType!T) { + float w = void; + ctx.textExtents(str, &w, null); + return w; +} + +/** Measures the specified text string. Parameter bounds should be a float[4], + * if the bounding box of the text should be returned. The bounds value are [xmin, ymin, xmax, ymax] + * Returns the horizontal advance of the measured text (i.e. where the next character should drawn). + * Measured values are returned in local coordinate space. + * + * Group: text_api + */ +public float textBounds(T) (NVGContext ctx, float x, float y, const(T)[] str, float[] bounds) nothrow @trusted @nogc +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; + } + + fonsSetSize(ctx.fs, state.fontSize*scale); + fonsSetSpacing(ctx.fs, state.letterSpacing*scale); + fonsSetBlur(ctx.fs, state.fontBlur*scale); + fonsSetAlign(ctx.fs, state.textAlign); + fonsSetFont(ctx.fs, state.fontId); + + float[4] b = void; + width = fonsTextBounds(ctx.fs, x*scale, y*scale, str, b[]); + if (bounds.length) { + // use line bounds for height + fonsLineBounds(ctx.fs, y*scale, b.ptr+1, b.ptr+3); + if (bounds.length > 0) bounds.ptr[0] = b.ptr[0]*invscale; + if (bounds.length > 1) bounds.ptr[1] = b.ptr[1]*invscale; + if (bounds.length > 2) bounds.ptr[2] = b.ptr[2]*invscale; + if (bounds.length > 3) bounds.ptr[3] = b.ptr[3]*invscale; + } + return width*invscale; +} + +/// Ditto. +public void textBoxBounds(T) (NVGContext ctx, float x, float y, float breakRowWidth, const(T)[] str, float[] bounds) if (isAnyCharType!T) { + NVGstate* state = nvg__getState(ctx); + NVGTextRow!T[2] rows; + float scale = nvg__getFontScale(state)*ctx.devicePxRatio; + float invscale = 1.0f/scale; + float lineh = 0, rminy = 0, rmaxy = 0; + float minx, miny, maxx, maxy; + + if (state.fontId == FONS_INVALID) { + bounds[] = 0; + return; + } + + auto oldAlign = state.textAlign; + scope(exit) state.textAlign = oldAlign; + auto halign = state.textAlign.horizontal; + + ctx.textMetrics(null, null, &lineh); + state.textAlign.horizontal = NVGTextAlign.H.Left; + + minx = maxx = x; + miny = maxy = y; + + fonsSetSize(ctx.fs, state.fontSize*scale); + fonsSetSpacing(ctx.fs, state.letterSpacing*scale); + fonsSetBlur(ctx.fs, state.fontBlur*scale); + fonsSetAlign(ctx.fs, state.textAlign); + fonsSetFont(ctx.fs, state.fontId); + fonsLineBounds(ctx.fs, 0, &rminy, &rmaxy); + rminy *= invscale; + rmaxy *= invscale; + + for (;;) { + auto rres = ctx.textBreakLines(str, breakRowWidth, rows[]); + if (rres.length == 0) break; + foreach (ref row; rres) { + float rminx, rmaxx, dx = 0; + // horizontal bounds + final switch (halign) { + case NVGTextAlign.H.Left: dx = 0; break; + case NVGTextAlign.H.Center: dx = breakRowWidth*0.5f-row.width*0.5f; break; + case NVGTextAlign.H.Right: dx = breakRowWidth-row.width; break; + } + rminx = x+row.minx+dx; + rmaxx = x+row.maxx+dx; + minx = nvg__min(minx, rminx); + maxx = nvg__max(maxx, rmaxx); + // vertical bounds + miny = nvg__min(miny, y+rminy); + maxy = nvg__max(maxy, y+rmaxy); + y += lineh*state.lineHeight; + } + str = rres[$-1].rest; + } + + if (bounds.length) { + if (bounds.length > 0) bounds.ptr[0] = minx; + if (bounds.length > 1) bounds.ptr[1] = miny; + if (bounds.length > 2) bounds.ptr[2] = maxx; + if (bounds.length > 3) bounds.ptr[3] = maxy; + } +} + +/// Returns the vertical metrics based on the current text style. Measured values are returned in local coordinate space. +/// Group: text_api +public void textMetrics (NVGContext ctx, float* ascender, float* descender, float* lineh) nothrow @trusted @nogc { + NVGstate* state = nvg__getState(ctx); + + if (state.fontId == FONS_INVALID) { + if (ascender !is null) *ascender *= 0; + if (descender !is null) *descender *= 0; + if (lineh !is null) *lineh *= 0; + return; + } + + immutable float scale = nvg__getFontScale(state)*ctx.devicePxRatio; + immutable float invscale = 1.0f/scale; + + fonsSetSize(ctx.fs, state.fontSize*scale); + fonsSetSpacing(ctx.fs, state.letterSpacing*scale); + fonsSetBlur(ctx.fs, state.fontBlur*scale); + fonsSetAlign(ctx.fs, state.textAlign); + fonsSetFont(ctx.fs, state.fontId); + + fonsVertMetrics(ctx.fs, ascender, descender, lineh); + if (ascender !is null) *ascender *= invscale; + if (descender !is null) *descender *= invscale; + if (lineh !is null) *lineh *= invscale; +} + + +// ////////////////////////////////////////////////////////////////////////// // +// fontstash +// ////////////////////////////////////////////////////////////////////////// // +import core.stdc.stdlib : malloc, realloc, free; +import core.stdc.string : memset, memcpy, strncpy, strcmp, strlen; +import core.stdc.stdio : FILE, fopen, fclose, fseek, ftell, fread, SEEK_END, SEEK_SET; + +public: +// welcome to version hell! +version(nanovg_force_detect) {} else version(nanovg_use_freetype) { version = nanovg_use_freetype_ii; } +version(nanovg_ignore_iv_stb_ttf) enum nanovg_ignore_iv_stb_ttf = true; else enum nanovg_ignore_iv_stb_ttf = false; +//version(nanovg_ignore_mono); + +version (nanovg_builtin_freetype_bindings) { + version(Posix) { + private enum NanoVegaForceFreeType = true; + } else { + private enum NanoVegaForceFreeType = false; + } +} else { + version(Posix) { + private enum NanoVegaForceFreeType = true; + } else { + private enum NanoVegaForceFreeType = false; + } +} + +version(nanovg_use_freetype_ii) { + enum NanoVegaIsUsingSTBTTF = false; + //pragma(msg, "iv.freetype: forced"); +} else { + static if (NanoVegaForceFreeType) { + enum NanoVegaIsUsingSTBTTF = false; + } else { + static if (!nanovg_ignore_iv_stb_ttf && __traits(compiles, { import iv.stb.ttf; })) { + import iv.stb.ttf; + enum NanoVegaIsUsingSTBTTF = true; + //pragma(msg, "iv.stb.ttf"); + } else static if (__traits(compiles, { import arsd.ttf; })) { + import arsd.ttf; + enum NanoVegaIsUsingSTBTTF = true; + //pragma(msg, "arsd.ttf"); + } else static if (__traits(compiles, { import stb_truetype; })) { + import stb_truetype; + enum NanoVegaIsUsingSTBTTF = true; + //pragma(msg, "stb_truetype"); + } else static if (__traits(compiles, { import iv.freetype; })) { + version (nanovg_builtin_freetype_bindings) { + enum NanoVegaIsUsingSTBTTF = false; + version = nanovg_builtin_freetype_bindings; + } else { + import iv.freetype; + enum NanoVegaIsUsingSTBTTF = false; + } + //pragma(msg, "iv.freetype"); + } else { + static assert(0, "no stb_ttf/iv.freetype found!"); + } + } +} + +//version = nanovg_kill_font_blur; + + +// ////////////////////////////////////////////////////////////////////////// // +//version = nanovg_ft_mono; + +enum FONS_INVALID = -1; + +alias FONSflags = int; +enum /*FONSflags*/ { + FONS_ZERO_TOPLEFT = 1<<0, + FONS_ZERO_BOTTOMLEFT = 1<<1, +} + +/+ +alias FONSalign = int; +enum /*FONSalign*/ { + // Horizontal align + FONS_ALIGN_LEFT = 1<<0, // Default + FONS_ALIGN_CENTER = 1<<1, + FONS_ALIGN_RIGHT = 1<<2, + // Vertical align + FONS_ALIGN_TOP = 1<<3, + FONS_ALIGN_MIDDLE = 1<<4, + FONS_ALIGN_BOTTOM = 1<<5, + FONS_ALIGN_BASELINE = 1<<6, // Default +} ++/ + +alias FONSglyphBitmap = int; +enum /*FONSglyphBitmap*/ { + FONS_GLYPH_BITMAP_OPTIONAL = 1, + FONS_GLYPH_BITMAP_REQUIRED = 2, +} + +alias FONSerrorCode = int; +enum /*FONSerrorCode*/ { + // Font atlas is full. + FONS_ATLAS_FULL = 1, + // Scratch memory used to render glyphs is full, requested size reported in 'val', you may need to bump up FONS_SCRATCH_BUF_SIZE. + FONS_SCRATCH_FULL = 2, + // Calls to fonsPushState has created too large stack, if you need deep state stack bump up FONS_MAX_STATES. + FONS_STATES_OVERFLOW = 3, + // Trying to pop too many states fonsPopState(). + FONS_STATES_UNDERFLOW = 4, +} + +struct FONSparams { + int width, height; + ubyte flags; + void* userPtr; + bool function (void* uptr, int width, int height) nothrow @trusted @nogc renderCreate; + int function (void* uptr, int width, int height) nothrow @trusted @nogc renderResize; + void function (void* uptr, int* rect, const(ubyte)* data) nothrow @trusted @nogc renderUpdate; + void function (void* uptr, const(float)* verts, const(float)* tcoords, const(uint)* colors, int nverts) nothrow @trusted @nogc renderDraw; + void function (void* uptr) nothrow @trusted @nogc renderDelete; +} + +struct FONSquad { + float x0=0, y0=0, s0=0, t0=0; + float x1=0, y1=0, s1=0, t1=0; +} + +struct FONStextIter(CT) if (isAnyCharType!CT) { + alias CharType = CT; + float x=0, y=0, nextx=0, nexty=0, scale=0, spacing=0; + uint codepoint; + short isize, iblur; + FONSfont* font; + int prevGlyphIndex; + const(CT)* s; // string + const(CT)* n; // next + const(CT)* e; // end + FONSglyphBitmap bitmapOption; + static if (is(CT == char)) { + uint utf8state; + } + ~this () nothrow @trusted @nogc { pragma(inline, true); static if (is(CT == char)) utf8state = 0; s = n = e = null; } + @property const(CT)* string () const pure nothrow @nogc { pragma(inline, true); return s; } + @property const(CT)* nextp () const pure nothrow @nogc { pragma(inline, true); return n; } + @property const(CT)* endp () const pure nothrow @nogc { pragma(inline, true); return e; } +} + + +// ////////////////////////////////////////////////////////////////////////// // +//static if (!HasAST) version = nanovg_use_freetype_ii_x; + +/*version(nanovg_use_freetype_ii_x)*/ static if (!NanoVegaIsUsingSTBTTF) { +version(nanovg_builtin_freetype_bindings) { +pragma(lib, "freetype"); +private extern(C) nothrow @trusted @nogc { +private import core.stdc.config : c_long, c_ulong; +alias FT_Pos = c_long; +// config/ftconfig.h +alias FT_Int16 = short; +alias FT_UInt16 = ushort; +alias FT_Int32 = int; +alias FT_UInt32 = uint; +alias FT_Fast = int; +alias FT_UFast = uint; +alias FT_Int64 = long; +alias FT_Uint64 = ulong; +// fttypes.h +alias FT_Bool = ubyte; +alias FT_FWord = short; +alias FT_UFWord = ushort; +alias FT_Char = char; +alias FT_Byte = ubyte; +alias FT_Bytes = FT_Byte*; +alias FT_Tag = FT_UInt32; +alias FT_String = char; +alias FT_Short = short; +alias FT_UShort = ushort; +alias FT_Int = int; +alias FT_UInt = uint; +alias FT_Long = c_long; +alias FT_ULong = c_ulong; +alias FT_F2Dot14 = short; +alias FT_F26Dot6 = c_long; +alias FT_Fixed = c_long; +alias FT_Error = int; +alias FT_Pointer = void*; +alias FT_Offset = usize; +alias FT_PtrDist = ptrdiff_t; + +struct FT_UnitVector { + FT_F2Dot14 x; + FT_F2Dot14 y; +} + +struct FT_Matrix { + FT_Fixed xx, xy; + FT_Fixed yx, yy; +} + +struct FT_Data { + const(FT_Byte)* pointer; + FT_Int length; +} +alias FT_Face = FT_FaceRec*; +struct FT_FaceRec { + FT_Long num_faces; + FT_Long face_index; + FT_Long face_flags; + FT_Long style_flags; + FT_Long num_glyphs; + FT_String* family_name; + FT_String* style_name; + FT_Int num_fixed_sizes; + FT_Bitmap_Size* available_sizes; + FT_Int num_charmaps; + FT_CharMap* charmaps; + FT_Generic generic; + FT_BBox bbox; + FT_UShort units_per_EM; + FT_Short ascender; + FT_Short descender; + FT_Short height; + FT_Short max_advance_width; + FT_Short max_advance_height; + FT_Short underline_position; + FT_Short underline_thickness; + FT_GlyphSlot glyph; + FT_Size size; + FT_CharMap charmap; + FT_Driver driver; + FT_Memory memory; + FT_Stream stream; + FT_ListRec sizes_list; + FT_Generic autohint; + void* extensions; + FT_Face_Internal internal; +} +struct FT_Bitmap_Size { + FT_Short height; + FT_Short width; + FT_Pos size; + FT_Pos x_ppem; + FT_Pos y_ppem; +} +alias FT_CharMap = FT_CharMapRec*; +struct FT_CharMapRec { + FT_Face face; + FT_Encoding encoding; + FT_UShort platform_id; + FT_UShort encoding_id; +} +extern(C) nothrow @nogc { alias FT_Generic_Finalizer = void function (void* object); } +struct FT_Generic { + void* data; + FT_Generic_Finalizer finalizer; +} +struct FT_Vector { + FT_Pos x; + FT_Pos y; +} +struct FT_BBox { + FT_Pos xMin, yMin; + FT_Pos xMax, yMax; +} +alias FT_Pixel_Mode = int; +enum { + FT_PIXEL_MODE_NONE = 0, + FT_PIXEL_MODE_MONO, + FT_PIXEL_MODE_GRAY, + FT_PIXEL_MODE_GRAY2, + FT_PIXEL_MODE_GRAY4, + FT_PIXEL_MODE_LCD, + FT_PIXEL_MODE_LCD_V, + FT_PIXEL_MODE_MAX +} +struct FT_Bitmap { + uint rows; + uint width; + int pitch; + ubyte* buffer; + ushort num_grays; + ubyte pixel_mode; + ubyte palette_mode; + void* palette; +} +struct FT_Outline { + short n_contours; + short n_points; + FT_Vector* points; + byte* tags; + short* contours; + int flags; +} +alias FT_GlyphSlot = FT_GlyphSlotRec*; +struct FT_GlyphSlotRec { + FT_Library library; + FT_Face face; + FT_GlyphSlot next; + FT_UInt reserved; + FT_Generic generic; + FT_Glyph_Metrics metrics; + FT_Fixed linearHoriAdvance; + FT_Fixed linearVertAdvance; + FT_Vector advance; + FT_Glyph_Format format; + FT_Bitmap bitmap; + FT_Int bitmap_left; + FT_Int bitmap_top; + FT_Outline outline; + FT_UInt num_subglyphs; + FT_SubGlyph subglyphs; + void* control_data; + c_long control_len; + FT_Pos lsb_delta; + FT_Pos rsb_delta; + void* other; + FT_Slot_Internal internal; +} +alias FT_Size = FT_SizeRec*; +struct FT_SizeRec { + FT_Face face; + FT_Generic generic; + FT_Size_Metrics metrics; + FT_Size_Internal internal; +} +alias FT_Encoding = FT_Tag; +alias FT_Face_Internal = void*; +alias FT_Driver = void*; +alias FT_Memory = void*; +alias FT_Stream = void*; +alias FT_Library = void*; +alias FT_SubGlyph = void*; +alias FT_Slot_Internal = void*; +alias FT_Size_Internal = void*; +alias FT_ListNode = FT_ListNodeRec*; +alias FT_List = FT_ListRec*; +struct FT_ListNodeRec { + FT_ListNode prev; + FT_ListNode next; + void* data; +} +struct FT_ListRec { + FT_ListNode head; + FT_ListNode tail; +} +struct FT_Glyph_Metrics { + FT_Pos width; + FT_Pos height; + FT_Pos horiBearingX; + FT_Pos horiBearingY; + FT_Pos horiAdvance; + FT_Pos vertBearingX; + FT_Pos vertBearingY; + FT_Pos vertAdvance; +} +alias FT_Glyph_Format = FT_Tag; +FT_Tag FT_MAKE_TAG (char x1, char x2, char x3, char x4) pure nothrow @safe @nogc { + pragma(inline, true); + return cast(FT_UInt32)((x1<<24)|(x2<<16)|(x3<<8)|x4); +} +enum : FT_Tag { + FT_GLYPH_FORMAT_NONE = 0, + FT_GLYPH_FORMAT_COMPOSITE = FT_MAKE_TAG('c','o','m','p'), + FT_GLYPH_FORMAT_BITMAP = FT_MAKE_TAG('b','i','t','s'), + FT_GLYPH_FORMAT_OUTLINE = FT_MAKE_TAG('o','u','t','l'), + FT_GLYPH_FORMAT_PLOTTER = FT_MAKE_TAG('p','l','o','t'), +} +struct FT_Size_Metrics { + FT_UShort x_ppem; + FT_UShort y_ppem; + + FT_Fixed x_scale; + FT_Fixed y_scale; + + FT_Pos ascender; + FT_Pos descender; + FT_Pos height; + FT_Pos max_advance; +} +enum FT_LOAD_DEFAULT = 0x0U; +enum FT_LOAD_NO_SCALE = 1U<<0; +enum FT_LOAD_NO_HINTING = 1U<<1; +enum FT_LOAD_RENDER = 1U<<2; +enum FT_LOAD_NO_BITMAP = 1U<<3; +enum FT_LOAD_VERTICAL_LAYOUT = 1U<<4; +enum FT_LOAD_FORCE_AUTOHINT = 1U<<5; +enum FT_LOAD_CROP_BITMAP = 1U<<6; +enum FT_LOAD_PEDANTIC = 1U<<7; +enum FT_LOAD_IGNORE_GLOBAL_ADVANCE_WIDTH = 1U<<9; +enum FT_LOAD_NO_RECURSE = 1U<<10; +enum FT_LOAD_IGNORE_TRANSFORM = 1U<<11; +enum FT_LOAD_MONOCHROME = 1U<<12; +enum FT_LOAD_LINEAR_DESIGN = 1U<<13; +enum FT_LOAD_NO_AUTOHINT = 1U<<15; +enum FT_LOAD_COLOR = 1U<<20; +enum FT_LOAD_COMPUTE_METRICS = 1U<<21; +enum FT_FACE_FLAG_KERNING = 1U<<6; +alias FT_Kerning_Mode = int; +enum /*FT_Kerning_Mode*/ { + FT_KERNING_DEFAULT = 0, + FT_KERNING_UNFITTED, + FT_KERNING_UNSCALED +} +extern(C) nothrow @nogc { + alias FT_Outline_MoveToFunc = int function (const(FT_Vector)*, void*); + alias FT_Outline_LineToFunc = int function (const(FT_Vector)*, void*); + alias FT_Outline_ConicToFunc = int function (const(FT_Vector)*, const(FT_Vector)*, void*); + alias FT_Outline_CubicToFunc = int function (const(FT_Vector)*, const(FT_Vector)*, const(FT_Vector)*, void*); +} +struct FT_Outline_Funcs { + FT_Outline_MoveToFunc move_to; + FT_Outline_LineToFunc line_to; + FT_Outline_ConicToFunc conic_to; + FT_Outline_CubicToFunc cubic_to; + int shift; + FT_Pos delta; +} + +FT_Error FT_Init_FreeType (FT_Library*); +FT_Error FT_New_Memory_Face (FT_Library, const(FT_Byte)*, FT_Long, FT_Long, FT_Face*); +FT_UInt FT_Get_Char_Index (FT_Face, FT_ULong); +FT_Error FT_Set_Pixel_Sizes (FT_Face, FT_UInt, FT_UInt); +FT_Error FT_Load_Glyph (FT_Face, FT_UInt, FT_Int32); +FT_Error FT_Get_Advance (FT_Face, FT_UInt, FT_Int32, FT_Fixed*); +FT_Error FT_Get_Kerning (FT_Face, FT_UInt, FT_UInt, FT_UInt, FT_Vector*); +void FT_Outline_Get_CBox (const(FT_Outline)*, FT_BBox*); +FT_Error FT_Outline_Decompose (FT_Outline*, const(FT_Outline_Funcs)*, void*); +} +} else { +import iv.freetype; +} + +struct FONSttFontImpl { + FT_Face font; + bool mono; // no aa? +} + +__gshared FT_Library ftLibrary; + +int fons__tt_init (FONScontext* context) nothrow @trusted @nogc { + FT_Error ftError; + //FONS_NOTUSED(context); + ftError = FT_Init_FreeType(&ftLibrary); + return (ftError == 0); +} + +void fons__tt_setMono (FONScontext* context, FONSttFontImpl* font, bool v) nothrow @trusted @nogc { + font.mono = v; +} + +bool fons__tt_getMono (FONScontext* context, FONSttFontImpl* font) nothrow @trusted @nogc { + return font.mono; +} + +int fons__tt_loadFont (FONScontext* context, FONSttFontImpl* font, ubyte* data, int dataSize) nothrow @trusted @nogc { + FT_Error ftError; + //font.font.userdata = stash; + ftError = FT_New_Memory_Face(ftLibrary, cast(const(FT_Byte)*)data, dataSize, 0, &font.font); + return ftError == 0; +} + +void fons__tt_getFontVMetrics (FONSttFontImpl* font, int* ascent, int* descent, int* lineGap) nothrow @trusted @nogc { + *ascent = font.font.ascender; + *descent = font.font.descender; + *lineGap = font.font.height-(*ascent - *descent); +} + +float fons__tt_getPixelHeightScale (FONSttFontImpl* font, float size) nothrow @trusted @nogc { + return size/(font.font.ascender-font.font.descender); +} + +int fons__tt_getGlyphIndex (FONSttFontImpl* font, int codepoint) nothrow @trusted @nogc { + return FT_Get_Char_Index(font.font, codepoint); +} + +int fons__tt_buildGlyphBitmap (FONSttFontImpl* font, int glyph, float size, float scale, int* advance, int* lsb, int* x0, int* y0, int* x1, int* y1) nothrow @trusted @nogc { + FT_Error ftError; + FT_GlyphSlot ftGlyph; + //version(nanovg_ignore_mono) enum exflags = 0; + //else version(nanovg_ft_mono) enum exflags = FT_LOAD_MONOCHROME; else enum exflags = 0; + uint exflags = (font.mono ? FT_LOAD_MONOCHROME : 0); + ftError = FT_Set_Pixel_Sizes(font.font, 0, cast(FT_UInt)(size*cast(float)font.font.units_per_EM/cast(float)(font.font.ascender-font.font.descender))); + if (ftError) return 0; + ftError = FT_Load_Glyph(font.font, glyph, FT_LOAD_RENDER|/*FT_LOAD_NO_AUTOHINT|*/exflags); + if (ftError) return 0; + ftError = FT_Get_Advance(font.font, glyph, FT_LOAD_NO_SCALE|/*FT_LOAD_NO_AUTOHINT|*/exflags, cast(FT_Fixed*)advance); + if (ftError) return 0; + ftGlyph = font.font.glyph; + *lsb = cast(int)ftGlyph.metrics.horiBearingX; + *x0 = ftGlyph.bitmap_left; + *x1 = *x0+ftGlyph.bitmap.width; + *y0 = -ftGlyph.bitmap_top; + *y1 = *y0+ftGlyph.bitmap.rows; + return 1; +} + +void fons__tt_renderGlyphBitmap (FONSttFontImpl* font, ubyte* output, int outWidth, int outHeight, int outStride, float scaleX, float scaleY, int glyph) nothrow @trusted @nogc { + FT_GlyphSlot ftGlyph = font.font.glyph; + //FONS_NOTUSED(glyph); // glyph has already been loaded by fons__tt_buildGlyphBitmap + //version(nanovg_ignore_mono) enum RenderAA = true; + //else version(nanovg_ft_mono) enum RenderAA = false; + //else enum RenderAA = true; + if (font.mono) { + auto src = ftGlyph.bitmap.buffer; + auto dst = output; + auto spt = ftGlyph.bitmap.pitch; + if (spt < 0) spt = -spt; + foreach (int y; 0..ftGlyph.bitmap.rows) { + ubyte count = 0, b = 0; + auto s = src; + auto d = dst; + foreach (int x; 0..ftGlyph.bitmap.width) { + if (count-- == 0) { count = 7; b = *s++; } else b <<= 1; + *d++ = (b&0x80 ? 255 : 0); + } + src += spt; + dst += outStride; + } + } else { + auto src = ftGlyph.bitmap.buffer; + auto dst = output; + auto spt = ftGlyph.bitmap.pitch; + if (spt < 0) spt = -spt; + foreach (int y; 0..ftGlyph.bitmap.rows) { + import core.stdc.string : memcpy; + //dst[0..ftGlyph.bitmap.width] = src[0..ftGlyph.bitmap.width]; + memcpy(dst, src, ftGlyph.bitmap.width); + src += spt; + dst += outStride; + } + } +} + +float fons__tt_getGlyphKernAdvance (FONSttFontImpl* font, float size, int glyph1, int glyph2) nothrow @trusted @nogc { + FT_Vector ftKerning; + version(none) { + // fitted kerning + FT_Get_Kerning(font.font, glyph1, glyph2, FT_KERNING_DEFAULT, &ftKerning); + //{ import core.stdc.stdio : printf; printf("kern for %u:%u: %d %d\n", glyph1, glyph2, ftKerning.x, ftKerning.y); } + return cast(int)ftKerning.x; // round up and convert to integer + } else { + // unfitted kerning + //FT_Get_Kerning(font.font, glyph1, glyph2, FT_KERNING_UNFITTED, &ftKerning); + if (glyph1 <= 0 || glyph2 <= 0 || (font.font.face_flags&FT_FACE_FLAG_KERNING) == 0) return 0; + if (FT_Set_Pixel_Sizes(font.font, 0, cast(FT_UInt)(size*cast(float)font.font.units_per_EM/cast(float)(font.font.ascender-font.font.descender)))) return 0; + if (FT_Get_Kerning(font.font, glyph1, glyph2, FT_KERNING_DEFAULT, &ftKerning)) return 0; + version(none) { + if (ftKerning.x) { + //{ import core.stdc.stdio : printf; printf("has kerning: %u\n", cast(uint)(font.font.face_flags&FT_FACE_FLAG_KERNING)); } + { import core.stdc.stdio : printf; printf("kern for %u:%u: %d %d (size=%g)\n", glyph1, glyph2, ftKerning.x, ftKerning.y, cast(double)size); } + } + } + version(none) { + FT_Vector kk; + if (FT_Get_Kerning(font.font, glyph1, glyph2, FT_KERNING_UNSCALED, &kk)) assert(0, "wtf?!"); + auto kadvfrac = FT_MulFix(kk.x, font.font.size.metrics.x_scale); // 1/64 of pixel + //return cast(int)((kadvfrac/*+(kadvfrac < 0 ? -32 : 32)*/)>>6); + //assert(ftKerning.x == kadvfrac); + if (ftKerning.x || kadvfrac) { + { import core.stdc.stdio : printf; printf("kern for %u:%u: %d %d (%d) (size=%g)\n", glyph1, glyph2, ftKerning.x, cast(int)kadvfrac, cast(int)(kadvfrac+(kadvfrac < 0 ? -31 : 32)>>6), cast(double)size); } + } + //return cast(int)(kadvfrac+(kadvfrac < 0 ? -31 : 32)>>6); // round up and convert to integer + return kadvfrac/64.0f; + } + //return cast(int)(ftKerning.x+(ftKerning.x < 0 ? -31 : 32)>>6); // round up and convert to integer + return ftKerning.x/64.0f; + } +} + +extern(C) nothrow @trusted @nogc { + static struct OutlinerData { + @disable this (this); + NVGContext vg; + NVGGlyphOutline* ol; + FT_BBox outlineBBox; + nothrow @trusted @nogc: + T transx(T) (T v) const pure { pragma(inline, true); return v; } + T transy(T) (T v) const pure { pragma(inline, true); return -v; } + void putBytes (const(void)[] b) { + assert(b.length <= 512); + if (b.length == 0) return; + if (ol.used+cast(uint)b.length > ol.size) { + import core.stdc.stdlib : realloc; + uint newsz = (ol.size == 0 ? 2048 : ol.size < 32768 ? ol.size*2 : ol.size+8192); + assert(ol.used+cast(uint)b.length <= newsz); + auto nd = cast(ubyte*)realloc(ol.data, newsz); + if (nd is null) assert(0, "FONS: out of memory"); + ol.size = newsz; + ol.data = nd; + } + import core.stdc.string : memcpy; + memcpy(ol.data+ol.used, b.ptr, b.length); + ol.used += cast(uint)b.length; + } + void newCommand (ubyte cmd) { pragma(inline, true); ++ol.ccount; putBytes((&cmd)[0..1]); } + void putArg (float f) { putBytes((&f)[0..1]); } + } + + int fons__nvg__moveto_cb (const(FT_Vector)* to, void* user) { + auto odata = cast(OutlinerData*)user; + if (odata.vg !is null) odata.vg.moveTo(odata.transx(to.x), odata.transy(to.y)); + if (odata.ol !is null) { + odata.newCommand(odata.ol.Command.Kind.MoveTo); + odata.putArg(odata.transx(to.x)); + odata.putArg(odata.transy(to.y)); + } + return 0; + } + + int fons__nvg__lineto_cb (const(FT_Vector)* to, void* user) { + auto odata = cast(OutlinerData*)user; + if (odata.vg !is null) odata.vg.lineTo(odata.transx(to.x), odata.transy(to.y)); + if (odata.ol !is null) { + odata.newCommand(odata.ol.Command.Kind.LineTo); + odata.putArg(odata.transx(to.x)); + odata.putArg(odata.transy(to.y)); + } + return 0; + } + + int fons__nvg__quadto_cb (const(FT_Vector)* c1, const(FT_Vector)* to, void* user) { + auto odata = cast(OutlinerData*)user; + if (odata.vg !is null) odata.vg.quadTo(odata.transx(c1.x), odata.transy(c1.y), odata.transx(to.x), odata.transy(to.y)); + if (odata.ol !is null) { + odata.newCommand(odata.ol.Command.Kind.QuadTo); + odata.putArg(odata.transx(c1.x)); + odata.putArg(odata.transy(c1.y)); + odata.putArg(odata.transx(to.x)); + odata.putArg(odata.transy(to.y)); + } + return 0; + } + + int fons__nvg__cubicto_cb (const(FT_Vector)* c1, const(FT_Vector)* c2, const(FT_Vector)* to, void* user) { + auto odata = cast(OutlinerData*)user; + if (odata.vg !is null) odata.vg.bezierTo(odata.transx(c1.x), odata.transy(c1.y), odata.transx(c2.x), odata.transy(c2.y), odata.transx(to.x), odata.transy(to.y)); + if (odata.ol !is null) { + odata.newCommand(odata.ol.Command.Kind.BezierTo); + odata.putArg(odata.transx(c1.x)); + odata.putArg(odata.transy(c1.y)); + odata.putArg(odata.transx(c2.x)); + odata.putArg(odata.transy(c2.y)); + odata.putArg(odata.transx(to.x)); + odata.putArg(odata.transy(to.y)); + } + return 0; + } +} + +bool fons__nvg__toPath (NVGContext vg, FONSttFontImpl* font, uint glyphidx, float[] bounds=null) nothrow @trusted @nogc { + if (bounds.length > 4) bounds = bounds.ptr[0..4]; + + FT_Outline_Funcs funcs; + funcs.move_to = &fons__nvg__moveto_cb; + funcs.line_to = &fons__nvg__lineto_cb; + funcs.conic_to = &fons__nvg__quadto_cb; + funcs.cubic_to = &fons__nvg__cubicto_cb; + + auto err = FT_Load_Glyph(font.font, glyphidx, FT_LOAD_NO_BITMAP|FT_LOAD_NO_SCALE); + if (err) { bounds[] = 0; return false; } + if (font.font.glyph.format != FT_GLYPH_FORMAT_OUTLINE) { bounds[] = 0; return false; } + + FT_Outline outline = font.font.glyph.outline; + + OutlinerData odata; + odata.vg = vg; + FT_Outline_Get_CBox(&outline, &odata.outlineBBox); + + err = FT_Outline_Decompose(&outline, &funcs, &odata); + if (err) { bounds[] = 0; return false; } + if (bounds.length > 0) bounds.ptr[0] = odata.outlineBBox.xMin; + if (bounds.length > 1) bounds.ptr[1] = -odata.outlineBBox.yMax; + if (bounds.length > 2) bounds.ptr[2] = odata.outlineBBox.xMax; + if (bounds.length > 3) bounds.ptr[3] = -odata.outlineBBox.yMin; + return true; +} + +bool fons__nvg__toOutline (FONSttFontImpl* font, uint glyphidx, NVGGlyphOutline* ol) nothrow @trusted @nogc { + FT_Outline_Funcs funcs; + funcs.move_to = &fons__nvg__moveto_cb; + funcs.line_to = &fons__nvg__lineto_cb; + funcs.conic_to = &fons__nvg__quadto_cb; + funcs.cubic_to = &fons__nvg__cubicto_cb; + + auto err = FT_Load_Glyph(font.font, glyphidx, FT_LOAD_NO_BITMAP|FT_LOAD_NO_SCALE); + if (err) return false; + if (font.font.glyph.format != FT_GLYPH_FORMAT_OUTLINE) return false; + + FT_Outline outline = font.font.glyph.outline; + + OutlinerData odata; + odata.ol = ol; + FT_Outline_Get_CBox(&outline, &odata.outlineBBox); + + err = FT_Outline_Decompose(&outline, &funcs, &odata); + if (err) return false; + ol.bounds.ptr[0] = odata.outlineBBox.xMin; + ol.bounds.ptr[1] = -odata.outlineBBox.yMax; + ol.bounds.ptr[2] = odata.outlineBBox.xMax; + ol.bounds.ptr[3] = -odata.outlineBBox.yMin; + return true; +} + +bool fons__nvg__bounds (FONSttFontImpl* font, uint glyphidx, float[] bounds) nothrow @trusted @nogc { + if (bounds.length > 4) bounds = bounds.ptr[0..4]; + + auto err = FT_Load_Glyph(font.font, glyphidx, FT_LOAD_NO_BITMAP|FT_LOAD_NO_SCALE); + if (err) return false; + if (font.font.glyph.format != FT_GLYPH_FORMAT_OUTLINE) { bounds[] = 0; return false; } + + FT_Outline outline = font.font.glyph.outline; + FT_BBox outlineBBox; + FT_Outline_Get_CBox(&outline, &outlineBBox); + if (bounds.length > 0) bounds.ptr[0] = outlineBBox.xMin; + if (bounds.length > 1) bounds.ptr[1] = -outlineBBox.yMax; + if (bounds.length > 2) bounds.ptr[2] = outlineBBox.xMax; + if (bounds.length > 3) bounds.ptr[3] = -outlineBBox.yMin; + return true; +} + + +} else { +// ////////////////////////////////////////////////////////////////////////// // +// sorry +import std.traits : isFunctionPointer, isDelegate; +private auto assumeNoThrowNoGC(T) (scope T t) if (isFunctionPointer!T || isDelegate!T) { + import std.traits; + enum attrs = functionAttributes!T|FunctionAttribute.nogc|FunctionAttribute.nothrow_; + return cast(SetFunctionAttributes!(T, functionLinkage!T, attrs)) t; +} + +private auto forceNoThrowNoGC(T) (scope T t) if (isFunctionPointer!T || isDelegate!T) { + try { + return assumeNoThrowNoGC(t)(); + } catch (Exception e) { + assert(0, "OOPS!"); + } +} + +struct FONSttFontImpl { + stbtt_fontinfo font; + bool mono; // no aa? +} + +int fons__tt_init (FONScontext* context) nothrow @trusted @nogc { + return 1; +} + +void fons__tt_setMono (FONScontext* context, FONSttFontImpl* font, bool v) nothrow @trusted @nogc { + font.mono = v; +} + +bool fons__tt_getMono (FONScontext* context, FONSttFontImpl* font) nothrow @trusted @nogc { + return font.mono; +} + +int fons__tt_loadFont (FONScontext* context, FONSttFontImpl* font, ubyte* data, int dataSize) nothrow @trusted @nogc { + int stbError; + font.font.userdata = context; + forceNoThrowNoGC({ stbError = stbtt_InitFont(&font.font, data, 0); }); + return stbError; +} + +void fons__tt_getFontVMetrics (FONSttFontImpl* font, int* ascent, int* descent, int* lineGap) nothrow @trusted @nogc { + forceNoThrowNoGC({ stbtt_GetFontVMetrics(&font.font, ascent, descent, lineGap); }); +} + +float fons__tt_getPixelHeightScale (FONSttFontImpl* font, float size) nothrow @trusted @nogc { + float res = void; + forceNoThrowNoGC({ res = stbtt_ScaleForPixelHeight(&font.font, size); }); + return res; +} + +int fons__tt_getGlyphIndex (FONSttFontImpl* font, int codepoint) nothrow @trusted @nogc { + int res; + forceNoThrowNoGC({ res = stbtt_FindGlyphIndex(&font.font, codepoint); }); + return res; +} + +int fons__tt_buildGlyphBitmap (FONSttFontImpl* font, int glyph, float size, float scale, int* advance, int* lsb, int* x0, int* y0, int* x1, int* y1) nothrow @trusted @nogc { + forceNoThrowNoGC({ stbtt_GetGlyphHMetrics(&font.font, glyph, advance, lsb); }); + forceNoThrowNoGC({ stbtt_GetGlyphBitmapBox(&font.font, glyph, scale, scale, x0, y0, x1, y1); }); + return 1; +} + +void fons__tt_renderGlyphBitmap (FONSttFontImpl* font, ubyte* output, int outWidth, int outHeight, int outStride, float scaleX, float scaleY, int glyph) nothrow @trusted @nogc { + forceNoThrowNoGC({ stbtt_MakeGlyphBitmap(&font.font, output, outWidth, outHeight, outStride, scaleX, scaleY, glyph); }); +} + +float fons__tt_getGlyphKernAdvance (FONSttFontImpl* font, float size, int glyph1, int glyph2) nothrow @trusted @nogc { + float res = void; + forceNoThrowNoGC({ res = stbtt_GetGlyphKernAdvance(&font.font, glyph1, glyph2); }); + return res; +} + +} // version + + +// ////////////////////////////////////////////////////////////////////////// // +private: +enum FONS_SCRATCH_BUF_SIZE = 64000; +enum FONS_HASH_LUT_SIZE = 256; +enum FONS_INIT_FONTS = 4; +enum FONS_INIT_GLYPHS = 256; +enum FONS_INIT_ATLAS_NODES = 256; +enum FONS_VERTEX_COUNT = 1024; +enum FONS_MAX_STATES = 20; +enum FONS_MAX_FALLBACKS = 20; + +uint fons__hashint() (uint a) pure nothrow @safe @nogc { + pragma(inline, true); + a += ~(a<<15); + a ^= (a>>10); + a += (a<<3); + a ^= (a>>6); + a += ~(a<<11); + a ^= (a>>16); + return a; +} + +uint fons__djbhash (const(void)[] s) pure nothrow @safe @nogc { + uint hash = 5381; + foreach (ubyte b; cast(const(ubyte)[])s) { + if (b >= 'A' && b <= 'Z') b += 32; // poor man's tolower + hash = ((hash<<5)+hash)+b; + } + return hash; +} + +private bool fons_strequci (const(char)[] s0, const(char)[] s1) nothrow @trusted @nogc { + if (s0.length != s1.length) return false; + const(char)* sp0 = s0.ptr; + const(char)* sp1 = s1.ptr; + foreach (immutable _; 0..s0.length) { + char c0 = *sp0++; + char c1 = *sp1++; + if (c0 != c1) { + if (c0 >= 'A' && c0 <= 'Z') c0 += 32; // poor man tolower + if (c1 >= 'A' && c1 <= 'Z') c1 += 32; // poor man tolower + if (c0 != c1) return false; + } + } + return true; +} + + +struct FONSglyph { + uint codepoint; + int index; + int next; + short size, blur; + short x0, y0, x1, y1; + short xadv, xoff, yoff; +} + +// refcounted +struct FONSfontData { + ubyte* data; + int dataSize; + bool freeData; + int rc; + + @disable this (this); // no copies +} + +// won't set rc to 1 +FONSfontData* fons__createFontData (ubyte* adata, int asize, bool afree) nothrow @trusted @nogc { + import core.stdc.stdlib : malloc; + assert(adata !is null); + assert(asize > 0); + auto res = cast(FONSfontData*)malloc(FONSfontData.sizeof); + if (res is null) assert(0, "FONS: out of memory"); + res.data = adata; + res.dataSize = asize; + res.freeData = afree; + res.rc = 0; + return res; +} + +void incref (FONSfontData* fd) pure nothrow @trusted @nogc { + pragma(inline, true); + if (fd !is null) ++fd.rc; +} + +void decref (ref FONSfontData* fd) nothrow @trusted @nogc { + if (fd !is null) { + if (--fd.rc == 0) { + import core.stdc.stdlib : free; + if (fd.freeData && fd.data !is null) { + free(fd.data); + fd.data = null; + } + free(fd); + fd = null; + } + } +} + +// as creating and destroying fonts is a rare operation, malloc some data +struct FONSfont { + FONSttFontImpl font; + char* name; // malloced, strz, always lowercase + uint namelen; + uint namehash; + char* path; // malloced, strz + FONSfontData* fdata; + float ascender; + float descender; + float lineh; + FONSglyph* glyphs; + int cglyphs; + int nglyphs; + int[FONS_HASH_LUT_SIZE] lut; + int[FONS_MAX_FALLBACKS] fallbacks; + int nfallbacks; + + // except glyphs + void freeMemory () nothrow @trusted @nogc { + import core.stdc.stdlib : free; + if (name !is null) { free(name); name = null; } + namelen = namehash = 0; + if (path !is null) { free(path); path = null; } + fdata.decref(); + } + + // this also calcs name hash + void setName (const(char)[] aname) nothrow @trusted @nogc { + //{ import core.stdc.stdio; printf("setname: [%.*s]\n", cast(uint)aname.length, aname.ptr); } + import core.stdc.stdlib : realloc; + if (aname.length > int.max/32) assert(0, "FONS: invalid font name"); + namelen = cast(uint)aname.length; + name = cast(char*)realloc(name, namelen+1); + if (name is null) assert(0, "FONS: out of memory"); + if (aname.length) name[0..aname.length] = aname[]; + name[namelen] = 0; + // lowercase it + foreach (ref char ch; name[0..namelen]) if (ch >= 'A' && ch <= 'Z') ch += 32; // poor man's tolower + namehash = fons__djbhash(name[0..namelen]); + //{ import core.stdc.stdio; printf(" [%s] [%.*s] [0x%08x]\n", name, namelen, name, namehash); } + } + + void setPath (const(char)[] apath) nothrow @trusted @nogc { + import core.stdc.stdlib : realloc; + if (apath.length > int.max/32) assert(0, "FONS: invalid font path"); + path = cast(char*)realloc(path, apath.length+1); + if (path is null) assert(0, "FONS: out of memory"); + if (apath.length) path[0..apath.length] = apath[]; + path[apath.length] = 0; + } + + // this won't check hash + bool nameEqu (const(char)[] aname) nothrow @trusted @nogc { + //{ import core.stdc.stdio; printf("nameEqu: aname=[%.*s]; namelen=%u; aslen=%u\n", cast(uint)aname.length, aname.ptr, namelen, cast(uint)aname.length); } + if (namelen != aname.length) return false; + const(char)* ns = name; + // name part + foreach (char ch; aname) { + if (ch >= 'A' && ch <= 'Z') ch += 32; // poor man's tolower + if (ch != *ns++) return false; + } + // done (length was checked earlier) + return true; + } +} + +struct FONSstate { + int font; + NVGTextAlign talign; + float size; + uint color; + float blur; + float spacing; +} + +struct FONSatlasNode { + short x, y, width; +} + +struct FONSatlas { + int width, height; + FONSatlasNode* nodes; + int nnodes; + int cnodes; +} + +public struct FONScontext { + FONSparams params; + float itw, ith; + ubyte* texData; + int[4] dirtyRect; + FONSfont** fonts; // actually, a simple hash table; can't grow yet + int cfonts; // allocated + int nfonts; // used (so we can track hash table stats) + int* hashidx; // [hsize] items; holds indicies in [fonts] array + int hused, hsize;// used items and total items in [hashidx] + FONSatlas* atlas; + float[FONS_VERTEX_COUNT*2] verts; + float[FONS_VERTEX_COUNT*2] tcoords; + uint[FONS_VERTEX_COUNT] colors; + int nverts; + ubyte* scratch; + int nscratch; + FONSstate[FONS_MAX_STATES] states; + int nstates; + void function (void* uptr, int error, int val) nothrow @trusted @nogc handleError; + void* errorUptr; + + // simple linear probing; returns [FONS_INVALID] if not found + int findNameInHash (const(char)[] name) nothrow @trusted @nogc { + if (nfonts == 0) return FONS_INVALID; + auto nhash = fons__djbhash(name); + //{ import core.stdc.stdio; printf("findinhash: name=[%.*s]; nhash=0x%08x\n", cast(uint)name.length, name.ptr, nhash); } + auto res = nhash%hsize; + // hash will never be 100% full, so this loop is safe + for (;;) { + int idx = hashidx[res]; + if (idx == -1) break; + auto font = fonts[idx]; + if (font is null) assert(0, "FONS internal error"); + if (font.namehash == nhash && font.nameEqu(name)) return idx; + //{ import core.stdc.stdio; printf("findinhash chained: name=[%.*s]; nhash=0x%08x\n", cast(uint)name.length, name.ptr, nhash); } + res = (res+1)%hsize; + } + return FONS_INVALID; + } + + // should be called $(B before) freeing `fonts[fidx]` + private void removeIndexFromHash (int fidx) nothrow @trusted @nogc { + if (fidx < 0 || fidx >= nfonts) assert(0, "FONS internal error"); + if (fonts[fidx] is null) assert(0, "FONS internal error"); + if (hused != nfonts) assert(0, "FONS internal error"); + auto nhash = fonts[fidx].namehash; + auto res = nhash%hsize; + // hash will never be 100% full, so this loop is safe + for (;;) { + int idx = hashidx[res]; + if (idx == -1) assert(0, "FONS INTERNAL ERROR"); + if (idx == fidx) { + // i found her! copy rest here + int nidx = (res+1)%hsize; + for (;;) { + if ((hashidx[res] = hashidx[nidx]) == -1) break; // so it will copy `-1` too + res = nidx; + nidx = (nidx+1)%hsize; + } + return; + } + res = (res+1)%hsize; + } + } + + // add font with the given index to hash + // prerequisite: font should not exists in hash + private void addIndexToHash (int idx) nothrow @trusted @nogc { + if (idx < 0 || idx >= nfonts) assert(0, "FONS internal error"); + if (fonts[idx] is null) assert(0, "FONS internal error"); + import core.stdc.stdlib : realloc; + auto nhash = fonts[idx].namehash; + //{ import core.stdc.stdio; printf("addtohash: name=[%.*s]; nhash=0x%08x\n", cast(uint)name.length, name.ptr, nhash); } + // allocate new hash table if there was none + if (hsize == 0) { + enum InitSize = 256; + auto newlist = cast(int*)realloc(null, InitSize*hashidx[0].sizeof); + if (newlist is null) assert(0, "FONS: out of memory"); + newlist[0..InitSize] = -1; + hsize = InitSize; + hused = 0; + hashidx = newlist; + } + int res = cast(int)(nhash%hsize); + // need to rehash? we want our hash table 50% full at max + if (hashidx[res] != -1 && hused >= hsize/2) { + uint nsz = hsize*2; + if (nsz > 1024*1024) assert(0, "FONS: out of memory for fonts"); + auto newlist = cast(int*)realloc(fonts, nsz*hashidx[0].sizeof); + if (newlist is null) assert(0, "FONS: out of memory"); + newlist[0..nsz] = -1; + hused = 0; + // rehash + foreach (immutable fidx, FONSfont* ff; fonts[0..nfonts]) { + if (ff is null) continue; + // find slot for this font (guaranteed to have one) + uint newslot = ff.namehash%nsz; + while (newlist[newslot] != -1) newslot = (newslot+1)%nsz; + newlist[newslot] = cast(int)fidx; + ++hused; + } + hsize = nsz; + hashidx = newlist; + // we added everything, including [idx], so nothing more to do here + } else { + // find slot (guaranteed to have one) + while (hashidx[res] != -1) res = (res+1)%hsize; + // i found her! + hashidx[res] = idx; + ++hused; + } + } +} + +void* fons__tmpalloc (usize size, void* up) nothrow @trusted @nogc { + ubyte* ptr; + FONScontext* stash = cast(FONScontext*)up; + // 16-byte align the returned pointer + size = (size+0xf)&~0xf; + if (stash.nscratch+cast(int)size > FONS_SCRATCH_BUF_SIZE) { + if (stash.handleError) stash.handleError(stash.errorUptr, FONS_SCRATCH_FULL, stash.nscratch+cast(int)size); + return null; + } + ptr = stash.scratch+stash.nscratch; + stash.nscratch += cast(int)size; + return ptr; +} + +void fons__tmpfree (void* ptr, void* up) nothrow @trusted @nogc { + // empty +} + +// Copyright (c) 2008-2010 Bjoern Hoehrmann +// See http://bjoern.hoehrmann.de/utf-8/decoder/dfa/ for details. + +enum FONS_UTF8_ACCEPT = 0; +enum FONS_UTF8_REJECT = 12; + +static immutable ubyte[364] utf8d = [ + // The first part of the table maps bytes to character classes that + // to reduce the size of the transition table and create bitmasks. + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, + 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, + 8, 8, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, + 10, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 4, 3, 3, 11, 6, 6, 6, 5, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, + + // The second part is a transition table that maps a combination + // of a state of the automaton and a character class to a state. + 0, 12, 24, 36, 60, 96, 84, 12, 12, 12, 48, 72, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, + 12, 0, 12, 12, 12, 12, 12, 0, 12, 0, 12, 12, 12, 24, 12, 12, 12, 12, 12, 24, 12, 24, 12, 12, + 12, 12, 12, 12, 12, 12, 12, 24, 12, 12, 12, 12, 12, 24, 12, 12, 12, 12, 12, 12, 12, 24, 12, 12, + 12, 12, 12, 12, 12, 12, 12, 36, 12, 36, 12, 12, 12, 36, 12, 12, 12, 12, 12, 36, 12, 36, 12, 12, + 12, 36, 12, 12, 12, 12, 12, 12, 12, 12, 12, 12, +]; + +private enum DecUtfMixin(string state, string codep, string byte_) = +`{ + uint type_ = utf8d.ptr[`~byte_~`]; + `~codep~` = (`~state~` != FONS_UTF8_ACCEPT ? (`~byte_~`&0x3fu)|(`~codep~`<<6) : (0xff>>type_)&`~byte_~`); + if ((`~state~` = utf8d.ptr[256+`~state~`+type_]) == FONS_UTF8_REJECT) { + `~state~` = FONS_UTF8_ACCEPT; + `~codep~` = 0xFFFD; + } + }`; + +/* +uint fons__decutf8 (uint* state, uint* codep, uint byte_) { + pragma(inline, true); + uint type = utf8d.ptr[byte_]; + *codep = (*state != FONS_UTF8_ACCEPT ? (byte_&0x3fu)|(*codep<<6) : (0xff>>type)&byte_); + *state = utf8d.ptr[256 + *state+type]; + return *state; +} +*/ + +// Atlas based on Skyline Bin Packer by Jukka Jylänki +void fons__deleteAtlas (FONSatlas* atlas) nothrow @trusted @nogc { + if (atlas is null) return; + if (atlas.nodes !is null) free(atlas.nodes); + free(atlas); +} + +FONSatlas* fons__allocAtlas (int w, int h, int nnodes) nothrow @trusted @nogc { + FONSatlas* atlas = null; + + // Allocate memory for the font stash. + atlas = cast(FONSatlas*)malloc(FONSatlas.sizeof); + if (atlas is null) goto error; + memset(atlas, 0, FONSatlas.sizeof); + + atlas.width = w; + atlas.height = h; + + // Allocate space for skyline nodes + atlas.nodes = cast(FONSatlasNode*)malloc(FONSatlasNode.sizeof*nnodes); + if (atlas.nodes is null) goto error; + memset(atlas.nodes, 0, FONSatlasNode.sizeof*nnodes); + atlas.nnodes = 0; + atlas.cnodes = nnodes; + + // Init root node. + atlas.nodes[0].x = 0; + atlas.nodes[0].y = 0; + atlas.nodes[0].width = cast(short)w; + ++atlas.nnodes; + + return atlas; + +error: + if (atlas !is null) fons__deleteAtlas(atlas); + return null; +} + +bool fons__atlasInsertNode (FONSatlas* atlas, int idx, int x, int y, int w) nothrow @trusted @nogc { + // Insert node + if (atlas.nnodes+1 > atlas.cnodes) { + atlas.cnodes = (atlas.cnodes == 0 ? 8 : atlas.cnodes*2); + atlas.nodes = cast(FONSatlasNode*)realloc(atlas.nodes, FONSatlasNode.sizeof*atlas.cnodes); + if (atlas.nodes is null) return false; + } + for (int i = atlas.nnodes; i > idx; --i) atlas.nodes[i] = atlas.nodes[i-1]; + atlas.nodes[idx].x = cast(short)x; + atlas.nodes[idx].y = cast(short)y; + atlas.nodes[idx].width = cast(short)w; + ++atlas.nnodes; + return 1; +} + +void fons__atlasRemoveNode (FONSatlas* atlas, int idx) nothrow @trusted @nogc { + if (atlas.nnodes == 0) return; + for (int i = idx; i < atlas.nnodes-1; ++i) atlas.nodes[i] = atlas.nodes[i+1]; + --atlas.nnodes; +} + +void fons__atlasExpand (FONSatlas* atlas, int w, int h) nothrow @trusted @nogc { + // Insert node for empty space + if (w > atlas.width) fons__atlasInsertNode(atlas, atlas.nnodes, atlas.width, 0, w-atlas.width); + atlas.width = w; + atlas.height = h; +} + +void fons__atlasReset (FONSatlas* atlas, int w, int h) nothrow @trusted @nogc { + atlas.width = w; + atlas.height = h; + atlas.nnodes = 0; + // Init root node. + atlas.nodes[0].x = 0; + atlas.nodes[0].y = 0; + atlas.nodes[0].width = cast(short)w; + ++atlas.nnodes; +} + +bool fons__atlasAddSkylineLevel (FONSatlas* atlas, int idx, int x, int y, int w, int h) nothrow @trusted @nogc { + // Insert new node + if (!fons__atlasInsertNode(atlas, idx, x, y+h, w)) return false; + + // Delete skyline segments that fall under the shadow of the new segment + for (int i = idx+1; i < atlas.nnodes; ++i) { + if (atlas.nodes[i].x < atlas.nodes[i-1].x+atlas.nodes[i-1].width) { + int shrink = atlas.nodes[i-1].x+atlas.nodes[i-1].width-atlas.nodes[i].x; + atlas.nodes[i].x += cast(short)shrink; + atlas.nodes[i].width -= cast(short)shrink; + if (atlas.nodes[i].width <= 0) { + fons__atlasRemoveNode(atlas, i); + --i; + } else { + break; + } + } else { + break; + } + } + + // Merge same height skyline segments that are next to each other + for (int i = 0; i < atlas.nnodes-1; ++i) { + if (atlas.nodes[i].y == atlas.nodes[i+1].y) { + atlas.nodes[i].width += atlas.nodes[i+1].width; + fons__atlasRemoveNode(atlas, i+1); + --i; + } + } + + return true; +} + +int fons__atlasRectFits (FONSatlas* atlas, int i, int w, int h) nothrow @trusted @nogc { + // Checks if there is enough space at the location of skyline span 'i', + // and return the max height of all skyline spans under that at that location, + // (think tetris block being dropped at that position). Or -1 if no space found. + int x = atlas.nodes[i].x; + int y = atlas.nodes[i].y; + int spaceLeft; + if (x+w > atlas.width) return -1; + spaceLeft = w; + while (spaceLeft > 0) { + if (i == atlas.nnodes) return -1; + y = nvg__max(y, atlas.nodes[i].y); + if (y+h > atlas.height) return -1; + spaceLeft -= atlas.nodes[i].width; + ++i; + } + return y; +} + +bool fons__atlasAddRect (FONSatlas* atlas, int rw, int rh, int* rx, int* ry) nothrow @trusted @nogc { + int besth = atlas.height, bestw = atlas.width, besti = -1; + int bestx = -1, besty = -1; + + // Bottom left fit heuristic. + for (int i = 0; i < atlas.nnodes; ++i) { + int y = fons__atlasRectFits(atlas, i, rw, rh); + if (y != -1) { + if (y+rh < besth || (y+rh == besth && atlas.nodes[i].width < bestw)) { + besti = i; + bestw = atlas.nodes[i].width; + besth = y+rh; + bestx = atlas.nodes[i].x; + besty = y; + } + } + } + + if (besti == -1) return false; + + // Perform the actual packing. + if (!fons__atlasAddSkylineLevel(atlas, besti, bestx, besty, rw, rh)) return false; + + *rx = bestx; + *ry = besty; + + return true; +} + +void fons__addWhiteRect (FONScontext* stash, int w, int h) nothrow @trusted @nogc { + int gx, gy; + ubyte* dst; + + if (!fons__atlasAddRect(stash.atlas, w, h, &gx, &gy)) return; + + // Rasterize + dst = &stash.texData[gx+gy*stash.params.width]; + foreach (int y; 0..h) { + foreach (int x; 0..w) { + dst[x] = 0xff; + } + dst += stash.params.width; + } + + stash.dirtyRect.ptr[0] = nvg__min(stash.dirtyRect.ptr[0], gx); + stash.dirtyRect.ptr[1] = nvg__min(stash.dirtyRect.ptr[1], gy); + stash.dirtyRect.ptr[2] = nvg__max(stash.dirtyRect.ptr[2], gx+w); + stash.dirtyRect.ptr[3] = nvg__max(stash.dirtyRect.ptr[3], gy+h); +} + +public FONScontext* fonsCreateInternal (FONSparams* params) nothrow @trusted @nogc { + FONScontext* stash = null; + + // Allocate memory for the font stash. + stash = cast(FONScontext*)malloc(FONScontext.sizeof); + if (stash is null) goto error; + memset(stash, 0, FONScontext.sizeof); + + stash.params = *params; + + // Allocate scratch buffer. + stash.scratch = cast(ubyte*)malloc(FONS_SCRATCH_BUF_SIZE); + if (stash.scratch is null) goto error; + + // Initialize implementation library + if (!fons__tt_init(stash)) goto error; + + if (stash.params.renderCreate !is null) { + if (!stash.params.renderCreate(stash.params.userPtr, stash.params.width, stash.params.height)) goto error; + } + + stash.atlas = fons__allocAtlas(stash.params.width, stash.params.height, FONS_INIT_ATLAS_NODES); + if (stash.atlas is null) goto error; + + // Don't allocate space for fonts: hash manager will do that for us later. + //stash.cfonts = 0; + //stash.nfonts = 0; + + // Create texture for the cache. + stash.itw = 1.0f/stash.params.width; + stash.ith = 1.0f/stash.params.height; + stash.texData = cast(ubyte*)malloc(stash.params.width*stash.params.height); + if (stash.texData is null) goto error; + memset(stash.texData, 0, stash.params.width*stash.params.height); + + stash.dirtyRect.ptr[0] = stash.params.width; + stash.dirtyRect.ptr[1] = stash.params.height; + stash.dirtyRect.ptr[2] = 0; + stash.dirtyRect.ptr[3] = 0; + + // Add white rect at 0, 0 for debug drawing. + fons__addWhiteRect(stash, 2, 2); + + fonsPushState(stash); + fonsClearState(stash); + + return stash; + +error: + fonsDeleteInternal(stash); + return null; +} + +FONSstate* fons__getState (FONScontext* stash) nothrow @trusted @nogc { + pragma(inline, true); + return &stash.states[stash.nstates-1]; +} + +bool fonsAddFallbackFont (FONScontext* stash, int base, int fallback) nothrow @trusted @nogc { + FONSfont* baseFont = stash.fonts[base]; + if (baseFont !is null && baseFont.nfallbacks < FONS_MAX_FALLBACKS) { + baseFont.fallbacks.ptr[baseFont.nfallbacks++] = fallback; + return true; + } + return false; +} + +public void fonsSetSize (FONScontext* stash, float size) nothrow @trusted @nogc { + pragma(inline, true); + fons__getState(stash).size = size; +} + +public void fonsSetColor (FONScontext* stash, uint color) nothrow @trusted @nogc { + pragma(inline, true); + fons__getState(stash).color = color; +} + +public void fonsSetSpacing (FONScontext* stash, float spacing) nothrow @trusted @nogc { + pragma(inline, true); + fons__getState(stash).spacing = spacing; +} + +public void fonsSetBlur (FONScontext* stash, float blur) nothrow @trusted @nogc { + pragma(inline, true); + version(nanovg_kill_font_blur) blur = 0; + fons__getState(stash).blur = blur; +} + +public void fonsSetAlign (FONScontext* stash, NVGTextAlign talign) nothrow @trusted @nogc { + pragma(inline, true); + fons__getState(stash).talign = talign; +} + +public void fonsSetFont (FONScontext* stash, int font) nothrow @trusted @nogc { + pragma(inline, true); + fons__getState(stash).font = font; +} + +// get AA for current font or for the specified font +public bool fonsGetFontAA (FONScontext* stash, int font=-1) nothrow @trusted @nogc { + FONSstate* state = fons__getState(stash); + if (font < 0) font = state.font; + if (font < 0 || font >= stash.nfonts) return false; + FONSfont* f = stash.fonts[font]; + return (f !is null ? !f.font.mono : false); +} + +public void fonsPushState (FONScontext* stash) nothrow @trusted @nogc { + if (stash.nstates >= FONS_MAX_STATES) { + if (stash.handleError) stash.handleError(stash.errorUptr, FONS_STATES_OVERFLOW, 0); + return; + } + if (stash.nstates > 0) memcpy(&stash.states[stash.nstates], &stash.states[stash.nstates-1], FONSstate.sizeof); + ++stash.nstates; +} + +public void fonsPopState (FONScontext* stash) nothrow @trusted @nogc { + if (stash.nstates <= 1) { + if (stash.handleError) stash.handleError(stash.errorUptr, FONS_STATES_UNDERFLOW, 0); + return; + } + --stash.nstates; +} + +public void fonsClearState (FONScontext* stash) nothrow @trusted @nogc { + FONSstate* state = fons__getState(stash); + state.size = 12.0f; + state.color = 0xffffffff; + state.font = 0; + state.blur = 0; + state.spacing = 0; + state.talign.reset; //FONS_ALIGN_LEFT|FONS_ALIGN_BASELINE; +} + +void fons__freeFont (FONSfont* font) nothrow @trusted @nogc { + if (font is null) return; + if (font.glyphs) free(font.glyphs); + font.freeMemory(); + free(font); +} + +// returns fid, not hash slot +int fons__allocFontAt (FONScontext* stash, int atidx) nothrow @trusted @nogc { + if (atidx >= 0 && atidx >= stash.nfonts) assert(0, "internal NanoVega fontstash error"); + + if (atidx < 0) { + if (stash.nfonts >= stash.cfonts) { + import core.stdc.stdlib : realloc; + import core.stdc.string : memset; + assert(stash.nfonts == stash.cfonts); + int newsz = stash.cfonts+64; + if (newsz > 65535) assert(0, "FONS: too many fonts"); + auto newlist = cast(FONSfont**)realloc(stash.fonts, newsz*(FONSfont*).sizeof); + if (newlist is null) assert(0, "FONS: out of memory"); + memset(newlist+stash.cfonts, 0, (newsz-stash.cfonts)*(FONSfont*).sizeof); + stash.fonts = newlist; + stash.cfonts = newsz; + } + assert(stash.nfonts < stash.cfonts); + } + + FONSfont* font = cast(FONSfont*)malloc(FONSfont.sizeof); + if (font is null) assert(0, "FONS: out of memory"); + memset(font, 0, FONSfont.sizeof); + + font.glyphs = cast(FONSglyph*)malloc(FONSglyph.sizeof*FONS_INIT_GLYPHS); + if (font.glyphs is null) assert(0, "FONS: out of memory"); + font.cglyphs = FONS_INIT_GLYPHS; + font.nglyphs = 0; + + if (atidx < 0) { + stash.fonts[stash.nfonts] = font; + return stash.nfonts++; + } else { + stash.fonts[atidx] = font; + return atidx; + } +} + +private enum NoAlias = ":noaa"; + +// defAA: antialias flag for fonts without ":noaa" +public int fonsAddFont (FONScontext* stash, const(char)[] name, const(char)[] path, bool defAA) nothrow @trusted { + if (path.length == 0 || name.length == 0 || fons_strequci(name, NoAlias)) return FONS_INVALID; + if (path.length > 32768) return FONS_INVALID; // arbitrary limit + + // if font path ends with ":noaa", turn off antialiasing + if (path.length >= NoAlias.length && fons_strequci(path[$-NoAlias.length..$], NoAlias)) { + path = path[0..$-NoAlias.length]; + if (path.length == 0) return FONS_INVALID; + defAA = false; + } + + // if font name ends with ":noaa", turn off antialiasing + if (name.length > NoAlias.length && fons_strequci(name[$-NoAlias.length..$], NoAlias)) { + name = name[0..$-NoAlias.length]; + defAA = false; + } + + // find a font with the given name + int fidx = stash.findNameInHash(name); + //{ import core.stdc.stdio; printf("loading font '%.*s' [%s] (fidx=%d)...\n", cast(uint)path.length, path.ptr, fontnamebuf.ptr, fidx); } + + int loadFontFile (const(char)[] path) { + // check if existing font (if any) has the same path + if (fidx >= 0) { + import core.stdc.string : strlen; + auto plen = (stash.fonts[fidx].path !is null ? strlen(stash.fonts[fidx].path) : 0); + version(Posix) { + //{ import core.stdc.stdio; printf("+++ font [%.*s] was loaded from [%.*s]\n", cast(uint)blen, fontnamebuf.ptr, cast(uint)stash.fonts[fidx].path.length, stash.fonts[fidx].path.ptr); } + if (plen == path.length && stash.fonts[fidx].path[0..plen] == path) { + //{ import core.stdc.stdio; printf("*** font [%.*s] already loaded from [%.*s]\n", cast(uint)blen, fontnamebuf.ptr, cast(uint)plen, path.ptr); } + // i found her! + return fidx; + } + } else { + if (plen == path.length && fons_strequci(stash.fonts[fidx].path[0..plen], path)) { + // i found her! + return fidx; + } + } + } + version(Windows) { + // special shitdows check + foreach (immutable char ch; path) if (ch == ':') return FONS_INVALID; + } + // either no such font, or different path + //{ import core.stdc.stdio; printf("trying font [%.*s] from file [%.*s]\n", cast(uint)blen, fontnamebuf.ptr, cast(uint)path.length, path.ptr); } + try { + import core.stdc.stdlib : free, malloc; + static if (NanoVegaHasIVVFS) { + auto fl = VFile(path); + auto dataSize = fl.size; + if (dataSize < 16 || dataSize > int.max/32) return FONS_INVALID; + ubyte* data = cast(ubyte*)malloc(cast(uint)dataSize); + if (data is null) assert(0, "out of memory in NanoVega fontstash"); + scope(failure) free(data); // oops + fl.rawReadExact(data[0..cast(uint)dataSize]); + fl.close(); + } else { + import core.stdc.stdio : FILE, fopen, fclose, fread, ftell, fseek; + import std.internal.cstring : tempCString; + auto fl = fopen(path.tempCString, "rb"); + if (fl is null) return FONS_INVALID; + scope(exit) fclose(fl); + if (fseek(fl, 0, 2/*SEEK_END*/) != 0) return FONS_INVALID; + auto dataSize = ftell(fl); + if (fseek(fl, 0, 0/*SEEK_SET*/) != 0) return FONS_INVALID; + if (dataSize < 16 || dataSize > int.max/32) return FONS_INVALID; + ubyte* data = cast(ubyte*)malloc(cast(uint)dataSize); + if (data is null) assert(0, "out of memory in NanoVega fontstash"); + scope(failure) free(data); // oops + ubyte* dptr = data; + auto left = cast(uint)dataSize; + while (left > 0) { + auto rd = fread(dptr, 1, left, fl); + if (rd == 0) { free(data); return FONS_INVALID; } // unexpected EOF or reading error, it doesn't matter + dptr += rd; + left -= rd; + } + } + scope(failure) free(data); // oops + // create font data + FONSfontData* fdata = fons__createFontData(data, cast(int)dataSize, true); // free data + fdata.incref(); + auto xres = fonsAddFontWithData(stash, name, fdata, defAA); + if (xres == FONS_INVALID) { + fdata.decref(); // this will free [data] and [fdata] + } else { + // remember path + stash.fonts[xres].setPath(path); + } + return xres; + } catch (Exception e) { + // oops; sorry + } + return FONS_INVALID; + } + + // first try direct path + auto res = loadFontFile(path); + // if loading failed, try fontconfig (if fontconfig is available) + static if (NanoVegaHasFontConfig) { + if (res == FONS_INVALID && fontconfigAvailable) { + import std.internal.cstring : tempCString; + FcPattern* pat = FcNameParse(path.tempCString); + if (pat !is null) { + scope(exit) FcPatternDestroy(pat); + if (FcConfigSubstitute(null, pat, FcMatchPattern)) { + FcDefaultSubstitute(pat); + // find the font + FcResult result; + FcPattern* font = FcFontMatch(null, pat, &result); + if (font !is null) { + scope(exit) FcPatternDestroy(font); + char* file = null; + if (FcPatternGetString(font, FC_FILE, 0, &file) == FcResultMatch) { + if (file !is null && file[0]) { + import core.stdc.string : strlen; + res = loadFontFile(file[0..strlen(file)]); + } + } + } + } + } + } + } + return res; +} + +// This will not free data on error! +public int fonsAddFontMem (FONScontext* stash, const(char)[] name, ubyte* data, int dataSize, bool freeData, bool defAA) nothrow @trusted @nogc { + FONSfontData* fdata = fons__createFontData(data, dataSize, freeData); + fdata.incref(); + auto res = fonsAddFontWithData(stash, name, fdata, defAA); + if (res == FONS_INVALID) { + // we promised to not free data on error + fdata.freeData = false; + fdata.decref(); // this will free [fdata] + } + return res; +} + +// Add fonts from another font stash +// This is more effective than reloading fonts, 'cause font data will be shared. +public void fonsAddStashFonts (FONScontext* stash, FONScontext* source) nothrow @trusted @nogc { + if (stash is null || source is null) return; + foreach (FONSfont* font; source.fonts[0..source.nfonts]) { + if (font !is null) { + auto newidx = fonsAddCookedFont(stash, font); + FONSfont* newfont = stash.fonts[newidx]; + assert(newfont !is null); + assert(newfont.path is null); + // copy path + if (font.path !is null && font.path[0]) { + import core.stdc.stdlib : malloc; + import core.stdc.string : strcpy, strlen; + newfont.path = cast(char*)malloc(strlen(font.path)+1); + if (newfont.path is null) assert(0, "FONS: out of memory"); + strcpy(newfont.path, font.path); + } + } + } +} + +// used to add font from another fontstash +int fonsAddCookedFont (FONScontext* stash, FONSfont* font) nothrow @trusted @nogc { + if (font is null || font.fdata is null) return FONS_INVALID; + font.fdata.incref(); + auto res = fonsAddFontWithData(stash, font.name[0..font.namelen], font.fdata, !font.font.mono); + if (res == FONS_INVALID) font.fdata.decref(); // oops + return res; +} + +// fdata refcount must be already increased; it won't be changed +int fonsAddFontWithData (FONScontext* stash, const(char)[] name, FONSfontData* fdata, bool defAA) nothrow @trusted @nogc { + int i, ascent, descent, fh, lineGap; + + if (name.length == 0 || fons_strequci(name, NoAlias)) return FONS_INVALID; + if (name.length > 32767) return FONS_INVALID; + if (fdata is null) return FONS_INVALID; + + // find a font with the given name + int newidx; + FONSfont* oldfont = null; + int oldidx = stash.findNameInHash(name); + if (oldidx != FONS_INVALID) { + // replacement font + oldfont = stash.fonts[oldidx]; + newidx = oldidx; + } else { + // new font, allocate new bucket + newidx = -1; + } + + newidx = fons__allocFontAt(stash, newidx); + FONSfont* font = stash.fonts[newidx]; + font.setName(name); + font.lut.ptr[0..FONS_HASH_LUT_SIZE] = -1; // init hash lookup + font.fdata = fdata; // set the font data (don't change reference count) + fons__tt_setMono(stash, &font.font, !defAA); + + // init font + stash.nscratch = 0; + if (!fons__tt_loadFont(stash, &font.font, fdata.data, fdata.dataSize)) { + // we promised to not free data on error, so just clear the data store (it will be freed by the caller) + font.fdata = null; + fons__freeFont(font); + if (oldidx != FONS_INVALID) { + assert(oldidx == newidx); + stash.fonts[oldidx] = oldfont; + } else { + assert(newidx == stash.nfonts-1); + stash.fonts[newidx] = null; + --stash.nfonts; + } + return FONS_INVALID; + } else { + // free old font data, if any + if (oldfont) fons__freeFont(oldfont); + } + + // add font to name hash + if (oldidx == FONS_INVALID) stash.addIndexToHash(newidx); + + // store normalized line height + // the real line height is got by multiplying the lineh by font size + fons__tt_getFontVMetrics(&font.font, &ascent, &descent, &lineGap); + fh = ascent-descent; + font.ascender = cast(float)ascent/cast(float)fh; + font.descender = cast(float)descent/cast(float)fh; + font.lineh = cast(float)(fh+lineGap)/cast(float)fh; + + //{ import core.stdc.stdio; printf("created font [%.*s] (idx=%d)...\n", cast(uint)name.length, name.ptr, idx); } + return newidx; +} + +// returns `null` on invalid index +// WARNING! copy name, as name buffer can be invalidated by next fontstash API call! +public const(char)[] fonsGetNameByIndex (FONScontext* stash, int idx) nothrow @trusted @nogc { + if (idx < 0 || idx >= stash.nfonts || stash.fonts[idx] is null) return null; + return stash.fonts[idx].name[0..stash.fonts[idx].namelen]; +} + +// allowSubstitutes: check AA variants if exact name wasn't found? +// return [FONS_INVALID] if no font was found +public int fonsGetFontByName (FONScontext* stash, const(char)[] name) nothrow @trusted @nogc { + //{ import core.stdc.stdio; printf("fonsGetFontByName: [%.*s]\n", cast(uint)name.length, name.ptr); } + // remove ":noaa" suffix + if (name.length >= NoAlias.length && fons_strequci(name[$-NoAlias.length..$], NoAlias)) { + name = name[0..$-NoAlias.length]; + } + if (name.length == 0) return FONS_INVALID; + return stash.findNameInHash(name); +} + +FONSglyph* fons__allocGlyph (FONSfont* font) nothrow @trusted @nogc { + if (font.nglyphs+1 > font.cglyphs) { + font.cglyphs = (font.cglyphs == 0 ? 8 : font.cglyphs*2); + font.glyphs = cast(FONSglyph*)realloc(font.glyphs, FONSglyph.sizeof*font.cglyphs); + if (font.glyphs is null) return null; + } + ++font.nglyphs; + return &font.glyphs[font.nglyphs-1]; +} + +// 0: ooops +int fons__findGlyphForCP (FONScontext* stash, FONSfont *font, dchar dch, FONSfont** renderfont) nothrow @trusted @nogc { + if (renderfont !is null) *renderfont = font; + if (stash is null) return 0; + if (font is null || font.fdata is null) return 0; + auto g = fons__tt_getGlyphIndex(&font.font, cast(uint)dch); + // try to find the glyph in fallback fonts + if (g == 0) { + foreach (immutable i; 0..font.nfallbacks) { + FONSfont* fallbackFont = stash.fonts[font.fallbacks.ptr[i]]; + if (fallbackFont !is null) { + int fallbackIndex = fons__tt_getGlyphIndex(&fallbackFont.font, cast(uint)dch); + if (fallbackIndex != 0) { + if (renderfont !is null) *renderfont = fallbackFont; + return g; + } + } + } + // no char, try to find replacement one + if (dch != 0xFFFD) { + g = fons__tt_getGlyphIndex(&font.font, 0xFFFD); + if (g == 0) { + foreach (immutable i; 0..font.nfallbacks) { + FONSfont* fallbackFont = stash.fonts[font.fallbacks.ptr[i]]; + if (fallbackFont !is null) { + int fallbackIndex = fons__tt_getGlyphIndex(&fallbackFont.font, 0xFFFD); + if (fallbackIndex != 0) { + if (renderfont !is null) *renderfont = fallbackFont; + return g; + } + } + } + } + } + } + return g; +} + +public bool fonsPathBounds (FONScontext* stash, dchar dch, float[] bounds) nothrow @trusted @nogc { + if (bounds.length > 4) bounds = bounds.ptr[0..4]; + static if (is(typeof(&fons__nvg__bounds))) { + if (stash is null) { bounds[] = 0; return false; } + FONSstate* state = fons__getState(stash); + if (state.font < 0 || state.font >= stash.nfonts) { bounds[] = 0; return false; } + FONSfont* font; + auto g = fons__findGlyphForCP(stash, stash.fonts[state.font], dch, &font); + if (g == 0) { bounds[] = 0; return false; } + assert(font !is null); + return fons__nvg__bounds(&font.font, g, bounds); + } else { + bounds[] = 0; + return false; + } +} + +public bool fonsToPath (FONScontext* stash, NVGContext vg, dchar dch, float[] bounds=null) nothrow @trusted @nogc { + if (bounds.length > 4) bounds = bounds.ptr[0..4]; + static if (is(typeof(&fons__nvg__toPath))) { + if (vg is null || stash is null) { bounds[] = 0; return false; } + FONSstate* state = fons__getState(stash); + if (state.font < 0 || state.font >= stash.nfonts) { bounds[] = 0; return false; } + FONSfont* font; + auto g = fons__findGlyphForCP(stash, stash.fonts[state.font], dch, &font); + if (g == 0) { bounds[] = 0; return false; } + assert(font !is null); + return fons__nvg__toPath(vg, &font.font, g, bounds); + } else { + bounds[] = 0; + return false; + } +} + +public bool fonsToOutline (FONScontext* stash, dchar dch, NVGGlyphOutline* ol) nothrow @trusted @nogc { + if (stash is null || ol is null) return false; + static if (is(typeof(&fons__nvg__toOutline))) { + FONSstate* state = fons__getState(stash); + if (state.font < 0 || state.font >= stash.nfonts) return false; + FONSfont* font; + auto g = fons__findGlyphForCP(stash, stash.fonts[state.font], dch, &font); + if (g == 0) return false; + assert(font !is null); + return fons__nvg__toOutline(&font.font, g, ol); + } else { + return false; + } +} + + +// Based on Exponential blur, Jani Huhtanen, 2006 + +enum APREC = 16; +enum ZPREC = 7; + +void fons__blurCols (ubyte* dst, int w, int h, int dstStride, int alpha) nothrow @trusted @nogc { + foreach (int y; 0..h) { + int z = 0; // force zero border + foreach (int x; 1..w) { + z += (alpha*((cast(int)(dst[x])<>APREC; + dst[x] = cast(ubyte)(z>>ZPREC); + } + dst[w-1] = 0; // force zero border + z = 0; + for (int x = w-2; x >= 0; --x) { + z += (alpha*((cast(int)(dst[x])<>APREC; + dst[x] = cast(ubyte)(z>>ZPREC); + } + dst[0] = 0; // force zero border + dst += dstStride; + } +} + +void fons__blurRows (ubyte* dst, int w, int h, int dstStride, int alpha) nothrow @trusted @nogc { + foreach (int x; 0..w) { + int z = 0; // force zero border + for (int y = dstStride; y < h*dstStride; y += dstStride) { + z += (alpha*((cast(int)(dst[y])<>APREC; + dst[y] = cast(ubyte)(z>>ZPREC); + } + dst[(h-1)*dstStride] = 0; // force zero border + z = 0; + for (int y = (h-2)*dstStride; y >= 0; y -= dstStride) { + z += (alpha*((cast(int)(dst[y])<>APREC; + dst[y] = cast(ubyte)(z>>ZPREC); + } + dst[0] = 0; // force zero border + ++dst; + } +} + + +void fons__blur (FONScontext* stash, ubyte* dst, int w, int h, int dstStride, int blur) nothrow @trusted @nogc { + import std.math : expf = exp; + int alpha; + float sigma; + if (blur < 1) return; + // Calculate the alpha such that 90% of the kernel is within the radius. (Kernel extends to infinity) + sigma = cast(float)blur*0.57735f; // 1/sqrt(3) + alpha = cast(int)((1< 20) iblur = 20; + pad = iblur+2; + + // Reset allocator. + stash.nscratch = 0; + + // Find code point and size. + h = fons__hashint(codepoint)&(FONS_HASH_LUT_SIZE-1); + i = font.lut.ptr[h]; + while (i != -1) { + //if (font.glyphs[i].codepoint == codepoint && font.glyphs[i].size == isize && font.glyphs[i].blur == iblur) return &font.glyphs[i]; + if (font.glyphs[i].codepoint == codepoint && font.glyphs[i].size == isize && font.glyphs[i].blur == iblur) { + glyph = &font.glyphs[i]; + // Negative coordinate indicates there is no bitmap data created. + if (bitmapOption == FONS_GLYPH_BITMAP_OPTIONAL || (glyph.x0 >= 0 && glyph.y0 >= 0)) return glyph; + // At this point, glyph exists but the bitmap data is not yet created. + break; + } + i = font.glyphs[i].next; + } + + // Create a new glyph or rasterize bitmap data for a cached glyph. + //scale = fons__tt_getPixelHeightScale(&font.font, size); + g = fons__findGlyphForCP(stash, font, cast(dchar)codepoint, &renderFont); + // It is possible that we did not find a fallback glyph. + // In that case the glyph index 'g' is 0, and we'll proceed below and cache empty glyph. + + scale = fons__tt_getPixelHeightScale(&renderFont.font, size); + fons__tt_buildGlyphBitmap(&renderFont.font, g, size, scale, &advance, &lsb, &x0, &y0, &x1, &y1); + gw = x1-x0+pad*2; + gh = y1-y0+pad*2; + + // Determines the spot to draw glyph in the atlas. + if (bitmapOption == FONS_GLYPH_BITMAP_REQUIRED) { + // Find free spot for the rect in the atlas. + added = fons__atlasAddRect(stash.atlas, gw, gh, &gx, &gy); + if (added == 0 && stash.handleError !is null) { + // Atlas is full, let the user to resize the atlas (or not), and try again. + stash.handleError(stash.errorUptr, FONS_ATLAS_FULL, 0); + added = fons__atlasAddRect(stash.atlas, gw, gh, &gx, &gy); + } + if (added == 0) return null; + } else { + // Negative coordinate indicates there is no bitmap data created. + gx = -1; + gy = -1; + } + + // Init glyph. + if (glyph is null) { + glyph = fons__allocGlyph(font); + glyph.codepoint = codepoint; + glyph.size = isize; + glyph.blur = iblur; + glyph.next = 0; + + // Insert char to hash lookup. + glyph.next = font.lut.ptr[h]; + font.lut.ptr[h] = font.nglyphs-1; + } + glyph.index = g; + glyph.x0 = cast(short)gx; + glyph.y0 = cast(short)gy; + glyph.x1 = cast(short)(glyph.x0+gw); + glyph.y1 = cast(short)(glyph.y0+gh); + glyph.xadv = cast(short)(scale*advance*10.0f); + glyph.xoff = cast(short)(x0-pad); + glyph.yoff = cast(short)(y0-pad); + + if (bitmapOption == FONS_GLYPH_BITMAP_OPTIONAL) return glyph; + + // Rasterize + dst = &stash.texData[(glyph.x0+pad)+(glyph.y0+pad)*stash.params.width]; + fons__tt_renderGlyphBitmap(&font.font, dst, gw-pad*2, gh-pad*2, stash.params.width, scale, scale, g); + + // Make sure there is one pixel empty border. + dst = &stash.texData[glyph.x0+glyph.y0*stash.params.width]; + for (y = 0; y < gh; y++) { + dst[y*stash.params.width] = 0; + dst[gw-1+y*stash.params.width] = 0; + } + for (x = 0; x < gw; x++) { + dst[x] = 0; + dst[x+(gh-1)*stash.params.width] = 0; + } + + // Debug code to color the glyph background + version(none) { + foreach (immutable yy; 0..gh) { + foreach (immutable xx; 0..gw) { + int a = cast(int)dst[xx+yy*stash.params.width]+42; + if (a > 255) a = 255; + dst[xx+yy*stash.params.width] = cast(ubyte)a; + } + } + } + + // Blur + if (iblur > 0) { + stash.nscratch = 0; + bdst = &stash.texData[glyph.x0+glyph.y0*stash.params.width]; + fons__blur(stash, bdst, gw, gh, stash.params.width, iblur); + } + + stash.dirtyRect.ptr[0] = nvg__min(stash.dirtyRect.ptr[0], glyph.x0); + stash.dirtyRect.ptr[1] = nvg__min(stash.dirtyRect.ptr[1], glyph.y0); + stash.dirtyRect.ptr[2] = nvg__max(stash.dirtyRect.ptr[2], glyph.x1); + stash.dirtyRect.ptr[3] = nvg__max(stash.dirtyRect.ptr[3], glyph.y1); + + return glyph; +} + +void fons__getQuad (FONScontext* stash, FONSfont* font, int prevGlyphIndex, FONSglyph* glyph, float size, float scale, float spacing, float* x, float* y, FONSquad* q) nothrow @trusted @nogc { + if (prevGlyphIndex >= 0) { + immutable float adv = fons__tt_getGlyphKernAdvance(&font.font, size, prevGlyphIndex, glyph.index)/**scale*/; //k8: do we really need scale here? + //if (adv != 0) { import core.stdc.stdio; printf("adv=%g (scale=%g; spacing=%g)\n", cast(double)adv, cast(double)scale, cast(double)spacing); } + *x += cast(int)(adv+spacing /*+0.5f*/); //k8: for me, it looks better this way (with non-aa fonts) + } + + // Each glyph has 2px border to allow good interpolation, + // one pixel to prevent leaking, and one to allow good interpolation for rendering. + // Inset the texture region by one pixel for correct interpolation. + immutable float xoff = cast(short)(glyph.xoff+1); + immutable float yoff = cast(short)(glyph.yoff+1); + immutable float x0 = cast(float)(glyph.x0+1); + immutable float y0 = cast(float)(glyph.y0+1); + immutable float x1 = cast(float)(glyph.x1-1); + immutable float y1 = cast(float)(glyph.y1-1); + + if (stash.params.flags&FONS_ZERO_TOPLEFT) { + immutable float rx = cast(float)cast(int)(*x+xoff); + immutable float ry = cast(float)cast(int)(*y+yoff); + + q.x0 = rx; + q.y0 = ry; + q.x1 = rx+x1-x0; + q.y1 = ry+y1-y0; + + q.s0 = x0*stash.itw; + q.t0 = y0*stash.ith; + q.s1 = x1*stash.itw; + q.t1 = y1*stash.ith; + } else { + immutable float rx = cast(float)cast(int)(*x+xoff); + immutable float ry = cast(float)cast(int)(*y-yoff); + + q.x0 = rx; + q.y0 = ry; + q.x1 = rx+x1-x0; + q.y1 = ry-y1+y0; + + q.s0 = x0*stash.itw; + q.t0 = y0*stash.ith; + q.s1 = x1*stash.itw; + q.t1 = y1*stash.ith; + } + + *x += cast(int)(glyph.xadv/10.0f+0.5f); +} + +void fons__flush (FONScontext* stash) nothrow @trusted @nogc { + // Flush texture + if (stash.dirtyRect.ptr[0] < stash.dirtyRect.ptr[2] && stash.dirtyRect.ptr[1] < stash.dirtyRect.ptr[3]) { + if (stash.params.renderUpdate !is null) stash.params.renderUpdate(stash.params.userPtr, stash.dirtyRect.ptr, stash.texData); + // Reset dirty rect + stash.dirtyRect.ptr[0] = stash.params.width; + stash.dirtyRect.ptr[1] = stash.params.height; + stash.dirtyRect.ptr[2] = 0; + stash.dirtyRect.ptr[3] = 0; + } + + // Flush triangles + if (stash.nverts > 0) { + if (stash.params.renderDraw !is null) stash.params.renderDraw(stash.params.userPtr, stash.verts.ptr, stash.tcoords.ptr, stash.colors.ptr, stash.nverts); + stash.nverts = 0; + } +} + +debug(nanovega) void fons__vertex (FONScontext* stash, float x, float y, float s, float t, uint c) nothrow @trusted @nogc { + stash.verts.ptr[stash.nverts*2+0] = x; + stash.verts.ptr[stash.nverts*2+1] = y; + stash.tcoords.ptr[stash.nverts*2+0] = s; + stash.tcoords.ptr[stash.nverts*2+1] = t; + stash.colors.ptr[stash.nverts] = c; + ++stash.nverts; +} + +float fons__getVertAlign (FONScontext* stash, FONSfont* font, NVGTextAlign talign, short isize) nothrow @trusted @nogc { + if (stash.params.flags&FONS_ZERO_TOPLEFT) { + final switch (talign.vertical) { + case NVGTextAlign.V.Top: return font.ascender*cast(float)isize/10.0f; + case NVGTextAlign.V.Middle: return (font.ascender+font.descender)/2.0f*cast(float)isize/10.0f; + case NVGTextAlign.V.Baseline: return 0.0f; + case NVGTextAlign.V.Bottom: return font.descender*cast(float)isize/10.0f; + } + } else { + final switch (talign.vertical) { + case NVGTextAlign.V.Top: return -font.ascender*cast(float)isize/10.0f; + case NVGTextAlign.V.Middle: return -(font.ascender+font.descender)/2.0f*cast(float)isize/10.0f; + case NVGTextAlign.V.Baseline: return 0.0f; + case NVGTextAlign.V.Bottom: return -font.descender*cast(float)isize/10.0f; + } + } + assert(0); +} + +public bool fonsTextIterInit(T) (FONScontext* stash, FONStextIter!T* iter, float x, float y, const(T)[] str, FONSglyphBitmap bitmapOption) if (isAnyCharType!T) { + if (stash is null || iter is null) return false; + + FONSstate* state = fons__getState(stash); + float width; + + memset(iter, 0, (*iter).sizeof); + + if (stash is null) return false; + if (state.font < 0 || state.font >= stash.nfonts) return false; + iter.font = stash.fonts[state.font]; + if (iter.font is null || iter.font.fdata is null) return false; + + iter.isize = cast(short)(state.size*10.0f); + iter.iblur = cast(short)state.blur; + iter.scale = fons__tt_getPixelHeightScale(&iter.font.font, cast(float)iter.isize/10.0f); + + // Align horizontally + if (state.talign.left) { + // empty + } else if (state.talign.right) { + width = fonsTextBounds(stash, x, y, str, null); + x -= width; + } else if (state.talign.center) { + width = fonsTextBounds(stash, x, y, str, null); + x -= width*0.5f; + } + // Align vertically. + y += fons__getVertAlign(stash, iter.font, state.talign, iter.isize); + + iter.x = iter.nextx = x; + iter.y = iter.nexty = y; + iter.spacing = state.spacing; + if (str.ptr is null) { + static if (is(T == char)) str = ""; + else static if (is(T == wchar)) str = ""w; + else static if (is(T == dchar)) str = ""d; + else static assert(0, "wtf?!"); + } + iter.s = str.ptr; + iter.n = str.ptr; + iter.e = str.ptr+str.length; + iter.codepoint = 0; + iter.prevGlyphIndex = -1; + iter.bitmapOption = bitmapOption; + + return true; +} + +public bool fonsTextIterGetDummyChar(FT) (FONScontext* stash, FT* iter, FONSquad* quad) nothrow @trusted @nogc if (is(FT : FONStextIter!CT, CT)) { + if (stash is null || iter is null) return false; + // Get glyph and quad + iter.x = iter.nextx; + iter.y = iter.nexty; + FONSglyph* glyph = fons__getGlyph(stash, iter.font, 0xFFFD, iter.isize, iter.iblur, iter.bitmapOption); + if (glyph !is null) { + fons__getQuad(stash, iter.font, iter.prevGlyphIndex, glyph, iter.isize/10.0f, iter.scale, iter.spacing, &iter.nextx, &iter.nexty, quad); + iter.prevGlyphIndex = glyph.index; + return true; + } else { + iter.prevGlyphIndex = -1; + return false; + } +} + +public bool fonsTextIterNext(FT) (FONScontext* stash, FT* iter, FONSquad* quad) nothrow @trusted @nogc if (is(FT : FONStextIter!CT, CT)) { + if (stash is null || iter is null) return false; + FONSglyph* glyph = null; + static if (is(FT.CharType == char)) { + const(char)* str = iter.n; + iter.s = iter.n; + if (str is iter.e) return false; + const(char)* e = iter.e; + for (; str !is e; ++str) { + /*if (fons__decutf8(&iter.utf8state, &iter.codepoint, *cast(const(ubyte)*)str)) continue;*/ + mixin(DecUtfMixin!("iter.utf8state", "iter.codepoint", "*cast(const(ubyte)*)str")); + if (iter.utf8state) continue; + ++str; // 'cause we'll break anyway + // get glyph and quad + iter.x = iter.nextx; + iter.y = iter.nexty; + glyph = fons__getGlyph(stash, iter.font, iter.codepoint, iter.isize, iter.iblur, iter.bitmapOption); + if (glyph !is null) { + fons__getQuad(stash, iter.font, iter.prevGlyphIndex, glyph, iter.isize/10.0f, iter.scale, iter.spacing, &iter.nextx, &iter.nexty, quad); + iter.prevGlyphIndex = glyph.index; + } else { + iter.prevGlyphIndex = -1; + } + break; + } + iter.n = str; + } else { + const(FT.CharType)* str = iter.n; + iter.s = iter.n; + if (str is iter.e) return false; + iter.codepoint = cast(uint)(*str++); + if (iter.codepoint > dchar.max) iter.codepoint = 0xFFFD; + // Get glyph and quad + iter.x = iter.nextx; + iter.y = iter.nexty; + glyph = fons__getGlyph(stash, iter.font, iter.codepoint, iter.isize, iter.iblur, iter.bitmapOption); + if (glyph !is null) { + fons__getQuad(stash, iter.font, iter.prevGlyphIndex, glyph, iter.isize/10.0f, iter.scale, iter.spacing, &iter.nextx, &iter.nexty, quad); + iter.prevGlyphIndex = glyph.index; + } else { + iter.prevGlyphIndex = -1; + } + iter.n = str; + } + return true; +} + +debug(nanovega) public void fonsDrawDebug (FONScontext* stash, float x, float y) nothrow @trusted @nogc { + int i; + int w = stash.params.width; + int h = stash.params.height; + float u = (w == 0 ? 0 : 1.0f/w); + float v = (h == 0 ? 0 : 1.0f/h); + + if (stash.nverts+6+6 > FONS_VERTEX_COUNT) fons__flush(stash); + + // Draw background + fons__vertex(stash, x+0, y+0, u, v, 0x0fffffff); + fons__vertex(stash, x+w, y+h, u, v, 0x0fffffff); + fons__vertex(stash, x+w, y+0, u, v, 0x0fffffff); + + fons__vertex(stash, x+0, y+0, u, v, 0x0fffffff); + fons__vertex(stash, x+0, y+h, u, v, 0x0fffffff); + fons__vertex(stash, x+w, y+h, u, v, 0x0fffffff); + + // Draw texture + fons__vertex(stash, x+0, y+0, 0, 0, 0xffffffff); + fons__vertex(stash, x+w, y+h, 1, 1, 0xffffffff); + fons__vertex(stash, x+w, y+0, 1, 0, 0xffffffff); + + fons__vertex(stash, x+0, y+0, 0, 0, 0xffffffff); + fons__vertex(stash, x+0, y+h, 0, 1, 0xffffffff); + fons__vertex(stash, x+w, y+h, 1, 1, 0xffffffff); + + // Drawbug draw atlas + for (i = 0; i < stash.atlas.nnodes; i++) { + FONSatlasNode* n = &stash.atlas.nodes[i]; + + if (stash.nverts+6 > FONS_VERTEX_COUNT) + fons__flush(stash); + + fons__vertex(stash, x+n.x+0, y+n.y+0, u, v, 0xc00000ff); + fons__vertex(stash, x+n.x+n.width, y+n.y+1, u, v, 0xc00000ff); + fons__vertex(stash, x+n.x+n.width, y+n.y+0, u, v, 0xc00000ff); + + fons__vertex(stash, x+n.x+0, y+n.y+0, u, v, 0xc00000ff); + fons__vertex(stash, x+n.x+0, y+n.y+1, u, v, 0xc00000ff); + fons__vertex(stash, x+n.x+n.width, y+n.y+1, u, v, 0xc00000ff); + } + + fons__flush(stash); +} + +public struct FonsTextBoundsIterator { +private: + FONScontext* stash; + FONSstate* state; + uint codepoint; + uint utf8state = 0; + FONSquad q; + FONSglyph* glyph = null; + int prevGlyphIndex = -1; + short isize, iblur; + float scale; + FONSfont* font; + float startx, x, y; + float minx, miny, maxx, maxy; + +public: + this (FONScontext* astash, float ax, float ay) nothrow @trusted @nogc { reset(astash, ax, ay); } + + void reset (FONScontext* astash, float ax, float ay) nothrow @trusted @nogc { + this = this.init; + if (astash is null) return; + stash = astash; + state = fons__getState(stash); + if (state is null) { stash = null; return; } // alas + + x = ax; + y = ay; + + isize = cast(short)(state.size*10.0f); + iblur = cast(short)state.blur; + + if (state.font < 0 || state.font >= stash.nfonts) { stash = null; return; } + font = stash.fonts[state.font]; + if (font is null || font.fdata is null) { stash = null; return; } + + scale = fons__tt_getPixelHeightScale(&font.font, cast(float)isize/10.0f); + + // align vertically + y += fons__getVertAlign(stash, font, state.talign, isize); + + minx = maxx = x; + miny = maxy = y; + startx = x; + //assert(prevGlyphIndex == -1); + } + +public: + @property bool valid () const pure nothrow @safe @nogc { pragma(inline, true); return (state !is null); } + + void put(T) (const(T)[] str...) nothrow @trusted @nogc if (isAnyCharType!T) { + enum DoCodePointMixin = q{ + 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; + } + }; + + if (state is null) return; // alas + static if (is(T == char)) { + foreach (char ch; str) { + mixin(DecUtfMixin!("utf8state", "codepoint", "cast(ubyte)ch")); + if (utf8state) continue; // full char is not collected yet + mixin(DoCodePointMixin); + } + } else { + if (str.length == 0) return; + if (utf8state) { + utf8state = 0; + codepoint = 0xFFFD; + mixin(DoCodePointMixin); + } + foreach (T dch; str) { + static if (is(T == dchar)) { + if (dch > dchar.max) dch = 0xFFFD; + } + codepoint = cast(uint)dch; + mixin(DoCodePointMixin); + } + } + } + + // return current advance + @property float advance () const pure nothrow @safe @nogc { pragma(inline, true); return (state !is null ? x-startx : 0); } + + void getBounds (ref float[4] bounds) const pure nothrow @safe @nogc { + if (state is null) { bounds[] = 0; return; } + float lminx = minx, lmaxx = maxx; + // align horizontally + if (state.talign.left) { + // empty + } else if (state.talign.right) { + float ca = advance; + lminx -= ca; + lmaxx -= ca; + } else if (state.talign.center) { + float ca = advance*0.5f; + lminx -= ca; + lmaxx -= ca; + } + bounds[0] = lminx; + bounds[1] = miny; + bounds[2] = lmaxx; + bounds[3] = maxy; + } + + // Returns current horizontal text bounds. + void getHBounds (out float xmin, out float xmax) nothrow @trusted @nogc { + if (state !is null) { + float lminx = minx, lmaxx = maxx; + // align horizontally + if (state.talign.left) { + // empty + } else if (state.talign.right) { + float ca = advance; + lminx -= ca; + lmaxx -= ca; + } else if (state.talign.center) { + float ca = advance*0.5f; + lminx -= ca; + lmaxx -= ca; + } + xmin = lminx; + xmax = lmaxx; + } + } + + // Returns current vertical text bounds. + void getVBounds (out float ymin, out float ymax) nothrow @trusted @nogc { + if (state !is null) { + ymin = miny; + ymax = maxy; + } + } +} + +public float fonsTextBounds(T) (FONScontext* stash, float x, float y, const(T)[] str, float[] bounds) nothrow @trusted @nogc +if (isAnyCharType!T) +{ + FONSstate* state = fons__getState(stash); + uint codepoint; + uint utf8state = 0; + FONSquad q; + FONSglyph* glyph = null; + int prevGlyphIndex = -1; + short isize = cast(short)(state.size*10.0f); + short iblur = cast(short)state.blur; + float scale; + FONSfont* font; + float startx, advance; + float minx, miny, maxx, maxy; + + if (stash is null) return 0; + if (state.font < 0 || state.font >= stash.nfonts) return 0; + font = stash.fonts[state.font]; + if (font is null || font.fdata is null) return 0; + + scale = fons__tt_getPixelHeightScale(&font.font, cast(float)isize/10.0f); + + // Align vertically. + y += fons__getVertAlign(stash, font, state.talign, isize); + + minx = maxx = x; + miny = maxy = y; + startx = x; + + static if (is(T == char)) { + foreach (char ch; str) { + //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)) { + 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; + } else { + prevGlyphIndex = -1; + } + } + } + + advance = x-startx; + + // Align horizontally + if (state.talign.left) { + // empty + } else if (state.talign.right) { + minx -= advance; + maxx -= advance; + } else if (state.talign.center) { + minx -= advance*0.5f; + maxx -= advance*0.5f; + } + + if (bounds.length) { + if (bounds.length > 0) bounds.ptr[0] = minx; + if (bounds.length > 1) bounds.ptr[1] = miny; + if (bounds.length > 2) bounds.ptr[2] = maxx; + if (bounds.length > 3) bounds.ptr[3] = maxy; + } + + return advance; +} + +public void fonsVertMetrics (FONScontext* stash, float* ascender, float* descender, float* lineh) nothrow @trusted @nogc { + FONSfont* font; + FONSstate* state = fons__getState(stash); + short isize; + + if (stash is null) return; + if (state.font < 0 || state.font >= stash.nfonts) return; + font = stash.fonts[state.font]; + isize = cast(short)(state.size*10.0f); + if (font is null || font.fdata is null) return; + + if (ascender) *ascender = font.ascender*isize/10.0f; + if (descender) *descender = font.descender*isize/10.0f; + if (lineh) *lineh = font.lineh*isize/10.0f; +} + +public void fonsLineBounds (FONScontext* stash, float y, float* minyp, float* maxyp) nothrow @trusted @nogc { + FONSfont* font; + FONSstate* state = fons__getState(stash); + short isize; + + if (minyp !is null) *minyp = 0; + if (maxyp !is null) *maxyp = 0; + + if (stash is null) return; + if (state.font < 0 || state.font >= stash.nfonts) return; + font = stash.fonts[state.font]; + isize = cast(short)(state.size*10.0f); + if (font is null || font.fdata is null) return; + + y += fons__getVertAlign(stash, font, state.talign, isize); + + if (stash.params.flags&FONS_ZERO_TOPLEFT) { + immutable float miny = y-font.ascender*cast(float)isize/10.0f; + immutable float maxy = miny+font.lineh*isize/10.0f; + if (minyp !is null) *minyp = miny; + if (maxyp !is null) *maxyp = maxy; + } else { + immutable float maxy = y+font.descender*cast(float)isize/10.0f; + immutable float miny = maxy-font.lineh*isize/10.0f; + if (minyp !is null) *minyp = miny; + if (maxyp !is null) *maxyp = maxy; + } +} + +public const(ubyte)* fonsGetTextureData (FONScontext* stash, int* width, int* height) nothrow @trusted @nogc { + if (width !is null) *width = stash.params.width; + if (height !is null) *height = stash.params.height; + return stash.texData; +} + +public int fonsValidateTexture (FONScontext* stash, int* dirty) nothrow @trusted @nogc { + if (stash.dirtyRect.ptr[0] < stash.dirtyRect.ptr[2] && stash.dirtyRect.ptr[1] < stash.dirtyRect.ptr[3]) { + dirty[0] = stash.dirtyRect.ptr[0]; + dirty[1] = stash.dirtyRect.ptr[1]; + dirty[2] = stash.dirtyRect.ptr[2]; + dirty[3] = stash.dirtyRect.ptr[3]; + // Reset dirty rect + stash.dirtyRect.ptr[0] = stash.params.width; + stash.dirtyRect.ptr[1] = stash.params.height; + stash.dirtyRect.ptr[2] = 0; + stash.dirtyRect.ptr[3] = 0; + return 1; + } + return 0; +} + +public void fonsDeleteInternal (FONScontext* stash) nothrow @trusted @nogc { + if (stash is null) return; + + if (stash.params.renderDelete !is null) stash.params.renderDelete(stash.params.userPtr); + + foreach (int i; 0..stash.nfonts) fons__freeFont(stash.fonts[i]); + + if (stash.atlas) fons__deleteAtlas(stash.atlas); + if (stash.fonts) free(stash.fonts); + if (stash.texData) free(stash.texData); + if (stash.scratch) free(stash.scratch); + if (stash.hashidx) free(stash.hashidx); + free(stash); +} + +public void fonsSetErrorCallback (FONScontext* stash, void function (void* uptr, int error, int val) nothrow @trusted @nogc callback, void* uptr) nothrow @trusted @nogc { + if (stash is null) return; + stash.handleError = callback; + stash.errorUptr = uptr; +} + +public void fonsGetAtlasSize (FONScontext* stash, int* width, int* height) nothrow @trusted @nogc { + if (stash is null) return; + *width = stash.params.width; + *height = stash.params.height; +} + +public int fonsExpandAtlas (FONScontext* stash, int width, int height) nothrow @trusted @nogc { + int i, maxy = 0; + ubyte* data = null; + if (stash is null) return 0; + + width = nvg__max(width, stash.params.width); + height = nvg__max(height, stash.params.height); + + if (width == stash.params.width && height == stash.params.height) return 1; + + // Flush pending glyphs. + fons__flush(stash); + + // Create new texture + if (stash.params.renderResize !is null) { + if (stash.params.renderResize(stash.params.userPtr, width, height) == 0) return 0; + } + // Copy old texture data over. + data = cast(ubyte*)malloc(width*height); + if (data is null) return 0; + for (i = 0; i < stash.params.height; i++) { + ubyte* dst = &data[i*width]; + ubyte* src = &stash.texData[i*stash.params.width]; + memcpy(dst, src, stash.params.width); + if (width > stash.params.width) + memset(dst+stash.params.width, 0, width-stash.params.width); + } + if (height > stash.params.height) memset(&data[stash.params.height*width], 0, (height-stash.params.height)*width); + + free(stash.texData); + stash.texData = data; + + // Increase atlas size + fons__atlasExpand(stash.atlas, width, height); + + // Add existing data as dirty. + for (i = 0; i < stash.atlas.nnodes; i++) maxy = nvg__max(maxy, stash.atlas.nodes[i].y); + stash.dirtyRect.ptr[0] = 0; + stash.dirtyRect.ptr[1] = 0; + stash.dirtyRect.ptr[2] = stash.params.width; + stash.dirtyRect.ptr[3] = maxy; + + stash.params.width = width; + stash.params.height = height; + stash.itw = 1.0f/stash.params.width; + stash.ith = 1.0f/stash.params.height; + + return 1; +} + +public bool fonsResetAtlas (FONScontext* stash, int width, int height) nothrow @trusted @nogc { + if (stash is null) return false; + + // Flush pending glyphs. + fons__flush(stash); + + // Create new texture + if (stash.params.renderResize !is null) { + if (stash.params.renderResize(stash.params.userPtr, width, height) == 0) return false; + } + + // Reset atlas + fons__atlasReset(stash.atlas, width, height); + + // Clear texture data. + stash.texData = cast(ubyte*)realloc(stash.texData, width*height); + if (stash.texData is null) return 0; + memset(stash.texData, 0, width*height); + + // Reset dirty rect + stash.dirtyRect.ptr[0] = width; + stash.dirtyRect.ptr[1] = height; + stash.dirtyRect.ptr[2] = 0; + stash.dirtyRect.ptr[3] = 0; + + // Reset cached glyphs + foreach (FONSfont* font; stash.fonts[0..stash.nfonts]) { + if (font !is null) { + font.nglyphs = 0; + font.lut.ptr[0..FONS_HASH_LUT_SIZE] = -1; + } + } + + stash.params.width = width; + stash.params.height = height; + stash.itw = 1.0f/stash.params.width; + stash.ith = 1.0f/stash.params.height; + + // Add white rect at 0, 0 for debug drawing. + fons__addWhiteRect(stash, 2, 2); + + return true; +} + + +// ////////////////////////////////////////////////////////////////////////// // +// backgl +// ////////////////////////////////////////////////////////////////////////// // +import core.stdc.stdlib : malloc, realloc, free; +import core.stdc.string : memcpy, memset; + +//import arsd.simpledisplay; +version(nanovg_builtin_opengl_bindings) { import arsd.simpledisplay; } else { import iv.glbinds; } + +private: +// sdpy is missing that yet +static if (!is(typeof(GL_STENCIL_BUFFER_BIT))) enum uint GL_STENCIL_BUFFER_BIT = 0x00000400; + + +// OpenGL API missing from simpledisplay +private extern(System) nothrow @nogc { + alias GLvoid = void; + alias GLboolean = ubyte; + alias GLuint = uint; + alias GLenum = uint; + alias GLchar = char; + alias GLsizei = int; + alias GLfloat = float; + alias GLsizeiptr = ptrdiff_t; + + enum uint GL_STENCIL_BUFFER_BIT = 0x00000400; + + enum uint GL_INVALID_ENUM = 0x0500; + + enum uint GL_ZERO = 0; + enum uint GL_ONE = 1; + + enum uint GL_FLOAT = 0x1406; + + enum uint GL_STREAM_DRAW = 0x88E0; + + enum uint GL_CCW = 0x0901; + + enum uint GL_STENCIL_TEST = 0x0B90; + enum uint GL_SCISSOR_TEST = 0x0C11; + + enum uint GL_EQUAL = 0x0202; + enum uint GL_NOTEQUAL = 0x0205; + + enum uint GL_ALWAYS = 0x0207; + enum uint GL_KEEP = 0x1E00; + + enum uint GL_INCR = 0x1E02; + + enum uint GL_INCR_WRAP = 0x8507; + enum uint GL_DECR_WRAP = 0x8508; + + enum uint GL_CULL_FACE = 0x0B44; + enum uint GL_BACK = 0x0405; + + enum uint GL_FRAGMENT_SHADER = 0x8B30; + enum uint GL_VERTEX_SHADER = 0x8B31; + + enum uint GL_COMPILE_STATUS = 0x8B81; + enum uint GL_LINK_STATUS = 0x8B82; + + enum uint GL_UNPACK_ALIGNMENT = 0x0CF5; + enum uint GL_UNPACK_ROW_LENGTH = 0x0CF2; + enum uint GL_UNPACK_SKIP_PIXELS = 0x0CF4; + enum uint GL_UNPACK_SKIP_ROWS = 0x0CF3; + + enum uint GL_GENERATE_MIPMAP = 0x8191; + enum uint GL_LINEAR_MIPMAP_LINEAR = 0x2703; + + enum uint GL_RED = 0x1903; + + enum uint GL_TEXTURE0 = 0x84C0; + + enum uint GL_ARRAY_BUFFER = 0x8892; + + enum uint GL_SRC_COLOR = 0x0300; + enum uint GL_ONE_MINUS_SRC_COLOR = 0x0301; + enum uint GL_SRC_ALPHA = 0x0302; + enum uint GL_ONE_MINUS_SRC_ALPHA = 0x0303; + enum uint GL_DST_ALPHA = 0x0304; + enum uint GL_ONE_MINUS_DST_ALPHA = 0x0305; + enum uint GL_DST_COLOR = 0x0306; + enum uint GL_ONE_MINUS_DST_COLOR = 0x0307; + enum uint GL_SRC_ALPHA_SATURATE = 0x0308; + + enum uint GL_INVERT = 0x150AU; + + /* + version(Windows) { + private void* kglLoad (const(char)* name) { + void* res = glGetProcAddress(name); + if (res is null) { + import core.sys.windows.windef, core.sys.windows.winbase; + static HINSTANCE dll = null; + if (dll is null) { + dll = LoadLibraryA("opengl32.dll"); + if (dll is null) return null; // <32, but idc + return GetProcAddress(dll, name); + } + } + } + } else { + alias kglLoad = glGetProcAddress; + } + */ + + alias glbfn_glStencilMask = void function(GLuint); + __gshared glbfn_glStencilMask glStencilMask_NVGLZ; alias glStencilMask = glStencilMask_NVGLZ; + alias glbfn_glStencilFunc = void function(GLenum, GLint, GLuint); + __gshared glbfn_glStencilFunc glStencilFunc_NVGLZ; alias glStencilFunc = glStencilFunc_NVGLZ; + alias glbfn_glGetShaderInfoLog = void function(GLuint, GLsizei, GLsizei*, GLchar*); + __gshared glbfn_glGetShaderInfoLog glGetShaderInfoLog_NVGLZ; alias glGetShaderInfoLog = glGetShaderInfoLog_NVGLZ; + alias glbfn_glGetProgramInfoLog = void function(GLuint, GLsizei, GLsizei*, GLchar*); + __gshared glbfn_glGetProgramInfoLog glGetProgramInfoLog_NVGLZ; alias glGetProgramInfoLog = glGetProgramInfoLog_NVGLZ; + alias glbfn_glCreateProgram = GLuint function(); + __gshared glbfn_glCreateProgram glCreateProgram_NVGLZ; alias glCreateProgram = glCreateProgram_NVGLZ; + alias glbfn_glCreateShader = GLuint function(GLenum); + __gshared glbfn_glCreateShader glCreateShader_NVGLZ; alias glCreateShader = glCreateShader_NVGLZ; + alias glbfn_glShaderSource = void function(GLuint, GLsizei, const(GLchar*)*, const(GLint)*); + __gshared glbfn_glShaderSource glShaderSource_NVGLZ; alias glShaderSource = glShaderSource_NVGLZ; + alias glbfn_glCompileShader = void function(GLuint); + __gshared glbfn_glCompileShader glCompileShader_NVGLZ; alias glCompileShader = glCompileShader_NVGLZ; + alias glbfn_glGetShaderiv = void function(GLuint, GLenum, GLint*); + __gshared glbfn_glGetShaderiv glGetShaderiv_NVGLZ; alias glGetShaderiv = glGetShaderiv_NVGLZ; + alias glbfn_glAttachShader = void function(GLuint, GLuint); + __gshared glbfn_glAttachShader glAttachShader_NVGLZ; alias glAttachShader = glAttachShader_NVGLZ; + alias glbfn_glBindAttribLocation = void function(GLuint, GLuint, const(GLchar)*); + __gshared glbfn_glBindAttribLocation glBindAttribLocation_NVGLZ; alias glBindAttribLocation = glBindAttribLocation_NVGLZ; + alias glbfn_glLinkProgram = void function(GLuint); + __gshared glbfn_glLinkProgram glLinkProgram_NVGLZ; alias glLinkProgram = glLinkProgram_NVGLZ; + alias glbfn_glGetProgramiv = void function(GLuint, GLenum, GLint*); + __gshared glbfn_glGetProgramiv glGetProgramiv_NVGLZ; alias glGetProgramiv = glGetProgramiv_NVGLZ; + alias glbfn_glDeleteProgram = void function(GLuint); + __gshared glbfn_glDeleteProgram glDeleteProgram_NVGLZ; alias glDeleteProgram = glDeleteProgram_NVGLZ; + alias glbfn_glDeleteShader = void function(GLuint); + __gshared glbfn_glDeleteShader glDeleteShader_NVGLZ; alias glDeleteShader = glDeleteShader_NVGLZ; + alias glbfn_glGetUniformLocation = GLint function(GLuint, const(GLchar)*); + __gshared glbfn_glGetUniformLocation glGetUniformLocation_NVGLZ; alias glGetUniformLocation = glGetUniformLocation_NVGLZ; + alias glbfn_glGenBuffers = void function(GLsizei, GLuint*); + __gshared glbfn_glGenBuffers glGenBuffers_NVGLZ; alias glGenBuffers = glGenBuffers_NVGLZ; + alias glbfn_glPixelStorei = void function(GLenum, GLint); + __gshared glbfn_glPixelStorei glPixelStorei_NVGLZ; alias glPixelStorei = glPixelStorei_NVGLZ; + alias glbfn_glUniform4fv = void function(GLint, GLsizei, const(GLfloat)*); + __gshared glbfn_glUniform4fv glUniform4fv_NVGLZ; alias glUniform4fv = glUniform4fv_NVGLZ; + alias glbfn_glColorMask = void function(GLboolean, GLboolean, GLboolean, GLboolean); + __gshared glbfn_glColorMask glColorMask_NVGLZ; alias glColorMask = glColorMask_NVGLZ; + alias glbfn_glStencilOpSeparate = void function(GLenum, GLenum, GLenum, GLenum); + __gshared glbfn_glStencilOpSeparate glStencilOpSeparate_NVGLZ; alias glStencilOpSeparate = glStencilOpSeparate_NVGLZ; + alias glbfn_glDrawArrays = void function(GLenum, GLint, GLsizei); + __gshared glbfn_glDrawArrays glDrawArrays_NVGLZ; alias glDrawArrays = glDrawArrays_NVGLZ; + alias glbfn_glStencilOp = void function(GLenum, GLenum, GLenum); + __gshared glbfn_glStencilOp glStencilOp_NVGLZ; alias glStencilOp = glStencilOp_NVGLZ; + alias glbfn_glUseProgram = void function(GLuint); + __gshared glbfn_glUseProgram glUseProgram_NVGLZ; alias glUseProgram = glUseProgram_NVGLZ; + alias glbfn_glCullFace = void function(GLenum); + __gshared glbfn_glCullFace glCullFace_NVGLZ; alias glCullFace = glCullFace_NVGLZ; + alias glbfn_glFrontFace = void function(GLenum); + __gshared glbfn_glFrontFace glFrontFace_NVGLZ; alias glFrontFace = glFrontFace_NVGLZ; + alias glbfn_glActiveTexture = void function(GLenum); + __gshared glbfn_glActiveTexture glActiveTexture_NVGLZ; alias glActiveTexture = glActiveTexture_NVGLZ; + alias glbfn_glBindBuffer = void function(GLenum, GLuint); + __gshared glbfn_glBindBuffer glBindBuffer_NVGLZ; alias glBindBuffer = glBindBuffer_NVGLZ; + alias glbfn_glBufferData = void function(GLenum, GLsizeiptr, const(void)*, GLenum); + __gshared glbfn_glBufferData glBufferData_NVGLZ; alias glBufferData = glBufferData_NVGLZ; + alias glbfn_glEnableVertexAttribArray = void function(GLuint); + __gshared glbfn_glEnableVertexAttribArray glEnableVertexAttribArray_NVGLZ; alias glEnableVertexAttribArray = glEnableVertexAttribArray_NVGLZ; + alias glbfn_glVertexAttribPointer = void function(GLuint, GLint, GLenum, GLboolean, GLsizei, const(void)*); + __gshared glbfn_glVertexAttribPointer glVertexAttribPointer_NVGLZ; alias glVertexAttribPointer = glVertexAttribPointer_NVGLZ; + alias glbfn_glUniform1i = void function(GLint, GLint); + __gshared glbfn_glUniform1i glUniform1i_NVGLZ; alias glUniform1i = glUniform1i_NVGLZ; + alias glbfn_glUniform2fv = void function(GLint, GLsizei, const(GLfloat)*); + __gshared glbfn_glUniform2fv glUniform2fv_NVGLZ; alias glUniform2fv = glUniform2fv_NVGLZ; + alias glbfn_glDisableVertexAttribArray = void function(GLuint); + __gshared glbfn_glDisableVertexAttribArray glDisableVertexAttribArray_NVGLZ; alias glDisableVertexAttribArray = glDisableVertexAttribArray_NVGLZ; + alias glbfn_glDeleteBuffers = void function(GLsizei, const(GLuint)*); + __gshared glbfn_glDeleteBuffers glDeleteBuffers_NVGLZ; alias glDeleteBuffers = glDeleteBuffers_NVGLZ; + alias glbfn_glBlendFuncSeparate = void function(GLenum, GLenum, GLenum, GLenum); + __gshared glbfn_glBlendFuncSeparate glBlendFuncSeparate_NVGLZ; alias glBlendFuncSeparate = glBlendFuncSeparate_NVGLZ; + + private void nanovgInitOpenGL () { + __gshared bool initialized = false; + if (initialized) return; + glStencilMask_NVGLZ = cast(glbfn_glStencilMask)glbindGetProcAddress(`glStencilMask`); + if (glStencilMask_NVGLZ is null) assert(0, `OpenGL function 'glStencilMask' not found!`); + glStencilFunc_NVGLZ = cast(glbfn_glStencilFunc)glbindGetProcAddress(`glStencilFunc`); + if (glStencilFunc_NVGLZ is null) assert(0, `OpenGL function 'glStencilFunc' not found!`); + glGetShaderInfoLog_NVGLZ = cast(glbfn_glGetShaderInfoLog)glbindGetProcAddress(`glGetShaderInfoLog`); + if (glGetShaderInfoLog_NVGLZ is null) assert(0, `OpenGL function 'glGetShaderInfoLog' not found!`); + glGetProgramInfoLog_NVGLZ = cast(glbfn_glGetProgramInfoLog)glbindGetProcAddress(`glGetProgramInfoLog`); + if (glGetProgramInfoLog_NVGLZ is null) assert(0, `OpenGL function 'glGetProgramInfoLog' not found!`); + glCreateProgram_NVGLZ = cast(glbfn_glCreateProgram)glbindGetProcAddress(`glCreateProgram`); + if (glCreateProgram_NVGLZ is null) assert(0, `OpenGL function 'glCreateProgram' not found!`); + glCreateShader_NVGLZ = cast(glbfn_glCreateShader)glbindGetProcAddress(`glCreateShader`); + if (glCreateShader_NVGLZ is null) assert(0, `OpenGL function 'glCreateShader' not found!`); + glShaderSource_NVGLZ = cast(glbfn_glShaderSource)glbindGetProcAddress(`glShaderSource`); + if (glShaderSource_NVGLZ is null) assert(0, `OpenGL function 'glShaderSource' not found!`); + glCompileShader_NVGLZ = cast(glbfn_glCompileShader)glbindGetProcAddress(`glCompileShader`); + if (glCompileShader_NVGLZ is null) assert(0, `OpenGL function 'glCompileShader' not found!`); + glGetShaderiv_NVGLZ = cast(glbfn_glGetShaderiv)glbindGetProcAddress(`glGetShaderiv`); + if (glGetShaderiv_NVGLZ is null) assert(0, `OpenGL function 'glGetShaderiv' not found!`); + glAttachShader_NVGLZ = cast(glbfn_glAttachShader)glbindGetProcAddress(`glAttachShader`); + if (glAttachShader_NVGLZ is null) assert(0, `OpenGL function 'glAttachShader' not found!`); + glBindAttribLocation_NVGLZ = cast(glbfn_glBindAttribLocation)glbindGetProcAddress(`glBindAttribLocation`); + if (glBindAttribLocation_NVGLZ is null) assert(0, `OpenGL function 'glBindAttribLocation' not found!`); + glLinkProgram_NVGLZ = cast(glbfn_glLinkProgram)glbindGetProcAddress(`glLinkProgram`); + if (glLinkProgram_NVGLZ is null) assert(0, `OpenGL function 'glLinkProgram' not found!`); + glGetProgramiv_NVGLZ = cast(glbfn_glGetProgramiv)glbindGetProcAddress(`glGetProgramiv`); + if (glGetProgramiv_NVGLZ is null) assert(0, `OpenGL function 'glGetProgramiv' not found!`); + glDeleteProgram_NVGLZ = cast(glbfn_glDeleteProgram)glbindGetProcAddress(`glDeleteProgram`); + if (glDeleteProgram_NVGLZ is null) assert(0, `OpenGL function 'glDeleteProgram' not found!`); + glDeleteShader_NVGLZ = cast(glbfn_glDeleteShader)glbindGetProcAddress(`glDeleteShader`); + if (glDeleteShader_NVGLZ is null) assert(0, `OpenGL function 'glDeleteShader' not found!`); + glGetUniformLocation_NVGLZ = cast(glbfn_glGetUniformLocation)glbindGetProcAddress(`glGetUniformLocation`); + if (glGetUniformLocation_NVGLZ is null) assert(0, `OpenGL function 'glGetUniformLocation' not found!`); + glGenBuffers_NVGLZ = cast(glbfn_glGenBuffers)glbindGetProcAddress(`glGenBuffers`); + if (glGenBuffers_NVGLZ is null) assert(0, `OpenGL function 'glGenBuffers' not found!`); + glPixelStorei_NVGLZ = cast(glbfn_glPixelStorei)glbindGetProcAddress(`glPixelStorei`); + if (glPixelStorei_NVGLZ is null) assert(0, `OpenGL function 'glPixelStorei' not found!`); + glUniform4fv_NVGLZ = cast(glbfn_glUniform4fv)glbindGetProcAddress(`glUniform4fv`); + if (glUniform4fv_NVGLZ is null) assert(0, `OpenGL function 'glUniform4fv' not found!`); + glColorMask_NVGLZ = cast(glbfn_glColorMask)glbindGetProcAddress(`glColorMask`); + if (glColorMask_NVGLZ is null) assert(0, `OpenGL function 'glColorMask' not found!`); + glStencilOpSeparate_NVGLZ = cast(glbfn_glStencilOpSeparate)glbindGetProcAddress(`glStencilOpSeparate`); + if (glStencilOpSeparate_NVGLZ is null) assert(0, `OpenGL function 'glStencilOpSeparate' not found!`); + glDrawArrays_NVGLZ = cast(glbfn_glDrawArrays)glbindGetProcAddress(`glDrawArrays`); + if (glDrawArrays_NVGLZ is null) assert(0, `OpenGL function 'glDrawArrays' not found!`); + glStencilOp_NVGLZ = cast(glbfn_glStencilOp)glbindGetProcAddress(`glStencilOp`); + if (glStencilOp_NVGLZ is null) assert(0, `OpenGL function 'glStencilOp' not found!`); + glUseProgram_NVGLZ = cast(glbfn_glUseProgram)glbindGetProcAddress(`glUseProgram`); + if (glUseProgram_NVGLZ is null) assert(0, `OpenGL function 'glUseProgram' not found!`); + glCullFace_NVGLZ = cast(glbfn_glCullFace)glbindGetProcAddress(`glCullFace`); + if (glCullFace_NVGLZ is null) assert(0, `OpenGL function 'glCullFace' not found!`); + glFrontFace_NVGLZ = cast(glbfn_glFrontFace)glbindGetProcAddress(`glFrontFace`); + if (glFrontFace_NVGLZ is null) assert(0, `OpenGL function 'glFrontFace' not found!`); + glActiveTexture_NVGLZ = cast(glbfn_glActiveTexture)glbindGetProcAddress(`glActiveTexture`); + if (glActiveTexture_NVGLZ is null) assert(0, `OpenGL function 'glActiveTexture' not found!`); + glBindBuffer_NVGLZ = cast(glbfn_glBindBuffer)glbindGetProcAddress(`glBindBuffer`); + if (glBindBuffer_NVGLZ is null) assert(0, `OpenGL function 'glBindBuffer' not found!`); + glBufferData_NVGLZ = cast(glbfn_glBufferData)glbindGetProcAddress(`glBufferData`); + if (glBufferData_NVGLZ is null) assert(0, `OpenGL function 'glBufferData' not found!`); + glEnableVertexAttribArray_NVGLZ = cast(glbfn_glEnableVertexAttribArray)glbindGetProcAddress(`glEnableVertexAttribArray`); + if (glEnableVertexAttribArray_NVGLZ is null) assert(0, `OpenGL function 'glEnableVertexAttribArray' not found!`); + glVertexAttribPointer_NVGLZ = cast(glbfn_glVertexAttribPointer)glbindGetProcAddress(`glVertexAttribPointer`); + if (glVertexAttribPointer_NVGLZ is null) assert(0, `OpenGL function 'glVertexAttribPointer' not found!`); + glUniform1i_NVGLZ = cast(glbfn_glUniform1i)glbindGetProcAddress(`glUniform1i`); + if (glUniform1i_NVGLZ is null) assert(0, `OpenGL function 'glUniform1i' not found!`); + glUniform2fv_NVGLZ = cast(glbfn_glUniform2fv)glbindGetProcAddress(`glUniform2fv`); + if (glUniform2fv_NVGLZ is null) assert(0, `OpenGL function 'glUniform2fv' not found!`); + glDisableVertexAttribArray_NVGLZ = cast(glbfn_glDisableVertexAttribArray)glbindGetProcAddress(`glDisableVertexAttribArray`); + if (glDisableVertexAttribArray_NVGLZ is null) assert(0, `OpenGL function 'glDisableVertexAttribArray' not found!`); + glDeleteBuffers_NVGLZ = cast(glbfn_glDeleteBuffers)glbindGetProcAddress(`glDeleteBuffers`); + if (glDeleteBuffers_NVGLZ is null) assert(0, `OpenGL function 'glDeleteBuffers' not found!`); + glBlendFuncSeparate_NVGLZ = cast(glbfn_glBlendFuncSeparate)glbindGetProcAddress(`glBlendFuncSeparate`); + if (glBlendFuncSeparate_NVGLZ is null) assert(0, `OpenGL function 'glBlendFuncSeparate' not found!`); + initialized = true; + } +} + + +/// Context creation flags. +/// Group: context_management +public enum NVGContextFlag : int { + /// Nothing special, i.e. empty flag. + None = 0, + /// Flag indicating if geometry based anti-aliasing is used (may not be needed when using MSAA). + Antialias = 1<<0, + /** Flag indicating if strokes should be drawn using stencil buffer. The rendering will be a little + * slower, but path overlaps (i.e. self-intersecting or sharp turns) will be drawn just once. */ + StencilStrokes = 1<<1, + /// Flag indicating that additional debug checks are done. + Debug = 1<<2, + /// Filter (antialias) fonts + FontAA = 1<<7, + /// Don't filter (antialias) fonts + FontNoAA = 1<<8, +} + +public enum NANOVG_GL_USE_STATE_FILTER = true; + +/// These are additional flags on top of NVGImageFlags. +/// Group: images +public enum NVGImageFlagsGL : int { + NoDelete = 1<<16, // Do not delete GL texture handle. +} + + +/// Returns flags for glClear(). +/// Group: context_management +public uint glNVGClearFlags () pure nothrow @safe @nogc { + pragma(inline, true); + return (GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT|GL_STENCIL_BUFFER_BIT); +} + + +// ////////////////////////////////////////////////////////////////////////// // +private: + +enum GLNVGuniformLoc { + ViewSize, + Tex, + Frag, + TMat, + TTr, +} + +alias GLNVGshaderType = int; +enum /*GLNVGshaderType*/ { + NSVG_SHADER_FILLGRAD, + NSVG_SHADER_FILLIMG, + NSVG_SHADER_SIMPLE, + NSVG_SHADER_IMG, +} + +struct GLNVGshader { + GLuint prog; + GLuint frag; + GLuint vert; + GLint[GLNVGuniformLoc.max+1] loc; +} + +struct GLNVGtexture { + int id; + GLuint tex; + int width, height; + NVGtexture type; + int flags; + int rc; + int nextfree; +} + +alias GLNVGcallType = int; +enum /*GLNVGcallType*/ { + GLNVG_NONE = 0, + GLNVG_FILL, + GLNVG_CONVEXFILL, + GLNVG_STROKE, + GLNVG_TRIANGLES, + GLNVG_AFFINE, // change affine transformation matrix +} + +struct GLNVGcall { + int type; + int evenOdd; // for fill + int image; + int pathOffset; + int pathCount; + int triangleOffset; + int triangleCount; + int uniformOffset; + NVGMatrix affine; +} + +struct GLNVGpath { + int fillOffset; + int fillCount; + int strokeOffset; + int strokeCount; +} + +align(1) struct GLNVGfragUniforms { +align(1): + enum UNIFORM_ARRAY_SIZE = 11; + // note: after modifying layout or size of uniform array, + // don't forget to also update the fragment shader source! + align(1) union { + align(1): + align(1) struct { + align(1): + float[12] scissorMat; // matrices are actually 3 vec4s + float[12] paintMat; + NVGColor innerCol; + NVGColor outerCol; + float[2] scissorExt; + float[2] scissorScale; + float[2] extent; + float radius; + float feather; + float strokeMult; + float strokeThr; + float texType; + float type; + } + float[4][UNIFORM_ARRAY_SIZE] uniformArray; + } +} + +struct GLNVGcontext { + GLNVGshader shader; + GLNVGtexture* textures; + float[2] view; + int freetexid; // -1: none + int ntextures; + int ctextures; + GLuint vertBuf; + int fragSize; + int flags; + + // Per frame buffers + GLNVGcall* calls; + int ccalls; + int ncalls; + GLNVGpath* paths; + int cpaths; + int npaths; + NVGvertex* verts; + int cverts; + int nverts; + ubyte* uniforms; + int cuniforms; + int nuniforms; + + // cached state + static if (NANOVG_GL_USE_STATE_FILTER) { + GLuint boundTexture; + GLuint stencilMask; + GLenum stencilFunc; + GLint stencilFuncRef; + GLuint stencilFuncMask; + } +} + +int glnvg__maxi() (int a, int b) { pragma(inline, true); return (a > b ? a : b); } + +void glnvg__bindTexture (GLNVGcontext* gl, GLuint tex) nothrow @trusted @nogc { + static if (NANOVG_GL_USE_STATE_FILTER) { + if (gl.boundTexture != tex) { + gl.boundTexture = tex; + glBindTexture(GL_TEXTURE_2D, tex); + } + } else { + glBindTexture(GL_TEXTURE_2D, tex); + } +} + +void glnvg__stencilMask (GLNVGcontext* gl, GLuint mask) nothrow @trusted @nogc { + static if (NANOVG_GL_USE_STATE_FILTER) { + if (gl.stencilMask != mask) { + gl.stencilMask = mask; + glStencilMask(mask); + } + } else { + glStencilMask(mask); + } +} + +void glnvg__stencilFunc (GLNVGcontext* gl, GLenum func, GLint ref_, GLuint mask) nothrow @trusted @nogc { + static if (NANOVG_GL_USE_STATE_FILTER) { + if (gl.stencilFunc != func || gl.stencilFuncRef != ref_ || gl.stencilFuncMask != mask) { + gl.stencilFunc = func; + gl.stencilFuncRef = ref_; + gl.stencilFuncMask = mask; + glStencilFunc(func, ref_, mask); + } + } else { + glStencilFunc(func, ref_, mask); + } +} + +// texture id is never zero +GLNVGtexture* glnvg__allocTexture (GLNVGcontext* gl) nothrow @trusted @nogc { + GLNVGtexture* tex = null; + + int tid = gl.freetexid; + if (tid == -1) { + if (gl.ntextures >= gl.ctextures) { + assert(gl.ntextures == gl.ctextures); + int ctextures = (gl.ctextures == 0 ? 16 : glnvg__maxi(tid+1, 4)+gl.ctextures/2); // 1.5x overallocate + GLNVGtexture* textures = cast(GLNVGtexture*)realloc(gl.textures, GLNVGtexture.sizeof*ctextures); + if (textures is null) return null; + memset(&textures[gl.ctextures], 0, (ctextures-gl.ctextures)*GLNVGtexture.sizeof); + version(nanovega_debug_textures) {{ import core.stdc.stdio; printf("allocated more textures (n=%d; c=%d; nc=%d)\n", gl.ntextures, gl.ctextures, ctextures); }} + gl.textures = textures; + gl.ctextures = ctextures; + } + tid = gl.ntextures++; + version(nanovega_debug_textures) {{ import core.stdc.stdio; printf(" got next free texture id %d, ntextures=%d\n", tid+1, gl.ntextures); }} + } else { + gl.freetexid = gl.textures[tid].nextfree; + } + assert(tid <= gl.ntextures); + + assert(gl.textures[tid].id == 0); + tex = &gl.textures[tid]; + memset(tex, 0, (*tex).sizeof); + tex.id = tid+1; + tex.rc = 1; + tex.nextfree = -1; + + version(nanovega_debug_textures) {{ import core.stdc.stdio; printf("allocated texture with id %d (%d)\n", tex.id, tid+1); }} + + return tex; +} + +GLNVGtexture* glnvg__findTexture (GLNVGcontext* gl, int id) nothrow @trusted @nogc { + if (id <= 0 || id > gl.ntextures) return null; + if (gl.textures[id-1].id == 0) return null; // free one + assert(gl.textures[id-1].id == id); + return &gl.textures[id-1]; +} + +bool glnvg__deleteTexture (GLNVGcontext* gl, int id) nothrow @trusted @nogc { + if (id <= 0 || id > gl.ntextures) return false; + auto tx = &gl.textures[id-1]; + if (tx.id == 0) return false; // free one + assert(tx.id == id); + assert(tx.tex != 0); + version(nanovega_debug_textures) {{ import core.stdc.stdio; printf("decrefing texture with id %d (%d)\n", tx.id, id); }} + if (--tx.rc == 0) { + if ((tx.flags&NVGImageFlagsGL.NoDelete) == 0) glDeleteTextures(1, &tx.tex); + version(nanovega_debug_textures) {{ import core.stdc.stdio; printf("deleted texture with id %d (%d); glid=%u\n", tx.id, id, tx.tex); }} + memset(tx, 0, (*tx).sizeof); + //{ import core.stdc.stdio; printf("deleting texture with id %d\n", id); } + tx.nextfree = gl.freetexid; + gl.freetexid = id-1; + } + return true; +} + +void glnvg__dumpShaderError (GLuint shader, const(char)* name, const(char)* type) nothrow @trusted @nogc { + import core.stdc.stdio : fprintf, stderr; + GLchar[512+1] str = 0; + GLsizei len = 0; + glGetShaderInfoLog(shader, 512, &len, str.ptr); + if (len > 512) len = 512; + str[len] = '\0'; + fprintf(stderr, "Shader %s/%s error:\n%s\n", name, type, str.ptr); +} + +void glnvg__dumpProgramError (GLuint prog, const(char)* name) nothrow @trusted @nogc { + import core.stdc.stdio : fprintf, stderr; + GLchar[512+1] str = 0; + GLsizei len = 0; + glGetProgramInfoLog(prog, 512, &len, str.ptr); + if (len > 512) len = 512; + str[len] = '\0'; + fprintf(stderr, "Program %s error:\n%s\n", name, str.ptr); +} + +void glnvg__checkError (GLNVGcontext* gl, const(char)* str) nothrow @trusted @nogc { + GLenum err; + if ((gl.flags&NVGContextFlag.Debug) == 0) return; + err = glGetError(); + if (err != GL_NO_ERROR) { + import core.stdc.stdio : fprintf, stderr; + fprintf(stderr, "Error %08x after %s\n", err, str); + return; + } +} + +bool glnvg__createShader (GLNVGshader* shader, const(char)* name, const(char)* header, const(char)* opts, const(char)* vshader, const(char)* fshader) nothrow @trusted @nogc { + GLint status; + GLuint prog, vert, frag; + const(char)*[3] str; + + memset(shader, 0, (*shader).sizeof); + + prog = glCreateProgram(); + vert = glCreateShader(GL_VERTEX_SHADER); + frag = glCreateShader(GL_FRAGMENT_SHADER); + str[0] = header; + str[1] = (opts !is null ? opts : ""); + str[2] = vshader; + glShaderSource(vert, 3, cast(const(char)**)str.ptr, null); + + glCompileShader(vert); + glGetShaderiv(vert, GL_COMPILE_STATUS, &status); + if (status != GL_TRUE) { + glnvg__dumpShaderError(vert, name, "vert"); + return false; + } + + str[0] = header; + str[1] = (opts !is null ? opts : ""); + str[2] = fshader; + glShaderSource(frag, 3, cast(const(char)**)str.ptr, null); + + glCompileShader(frag); + glGetShaderiv(frag, GL_COMPILE_STATUS, &status); + if (status != GL_TRUE) { + glnvg__dumpShaderError(frag, name, "frag"); + return false; + } + + glAttachShader(prog, vert); + glAttachShader(prog, frag); + + glBindAttribLocation(prog, 0, "vertex"); + glBindAttribLocation(prog, 1, "tcoord"); + + glLinkProgram(prog); + glGetProgramiv(prog, GL_LINK_STATUS, &status); + if (status != GL_TRUE) { + glnvg__dumpProgramError(prog, name); + return false; + } + + shader.prog = prog; + shader.vert = vert; + shader.frag = frag; + + return true; +} + +void glnvg__deleteShader (GLNVGshader* shader) nothrow @trusted @nogc { + if (shader.prog != 0) glDeleteProgram(shader.prog); + if (shader.vert != 0) glDeleteShader(shader.vert); + if (shader.frag != 0) glDeleteShader(shader.frag); +} + +void glnvg__getUniforms (GLNVGshader* shader) nothrow @trusted @nogc { + shader.loc[GLNVGuniformLoc.ViewSize] = glGetUniformLocation(shader.prog, "viewSize"); + shader.loc[GLNVGuniformLoc.Tex] = glGetUniformLocation(shader.prog, "tex"); + shader.loc[GLNVGuniformLoc.Frag] = glGetUniformLocation(shader.prog, "frag"); + shader.loc[GLNVGuniformLoc.TMat] = glGetUniformLocation(shader.prog, "tmat"); + shader.loc[GLNVGuniformLoc.TTr] = glGetUniformLocation(shader.prog, "ttr"); +} + +bool glnvg__renderCreate (void* uptr) nothrow @trusted @nogc { + GLNVGcontext* gl = cast(GLNVGcontext*)uptr; + enum align_ = 4; + + enum shaderHeader = "#define UNIFORM_ARRAY_SIZE 11\n"; + + enum fillVertShader = q{ + uniform vec2 viewSize; + attribute vec2 vertex; + attribute vec2 tcoord; + varying vec2 ftcoord; + varying vec2 fpos; + uniform vec4 tmat; /* abcd of affine matrix: xyzw */ + uniform vec2 ttr; /* tx and ty of affine matrix */ + void main (void) { + /* affine transformation */ + float nx = vertex.x*tmat.x+vertex.y*tmat.z+ttr.x; + float ny = vertex.x*tmat.y+vertex.y*tmat.w+ttr.y; + ftcoord = tcoord; + fpos = vec2(nx, ny); + gl_Position = vec4(2.0*nx/viewSize.x-1.0, 1.0-2.0*ny/viewSize.y, 0, 1); + } + }; + + enum fillFragShader = q{ + uniform vec4 frag[UNIFORM_ARRAY_SIZE]; + uniform sampler2D tex; + varying vec2 ftcoord; + varying vec2 fpos; + #define scissorMat mat3(frag[0].xyz, frag[1].xyz, frag[2].xyz) + #define paintMat mat3(frag[3].xyz, frag[4].xyz, frag[5].xyz) + #define innerCol frag[6] + #define outerCol frag[7] + #define scissorExt frag[8].xy + #define scissorScale frag[8].zw + #define extent frag[9].xy + #define radius frag[9].z + #define feather frag[9].w + #define strokeMult frag[10].x + #define strokeThr frag[10].y + #define texType int(frag[10].z) + #define type int(frag[10].w) + + float sdroundrect (in vec2 pt, in vec2 ext, in float rad) { + vec2 ext2 = ext-vec2(rad, rad); + vec2 d = abs(pt)-ext2; + return min(max(d.x, d.y), 0.0)+length(max(d, 0.0))-rad; + } + + // Scissoring + float scissorMask (in vec2 p) { + vec2 sc = (abs((scissorMat*vec3(p, 1.0)).xy)-scissorExt); + sc = vec2(0.5, 0.5)-sc*scissorScale; + return clamp(sc.x, 0.0, 1.0)*clamp(sc.y, 0.0, 1.0); + } + + #ifdef EDGE_AA + // Stroke - from [0..1] to clipped pyramid, where the slope is 1px. + float strokeMask () { + return min(1.0, (1.0-abs(ftcoord.x*2.0-1.0))*strokeMult)*min(1.0, ftcoord.y); + } + #endif + + void main (void) { + vec4 result; + float scissor = scissorMask(fpos); + #ifdef EDGE_AA + float strokeAlpha = strokeMask(); + if (strokeAlpha < strokeThr) discard; + #else + float strokeAlpha = 1.0; + #endif + if (type == 0) { /* NSVG_SHADER_FILLGRAD */ + // Gradient + // Calculate gradient color using box gradient + vec2 pt = (paintMat*vec3(fpos, 1.0)).xy; + float d = clamp((sdroundrect(pt, extent, radius)+feather*0.5)/feather, 0.0, 1.0); + vec4 color = mix(innerCol, outerCol, d); + // Combine alpha + color *= strokeAlpha*scissor; + result = color; + } else if (type == 1) { /* NSVG_SHADER_FILLIMG */ + // Image + // Calculate color from texture + vec2 pt = (paintMat*vec3(fpos, 1.0)).xy/extent; + vec4 color = texture2D(tex, pt); + if (texType == 1) color = vec4(color.xyz*color.w, color.w); + if (texType == 2) color = vec4(color.x); + // Apply color tint and alpha + color *= innerCol; + // Combine alpha + color *= strokeAlpha*scissor; + result = color; + } else if (type == 2) { /* NSVG_SHADER_SIMPLE */ + // Stencil fill + result = vec4(1, 1, 1, 1); + } else if (type == 3) { /* NSVG_SHADER_IMG */ + // Textured tris + vec4 color = texture2D(tex, ftcoord); + if (texType == 1) color = vec4(color.xyz*color.w, color.w); + if (texType == 2) color = vec4(color.x); + color *= scissor; + result = color*innerCol; // Apply color tint + } + gl_FragColor = result; + } + }; + + glnvg__checkError(gl, "init"); + + if (gl.flags&NVGContextFlag.Antialias) { + if (!glnvg__createShader(&gl.shader, "shader", shaderHeader, "#define EDGE_AA 1\n", fillVertShader, fillFragShader)) return false; + } else { + if (!glnvg__createShader(&gl.shader, "shader", shaderHeader, null, fillVertShader, fillFragShader)) return false; + } + + glnvg__checkError(gl, "uniform locations"); + glnvg__getUniforms(&gl.shader); + + // Create dynamic vertex array + glGenBuffers(1, &gl.vertBuf); + + gl.fragSize = GLNVGfragUniforms.sizeof+align_-GLNVGfragUniforms.sizeof%align_; + + glnvg__checkError(gl, "create done"); + + glFinish(); + + return true; +} + +int glnvg__renderCreateTexture (void* uptr, NVGtexture type, int w, int h, int imageFlags, const(ubyte)* data) nothrow @trusted @nogc { + GLNVGcontext* gl = cast(GLNVGcontext*)uptr; + GLNVGtexture* tex = glnvg__allocTexture(gl); + + if (tex is null) return 0; + + glGenTextures(1, &tex.tex); + tex.width = w; + tex.height = h; + tex.type = type; + tex.flags = imageFlags; + glnvg__bindTexture(gl, tex.tex); + + version(nanovega_debug_textures) {{ import core.stdc.stdio; printf("created texture with id %d; glid=%u\n", tex.id, tex.tex); }} + + glPixelStorei(GL_UNPACK_ALIGNMENT, 1); + glPixelStorei(GL_UNPACK_ROW_LENGTH, tex.width); + glPixelStorei(GL_UNPACK_SKIP_PIXELS, 0); + glPixelStorei(GL_UNPACK_SKIP_ROWS, 0); + + // GL 1.4 and later has support for generating mipmaps using a tex parameter. + if ((imageFlags&(NVGImageFlags.GenerateMipmaps|NVGImageFlags.NoFiltering)) == NVGImageFlags.GenerateMipmaps) glTexParameteri(GL_TEXTURE_2D, GL_GENERATE_MIPMAP, GL_TRUE); + + if (type == NVGtexture.RGBA) { + glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, w, h, 0, GL_RGBA, GL_UNSIGNED_BYTE, data); + } else { + glTexImage2D(GL_TEXTURE_2D, 0, GL_RED, w, h, 0, GL_RED, GL_UNSIGNED_BYTE, data); + } + + if ((imageFlags&(NVGImageFlags.GenerateMipmaps|NVGImageFlags.NoFiltering)) == NVGImageFlags.GenerateMipmaps) { + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR_MIPMAP_LINEAR); + } else { + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, (imageFlags&NVGImageFlags.NoFiltering ? GL_NEAREST : GL_LINEAR)); + } + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, (imageFlags&NVGImageFlags.NoFiltering ? GL_NEAREST : GL_LINEAR)); + + if (imageFlags&NVGImageFlags.RepeatX) { + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT); + } else { + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE); + } + + if (imageFlags&NVGImageFlags.RepeatY) { + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT); + } else { + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE); + } + + glPixelStorei(GL_UNPACK_ALIGNMENT, 4); + glPixelStorei(GL_UNPACK_ROW_LENGTH, 0); + glPixelStorei(GL_UNPACK_SKIP_PIXELS, 0); + glPixelStorei(GL_UNPACK_SKIP_ROWS, 0); + + glnvg__checkError(gl, "create tex"); + glnvg__bindTexture(gl, 0); + + return tex.id; +} + +bool glnvg__renderDeleteTexture (void* uptr, int image) nothrow @trusted @nogc { + GLNVGcontext* gl = cast(GLNVGcontext*)uptr; + return glnvg__deleteTexture(gl, image); +} + +bool glnvg__renderTextureIncRef (void* uptr, int image) nothrow @trusted @nogc { + GLNVGcontext* gl = cast(GLNVGcontext*)uptr; + GLNVGtexture* tex = glnvg__findTexture(gl, image); + if (tex is null) { + version(nanovega_debug_textures) {{ import core.stdc.stdio; printf("CANNOT incref texture with id %d\n", image); }} + return false; + } + ++tex.rc; + version(nanovega_debug_textures) {{ import core.stdc.stdio; printf("texture #%d: incref; newref=%d\n", image, tex.rc); }} + return true; +} + +bool glnvg__renderUpdateTexture (void* uptr, int image, int x, int y, int w, int h, const(ubyte)* data) nothrow @trusted @nogc { + GLNVGcontext* gl = cast(GLNVGcontext*)uptr; + GLNVGtexture* tex = glnvg__findTexture(gl, image); + + if (tex is null) { + version(nanovega_debug_textures) {{ import core.stdc.stdio; printf("CANNOT update texture with id %d\n", image); }} + return false; + } + + version(nanovega_debug_textures) {{ import core.stdc.stdio; printf("updated texture with id %d; glid=%u\n", tex.id, image, tex.tex); }} + + glnvg__bindTexture(gl, tex.tex); + + glPixelStorei(GL_UNPACK_ALIGNMENT, 1); + glPixelStorei(GL_UNPACK_ROW_LENGTH, tex.width); + glPixelStorei(GL_UNPACK_SKIP_PIXELS, x); + glPixelStorei(GL_UNPACK_SKIP_ROWS, y); + + if (tex.type == NVGtexture.RGBA) { + glTexSubImage2D(GL_TEXTURE_2D, 0, x, y, w, h, GL_RGBA, GL_UNSIGNED_BYTE, data); + } else { + glTexSubImage2D(GL_TEXTURE_2D, 0, x, y, w, h, GL_RED, GL_UNSIGNED_BYTE, data); + } + + glPixelStorei(GL_UNPACK_ALIGNMENT, 4); + glPixelStorei(GL_UNPACK_ROW_LENGTH, 0); + glPixelStorei(GL_UNPACK_SKIP_PIXELS, 0); + glPixelStorei(GL_UNPACK_SKIP_ROWS, 0); + + glnvg__bindTexture(gl, 0); + + return true; +} + +bool glnvg__renderGetTextureSize (void* uptr, int image, int* w, int* h) nothrow @trusted @nogc { + GLNVGcontext* gl = cast(GLNVGcontext*)uptr; + GLNVGtexture* tex = glnvg__findTexture(gl, image); + if (tex is null) { + if (w !is null) *w = 0; + if (h !is null) *h = 0; + return false; + } else { + if (w !is null) *w = tex.width; + if (h !is null) *h = tex.height; + return true; + } +} + +void glnvg__xformToMat3x4 (float[] m3, const(float)[] t) nothrow @trusted @nogc { + assert(t.length >= 6); + assert(m3.length >= 12); + m3.ptr[0] = t.ptr[0]; + m3.ptr[1] = t.ptr[1]; + m3.ptr[2] = 0.0f; + m3.ptr[3] = 0.0f; + m3.ptr[4] = t.ptr[2]; + m3.ptr[5] = t.ptr[3]; + m3.ptr[6] = 0.0f; + m3.ptr[7] = 0.0f; + m3.ptr[8] = t.ptr[4]; + m3.ptr[9] = t.ptr[5]; + m3.ptr[10] = 1.0f; + m3.ptr[11] = 0.0f; +} + +NVGColor glnvg__premulColor (NVGColor c) nothrow @trusted @nogc { + //pragma(inline, true); + c.r *= c.a; + c.g *= c.a; + c.b *= c.a; + return c; +} + +bool glnvg__convertPaint (GLNVGcontext* gl, GLNVGfragUniforms* frag, NVGPaint* paint, NVGscissor* scissor, float width, float fringe, float strokeThr) nothrow @trusted @nogc { + import core.stdc.math : sqrtf; + GLNVGtexture* tex = null; + NVGMatrix invxform = void; + + memset(frag, 0, (*frag).sizeof); + + frag.innerCol = glnvg__premulColor(paint.innerColor); + frag.outerCol = glnvg__premulColor(paint.outerColor); + + if (scissor.extent.ptr[0] < -0.5f || scissor.extent.ptr[1] < -0.5f) { + memset(frag.scissorMat.ptr, 0, frag.scissorMat.sizeof); + frag.scissorExt.ptr[0] = 1.0f; + frag.scissorExt.ptr[1] = 1.0f; + frag.scissorScale.ptr[0] = 1.0f; + frag.scissorScale.ptr[1] = 1.0f; + } else { + //nvgTransformInverse(invxform[], scissor.xform[]); + invxform = scissor.xform.inverted; + glnvg__xformToMat3x4(frag.scissorMat[], invxform.mat[]); + frag.scissorExt.ptr[0] = scissor.extent.ptr[0]; + frag.scissorExt.ptr[1] = scissor.extent.ptr[1]; + frag.scissorScale.ptr[0] = sqrtf(scissor.xform.mat.ptr[0]*scissor.xform.mat.ptr[0]+scissor.xform.mat.ptr[2]*scissor.xform.mat.ptr[2])/fringe; + frag.scissorScale.ptr[1] = sqrtf(scissor.xform.mat.ptr[1]*scissor.xform.mat.ptr[1]+scissor.xform.mat.ptr[3]*scissor.xform.mat.ptr[3])/fringe; + } + + memcpy(frag.extent.ptr, paint.extent.ptr, frag.extent.sizeof); + frag.strokeMult = (width*0.5f+fringe*0.5f)/fringe; + frag.strokeThr = strokeThr; + + if (paint.image != 0) { + tex = glnvg__findTexture(gl, paint.image); + if (tex is null) return false; + if ((tex.flags&NVGImageFlags.FlipY) != 0) { + /* + NVGMatrix flipped; + nvgTransformScale(flipped[], 1.0f, -1.0f); + nvgTransformMultiply(flipped[], paint.xform[]); + nvgTransformInverse(invxform[], flipped[]); + */ + /* + NVGMatrix m1 = void, m2 = void; + nvgTransformTranslate(m1[], 0.0f, frag.extent.ptr[1]*0.5f); + nvgTransformMultiply(m1[], paint.xform[]); + nvgTransformScale(m2[], 1.0f, -1.0f); + nvgTransformMultiply(m2[], m1[]); + nvgTransformTranslate(m1[], 0.0f, -frag.extent.ptr[1]*0.5f); + nvgTransformMultiply(m1[], m2[]); + nvgTransformInverse(invxform[], m1[]); + */ + NVGMatrix m1 = NVGMatrix.Translated(0.0f, frag.extent.ptr[1]*0.5f); + m1.mul(paint.xform); + NVGMatrix m2 = NVGMatrix.Scaled(1.0f, -1.0f); + m2.mul(m1); + m1 = NVGMatrix.Translated(0.0f, -frag.extent.ptr[1]*0.5f); + m1.mul(m2); + invxform = m1.inverted; + } else { + //nvgTransformInverse(invxform[], paint.xform[]); + invxform = paint.xform.inverted; + } + frag.type = NSVG_SHADER_FILLIMG; + + if (tex.type == NVGtexture.RGBA) { + frag.texType = (tex.flags&NVGImageFlags.Premultiplied ? 0 : 1); + } else { + frag.texType = 2; + } + //printf("frag.texType = %d\n", frag.texType); + } else { + frag.type = NSVG_SHADER_FILLGRAD; + frag.radius = paint.radius; + frag.feather = paint.feather; + //nvgTransformInverse(invxform[], paint.xform[]); + invxform = paint.xform.inverted; + } + + glnvg__xformToMat3x4(frag.paintMat[], invxform.mat[]); + + return true; +} + +void glnvg__setUniforms (GLNVGcontext* gl, int uniformOffset, int image) nothrow @trusted @nogc { + GLNVGfragUniforms* frag = nvg__fragUniformPtr(gl, uniformOffset); + glUniform4fv(gl.shader.loc[GLNVGuniformLoc.Frag], frag.UNIFORM_ARRAY_SIZE, &(frag.uniformArray.ptr[0].ptr[0])); + if (image != 0) { + GLNVGtexture* tex = glnvg__findTexture(gl, image); + glnvg__bindTexture(gl, tex !is null ? tex.tex : 0); + glnvg__checkError(gl, "tex paint tex"); + } else { + glnvg__bindTexture(gl, 0); + } +} + +void glnvg__renderViewport (void* uptr, int width, int height) nothrow @trusted @nogc { + GLNVGcontext* gl = cast(GLNVGcontext*)uptr; + gl.view.ptr[0] = cast(float)width; + gl.view.ptr[1] = cast(float)height; +} + +void glnvg__fill (GLNVGcontext* gl, GLNVGcall* call) nothrow @trusted @nogc { + GLNVGpath* paths = &gl.paths[call.pathOffset]; + int npaths = call.pathCount; + + // Draw shapes + glEnable(GL_STENCIL_TEST); + glnvg__stencilMask(gl, 0xffffffffU); + glnvg__stencilFunc(gl, GL_ALWAYS, 0, 0xffffffffU); + glColorMask(GL_FALSE, GL_FALSE, GL_FALSE, GL_FALSE); + + // set bindpoint for solid loc + glnvg__setUniforms(gl, call.uniformOffset, 0); + glnvg__checkError(gl, "fill simple"); + + if (call.evenOdd) { + glStencilOpSeparate(GL_FRONT, GL_KEEP, GL_KEEP, GL_INVERT); + glStencilOpSeparate(GL_BACK, GL_KEEP, GL_KEEP, GL_INVERT); + } else { + glStencilOpSeparate(GL_FRONT, GL_KEEP, GL_KEEP, GL_INCR_WRAP); + glStencilOpSeparate(GL_BACK, GL_KEEP, GL_KEEP, GL_DECR_WRAP); + } + glDisable(GL_CULL_FACE); + foreach (int i; 0..npaths) glDrawArrays(GL_TRIANGLE_FAN, paths[i].fillOffset, paths[i].fillCount); + glEnable(GL_CULL_FACE); + + // Draw anti-aliased pixels + glColorMask(GL_TRUE, GL_TRUE, GL_TRUE, GL_TRUE); + + glnvg__setUniforms(gl, call.uniformOffset+gl.fragSize, call.image); + glnvg__checkError(gl, "fill fill"); + + if (gl.flags&NVGContextFlag.Antialias) { + glnvg__stencilFunc(gl, GL_EQUAL, 0x00, 0xffffffffU); + glStencilOp(GL_KEEP, GL_KEEP, GL_KEEP); + // Draw fringes + foreach (int i; 0..npaths) glDrawArrays(GL_TRIANGLE_STRIP, paths[i].strokeOffset, paths[i].strokeCount); + } + + // Draw fill + glnvg__stencilFunc(gl, GL_NOTEQUAL, 0x0, 0xffffffffU); + glStencilOp(GL_ZERO, GL_ZERO, GL_ZERO); + if (call.evenOdd) { + glDisable(GL_CULL_FACE); + glDrawArrays(GL_TRIANGLE_STRIP, call.triangleOffset, call.triangleCount); + //foreach (int i; 0..npaths) glDrawArrays(GL_TRIANGLE_FAN, paths[i].fillOffset, paths[i].fillCount); + glEnable(GL_CULL_FACE); + } else { + glDrawArrays(GL_TRIANGLE_STRIP, call.triangleOffset, call.triangleCount); + } + + glDisable(GL_STENCIL_TEST); +} + +void glnvg__convexFill (GLNVGcontext* gl, GLNVGcall* call) nothrow @trusted @nogc { + GLNVGpath* paths = &gl.paths[call.pathOffset]; + int npaths = call.pathCount; + + glnvg__setUniforms(gl, call.uniformOffset, call.image); + glnvg__checkError(gl, "convex fill"); + + foreach (int i; 0..npaths) glDrawArrays(GL_TRIANGLE_FAN, paths[i].fillOffset, paths[i].fillCount); + if (gl.flags&NVGContextFlag.Antialias) { + // Draw fringes + foreach (int i; 0..npaths) glDrawArrays(GL_TRIANGLE_STRIP, paths[i].strokeOffset, paths[i].strokeCount); + } +} + +void glnvg__stroke (GLNVGcontext* gl, GLNVGcall* call) nothrow @trusted @nogc { + GLNVGpath* paths = &gl.paths[call.pathOffset]; + int npaths = call.pathCount; + + if (gl.flags&NVGContextFlag.StencilStrokes) { + glEnable(GL_STENCIL_TEST); + glnvg__stencilMask(gl, 0xff); + + // Fill the stroke base without overlap + glnvg__stencilFunc(gl, GL_EQUAL, 0x0, 0xff); + glStencilOp(GL_KEEP, GL_KEEP, GL_INCR); + glnvg__setUniforms(gl, call.uniformOffset+gl.fragSize, call.image); + glnvg__checkError(gl, "stroke fill 0"); + foreach (int i; 0..npaths) glDrawArrays(GL_TRIANGLE_STRIP, paths[i].strokeOffset, paths[i].strokeCount); + + // Draw anti-aliased pixels. + glnvg__setUniforms(gl, call.uniformOffset, call.image); + glnvg__stencilFunc(gl, GL_EQUAL, 0x00, 0xff); + glStencilOp(GL_KEEP, GL_KEEP, GL_KEEP); + foreach (int i; 0..npaths) glDrawArrays(GL_TRIANGLE_STRIP, paths[i].strokeOffset, paths[i].strokeCount); + + // Clear stencil buffer. + glColorMask(GL_FALSE, GL_FALSE, GL_FALSE, GL_FALSE); + glnvg__stencilFunc(gl, GL_ALWAYS, 0x0, 0xff); + glStencilOp(GL_ZERO, GL_ZERO, GL_ZERO); + glnvg__checkError(gl, "stroke fill 1"); + foreach (int i; 0..npaths) glDrawArrays(GL_TRIANGLE_STRIP, paths[i].strokeOffset, paths[i].strokeCount); + glColorMask(GL_TRUE, GL_TRUE, GL_TRUE, GL_TRUE); + + glDisable(GL_STENCIL_TEST); + + //glnvg__convertPaint(gl, nvg__fragUniformPtr(gl, call.uniformOffset+gl.fragSize), paint, scissor, strokeWidth, fringe, 1.0f-0.5f/255.0f); + } else { + glnvg__setUniforms(gl, call.uniformOffset, call.image); + glnvg__checkError(gl, "stroke fill"); + // Draw Strokes + foreach (int i; 0..npaths) glDrawArrays(GL_TRIANGLE_STRIP, paths[i].strokeOffset, paths[i].strokeCount); + } +} + +void glnvg__triangles (GLNVGcontext* gl, GLNVGcall* call) nothrow @trusted @nogc { + glnvg__setUniforms(gl, call.uniformOffset, call.image); + glnvg__checkError(gl, "triangles fill"); + glDrawArrays(GL_TRIANGLES, call.triangleOffset, call.triangleCount); +} + +void glnvg__affine (GLNVGcontext* gl, GLNVGcall* call) nothrow @trusted @nogc { + glUniform4fv(gl.shader.loc[GLNVGuniformLoc.TMat], 1, call.affine.mat.ptr); + glnvg__checkError(gl, "affine"); + glUniform2fv(gl.shader.loc[GLNVGuniformLoc.TTr], 1, call.affine.mat.ptr+4); + glnvg__checkError(gl, "affine"); + glnvg__setUniforms(gl, call.uniformOffset, call.image); +} + +void glnvg__renderCancel (void* uptr) nothrow @trusted @nogc { + GLNVGcontext* gl = cast(GLNVGcontext*)uptr; + gl.nverts = 0; + gl.npaths = 0; + gl.ncalls = 0; + gl.nuniforms = 0; +} + +GLenum glnvg_convertBlendFuncFactor (NVGBlendFactor factor) nothrow @trusted @nogc { + if (factor == NVGBlendFactor.Zero) return GL_ZERO; + if (factor == NVGBlendFactor.One) return GL_ONE; + if (factor == NVGBlendFactor.SrcColor) return GL_SRC_COLOR; + if (factor == NVGBlendFactor.OneMinusSrcColor) return GL_ONE_MINUS_SRC_COLOR; + if (factor == NVGBlendFactor.DstColor) return GL_DST_COLOR; + if (factor == NVGBlendFactor.OneMinusDstColor) return GL_ONE_MINUS_DST_COLOR; + if (factor == NVGBlendFactor.SrcAlpha) return GL_SRC_ALPHA; + if (factor == NVGBlendFactor.OneMinusSrcAlpha) return GL_ONE_MINUS_SRC_ALPHA; + if (factor == NVGBlendFactor.DstAlpha) return GL_DST_ALPHA; + if (factor == NVGBlendFactor.OneMinusDstAlpha) return GL_ONE_MINUS_DST_ALPHA; + if (factor == NVGBlendFactor.SrcAlphaSaturate) return GL_SRC_ALPHA_SATURATE; + return GL_INVALID_ENUM; +} + +void glnvg__blendCompositeOperation (NVGCompositeOperationState op) nothrow @trusted @nogc { + //glBlendFuncSeparate(glnvg_convertBlendFuncFactor(op.srcRGB), glnvg_convertBlendFuncFactor(op.dstRGB), glnvg_convertBlendFuncFactor(op.srcAlpha), glnvg_convertBlendFuncFactor(op.dstAlpha)); + GLenum srcRGB = glnvg_convertBlendFuncFactor(op.srcRGB); + GLenum dstRGB = glnvg_convertBlendFuncFactor(op.dstRGB); + GLenum srcAlpha = glnvg_convertBlendFuncFactor(op.srcAlpha); + GLenum dstAlpha = glnvg_convertBlendFuncFactor(op.dstAlpha); + if (srcRGB == GL_INVALID_ENUM || dstRGB == GL_INVALID_ENUM || srcAlpha == GL_INVALID_ENUM || dstAlpha == GL_INVALID_ENUM) { + glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA); + } else { + glBlendFuncSeparate(srcRGB, dstRGB, srcAlpha, dstAlpha); + } +} + +void glnvg__renderSetAffine (void* uptr, const(float)[] mat...) nothrow @trusted @nogc { + assert(mat.length == 4 || mat.length == 6); + GLNVGcontext* gl = cast(GLNVGcontext*)uptr; + GLNVGcall* call; + // if last operation was GLNVG_AFFINE, simply replace the matrix + if (gl.ncalls > 0 && gl.calls[gl.ncalls-1].type == GLNVG_AFFINE) { + call = &gl.calls[gl.ncalls-1]; + } else { + call = glnvg__allocCall(gl); + if (call is null) return; + call.type = GLNVG_AFFINE; + } + if (mat.length == 4) { + call.affine.mat.ptr[0..4] = mat.ptr[0..4]; + call.affine.mat.ptr[4..6] = 0.0f; + } else if (mat.length >= 6) { + call.affine.mat.ptr[0..6] = mat.ptr[0..6]; + } +} + +void glnvg__renderFlush (void* uptr, NVGCompositeOperationState compositeOperation) nothrow @trusted @nogc { + GLNVGcontext* gl = cast(GLNVGcontext*)uptr; + if (gl.ncalls > 0) { + // Setup require GL state. + glUseProgram(gl.shader.prog); + + //glBlendFunc(GL_ONE, GL_ONE_MINUS_SRC_ALPHA); + glnvg__blendCompositeOperation(compositeOperation); + glEnable(GL_CULL_FACE); + glCullFace(GL_BACK); + glFrontFace(GL_CCW); + glEnable(GL_BLEND); + glDisable(GL_DEPTH_TEST); + glDisable(GL_SCISSOR_TEST); + glColorMask(GL_TRUE, GL_TRUE, GL_TRUE, GL_TRUE); + glStencilMask(0xffffffff); + glStencilOp(GL_KEEP, GL_KEEP, GL_KEEP); + glStencilFunc(GL_ALWAYS, 0, 0xffffffff); + glActiveTexture(GL_TEXTURE0); + glBindTexture(GL_TEXTURE_2D, 0); + static if (NANOVG_GL_USE_STATE_FILTER) { + gl.boundTexture = 0; + gl.stencilMask = 0xffffffff; + gl.stencilFunc = GL_ALWAYS; + gl.stencilFuncRef = 0; + gl.stencilFuncMask = 0xffffffff; + } + + // Upload vertex data + glBindBuffer(GL_ARRAY_BUFFER, gl.vertBuf); + glBufferData(GL_ARRAY_BUFFER, gl.nverts*NVGvertex.sizeof, gl.verts, GL_STREAM_DRAW); + glEnableVertexAttribArray(0); + glEnableVertexAttribArray(1); + glVertexAttribPointer(0, 2, GL_FLOAT, GL_FALSE, NVGvertex.sizeof, cast(const(GLvoid)*)cast(usize)0); + glVertexAttribPointer(1, 2, GL_FLOAT, GL_FALSE, NVGvertex.sizeof, cast(const(GLvoid)*)(0+2*float.sizeof)); + + // Set view and texture just once per frame. + glUniform1i(gl.shader.loc[GLNVGuniformLoc.Tex], 0); + glUniform2fv(gl.shader.loc[GLNVGuniformLoc.ViewSize], 1, gl.view.ptr); + // Reset affine transformations. + glUniform4fv(gl.shader.loc[GLNVGuniformLoc.TMat], 1, NVGMatrix.IdentityMat.ptr); + glUniform2fv(gl.shader.loc[GLNVGuniformLoc.TTr], 1, NVGMatrix.IdentityMat.ptr+4); + + foreach (int i; 0..gl.ncalls) { + GLNVGcall* call = &gl.calls[i]; + switch (call.type) { + case GLNVG_FILL: glnvg__fill(gl, call); break; + case GLNVG_CONVEXFILL: glnvg__convexFill(gl, call); break; + case GLNVG_STROKE: glnvg__stroke(gl, call); break; + case GLNVG_TRIANGLES: glnvg__triangles(gl, call); break; + case GLNVG_AFFINE: glnvg__affine(gl, call); break; + default: break; + } + } + + glDisableVertexAttribArray(0); + glDisableVertexAttribArray(1); + glDisable(GL_CULL_FACE); + glBindBuffer(GL_ARRAY_BUFFER, 0); + glUseProgram(0); + glnvg__bindTexture(gl, 0); + } + + // Reset calls + gl.nverts = 0; + gl.npaths = 0; + gl.ncalls = 0; + gl.nuniforms = 0; +} + +int glnvg__maxVertCount (const(NVGpath)* paths, int npaths) nothrow @trusted @nogc { + int count = 0; + foreach (int i; 0..npaths) { + count += paths[i].nfill; + count += paths[i].nstroke; + } + return count; +} + +GLNVGcall* glnvg__allocCall (GLNVGcontext* gl) nothrow @trusted @nogc { + GLNVGcall* ret = null; + if (gl.ncalls+1 > gl.ccalls) { + GLNVGcall* calls; + int ccalls = glnvg__maxi(gl.ncalls+1, 128)+gl.ccalls/2; // 1.5x Overallocate + calls = cast(GLNVGcall*)realloc(gl.calls, GLNVGcall.sizeof*ccalls); + if (calls is null) return null; + gl.calls = calls; + gl.ccalls = ccalls; + } + ret = &gl.calls[gl.ncalls++]; + memset(ret, 0, GLNVGcall.sizeof); + return ret; +} + +int glnvg__allocPaths (GLNVGcontext* gl, int n) nothrow @trusted @nogc { + int ret = 0; + if (gl.npaths+n > gl.cpaths) { + GLNVGpath* paths; + int cpaths = glnvg__maxi(gl.npaths+n, 128)+gl.cpaths/2; // 1.5x Overallocate + paths = cast(GLNVGpath*)realloc(gl.paths, GLNVGpath.sizeof*cpaths); + if (paths is null) return -1; + gl.paths = paths; + gl.cpaths = cpaths; + } + ret = gl.npaths; + gl.npaths += n; + return ret; +} + +int glnvg__allocVerts (GLNVGcontext* gl, int n) nothrow @trusted @nogc { + int ret = 0; + if (gl.nverts+n > gl.cverts) { + NVGvertex* verts; + int cverts = glnvg__maxi(gl.nverts+n, 4096)+gl.cverts/2; // 1.5x Overallocate + verts = cast(NVGvertex*)realloc(gl.verts, NVGvertex.sizeof*cverts); + if (verts is null) return -1; + gl.verts = verts; + gl.cverts = cverts; + } + ret = gl.nverts; + gl.nverts += n; + return ret; +} + +int glnvg__allocFragUniforms (GLNVGcontext* gl, int n) nothrow @trusted @nogc { + int ret = 0, structSize = gl.fragSize; + if (gl.nuniforms+n > gl.cuniforms) { + ubyte* uniforms; + int cuniforms = glnvg__maxi(gl.nuniforms+n, 128)+gl.cuniforms/2; // 1.5x Overallocate + uniforms = cast(ubyte*)realloc(gl.uniforms, structSize*cuniforms); + if (uniforms is null) return -1; + gl.uniforms = uniforms; + gl.cuniforms = cuniforms; + } + ret = gl.nuniforms*structSize; + gl.nuniforms += n; + return ret; +} + +GLNVGfragUniforms* nvg__fragUniformPtr (GLNVGcontext* gl, int i) nothrow @trusted @nogc { + return cast(GLNVGfragUniforms*)&gl.uniforms[i]; +} + +void glnvg__vset (NVGvertex* vtx, float x, float y, float u, float v) nothrow @trusted @nogc { + vtx.x = x; + vtx.y = y; + vtx.u = u; + vtx.v = v; +} + +void glnvg__renderFill (void* uptr, NVGPaint* paint, NVGscissor* scissor, float fringe, const(float)* bounds, const(NVGpath)* paths, int npaths, bool evenOdd) nothrow @trusted @nogc { + GLNVGcontext* gl = cast(GLNVGcontext*)uptr; + GLNVGcall* call = glnvg__allocCall(gl); + NVGvertex* quad; + GLNVGfragUniforms* frag; + int maxverts, offset; + + if (call is null || npaths < 1) return; + + call.type = GLNVG_FILL; + call.evenOdd = evenOdd; + call.triangleCount = 4; + call.pathOffset = glnvg__allocPaths(gl, npaths); + if (call.pathOffset == -1) goto error; + call.pathCount = npaths; + call.image = paint.image; + + if (npaths == 1 && paths[0].convex) { + call.type = GLNVG_CONVEXFILL; + call.triangleCount = 0; // Bounding box fill quad not needed for convex fill + } + + // Allocate vertices for all the paths. + maxverts = glnvg__maxVertCount(paths, npaths)+call.triangleCount; + offset = glnvg__allocVerts(gl, maxverts); + if (offset == -1) goto error; + + foreach (int i; 0..npaths) { + GLNVGpath* copy = &gl.paths[call.pathOffset+i]; + const(NVGpath)* path = &paths[i]; + memset(copy, 0, GLNVGpath.sizeof); + if (path.nfill > 0) { + copy.fillOffset = offset; + copy.fillCount = path.nfill; + memcpy(&gl.verts[offset], path.fill, NVGvertex.sizeof*path.nfill); + offset += path.nfill; + } + if (path.nstroke > 0) { + copy.strokeOffset = offset; + copy.strokeCount = path.nstroke; + memcpy(&gl.verts[offset], path.stroke, NVGvertex.sizeof*path.nstroke); + offset += path.nstroke; + } + } + + // Setup uniforms for draw calls + if (call.type == GLNVG_FILL) { + // Quad + call.triangleOffset = offset; + quad = &gl.verts[call.triangleOffset]; + glnvg__vset(&quad[0], bounds[2], bounds[3], 0.5f, 1.0f); + glnvg__vset(&quad[1], bounds[2], bounds[1], 0.5f, 1.0f); + glnvg__vset(&quad[2], bounds[0], bounds[3], 0.5f, 1.0f); + glnvg__vset(&quad[3], bounds[0], bounds[1], 0.5f, 1.0f); + // Get uniform + call.uniformOffset = glnvg__allocFragUniforms(gl, 2); + if (call.uniformOffset == -1) goto error; + // Simple shader for stencil + frag = nvg__fragUniformPtr(gl, call.uniformOffset); + memset(frag, 0, (*frag).sizeof); + frag.strokeThr = -1.0f; + frag.type = NSVG_SHADER_SIMPLE; + // Fill shader + glnvg__convertPaint(gl, nvg__fragUniformPtr(gl, call.uniformOffset+gl.fragSize), paint, scissor, fringe, fringe, -1.0f); + } else { + call.uniformOffset = glnvg__allocFragUniforms(gl, 1); + if (call.uniformOffset == -1) goto error; + // Fill shader + glnvg__convertPaint(gl, nvg__fragUniformPtr(gl, call.uniformOffset), paint, scissor, fringe, fringe, -1.0f); + } + + return; + +error: + // We get here if call alloc was ok, but something else is not. + // Roll back the last call to prevent drawing it. + if (gl.ncalls > 0) --gl.ncalls; +} + +void glnvg__renderStroke (void* uptr, NVGPaint* paint, NVGscissor* scissor, float fringe, float strokeWidth, const(NVGpath)* paths, int npaths) nothrow @trusted @nogc { + GLNVGcontext* gl = cast(GLNVGcontext*)uptr; + GLNVGcall* call = glnvg__allocCall(gl); + int maxverts, offset; + + if (call is null || npaths < 1) return; + + call.type = GLNVG_STROKE; + call.pathOffset = glnvg__allocPaths(gl, npaths); + if (call.pathOffset == -1) goto error; + call.pathCount = npaths; + call.image = paint.image; + + // Allocate vertices for all the paths. + maxverts = glnvg__maxVertCount(paths, npaths); + offset = glnvg__allocVerts(gl, maxverts); + if (offset == -1) goto error; + + foreach (int i; 0..npaths) { + GLNVGpath* copy = &gl.paths[call.pathOffset+i]; + const(NVGpath)* path = &paths[i]; + memset(copy, 0, GLNVGpath.sizeof); + if (path.nstroke) { + copy.strokeOffset = offset; + copy.strokeCount = path.nstroke; + memcpy(&gl.verts[offset], path.stroke, NVGvertex.sizeof*path.nstroke); + offset += path.nstroke; + } + } + + if (gl.flags&NVGContextFlag.StencilStrokes) { + // Fill shader + call.uniformOffset = glnvg__allocFragUniforms(gl, 2); + if (call.uniformOffset == -1) goto error; + glnvg__convertPaint(gl, nvg__fragUniformPtr(gl, call.uniformOffset), paint, scissor, strokeWidth, fringe, -1.0f); + glnvg__convertPaint(gl, nvg__fragUniformPtr(gl, call.uniformOffset+gl.fragSize), paint, scissor, strokeWidth, fringe, 1.0f-0.5f/255.0f); + } else { + // Fill shader + call.uniformOffset = glnvg__allocFragUniforms(gl, 1); + if (call.uniformOffset == -1) goto error; + glnvg__convertPaint(gl, nvg__fragUniformPtr(gl, call.uniformOffset), paint, scissor, strokeWidth, fringe, -1.0f); + } + + return; + +error: + // We get here if call alloc was ok, but something else is not. + // Roll back the last call to prevent drawing it. + if (gl.ncalls > 0) --gl.ncalls; +} + +void glnvg__renderTriangles (void* uptr, NVGPaint* paint, NVGscissor* scissor, const(NVGvertex)* verts, int nverts) nothrow @trusted @nogc { + GLNVGcontext* gl = cast(GLNVGcontext*)uptr; + GLNVGcall* call = glnvg__allocCall(gl); + GLNVGfragUniforms* frag; + + if (call is null) return; + + call.type = GLNVG_TRIANGLES; + call.image = paint.image; + + // Allocate vertices for all the paths. + call.triangleOffset = glnvg__allocVerts(gl, nverts); + if (call.triangleOffset == -1) goto error; + call.triangleCount = nverts; + + memcpy(&gl.verts[call.triangleOffset], verts, NVGvertex.sizeof*nverts); + + // Fill shader + call.uniformOffset = glnvg__allocFragUniforms(gl, 1); + if (call.uniformOffset == -1) goto error; + frag = nvg__fragUniformPtr(gl, call.uniformOffset); + glnvg__convertPaint(gl, frag, paint, scissor, 1.0f, 1.0f, -1.0f); + frag.type = NSVG_SHADER_IMG; + + return; + +error: + // We get here if call alloc was ok, but something else is not. + // Roll back the last call to prevent drawing it. + if (gl.ncalls > 0) --gl.ncalls; +} + +void glnvg__renderDelete (void* uptr) nothrow @trusted @nogc { + GLNVGcontext* gl = cast(GLNVGcontext*)uptr; + if (gl is null) return; + + glnvg__deleteShader(&gl.shader); + + if (gl.vertBuf != 0) glDeleteBuffers(1, &gl.vertBuf); + + foreach (ref GLNVGtexture tex; gl.textures[0..gl.ntextures]) { + if (tex.id != 0 && (tex.flags&NVGImageFlagsGL.NoDelete) == 0) { + assert(tex.tex != 0); + glDeleteTextures(1, &tex.tex); + } + } + free(gl.textures); + + free(gl.paths); + free(gl.verts); + free(gl.uniforms); + free(gl.calls); + + free(gl); +} + + +version(aliced) { + private enum NVGDefaultContextFlagsXX = NVGContextFlag.Antialias|NVGContextFlag.StencilStrokes|NVGContextFlag.FontNoAA; +} else { + private enum NVGDefaultContextFlagsXX = NVGContextFlag.Antialias|NVGContextFlag.StencilStrokes; +} + +/// Default flags for [nvgCreateContext]. +/// Group: context_management +public enum NVGDefaultContextFlags = NVGDefaultContextFlagsXX; + +/// Creates NanoVega contexts for OpenGL2+. +/// Flags should be combination (bitwise or) of the [NVGContextFlag] members. +/// Group: context_management +public NVGContext nvgCreateContext (int flags=NVGDefaultContextFlags) nothrow @trusted @nogc { + NVGparams params = void; + NVGContext ctx = null; + version(nanovg_builtin_opengl_bindings) nanovgInitOpenGL(); // why not? + GLNVGcontext* gl = cast(GLNVGcontext*)malloc(GLNVGcontext.sizeof); + if (gl is null) goto error; + memset(gl, 0, GLNVGcontext.sizeof); + + memset(¶ms, 0, params.sizeof); + params.renderCreate = &glnvg__renderCreate; + params.renderCreateTexture = &glnvg__renderCreateTexture; + params.renderTextureIncRef = &glnvg__renderTextureIncRef; + params.renderDeleteTexture = &glnvg__renderDeleteTexture; + params.renderUpdateTexture = &glnvg__renderUpdateTexture; + params.renderGetTextureSize = &glnvg__renderGetTextureSize; + params.renderViewport = &glnvg__renderViewport; + params.renderCancel = &glnvg__renderCancel; + params.renderFlush = &glnvg__renderFlush; + params.renderFill = &glnvg__renderFill; + params.renderStroke = &glnvg__renderStroke; + params.renderTriangles = &glnvg__renderTriangles; + params.renderSetAffine = &glnvg__renderSetAffine; + params.renderDelete = &glnvg__renderDelete; + params.userPtr = gl; + params.edgeAntiAlias = (flags&NVGContextFlag.Antialias ? true : false); + if (flags&(NVGContextFlag.FontAA|NVGContextFlag.FontNoAA)) { + params.fontAA = (flags&NVGContextFlag.FontNoAA ? NVG_INVERT_FONT_AA : !NVG_INVERT_FONT_AA); + } else { + params.fontAA = NVG_INVERT_FONT_AA; + } + + gl.flags = flags; + gl.freetexid = -1; + + ctx = createInternal(¶ms); + if (ctx is null) goto error; + + return ctx; + +error: + // 'gl' is freed by nvgDeleteInternal. + if (ctx !is null) ctx.deleteInternal(); + return null; +} + +/// Create NanoVega OpenGL image from texture id. +/// Group: images +public int glCreateImageFromHandleGL2 (NVGContext ctx, GLuint textureId, int w, int h, int imageFlags) nothrow @trusted @nogc { + GLNVGcontext* gl = cast(GLNVGcontext*)ctx.internalParams().userPtr; + GLNVGtexture* tex = glnvg__allocTexture(gl); + + if (tex is null) return 0; + + tex.type = NVGtexture.RGBA; + tex.tex = textureId; + tex.flags = imageFlags; + tex.width = w; + tex.height = h; + + return tex.id; +} + +/// Returns OpenGL texture id for NanoVega image. +/// Group: images +public GLuint glImageHandleGL2 (NVGContext ctx, int image) nothrow @trusted @nogc { + GLNVGcontext* gl = cast(GLNVGcontext*)ctx.internalParams().userPtr; + GLNVGtexture* tex = glnvg__findTexture(gl, image); + return tex.tex; +} + + +// ////////////////////////////////////////////////////////////////////////// // +private: + +static if (NanoVegaHasFontConfig) { + version(nanovg_builtin_fontconfig_bindings) { + pragma(lib, "fontconfig"); + + private extern(C) nothrow @trusted @nogc { + enum FC_FILE = "file"; /* String */ + alias FcBool = int; + alias FcChar8 = char; + struct FcConfig; + struct FcPattern; + alias FcMatchKind = int; + enum : FcMatchKind { + FcMatchPattern, + FcMatchFont, + FcMatchScan + } + alias FcResult = int; + enum : FcResult { + FcResultMatch, + FcResultNoMatch, + FcResultTypeMismatch, + FcResultNoId, + FcResultOutOfMemory + } + FcBool FcInit (); + FcBool FcConfigSubstituteWithPat (FcConfig* config, FcPattern* p, FcPattern* p_pat, FcMatchKind kind); + void FcDefaultSubstitute (FcPattern* pattern); + FcBool FcConfigSubstitute (FcConfig* config, FcPattern* p, FcMatchKind kind); + FcPattern* FcFontMatch (FcConfig* config, FcPattern* p, FcResult* result); + FcPattern* FcNameParse (const(FcChar8)* name); + void FcPatternDestroy (FcPattern* p); + FcResult FcPatternGetString (const(FcPattern)* p, const(char)* object, int n, FcChar8** s); + } + } + + __gshared bool fontconfigAvailable = false; + // initialize fontconfig + shared static this () { + if (FcInit()) { + fontconfigAvailable = true; + } else { + import core.stdc.stdio : stderr, fprintf; + stderr.fprintf("***NanoVega WARNING: cannot init fontconfig!\n"); + } + } +} + + +// ////////////////////////////////////////////////////////////////////////// // +public enum BaphometDims = 512.0f; // baphomet icon is 512x512 ([0..511]) + +private static immutable ubyte[7641] baphometPath = [ + 0x01,0x04,0x06,0x30,0x89,0x7f,0x43,0x00,0x80,0xff,0x43,0x08,0xa0,0x1d,0xc6,0x43,0x00,0x80,0xff,0x43, + 0x00,0x80,0xff,0x43,0xa2,0x1d,0xc6,0x43,0x00,0x80,0xff,0x43,0x30,0x89,0x7f,0x43,0x08,0x00,0x80,0xff, + 0x43,0x7a,0x89,0xe5,0x42,0xa0,0x1d,0xc6,0x43,0x00,0x00,0x00,0x00,0x30,0x89,0x7f,0x43,0x00,0x00,0x00, + 0x00,0x08,0x7a,0x89,0xe5,0x42,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x7a,0x89,0xe5,0x42,0x00,0x00, + 0x00,0x00,0x30,0x89,0x7f,0x43,0x08,0x00,0x00,0x00,0x00,0xa2,0x1d,0xc6,0x43,0x7a,0x89,0xe5,0x42,0x00, + 0x80,0xff,0x43,0x30,0x89,0x7f,0x43,0x00,0x80,0xff,0x43,0x09,0x06,0x30,0x89,0x7f,0x43,0x72,0x87,0xdd, + 0x43,0x08,0x16,0x68,0xb3,0x43,0x72,0x87,0xdd,0x43,0x71,0x87,0xdd,0x43,0x17,0x68,0xb3,0x43,0x71,0x87, + 0xdd,0x43,0x30,0x89,0x7f,0x43,0x08,0x71,0x87,0xdd,0x43,0xd2,0x2f,0x18,0x43,0x16,0x68,0xb3,0x43,0x35, + 0xe2,0x87,0x42,0x30,0x89,0x7f,0x43,0x35,0xe2,0x87,0x42,0x08,0xd1,0x2f,0x18,0x43,0x35,0xe2,0x87,0x42, + 0x35,0xe2,0x87,0x42,0xd2,0x2f,0x18,0x43,0x35,0xe2,0x87,0x42,0x30,0x89,0x7f,0x43,0x08,0x35,0xe2,0x87, + 0x42,0x17,0x68,0xb3,0x43,0xd1,0x2f,0x18,0x43,0x72,0x87,0xdd,0x43,0x30,0x89,0x7f,0x43,0x72,0x87,0xdd, + 0x43,0x09,0x06,0x79,0xcb,0x11,0x43,0x62,0xbf,0xd7,0x42,0x07,0xa4,0x3f,0x7f,0x43,0x0b,0x86,0xdc,0x43, + 0x07,0x6c,0xb9,0xb2,0x43,0xe8,0xd1,0xca,0x42,0x07,0x6e,0x4d,0xa0,0x42,0xa9,0x10,0x9c,0x43,0x07,0xb7, + 0x40,0xd7,0x43,0xa9,0x10,0x9c,0x43,0x07,0x79,0xcb,0x11,0x43,0x62,0xbf,0xd7,0x42,0x09,0x06,0x98,0x42, + 0x74,0x43,0xb1,0x8d,0x68,0x43,0x08,0xd7,0x24,0x79,0x43,0xba,0x83,0x6e,0x43,0xa9,0x16,0x7c,0x43,0x56, + 0xa1,0x76,0x43,0x74,0x2a,0x7d,0x43,0x44,0x73,0x80,0x43,0x08,0x55,0xd1,0x7e,0x43,0xe3,0xea,0x76,0x43, + 0xbc,0x18,0x81,0x43,0x7f,0xa8,0x6e,0x43,0x8f,0x0a,0x84,0x43,0x02,0xfc,0x68,0x43,0x09,0x06,0x92,0x29, + 0x8d,0x43,0x73,0xc3,0x67,0x43,0x08,0xa4,0xd9,0x8e,0x43,0xf2,0xa6,0x7a,0x43,0x8f,0x22,0x88,0x43,0x75, + 0x2a,0x7d,0x43,0x42,0x7f,0x82,0x43,0x08,0xc8,0x88,0x43,0x09,0x06,0xc1,0x79,0x74,0x43,0x50,0x64,0x89, + 0x43,0x08,0x68,0x2d,0x72,0x43,0xee,0x21,0x81,0x43,0xcd,0x97,0x55,0x43,0xe6,0xf1,0x7b,0x43,0x91,0xec, + 0x5d,0x43,0xa8,0xc7,0x6a,0x43,0x09,0x06,0xfa,0xa5,0x52,0x43,0x60,0x97,0x7c,0x43,0x08,0x19,0xff,0x50, + 0x43,0xe9,0x6e,0x8a,0x43,0xb0,0xbd,0x70,0x43,0x4c,0x51,0x82,0x43,0x04,0xeb,0x69,0x43,0x66,0x0f,0x8e, + 0x43,0x09,0x06,0x17,0xbf,0x71,0x43,0x2c,0x58,0x94,0x43,0x08,0x1c,0x96,0x6e,0x43,0x61,0x68,0x99,0x43, + 0x2d,0x3a,0x6e,0x43,0xc8,0x81,0x9e,0x43,0xb7,0x9b,0x72,0x43,0x61,0xa4,0xa3,0x43,0x09,0x06,0x30,0xdb, + 0x82,0x43,0xdb,0xe9,0x93,0x43,0x08,0x11,0x82,0x84,0x43,0x61,0x68,0x99,0x43,0xe8,0x4a,0x84,0x43,0x8e, + 0xa6,0x9e,0x43,0x42,0x7f,0x82,0x43,0x61,0xa4,0xa3,0x43,0x09,0x06,0xc4,0x02,0x85,0x43,0xd1,0x0b,0x92, + 0x43,0x08,0xd6,0xb2,0x86,0x43,0x34,0x1e,0x92,0x43,0x4f,0x58,0x87,0x43,0xa4,0xf1,0x92,0x43,0x03,0xd9, + 0x87,0x43,0x7b,0xc6,0x94,0x43,0x09,0x06,0x87,0x3e,0x64,0x43,0x31,0x3b,0x93,0x43,0x08,0x3b,0xbf,0x64, + 0x43,0x6f,0xf9,0x91,0x43,0x96,0x0b,0x67,0x43,0xc5,0x4a,0x91,0x43,0xcf,0xfe,0x6a,0x43,0x31,0x2f,0x91, + 0x43,0x09,0x06,0x16,0x74,0xb5,0x43,0x08,0xec,0x8e,0x43,0x08,0x1b,0x4b,0xb2,0x43,0xee,0x5d,0x8b,0x43, + 0x48,0x4d,0xad,0x43,0x12,0xa6,0x8a,0x43,0xf3,0xd7,0xa7,0x43,0x74,0xb8,0x8a,0x43,0x08,0x8c,0xb2,0xa0, + 0x43,0xcd,0xf8,0x8a,0x43,0x68,0x46,0x9b,0x43,0x79,0x8f,0x87,0x43,0x49,0xc9,0x96,0x43,0xe9,0x3e,0x82, + 0x43,0x08,0x60,0x5c,0x97,0x43,0xa1,0xde,0x8b,0x43,0x4e,0xa0,0x93,0x43,0x31,0x3b,0x93,0x43,0x9f,0xea, + 0x8d,0x43,0x27,0x8d,0x99,0x43,0x08,0x07,0xe0,0x8c,0x43,0x06,0x34,0x9b,0x43,0x38,0xe9,0x8c,0x43,0x46, + 0x0a,0x9e,0x43,0x3d,0xcc,0x8b,0x43,0xb2,0x06,0xa2,0x43,0x08,0xf1,0x40,0x8a,0x43,0xb0,0x12,0xa4,0x43, + 0x39,0xd1,0x88,0x43,0x76,0x43,0xa6,0x43,0xfa,0x06,0x88,0x43,0xa4,0x75,0xa9,0x43,0x08,0x19,0x6c,0x88, + 0x43,0x9f,0x9e,0xac,0x43,0x66,0xeb,0x87,0x43,0x44,0x76,0xb0,0x43,0x6b,0xce,0x86,0x43,0x3b,0xbc,0xb4, + 0x43,0x08,0xa9,0x8c,0x85,0x43,0x06,0xd0,0xb5,0x43,0xfa,0xee,0x83,0x43,0x74,0xa3,0xb6,0x43,0x3d,0x90, + 0x81,0x43,0x31,0xf6,0xb6,0x43,0x08,0x9d,0x61,0x7d,0x43,0xee,0x48,0xb7,0x43,0x3b,0x1f,0x75,0x43,0xcf, + 0xe3,0xb6,0x43,0xee,0x6f,0x6d,0x43,0x68,0xe2,0xb5,0x43,0x08,0xd4,0xed,0x6b,0x43,0x87,0x2f,0xb2,0x43, + 0x0e,0xc9,0x6b,0x43,0xa7,0x7c,0xae,0x43,0x98,0xfa,0x67,0x43,0xab,0x53,0xab,0x43,0x08,0x25,0x2c,0x64, + 0x43,0x33,0xa2,0xa8,0x43,0x40,0x96,0x61,0x43,0xc3,0xc2,0xa5,0x43,0x64,0xde,0x60,0x43,0xfa,0xa2,0xa2, + 0x43,0x08,0xb0,0x5d,0x60,0x43,0x06,0x4c,0x9f,0x43,0x9a,0xca,0x5f,0x43,0x38,0x3d,0x9b,0x43,0x3b,0x8f, + 0x5c,0x43,0x85,0xb0,0x98,0x43,0x08,0x42,0x36,0x51,0x43,0x3d,0xf0,0x91,0x43,0xcd,0x4f,0x49,0x43,0xdb, + 0xb9,0x8b,0x43,0xe0,0xdb,0x44,0x43,0x42,0x8b,0x84,0x43,0x08,0x7e,0xc9,0x44,0x43,0x8a,0x57,0x8d,0x43, + 0xbc,0x6c,0x0f,0x43,0x23,0x62,0x8e,0x43,0xf5,0x17,0x07,0x43,0xc5,0x3e,0x8f,0x43,0x09,0x06,0xe0,0xea, + 0x76,0x43,0xab,0xef,0xc5,0x43,0x08,0x12,0x00,0x79,0x43,0xab,0xcb,0xbf,0x43,0x79,0xb9,0x6d,0x43,0x7e, + 0x8d,0xba,0x43,0xee,0x6f,0x6d,0x43,0x98,0xeb,0xb5,0x43,0x08,0xe0,0x02,0x7b,0x43,0x5f,0x1c,0xb8,0x43, + 0x85,0x2c,0x82,0x43,0xe9,0x65,0xb8,0x43,0xd6,0xb2,0x86,0x43,0xc6,0x05,0xb5,0x43,0x08,0x03,0xcd,0x85, + 0x43,0x5a,0x39,0xb9,0x43,0xe4,0x4f,0x81,0x43,0xdb,0xd4,0xbf,0x43,0xdf,0x6c,0x82,0x43,0xbc,0x93,0xc5, + 0x43,0x09,0x06,0xf0,0xd0,0x22,0x43,0x5d,0x19,0x08,0x43,0x08,0xbc,0xab,0x49,0x43,0x4a,0x35,0x29,0x43, + 0xcb,0xf7,0x65,0x43,0xce,0x37,0x45,0x43,0x0e,0x99,0x63,0x43,0x67,0xc6,0x5c,0x43,0x09,0x06,0x05,0x94, + 0xab,0x43,0xc2,0x13,0x04,0x43,0x08,0x9f,0x26,0x98,0x43,0x11,0x42,0x25,0x43,0x97,0x00,0x8a,0x43,0x32, + 0x32,0x41,0x43,0xf5,0x2f,0x8b,0x43,0xc7,0xc0,0x58,0x43,0x09,0x06,0x8f,0x85,0x48,0x43,0xe0,0xa8,0x8c, + 0x43,0x08,0x55,0xaa,0x48,0x43,0xe0,0xa8,0x8c,0x43,0x6b,0x3d,0x49,0x43,0xc1,0x43,0x8c,0x43,0x31,0x62, + 0x49,0x43,0xc1,0x43,0x8c,0x43,0x08,0x2f,0xe3,0x2f,0x43,0xad,0xe7,0x98,0x43,0xff,0x0d,0x0d,0x43,0xad, + 0xf3,0x9a,0x43,0xf0,0xaf,0xcc,0x42,0x74,0x00,0x97,0x43,0x08,0xbb,0xa2,0xf7,0x42,0x93,0x4d,0x93,0x43, + 0x5e,0x19,0x08,0x43,0x5a,0x2a,0x87,0x43,0x23,0x6e,0x10,0x43,0x42,0x97,0x86,0x43,0x08,0xca,0xe8,0x33, + 0x43,0x1b,0x3c,0x80,0x43,0x80,0xe8,0x4d,0x43,0xda,0xf4,0x70,0x43,0xae,0x0e,0x4f,0x43,0x2b,0x1b,0x65, + 0x43,0x08,0x66,0x96,0x54,0x43,0xa3,0xe1,0x3b,0x43,0x4e,0xc4,0x19,0x43,0xa0,0x1a,0x16,0x43,0x10,0xe2, + 0x14,0x43,0x26,0x14,0xe0,0x42,0x08,0x5c,0x91,0x1c,0x43,0xcb,0x27,0xee,0x42,0xa9,0x40,0x24,0x43,0x71, + 0x3b,0xfc,0x42,0xf3,0xef,0x2b,0x43,0x8b,0x27,0x05,0x43,0x08,0xe2,0x4b,0x2c,0x43,0x48,0x86,0x07,0x43, + 0x79,0x62,0x2f,0x43,0x05,0xe5,0x09,0x43,0x55,0x32,0x34,0x43,0xa0,0xd2,0x09,0x43,0x08,0x74,0xa3,0x36, + 0x43,0x3a,0xd1,0x08,0x43,0x7e,0x81,0x38,0x43,0x09,0xd4,0x0a,0x43,0x0d,0xba,0x39,0x43,0xa0,0xea,0x0d, + 0x43,0x08,0x6f,0xe4,0x3d,0x43,0x43,0xc7,0x0e,0x43,0xd6,0xe5,0x3e,0x43,0xc4,0x4a,0x11,0x43,0x55,0x7a, + 0x40,0x43,0x59,0x72,0x13,0x43,0x08,0x55,0x92,0x44,0x43,0xbf,0x73,0x14,0x43,0x23,0x95,0x46,0x43,0xa5, + 0x09,0x17,0x43,0xe0,0xf3,0x48,0x43,0xfe,0x55,0x19,0x43,0x08,0xcd,0x4f,0x49,0x43,0xaa,0x10,0x1c,0x43, + 0x61,0x77,0x4b,0x43,0xfe,0x6d,0x1d,0x43,0x80,0xe8,0x4d,0x43,0x2b,0x94,0x1e,0x43,0x08,0x58,0xc9,0x51, + 0x43,0x41,0x27,0x1f,0x43,0x9b,0x82,0x53,0x43,0x35,0x72,0x20,0x43,0x53,0xf2,0x54,0x43,0x88,0xcf,0x21, + 0x43,0x08,0x7b,0x29,0x55,0x43,0xe8,0x0a,0x25,0x43,0xb2,0x2d,0x58,0x43,0xef,0xe8,0x26,0x43,0x9b,0xb2, + 0x5b,0x43,0xd0,0x8f,0x28,0x43,0x08,0x5f,0xef,0x5f,0x43,0xeb,0x11,0x2a,0x43,0xfd,0xdc,0x5f,0x43,0x6e, + 0x95,0x2c,0x43,0x3b,0xa7,0x60,0x43,0x2b,0xf4,0x2e,0x43,0x08,0x06,0xbb,0x61,0x43,0xfd,0xe5,0x31,0x43, + 0xe7,0x61,0x63,0x43,0xef,0x30,0x33,0x43,0x53,0x52,0x65,0x43,0xa3,0xb1,0x33,0x43,0x08,0x12,0xa0,0x68, + 0x43,0x7f,0x69,0x34,0x43,0x40,0xc6,0x69,0x43,0x64,0xff,0x36,0x43,0x7e,0x90,0x6a,0x43,0x71,0xcc,0x39, + 0x43,0x08,0xbc,0x5a,0x6b,0x43,0x51,0x73,0x3b,0x43,0xc1,0x49,0x6c,0x43,0xa5,0xd0,0x3c,0x43,0xe0,0xba, + 0x6e,0x43,0xb8,0x74,0x3c,0x43,0x08,0x6b,0x1c,0x73,0x43,0x13,0xc1,0x3e,0x43,0x40,0xf6,0x71,0x43,0xce, + 0x1f,0x41,0x43,0x55,0x89,0x72,0x43,0x8d,0x7e,0x43,0x43,0x08,0x68,0x2d,0x72,0x43,0x89,0xae,0x4b,0x43, + 0xc1,0x79,0x74,0x43,0xcb,0x78,0x4c,0x43,0x55,0xa1,0x76,0x43,0x5b,0xb1,0x4d,0x43,0x08,0xa2,0x38,0x7a, + 0x43,0xd1,0x56,0x4e,0x43,0x85,0xb6,0x78,0x43,0xb1,0x15,0x54,0x43,0x83,0xc7,0x77,0x43,0x89,0x0e,0x5c, + 0x43,0x08,0xcf,0x46,0x77,0x43,0x0f,0x81,0x5f,0x43,0x1a,0xde,0x7a,0x43,0xce,0xc7,0x5d,0x43,0x42,0x73, + 0x80,0x43,0x99,0xc3,0x5a,0x43,0x08,0x85,0x2c,0x82,0x43,0xf6,0xe6,0x59,0x43,0x81,0x3d,0x81,0x43,0x16, + 0x10,0x50,0x43,0xd6,0x8e,0x80,0x43,0x5b,0x99,0x49,0x43,0x08,0xc4,0xea,0x80,0x43,0x22,0x95,0x46,0x43, + 0xfa,0xe2,0x81,0x43,0xda,0xec,0x43,0x43,0x78,0x77,0x83,0x43,0xe4,0xb2,0x41,0x43,0x08,0x8a,0x27,0x85, + 0x43,0x86,0x77,0x3e,0x43,0x0c,0x9f,0x85,0x43,0x07,0xf4,0x3b,0x43,0x8f,0x16,0x86,0x43,0xe6,0x82,0x39, + 0x43,0x08,0x85,0x44,0x86,0x43,0x37,0xd9,0x35,0x43,0x1e,0x4f,0x87,0x43,0xe1,0x7b,0x34,0x43,0xdf,0x90, + 0x88,0x43,0xb6,0x55,0x33,0x43,0x08,0xae,0x93,0x8a,0x43,0xfd,0xe5,0x31,0x43,0xfa,0x12,0x8a,0x43,0xbf, + 0x03,0x2d,0x43,0x19,0x78,0x8a,0x43,0x45,0x5e,0x2c,0x43,0x08,0x03,0xf1,0x8b,0x43,0xac,0x47,0x29,0x43, + 0x2f,0x17,0x8d,0x43,0x45,0x46,0x28,0x43,0xc8,0x21,0x8e,0x43,0x30,0xb3,0x27,0x43,0x08,0xa9,0xc8,0x8f, + 0x43,0xef,0xe8,0x26,0x43,0xbf,0x5b,0x90,0x43,0x5b,0xc1,0x24,0x43,0x10,0xca,0x90,0x43,0xa0,0x62,0x22, + 0x43,0x08,0x26,0x5d,0x91,0x43,0xbb,0xcc,0x1f,0x43,0xf0,0x70,0x92,0x43,0x78,0x13,0x1e,0x43,0x77,0xd7, + 0x93,0x43,0x73,0x24,0x1d,0x43,0x08,0x65,0x3f,0x96,0x43,0xce,0x58,0x1b,0x43,0xbe,0x7f,0x96,0x43,0xbf, + 0x8b,0x18,0x43,0x60,0x5c,0x97,0x43,0xb6,0xad,0x16,0x43,0x08,0xba,0xa8,0x99,0x43,0x78,0xcb,0x11,0x43, + 0x49,0xe1,0x9a,0x43,0x78,0xcb,0x11,0x43,0x01,0x51,0x9c,0x43,0x73,0xdc,0x10,0x43,0x08,0x72,0x24,0x9d, + 0x43,0xd2,0xff,0x0f,0x43,0x1c,0xd3,0x9d,0x43,0x07,0xec,0x0e,0x43,0xeb,0xc9,0x9d,0x43,0xe8,0x7a,0x0c, + 0x43,0x08,0x60,0x80,0x9d,0x43,0xd7,0xbe,0x08,0x43,0x4d,0xe8,0x9f,0x43,0x86,0x50,0x08,0x43,0x25,0xbd, + 0xa1,0x43,0x5b,0x2a,0x07,0x43,0x08,0x99,0x7f,0xa3,0x43,0xc9,0xf1,0x05,0x43,0x48,0x1d,0xa5,0x43,0x86, + 0x38,0x04,0x43,0x6c,0x71,0xa6,0x43,0x18,0x59,0x01,0x43,0x08,0x32,0x96,0xa6,0x43,0x6e,0x64,0xff,0x42, + 0x48,0x29,0xa7,0x43,0xed,0xcf,0xfd,0x42,0x5f,0xbc,0xa7,0x43,0x71,0x3b,0xfc,0x42,0x08,0xf3,0xe3,0xa9, + 0x43,0xf7,0x7d,0xf7,0x42,0xd8,0x6d,0xaa,0x43,0x45,0xe5,0xf2,0x42,0x48,0x41,0xab,0x43,0xcb,0x27,0xee, + 0x42,0x08,0x24,0xf9,0xab,0x43,0x52,0x6a,0xe9,0x42,0xee,0x0c,0xad,0x43,0x4c,0x8c,0xe7,0x42,0x1b,0x33, + 0xae,0x43,0xcc,0xf7,0xe5,0x42,0x08,0xaa,0x6b,0xaf,0x43,0xe8,0x61,0xe3,0x42,0x90,0xf5,0xaf,0x43,0xc9, + 0xf0,0xe0,0x42,0xe0,0x63,0xb0,0x43,0xe5,0x5a,0xde,0x42,0x08,0xaa,0x83,0xb3,0x43,0x29,0x2d,0x09,0x43, + 0x6a,0xfe,0x8e,0x43,0xb8,0x74,0x3c,0x43,0xd5,0x06,0x95,0x43,0xe6,0x79,0x67,0x43,0x08,0x2f,0x53,0x97, + 0x43,0xe9,0xb0,0x74,0x43,0xa8,0x28,0xa0,0x43,0x43,0xfd,0x76,0x43,0x83,0x28,0xad,0x43,0x17,0x59,0x81, + 0x43,0x08,0x3d,0xe7,0xbf,0x43,0x4b,0x8d,0x8c,0x43,0xae,0x96,0xba,0x43,0x66,0x27,0x92,0x43,0x15,0xe0, + 0xc7,0x43,0x6f,0x11,0x96,0x43,0x08,0x7e,0x5d,0xb2,0x43,0xdb,0x01,0x98,0x43,0x9e,0x56,0xa0,0x43,0x80, + 0xc1,0x97,0x43,0x69,0x2e,0x97,0x43,0x31,0x17,0x8d,0x43,0x09,0x06,0xab,0xa7,0x39,0x43,0x67,0x0f,0x0e, + 0x43,0x08,0xdb,0xbc,0x3b,0x43,0xe8,0x92,0x10,0x43,0xb5,0x85,0x3b,0x43,0x97,0x3c,0x14,0x43,0xab,0xa7, + 0x39,0x43,0x0c,0x0b,0x18,0x43,0x09,0x06,0xca,0x30,0x40,0x43,0x30,0x3b,0x13,0x43,0x08,0x17,0xc8,0x43, + 0x43,0xa5,0x09,0x17,0x43,0x7e,0xc9,0x44,0x43,0x1a,0xd8,0x1a,0x43,0x9d,0x22,0x43,0x43,0x8d,0xa6,0x1e, + 0x43,0x09,0x06,0xc8,0x78,0x4c,0x43,0xed,0xc9,0x1d,0x43,0x08,0x0b,0x32,0x4e,0x43,0x22,0xce,0x20,0x43, + 0x23,0xc5,0x4e,0x43,0x58,0xd2,0x23,0x43,0x0b,0x32,0x4e,0x43,0x2b,0xc4,0x26,0x43,0x09,0x06,0xec,0x08, + 0x58,0x43,0xc7,0xb1,0x26,0x43,0x08,0x02,0x9c,0x58,0x43,0xef,0x00,0x2b,0x43,0xd9,0x64,0x58,0x43,0x02, + 0xbd,0x2e,0x43,0x10,0x51,0x57,0x43,0x37,0xc1,0x31,0x43,0x09,0x06,0xcb,0xdf,0x61,0x43,0x4a,0x65,0x31, + 0x43,0x08,0xbe,0x2a,0x63,0x43,0xbd,0x33,0x35,0x43,0x32,0xe1,0x62,0x43,0x56,0x4a,0x38,0x43,0xde,0x83, + 0x61,0x43,0x3c,0xe0,0x3a,0x43,0x09,0x06,0x1c,0x7e,0x6a,0x43,0x5b,0x39,0x39,0x43,0x08,0x31,0x11,0x6b, + 0x43,0x0c,0xd2,0x3d,0x43,0x1c,0x7e,0x6a,0x43,0x13,0xd9,0x42,0x43,0xd9,0xc4,0x68,0x43,0xcb,0x60,0x48, + 0x43,0x09,0x06,0xe5,0xc1,0x73,0x43,0x16,0xf8,0x4b,0x43,0x08,0xa6,0xf7,0x72,0x43,0xb1,0xfd,0x4f,0x43, + 0x3b,0x07,0x71,0x43,0x4a,0x14,0x53,0x43,0xa2,0xf0,0x6d,0x43,0x7c,0x29,0x55,0x43,0x09,0x06,0x00,0x8d, + 0xa6,0x43,0xef,0x21,0x01,0x43,0x08,0x52,0xfb,0xa6,0x43,0xce,0xc8,0x02,0x43,0xe6,0x16,0xa7,0x43,0x51, + 0x4c,0x05,0x43,0x3b,0x68,0xa6,0x43,0x4c,0x75,0x08,0x43,0x09,0x06,0xde,0x20,0xa1,0x43,0x86,0x50,0x08, + 0x43,0x08,0xd4,0x4e,0xa1,0x43,0xd3,0xe7,0x0b,0x43,0xb5,0xe9,0xa0,0x43,0x59,0x5a,0x0f,0x43,0xba,0xcc, + 0x9f,0x43,0x54,0x83,0x12,0x43,0x09,0x06,0x77,0xfb,0x99,0x43,0x6c,0x16,0x13,0x43,0x08,0xde,0xfc,0x9a, + 0x43,0x4a,0xbd,0x14,0x43,0x06,0x34,0x9b,0x43,0xfe,0x55,0x19,0x43,0x13,0xe9,0x99,0x43,0x41,0x27,0x1f, + 0x43,0x09,0x06,0x46,0xce,0x93,0x43,0x26,0xa5,0x1d,0x43,0x08,0xe7,0xaa,0x94,0x43,0xbb,0xcc,0x1f,0x43, + 0x18,0xb4,0x94,0x43,0xa8,0x40,0x24,0x43,0xe2,0xbb,0x93,0x43,0x21,0xfe,0x28,0x43,0x09,0x06,0xb1,0x8e, + 0x8d,0x43,0xa8,0x58,0x28,0x43,0x08,0x19,0x90,0x8e,0x43,0x54,0x13,0x2b,0x43,0xa4,0xd9,0x8e,0x43,0x84, + 0x40,0x31,0x43,0x46,0xaa,0x8d,0x43,0x29,0x24,0x37,0x43,0x09,0x06,0xd6,0xbe,0x88,0x43,0xef,0x30,0x33, + 0x43,0x08,0x0c,0xb7,0x89,0x43,0x0e,0xa2,0x35,0x43,0xc0,0x37,0x8a,0x43,0x7a,0xaa,0x3b,0x43,0xbb,0x48, + 0x89,0x43,0xbb,0x7b,0x41,0x43,0x09,0x06,0x3a,0xad,0x82,0x43,0xc4,0x59,0x43,0x43,0x08,0xd2,0xb7,0x83, + 0x43,0x2b,0x5b,0x44,0x43,0x35,0xd6,0x85,0x43,0x48,0xf5,0x49,0x43,0x42,0x97,0x86,0x43,0xc4,0xa1,0x4f, + 0x43,0x09,0x06,0x9c,0xb3,0x80,0x43,0x48,0x55,0x5a,0x43,0x08,0xff,0xc5,0x80,0x43,0x09,0x73,0x55,0x43, + 0x93,0xe1,0x80,0x43,0x0f,0x39,0x53,0x43,0xf1,0xbe,0x7e,0x43,0x18,0xe7,0x4c,0x43,0x09,0x06,0xe0,0x02, + 0x7b,0x43,0x92,0xec,0x5d,0x43,0x08,0x09,0x3a,0x7b,0x43,0xf0,0xf7,0x58,0x43,0x09,0x3a,0x7b,0x43,0xe6, + 0x31,0x5b,0x43,0xe0,0x02,0x7b,0x43,0xa8,0x4f,0x56,0x43,0x09,0x06,0x39,0x4f,0x7d,0x43,0x3e,0x8f,0x5c, + 0x43,0x08,0xe9,0xe0,0x7c,0x43,0x03,0x9c,0x58,0x43,0x1e,0x2b,0x81,0x43,0x7f,0x30,0x5a,0x43,0xff,0x73, + 0x7d,0x43,0xf6,0xb6,0x51,0x43,0x09,0x06,0x5c,0xb8,0x52,0x43,0x28,0x21,0x87,0x43,0x08,0xae,0x3e,0x57, + 0x43,0x12,0x9a,0x88,0x43,0x23,0xf5,0x56,0x43,0x04,0xf1,0x8b,0x43,0x25,0xfc,0x5b,0x43,0x85,0x74,0x8e, + 0x43,0x08,0x2f,0xf2,0x61,0x43,0x8e,0x52,0x90,0x43,0xd9,0xdc,0x6c,0x43,0x85,0x74,0x8e,0x43,0xc6,0x20, + 0x69,0x43,0x3d,0xd8,0x8d,0x43,0x08,0x6d,0x8c,0x5a,0x43,0xf5,0x3b,0x8d,0x43,0x3d,0x77,0x58,0x43,0xa1, + 0xc6,0x87,0x43,0xf8,0xed,0x5e,0x43,0x5e,0x0d,0x86,0x43,0x09,0x06,0xde,0xcc,0x92,0x43,0xf7,0x17,0x87, + 0x43,0x08,0xb6,0x89,0x90,0x43,0xae,0x87,0x88,0x43,0x4a,0xa5,0x90,0x43,0xa1,0xde,0x8b,0x43,0xf9,0x2a, + 0x8e,0x43,0x23,0x62,0x8e,0x43,0x08,0xf5,0x2f,0x8b,0x43,0x5c,0x49,0x90,0x43,0x35,0xd6,0x85,0x43,0x8e, + 0x46,0x8e,0x43,0x3d,0xb4,0x87,0x43,0x47,0xaa,0x8d,0x43,0x08,0x6a,0xfe,0x8e,0x43,0xff,0x0d,0x8d,0x43, + 0xbb,0x6c,0x8f,0x43,0xf7,0x17,0x87,0x43,0x5c,0x31,0x8c,0x43,0xb2,0x5e,0x85,0x43,0x09,0x06,0x60,0x38, + 0x91,0x43,0x69,0x5d,0x7a,0x43,0x08,0x34,0x1e,0x92,0x43,0x1e,0x5b,0x89,0x43,0x04,0x63,0x7e,0x43,0x5e, + 0x01,0x84,0x43,0x59,0x2a,0x87,0x43,0x0d,0xcf,0x8d,0x43,0x09,0x03,0x04,0x06,0x5a,0x18,0x63,0x43,0x82, + 0x79,0x8b,0x43,0x08,0x25,0x2c,0x64,0x43,0x82,0x79,0x8b,0x43,0x2a,0x1b,0x65,0x43,0x9d,0xef,0x8a,0x43, + 0x2a,0x1b,0x65,0x43,0xc1,0x37,0x8a,0x43,0x08,0x2a,0x1b,0x65,0x43,0x17,0x89,0x89,0x43,0x25,0x2c,0x64, + 0x43,0x31,0xff,0x88,0x43,0x5a,0x18,0x63,0x43,0x31,0xff,0x88,0x43,0x08,0xf3,0x16,0x62,0x43,0x31,0xff, + 0x88,0x43,0xee,0x27,0x61,0x43,0x17,0x89,0x89,0x43,0xee,0x27,0x61,0x43,0xc1,0x37,0x8a,0x43,0x08,0xee, + 0x27,0x61,0x43,0x9d,0xef,0x8a,0x43,0xf3,0x16,0x62,0x43,0x82,0x79,0x8b,0x43,0x5a,0x18,0x63,0x43,0x82, + 0x79,0x8b,0x43,0x09,0x06,0x4f,0x64,0x89,0x43,0x82,0x79,0x8b,0x43,0x08,0x34,0xee,0x89,0x43,0x82,0x79, + 0x8b,0x43,0x85,0x5c,0x8a,0x43,0x9d,0xef,0x8a,0x43,0x85,0x5c,0x8a,0x43,0xc1,0x37,0x8a,0x43,0x08,0x85, + 0x5c,0x8a,0x43,0x17,0x89,0x89,0x43,0x34,0xee,0x89,0x43,0x31,0xff,0x88,0x43,0x4f,0x64,0x89,0x43,0x31, + 0xff,0x88,0x43,0x08,0x9c,0xe3,0x88,0x43,0x31,0xff,0x88,0x43,0x19,0x6c,0x88,0x43,0x17,0x89,0x89,0x43, + 0x19,0x6c,0x88,0x43,0xc1,0x37,0x8a,0x43,0x08,0x19,0x6c,0x88,0x43,0x9d,0xef,0x8a,0x43,0x9c,0xe3,0x88, + 0x43,0x82,0x79,0x8b,0x43,0x4f,0x64,0x89,0x43,0x82,0x79,0x8b,0x43,0x09,0x02,0x04,0x06,0x19,0x60,0x86, + 0x43,0xec,0xed,0xa3,0x43,0x08,0x35,0xd6,0x85,0x43,0x76,0x43,0xa6,0x43,0x93,0xe1,0x80,0x43,0x57,0x02, + 0xac,0x43,0x61,0xd8,0x80,0x43,0x87,0x17,0xae,0x43,0x08,0xa5,0x85,0x80,0x43,0xc3,0xfe,0xaf,0x43,0xce, + 0xbc,0x80,0x43,0x83,0x40,0xb1,0x43,0xa5,0x91,0x82,0x43,0x79,0x6e,0xb1,0x43,0x08,0x23,0x26,0x84,0x43, + 0x40,0x93,0xb1,0x43,0x30,0xe7,0x84,0x43,0xbe,0x1b,0xb1,0x43,0x11,0x82,0x84,0x43,0xab,0x6b,0xaf,0x43, + 0x08,0xb7,0x41,0x84,0x43,0x3b,0x98,0xae,0x43,0xb7,0x41,0x84,0x43,0xc3,0xf2,0xad,0x43,0xa1,0xae,0x83, + 0x43,0x83,0x28,0xad,0x43,0x08,0xb2,0x52,0x83,0x43,0x80,0x39,0xac,0x43,0x81,0x49,0x83,0x43,0xf0,0x00, + 0xab,0x43,0xe4,0x67,0x85,0x43,0x76,0x4f,0xa8,0x43,0x08,0x9c,0xd7,0x86,0x43,0xd1,0x83,0xa6,0x43,0xec, + 0x45,0x87,0x43,0x01,0x75,0xa2,0x43,0x19,0x60,0x86,0x43,0xec,0xed,0xa3,0x43,0x09,0x06,0xd9,0xdc,0x6c, + 0x43,0x14,0x25,0xa4,0x43,0x08,0xa2,0xf0,0x6d,0x43,0x9f,0x7a,0xa6,0x43,0x47,0xec,0x77,0x43,0x80,0x39, + 0xac,0x43,0xa9,0xfe,0x77,0x43,0xb0,0x4e,0xae,0x43,0x08,0x23,0xa4,0x78,0x43,0xea,0x35,0xb0,0x43,0xd2, + 0x35,0x78,0x43,0xab,0x77,0xb1,0x43,0xc1,0x79,0x74,0x43,0xa2,0xa5,0xb1,0x43,0x08,0xc6,0x50,0x71,0x43, + 0x68,0xca,0xb1,0x43,0xab,0xce,0x6f,0x43,0xe7,0x52,0xb1,0x43,0xea,0x98,0x70,0x43,0xd4,0xa2,0xaf,0x43, + 0x08,0x9d,0x19,0x71,0x43,0x96,0xd8,0xae,0x43,0x9d,0x19,0x71,0x43,0xec,0x29,0xae,0x43,0xca,0x3f,0x72, + 0x43,0xab,0x5f,0xad,0x43,0x08,0xa6,0xf7,0x72,0x43,0xa7,0x70,0xac,0x43,0x09,0x0a,0x73,0x43,0x17,0x38, + 0xab,0x43,0x44,0xcd,0x6e,0x43,0x9f,0x86,0xa8,0x43,0x08,0xd4,0xed,0x6b,0x43,0xf8,0xba,0xa6,0x43,0x31, + 0x11,0x6b,0x43,0x2a,0xac,0xa2,0x43,0xd9,0xdc,0x6c,0x43,0x14,0x25,0xa4,0x43,0x09,0x01,0x05,0x06,0x66, + 0x5d,0x7a,0x43,0x74,0xeb,0xc2,0x43,0x08,0x09,0x22,0x77,0x43,0x50,0xbb,0xc7,0x43,0xe9,0xe0,0x7c,0x43, + 0xf5,0x86,0xc9,0x43,0x8f,0x94,0x7a,0x43,0xc5,0x95,0xcd,0x43,0x09,0x06,0x08,0x98,0x80,0x43,0x6b,0x19, + 0xc3,0x43,0x08,0xb7,0x35,0x82,0x43,0x79,0xf2,0xc7,0x43,0xf1,0xbe,0x7e,0x43,0x1e,0xbe,0xc9,0x43,0x73, + 0x7c,0x80,0x43,0xec,0xcc,0xcd,0x43,0x09,0x06,0x28,0xab,0x7d,0x43,0xae,0xde,0xc6,0x43,0x08,0x1e,0xcd, + 0x7b,0x43,0x8a,0xa2,0xc9,0x43,0x30,0x89,0x7f,0x43,0x5c,0x94,0xcc,0x43,0x28,0xab,0x7d,0x43,0x42,0x2a, + 0xcf,0x43,0x09,0x01,0x05,0x06,0x24,0x14,0xe0,0x42,0xf5,0x77,0x97,0x43,0x08,0xf7,0x1d,0xe7,0x42,0x74, + 0x00,0x97,0x43,0x4d,0x93,0xec,0x42,0xdb,0xf5,0x95,0x43,0x29,0x4b,0xed,0x42,0xcd,0x34,0x95,0x43,0x09, + 0x06,0x29,0x7b,0xf5,0x42,0x6f,0x1d,0x98,0x43,0x08,0xe4,0xf1,0xfb,0x42,0x61,0x5c,0x97,0x43,0xdb,0x7d, + 0x01,0x43,0xb2,0xbe,0x95,0x43,0x55,0x23,0x02,0x43,0xe7,0xaa,0x94,0x43,0x09,0x06,0x98,0xdc,0x03,0x43, + 0xbe,0x8b,0x98,0x43,0x08,0x66,0xdf,0x05,0x43,0x47,0xe6,0x97,0x43,0xae,0x87,0x08,0x43,0x98,0x48,0x96, + 0x43,0x61,0x08,0x09,0x43,0xd6,0x06,0x95,0x43,0x09,0x06,0x31,0x0b,0x0b,0x43,0x8e,0x82,0x98,0x43,0x08, + 0xdb,0xc5,0x0d,0x43,0x80,0xc1,0x97,0x43,0xd6,0xee,0x10,0x43,0xa9,0xec,0x95,0x43,0x79,0xcb,0x11,0x43, + 0x55,0x8f,0x94,0x43,0x09,0x06,0xd1,0x2f,0x18,0x43,0xdb,0x01,0x98,0x43,0x08,0xad,0xe7,0x18,0x43,0x38, + 0x25,0x97,0x43,0x8a,0x9f,0x19,0x43,0x80,0xb5,0x95,0x43,0xd6,0x1e,0x19,0x43,0xe0,0xd8,0x94,0x43,0x09, + 0x06,0x9a,0x5b,0x1d,0x43,0x58,0x8a,0x97,0x43,0x08,0x01,0x5d,0x1e,0x43,0xf1,0x88,0x96,0x43,0x2f,0x83, + 0x1f,0x43,0x19,0xb4,0x94,0x43,0x19,0xf0,0x1e,0x43,0x6f,0x05,0x94,0x43,0x09,0x06,0x0b,0x53,0x24,0x43, + 0xae,0xdb,0x96,0x43,0x08,0x25,0xd5,0x25,0x43,0x50,0xac,0x95,0x43,0x53,0xfb,0x26,0x43,0x8a,0x7b,0x93, + 0x43,0x76,0x43,0x26,0x43,0xb7,0x95,0x92,0x43,0x09,0x06,0x76,0x5b,0x2a,0x43,0x47,0xda,0x95,0x43,0x08, + 0xf3,0xef,0x2b,0x43,0x10,0xe2,0x94,0x43,0x6d,0x95,0x2c,0x43,0xae,0xc3,0x92,0x43,0x68,0xa6,0x2b,0x43, + 0x47,0xc2,0x91,0x43,0x09,0x06,0x36,0xc1,0x31,0x43,0x2c,0x58,0x94,0x43,0x08,0x8c,0x1e,0x33,0x43,0x31, + 0x3b,0x93,0x43,0x79,0x7a,0x33,0x43,0xff,0x25,0x91,0x43,0xd9,0x9d,0x32,0x43,0xc1,0x5b,0x90,0x43,0x09, + 0x06,0x25,0x35,0x36,0x43,0x31,0x3b,0x93,0x43,0x08,0x3f,0xb7,0x37,0x43,0xc1,0x67,0x92,0x43,0xe0,0x93, + 0x38,0x43,0xae,0xb7,0x90,0x43,0x7e,0x81,0x38,0x43,0x0d,0xdb,0x8f,0x43,0x09,0x06,0xb5,0x85,0x3b,0x43, + 0xe4,0xaf,0x91,0x43,0x08,0xcf,0x07,0x3d,0x43,0x9d,0x13,0x91,0x43,0xbc,0x63,0x3d,0x43,0x47,0xb6,0x8f, + 0x43,0xe5,0x9a,0x3d,0x43,0x74,0xd0,0x8e,0x43,0x09,0x06,0xae,0xc6,0x42,0x43,0xa4,0xd9,0x8e,0x43,0x08, + 0xca,0x48,0x44,0x43,0xfa,0x2a,0x8e,0x43,0xa2,0x11,0x44,0x43,0x9d,0xfb,0x8c,0x43,0x55,0x92,0x44,0x43, + 0x0d,0xc3,0x8b,0x43,0x09,0x06,0x39,0x10,0xc3,0x43,0x34,0x36,0x96,0x43,0x08,0x92,0x44,0xc1,0x43,0xe4, + 0xc7,0x95,0x43,0x6f,0xf0,0xbf,0x43,0x4b,0xbd,0x94,0x43,0x47,0xb9,0xbf,0x43,0x0b,0xf3,0x93,0x43,0x09, + 0x06,0x8f,0x49,0xbe,0x43,0xb7,0xad,0x96,0x43,0x08,0x11,0xb5,0xbc,0x43,0x77,0xe3,0x95,0x43,0x9c,0xf2, + 0xba,0x43,0xfa,0x4e,0x94,0x43,0xae,0x96,0xba,0x43,0x31,0x3b,0x93,0x43,0x09,0x06,0xdb,0xb0,0xb9,0x43, + 0x10,0xee,0x96,0x43,0x08,0x42,0xa6,0xb8,0x43,0xc8,0x51,0x96,0x43,0x50,0x5b,0xb7,0x43,0x19,0xb4,0x94, + 0x43,0xf7,0x1a,0xb7,0x43,0x58,0x72,0x93,0x43,0x09,0x06,0xf2,0x2b,0xb6,0x43,0x10,0xee,0x96,0x43,0x08, + 0x9d,0xce,0xb4,0x43,0x04,0x2d,0x96,0x43,0xed,0x30,0xb3,0x43,0x2c,0x58,0x94,0x43,0xce,0xcb,0xb2,0x43, + 0xd6,0xfa,0x92,0x43,0x09,0x06,0x5a,0x09,0xb1,0x43,0x19,0xc0,0x96,0x43,0x08,0x6c,0xad,0xb0,0x43,0x77, + 0xe3,0x95,0x43,0x7e,0x51,0xb0,0x43,0xc0,0x73,0x94,0x43,0xd8,0x91,0xb0,0x43,0x1e,0x97,0x93,0x43,0x09, + 0x06,0x48,0x4d,0xad,0x43,0xbe,0x7f,0x96,0x43,0x08,0x95,0xcc,0xac,0x43,0x58,0x7e,0x95,0x43,0x4d,0x30, + 0xac,0x43,0x80,0xa9,0x93,0x43,0xd8,0x79,0xac,0x43,0xd6,0xfa,0x92,0x43,0x09,0x06,0x90,0xd1,0xa9,0x43, + 0x14,0xd1,0x95,0x43,0x08,0x83,0x10,0xa9,0x43,0xb7,0xa1,0x94,0x43,0x3b,0x74,0xa8,0x43,0xf1,0x70,0x92, + 0x43,0x29,0xd0,0xa8,0x43,0x1e,0x8b,0x91,0x43,0x09,0x06,0x5a,0xcd,0xa6,0x43,0x8a,0x87,0x95,0x43,0x08, + 0x1c,0x03,0xa6,0x43,0x23,0x86,0x94,0x43,0x5f,0xb0,0xa5,0x43,0xc1,0x67,0x92,0x43,0xe1,0x27,0xa6,0x43, + 0x8a,0x6f,0x91,0x43,0x09,0x06,0xd4,0x5a,0xa3,0x43,0x2c,0x58,0x94,0x43,0x08,0x29,0xac,0xa2,0x43,0x31, + 0x3b,0x93,0x43,0x32,0x7e,0xa2,0x43,0xff,0x25,0x91,0x43,0x83,0xec,0xa2,0x43,0x8e,0x52,0x90,0x43,0x09, + 0x06,0xf8,0x96,0xa0,0x43,0x1e,0x97,0x93,0x43,0x08,0xeb,0xd5,0x9f,0x43,0x7b,0xba,0x92,0x43,0x99,0x67, + 0x9f,0x43,0x9d,0x13,0x91,0x43,0x99,0x67,0x9f,0x43,0xfa,0x36,0x90,0x43,0x09,0x06,0xeb,0xc9,0x9d,0x43, + 0xc8,0x39,0x92,0x43,0x08,0xde,0x08,0x9d,0x43,0xb2,0xa6,0x91,0x43,0xe6,0xda,0x9c,0x43,0x2c,0x40,0x90, + 0x43,0x52,0xbf,0x9c,0x43,0x5a,0x5a,0x8f,0x43,0x09,0x06,0x37,0x3d,0x9b,0x43,0x85,0x80,0x90,0x43,0x08, + 0x2a,0x7c,0x9a,0x43,0xdb,0xd1,0x8f,0x43,0xf0,0xa0,0x9a,0x43,0x7d,0xa2,0x8e,0x43,0x65,0x57,0x9a,0x43, + 0xee,0x69,0x8d,0x43,0x09,0x02,0x04,0x06,0x2a,0xf4,0x2e,0x42,0x04,0x21,0x94,0x43,0x08,0x0d,0x8a,0x31, + 0x42,0x9f,0x0e,0x94,0x43,0xf3,0x1f,0x34,0x42,0x3d,0xfc,0x93,0x43,0x63,0xff,0x36,0x42,0xa9,0xe0,0x93, + 0x43,0x08,0xb5,0x34,0x5d,0x42,0x0b,0xf3,0x93,0x43,0x6d,0xa4,0x5e,0x42,0x03,0x39,0x98,0x43,0xe7,0x31, + 0x5b,0x42,0x93,0x89,0x9d,0x43,0x08,0x02,0x9c,0x58,0x42,0xd4,0x5a,0xa3,0x43,0x38,0x70,0x53,0x42,0x14, + 0x49,0xaa,0x43,0xf8,0xed,0x5e,0x42,0x83,0x28,0xad,0x43,0x08,0xea,0x68,0x68,0x42,0x20,0x22,0xaf,0x43, + 0x12,0xb8,0x6c,0x42,0xb5,0x49,0xb1,0x43,0x2a,0x4b,0x6d,0x42,0x0d,0x96,0xb3,0x43,0x07,0x2a,0x4b,0x6d, + 0x42,0xc6,0x05,0xb5,0x43,0x08,0x87,0x6e,0x6c,0x42,0x68,0xee,0xb7,0x43,0x1c,0x66,0x66,0x42,0x31,0x0e, + 0xbb,0x43,0x57,0x11,0x5e,0x42,0x8f,0x49,0xbe,0x43,0x08,0x66,0x96,0x54,0x42,0xb9,0x5c,0xb8,0x43,0x2c, + 0x2b,0x3c,0x42,0x68,0xd6,0xb3,0x43,0x2a,0xf4,0x2e,0x42,0x6d,0xad,0xb0,0x43,0x07,0x2a,0xf4,0x2e,0x42, + 0x61,0xa4,0xa3,0x43,0x08,0x55,0x1a,0x30,0x42,0xf0,0xd0,0xa2,0x43,0xf8,0xf6,0x30,0x42,0xb2,0x06,0xa2, + 0x43,0x98,0xd3,0x31,0x42,0xd6,0x4e,0xa1,0x43,0x08,0x1c,0x6f,0x38,0x42,0x2a,0x94,0x9e,0x43,0xc1,0x22, + 0x36,0x42,0xf5,0x9b,0x9d,0x43,0x2a,0xf4,0x2e,0x42,0x6a,0x52,0x9d,0x43,0x07,0x2a,0xf4,0x2e,0x42,0x57, + 0xa2,0x9b,0x43,0x08,0xab,0x8f,0x35,0x42,0x8a,0xab,0x9b,0x43,0xe9,0x71,0x3a,0x42,0xb2,0xe2,0x9b,0x43, + 0xb7,0x74,0x3c,0x42,0x34,0x5a,0x9c,0x43,0x08,0x23,0x7d,0x42,0x42,0x0b,0x2f,0x9e,0x43,0xe5,0x9a,0x3d, + 0x42,0x38,0x6d,0xa3,0x43,0x36,0xd9,0x35,0x42,0xf3,0xd7,0xa7,0x43,0x08,0x12,0x61,0x2e,0x42,0xb0,0x42, + 0xac,0x43,0x63,0xff,0x36,0x42,0xdd,0x74,0xaf,0x43,0x1e,0xa6,0x45,0x42,0x44,0x82,0xb2,0x43,0x08,0x74, + 0x1b,0x4b,0x42,0x79,0x7a,0xb3,0x43,0x10,0x21,0x4f,0x42,0x2a,0x18,0xb5,0x43,0xdb,0x4c,0x54,0x42,0x91, + 0x19,0xb6,0x43,0x08,0xee,0x3f,0x65,0x42,0x5f,0x28,0xba,0x43,0xa7,0xaf,0x66,0x42,0xb9,0x50,0xb6,0x43, + 0x14,0x58,0x5c,0x42,0xca,0xdc,0xb1,0x43,0x08,0x2c,0x8b,0x4c,0x42,0x4e,0x30,0xac,0x43,0x19,0xcf,0x48, + 0x42,0x2a,0xd0,0xa8,0x43,0xbc,0xab,0x49,0x42,0xa9,0x4c,0xa6,0x43,0x08,0x61,0x5f,0x47,0x42,0xfa,0xa2, + 0xa2,0x43,0xa7,0xaf,0x66,0x42,0x85,0x98,0x94,0x43,0x2a,0xf4,0x2e,0x42,0xc3,0x62,0x95,0x43,0x07,0x2a, + 0xf4,0x2e,0x42,0x04,0x21,0x94,0x43,0x09,0x06,0xd0,0xfe,0xea,0x41,0x9f,0x0e,0x94,0x43,0x08,0xdc,0xe3, + 0xf1,0x41,0xe9,0x9e,0x92,0x43,0xd2,0xe7,0x0b,0x42,0xd6,0x06,0x95,0x43,0x2a,0xf4,0x2e,0x42,0x04,0x21, + 0x94,0x43,0x07,0x2a,0xf4,0x2e,0x42,0xc3,0x62,0x95,0x43,0x08,0x87,0x17,0x2e,0x42,0xc3,0x62,0x95,0x43, + 0xe7,0x3a,0x2d,0x42,0xf5,0x6b,0x95,0x43,0x44,0x5e,0x2c,0x42,0xf5,0x6b,0x95,0x43,0x08,0xd1,0x47,0x1c, + 0x42,0x19,0xc0,0x96,0x43,0x66,0xdf,0x05,0x42,0x38,0x19,0x95,0x43,0x12,0x6a,0x00,0x42,0xb2,0xbe,0x95, + 0x43,0x08,0xbb,0x6b,0xea,0x41,0xd6,0x12,0x97,0x43,0x2d,0x82,0xfa,0x41,0x61,0x74,0x9b,0x43,0x7e,0x72, + 0x06,0x42,0x8a,0xab,0x9b,0x43,0x08,0xc8,0x39,0x12,0x42,0x4e,0xd0,0x9b,0x43,0x53,0xe3,0x22,0x42,0xc3, + 0x86,0x9b,0x43,0x2a,0xf4,0x2e,0x42,0x57,0xa2,0x9b,0x43,0x07,0x2a,0xf4,0x2e,0x42,0x6a,0x52,0x9d,0x43, + 0x08,0x01,0xa5,0x2a,0x42,0xa4,0x2d,0x9d,0x43,0x96,0x9c,0x24,0x42,0x06,0x40,0x9d,0x43,0x8a,0xb7,0x1d, + 0x42,0x9a,0x5b,0x9d,0x43,0x08,0x6b,0x16,0x13,0x42,0xcd,0x64,0x9d,0x43,0x42,0xc7,0x0e,0x42,0x9a,0x5b, + 0x9d,0x43,0x23,0x26,0x04,0x42,0xcd,0x64,0x9d,0x43,0x08,0xe6,0x91,0xeb,0x41,0x38,0x49,0x9d,0x43,0x73, + 0x7b,0xdb,0x41,0xf5,0x83,0x99,0x43,0x7f,0x60,0xe2,0x41,0x0b,0x0b,0x98,0x43,0x08,0x7f,0x60,0xe2,0x41, + 0xec,0x99,0x95,0x43,0xe3,0x5a,0xde,0x41,0xbe,0x7f,0x96,0x43,0xd0,0xfe,0xea,0x41,0x9f,0x0e,0x94,0x43, + 0x07,0xd0,0xfe,0xea,0x41,0x9f,0x0e,0x94,0x43,0x09,0x06,0x2a,0xf4,0x2e,0x42,0x6d,0xad,0xb0,0x43,0x08, + 0xd4,0x7e,0x29,0x42,0xab,0x6b,0xaf,0x43,0x4e,0x0c,0x26,0x42,0x44,0x6a,0xae,0x43,0x38,0x79,0x25,0x42, + 0xd4,0x96,0xad,0x43,0x08,0x25,0xbd,0x21,0x42,0xe2,0x4b,0xac,0x43,0x49,0x35,0x29,0x42,0x9a,0x97,0xa7, + 0x43,0x2a,0xf4,0x2e,0x42,0x61,0xa4,0xa3,0x43,0x07,0x2a,0xf4,0x2e,0x42,0x6d,0xad,0xb0,0x43,0x09,0x06, + 0x1d,0xe5,0x7f,0x43,0x87,0x4a,0xe6,0x43,0x08,0x86,0x20,0x80,0x43,0x57,0x41,0xe6,0x43,0x7d,0x4e,0x80, + 0x43,0x25,0x38,0xe6,0x43,0xa5,0x85,0x80,0x43,0xf3,0x2e,0xe6,0x43,0x08,0x35,0xca,0x83,0x43,0xd4,0xc9, + 0xe5,0x43,0x9c,0xd7,0x86,0x43,0x44,0x91,0xe4,0x43,0xd5,0xca,0x8a,0x43,0x91,0x1c,0xe6,0x43,0x08,0x53, + 0x5f,0x8c,0x43,0xf8,0x1d,0xe7,0x43,0x2f,0x17,0x8d,0x43,0x4e,0x7b,0xe8,0x43,0x92,0x29,0x8d,0x43,0x2f, + 0x22,0xea,0x43,0x07,0x92,0x29,0x8d,0x43,0x44,0xb5,0xea,0x43,0x08,0xfe,0x0d,0x8d,0x43,0x2a,0x4b,0xed, + 0x43,0xe3,0x8b,0x8b,0x43,0x55,0x7d,0xf0,0x43,0xec,0x51,0x89,0x43,0x72,0x0b,0xf4,0x43,0x08,0xcd,0xd4, + 0x84,0x43,0x9d,0x55,0xfb,0x43,0xc9,0xe5,0x83,0x43,0x74,0x1e,0xfb,0x43,0x73,0x94,0x84,0x43,0x5a,0x90, + 0xf7,0x43,0x08,0xe8,0x62,0x88,0x43,0xfd,0x30,0xee,0x43,0x39,0xc5,0x86,0x43,0xdd,0xbf,0xeb,0x43,0x35, + 0xbe,0x81,0x43,0x40,0xde,0xed,0x43,0x08,0x4f,0x34,0x81,0x43,0x36,0x0c,0xee,0x43,0x08,0x98,0x80,0x43, + 0xfd,0x30,0xee,0x43,0x1d,0xe5,0x7f,0x43,0x91,0x4c,0xee,0x43,0x07,0x1d,0xe5,0x7f,0x43,0x91,0x40,0xec, + 0x43,0x08,0x35,0xbe,0x81,0x43,0x06,0xf7,0xeb,0x43,0x15,0x65,0x83,0x43,0x49,0xa4,0xeb,0x43,0x1e,0x43, + 0x85,0x43,0xbe,0x5a,0xeb,0x43,0x08,0xae,0x93,0x8a,0x43,0xfd,0x18,0xea,0x43,0x42,0x97,0x86,0x43,0x5f, + 0x67,0xf4,0x43,0xa9,0x98,0x87,0x43,0xd4,0x1d,0xf4,0x43,0x08,0x5c,0x25,0x8a,0x43,0xcf,0x16,0xef,0x43, + 0x46,0xaa,0x8d,0x43,0x5a,0x3c,0xe9,0x43,0x19,0x6c,0x88,0x43,0x53,0x5e,0xe7,0x43,0x08,0xc4,0x02,0x85, + 0x43,0x96,0x0b,0xe7,0x43,0x85,0x2c,0x82,0x43,0x83,0x67,0xe7,0x43,0x1d,0xe5,0x7f,0x43,0x72,0xc3,0xe7, + 0x43,0x07,0x1d,0xe5,0x7f,0x43,0x87,0x4a,0xe6,0x43,0x09,0x06,0xfd,0x24,0x6c,0x43,0xd9,0x94,0xe0,0x43, + 0x08,0xfa,0x6c,0x78,0x43,0xd1,0xc2,0xe0,0x43,0x25,0x5c,0x6c,0x43,0x25,0x44,0xe8,0x43,0x1d,0xe5,0x7f, + 0x43,0x87,0x4a,0xe6,0x43,0x07,0x1d,0xe5,0x7f,0x43,0x72,0xc3,0xe7,0x43,0x08,0xa6,0x27,0x7b,0x43,0x91, + 0x28,0xe8,0x43,0xbc,0xa2,0x77,0x43,0xb0,0x8d,0xe8,0x43,0xc6,0x68,0x75,0x43,0x57,0x4d,0xe8,0x43,0x08, + 0xe0,0xd2,0x72,0x43,0xab,0x9e,0xe7,0x43,0x50,0x9a,0x71,0x43,0x2a,0x27,0xe7,0x43,0xea,0x98,0x70,0x43, + 0x57,0x35,0xe4,0x43,0x08,0x94,0x3b,0x6f,0x43,0x14,0x7c,0xe2,0x43,0xff,0x13,0x6d,0x43,0x06,0xbb,0xe1, + 0x43,0xcf,0xfe,0x6a,0x43,0x06,0xbb,0xe1,0x43,0x08,0x44,0x9d,0x66,0x43,0x77,0x8e,0xe2,0x43,0x3b,0xef, + 0x6c,0x43,0x91,0x10,0xe4,0x43,0xfd,0x24,0x6c,0x43,0xb0,0x81,0xe6,0x43,0x08,0x96,0x23,0x6b,0x43,0xee, + 0x57,0xe9,0x43,0xca,0x0f,0x6a,0x43,0x5f,0x37,0xec,0x43,0x55,0x71,0x6e,0x43,0x9f,0x01,0xed,0x43,0x08, + 0xdb,0xfb,0x75,0x43,0x3b,0xef,0xec,0x43,0x09,0x3a,0x7b,0x43,0xb0,0xa5,0xec,0x43,0x1d,0xe5,0x7f,0x43, + 0x91,0x40,0xec,0x43,0x07,0x1d,0xe5,0x7f,0x43,0x91,0x4c,0xee,0x43,0x08,0xa9,0x16,0x7c,0x43,0xb0,0xb1, + 0xee,0x43,0x47,0xec,0x77,0x43,0xd9,0xe8,0xee,0x43,0x1e,0x9d,0x73,0x43,0xcf,0x16,0xef,0x43,0x08,0x0e, + 0xc9,0x6b,0x43,0xee,0x7b,0xef,0x43,0x7e,0x90,0x6a,0x43,0xfd,0x30,0xee,0x43,0x01,0xfc,0x68,0x43,0x4e, + 0x93,0xec,0x43,0x08,0x31,0xf9,0x66,0x43,0x4e,0x87,0xea,0x43,0x31,0x11,0x6b,0x43,0xd4,0xd5,0xe7,0x43, + 0xd9,0xc4,0x68,0x43,0xd4,0xc9,0xe5,0x43,0x08,0xe5,0x79,0x67,0x43,0x77,0x9a,0xe4,0x43,0x44,0x9d,0x66, + 0x43,0xab,0x86,0xe3,0x43,0x7e,0x78,0x66,0x43,0x0b,0xaa,0xe2,0x43,0x07,0x7e,0x78,0x66,0x43,0x57,0x29, + 0xe2,0x43,0x08,0xa7,0xaf,0x66,0x43,0xbe,0x1e,0xe1,0x43,0x87,0x56,0x68,0x43,0x77,0x82,0xe0,0x43,0xfd, + 0x24,0x6c,0x43,0xd9,0x94,0xe0,0x43,0x09,0x06,0xc4,0x41,0xbf,0x43,0x85,0xc0,0x72,0x42,0x08,0x73,0xdf, + 0xc0,0x43,0xf4,0x76,0x72,0x42,0x97,0x33,0xc2,0x43,0x85,0xc0,0x72,0x42,0xb2,0xb5,0xc3,0x43,0x64,0x56, + 0x75,0x42,0x08,0x03,0x24,0xc4,0x43,0x5e,0x7f,0x78,0x42,0xfa,0x51,0xc4,0x43,0x01,0x85,0x7c,0x42,0x5c, + 0x64,0xc4,0x43,0xa0,0xb3,0x80,0x42,0x07,0x5c,0x64,0xc4,0x43,0x10,0x93,0x83,0x42,0x08,0xc8,0x48,0xc4, + 0x43,0x1c,0x78,0x8a,0x42,0x27,0x6c,0xc3,0x43,0xaf,0xcf,0x94,0x42,0x23,0x7d,0xc2,0x43,0x99,0x9c,0xa4, + 0x42,0x08,0x3d,0xe7,0xbf,0x43,0xfb,0xfd,0xb5,0x42,0xb3,0x9d,0xbf,0x43,0x88,0x17,0xae,0x42,0xc4,0x41, + 0xbf,0x43,0x69,0x76,0xa3,0x42,0x07,0xc4,0x41,0xbf,0x43,0xac,0xc8,0x8f,0x42,0x08,0x4f,0x8b,0xbf,0x43, + 0xed,0x81,0x91,0x42,0xe4,0xa6,0xbf,0x43,0x5d,0x61,0x94,0x42,0xfa,0x39,0xc0,0x43,0x3b,0x49,0x9d,0x42, + 0x08,0x2b,0x43,0xc0,0x43,0x28,0xed,0xa9,0x42,0x61,0x3b,0xc1,0x43,0x00,0x9e,0xa5,0x42,0xe4,0xb2,0xc1, + 0x43,0x5d,0x91,0x9c,0x42,0x08,0x78,0xce,0xc1,0x43,0xfd,0x36,0x90,0x42,0x22,0x89,0xc4,0x43,0x81,0x72, + 0x86,0x42,0xae,0xc6,0xc2,0x43,0xa0,0xb3,0x80,0x42,0x08,0x54,0x86,0xc2,0x43,0x58,0xd1,0x7e,0x42,0x30, + 0x32,0xc1,0x43,0xce,0x5e,0x7b,0x42,0xc4,0x41,0xbf,0x43,0xe8,0xf1,0x7b,0x42,0x07,0xc4,0x41,0xbf,0x43, + 0x85,0xc0,0x72,0x42,0x09,0x06,0xf6,0x32,0xbb,0x43,0x40,0xa7,0x60,0x42,0x08,0x35,0xfd,0xbb,0x43,0xa4, + 0xa1,0x5c,0x42,0x5e,0x34,0xbc,0x43,0x9d,0x2a,0x70,0x42,0x5e,0x40,0xbe,0x43,0x0e,0x0a,0x73,0x42,0x08, + 0x4c,0x9c,0xbe,0x43,0x0e,0x0a,0x73,0x42,0x08,0xef,0xbe,0x43,0x0e,0x0a,0x73,0x42,0xc4,0x41,0xbf,0x43, + 0x85,0xc0,0x72,0x42,0x07,0xc4,0x41,0xbf,0x43,0xe8,0xf1,0x7b,0x42,0x08,0xcd,0x13,0xbf,0x43,0xe8,0xf1, + 0x7b,0x42,0xd6,0xe5,0xbe,0x43,0x71,0x3b,0x7c,0x42,0xdf,0xb7,0xbe,0x43,0x71,0x3b,0x7c,0x42,0x08,0x08, + 0xe3,0xbc,0x43,0xa4,0x61,0x7d,0x42,0x28,0x3c,0xbb,0x43,0x91,0x45,0x69,0x42,0x28,0x3c,0xbb,0x43,0x58, + 0x71,0x6e,0x42,0x08,0xce,0xfb,0xba,0x43,0xd5,0x35,0x78,0x42,0x59,0x45,0xbb,0x43,0x58,0x23,0x82,0x42, + 0xa1,0xe1,0xbb,0x43,0xd7,0xbe,0x88,0x42,0x08,0xc9,0x18,0xbc,0x43,0xaf,0x9f,0x8c,0x42,0x1e,0x76,0xbd, + 0x43,0x51,0x7c,0x8d,0x42,0xd6,0xe5,0xbe,0x43,0xf4,0x58,0x8e,0x42,0x08,0x9c,0x0a,0xbf,0x43,0x45,0xc7, + 0x8e,0x42,0x30,0x26,0xbf,0x43,0x96,0x35,0x8f,0x42,0xc4,0x41,0xbf,0x43,0xac,0xc8,0x8f,0x42,0x07,0xc4, + 0x41,0xbf,0x43,0x69,0x76,0xa3,0x42,0x08,0x08,0xef,0xbe,0x43,0xb1,0xd6,0x99,0x42,0xe8,0x89,0xbe,0x43, + 0xde,0xc5,0x8d,0x42,0xc0,0x46,0xbc,0x43,0xc2,0x5b,0x90,0x42,0x08,0x9c,0xf2,0xba,0x43,0x86,0x80,0x90, + 0x42,0xf2,0x43,0xba,0x43,0xe8,0x73,0x87,0x42,0x8f,0x31,0xba,0x43,0xb6,0xf4,0x7d,0x42,0x07,0x8f,0x31, + 0xba,0x43,0x21,0xc6,0x76,0x42,0x08,0xc0,0x3a,0xba,0x43,0x5f,0x48,0x6b,0x42,0xae,0x96,0xba,0x43,0xe3, + 0x83,0x61,0x42,0xf6,0x32,0xbb,0x43,0x40,0xa7,0x60,0x42,0x09,0x06,0xea,0x74,0xea,0x43,0x61,0x44,0x93, + 0x43,0x08,0x24,0x5c,0xec,0x43,0x31,0x3b,0x93,0x43,0xfb,0x30,0xee,0x43,0x93,0x4d,0x93,0x43,0x0d,0xe1, + 0xef,0x43,0x80,0xa9,0x93,0x43,0x08,0x8f,0x58,0xf0,0x43,0xd1,0x17,0x94,0x43,0xb7,0x8f,0xf0,0x43,0x10, + 0xe2,0x94,0x43,0xea,0x98,0xf0,0x43,0xa9,0xec,0x95,0x43,0x07,0xea,0x98,0xf0,0x43,0x38,0x25,0x97,0x43, + 0x08,0x23,0x74,0xf0,0x43,0x9f,0x32,0x9a,0x43,0x5a,0x60,0xef,0x43,0x53,0xcb,0x9e,0x43,0x2d,0x3a,0xee, + 0x43,0xfd,0x91,0xa3,0x43,0x08,0xa2,0xf0,0xed,0x43,0xdd,0x38,0xa5,0x43,0x17,0xa7,0xed,0x43,0xbe,0xdf, + 0xa6,0x43,0x5a,0x54,0xed,0x43,0x9f,0x86,0xa8,0x43,0x08,0xfc,0x24,0xec,0x43,0xca,0xc4,0xad,0x43,0x48, + 0xa4,0xeb,0x43,0x40,0x6f,0xab,0x43,0x28,0x3f,0xeb,0x43,0x1c,0x0f,0xa8,0x43,0x08,0x1f,0x6d,0xeb,0x43, + 0x72,0x48,0xa3,0x43,0x67,0x09,0xec,0x43,0xd1,0x53,0x9e,0x43,0xea,0x74,0xea,0x43,0x1e,0xc7,0x9b,0x43, + 0x07,0xea,0x74,0xea,0x43,0x8a,0x9f,0x99,0x43,0x08,0x7e,0x90,0xea,0x43,0x8a,0x9f,0x99,0x43,0x12,0xac, + 0xea,0x43,0xbc,0xa8,0x99,0x43,0xa7,0xc7,0xea,0x43,0xbc,0xa8,0x99,0x43,0x08,0x51,0x76,0xeb,0x43,0x9f, + 0x32,0x9a,0x43,0x5e,0x37,0xec,0x43,0x49,0xed,0x9c,0x43,0xb0,0xa5,0xec,0x43,0x2a,0xa0,0xa0,0x43,0x08, + 0x09,0xe6,0xec,0x43,0xd1,0x77,0xa4,0x43,0x28,0x4b,0xed,0x43,0x61,0xa4,0xa3,0x43,0xab,0xc2,0xed,0x43, + 0x8e,0xb2,0xa0,0x43,0x08,0x70,0xe7,0xed,0x43,0xde,0x08,0x9d,0x43,0x87,0x86,0xf0,0x43,0x2f,0x53,0x97, + 0x43,0x87,0x7a,0xee,0x43,0xec,0x99,0x95,0x43,0x08,0xca,0x27,0xee,0x43,0xff,0x3d,0x95,0x43,0x74,0xca, + 0xec,0x43,0x55,0x8f,0x94,0x43,0xea,0x74,0xea,0x43,0xe7,0xaa,0x94,0x43,0x07,0xea,0x74,0xea,0x43,0x61, + 0x44,0x93,0x43,0x09,0x06,0x05,0xd3,0xe5,0x43,0x19,0x9c,0x90,0x43,0x08,0x09,0xc2,0xe6,0x43,0xd1,0xff, + 0x8f,0x43,0x4d,0x6f,0xe6,0x43,0x74,0xe8,0x92,0x43,0x3b,0xd7,0xe8,0x43,0xc3,0x56,0x93,0x43,0x08,0x1f, + 0x61,0xe9,0x43,0x93,0x4d,0x93,0x43,0x05,0xeb,0xe9,0x43,0x93,0x4d,0x93,0x43,0xea,0x74,0xea,0x43,0x61, + 0x44,0x93,0x43,0x07,0xea,0x74,0xea,0x43,0xe7,0xaa,0x94,0x43,0x08,0x24,0x50,0xea,0x43,0xe7,0xaa,0x94, + 0x43,0x2d,0x22,0xea,0x43,0xe7,0xaa,0x94,0x43,0x36,0xf4,0xe9,0x43,0xe7,0xaa,0x94,0x43,0x08,0xa2,0xcc, + 0xe7,0x43,0xe0,0xd8,0x94,0x43,0xd4,0xc9,0xe5,0x43,0x19,0xa8,0x92,0x43,0xd4,0xc9,0xe5,0x43,0x27,0x69, + 0x93,0x43,0x08,0x17,0x77,0xe5,0x43,0xe0,0xd8,0x94,0x43,0x67,0xe5,0xe5,0x43,0x47,0xda,0x95,0x43,0x43, + 0x9d,0xe6,0x43,0xe2,0xd3,0x97,0x43,0x08,0x9d,0xdd,0xe6,0x43,0xad,0xe7,0x98,0x43,0x09,0xce,0xe8,0x43, + 0xff,0x55,0x99,0x43,0xea,0x74,0xea,0x43,0x8a,0x9f,0x99,0x43,0x07,0xea,0x74,0xea,0x43,0x1e,0xc7,0x9b, + 0x43,0x08,0x71,0xcf,0xe9,0x43,0x53,0xb3,0x9a,0x43,0xa7,0xbb,0xe8,0x43,0xdb,0x0d,0x9a,0x43,0xc6,0x14, + 0xe7,0x43,0xdb,0x0d,0x9a,0x43,0x08,0x48,0x80,0xe5,0x43,0xdb,0x0d,0x9a,0x43,0x0a,0xb6,0xe4,0x43,0xc3, + 0x6e,0x97,0x43,0x76,0x9a,0xe4,0x43,0x74,0xf4,0x94,0x43,0x07,0x76,0x9a,0xe4,0x43,0x79,0xd7,0x93,0x43, + 0x08,0xd8,0xac,0xe4,0x43,0x66,0x27,0x92,0x43,0x29,0x1b,0xe5,0x43,0xe0,0xc0,0x90,0x43,0x05,0xd3,0xe5, + 0x43,0x19,0x9c,0x90,0x43,0x09,0x06,0x1b,0x66,0xe6,0x42,0xe3,0xa3,0x8f,0x42,0x08,0x71,0x0b,0xf4,0x42, + 0x00,0x0e,0x8d,0x42,0x8c,0x0f,0x01,0x43,0x3e,0xc0,0x89,0x42,0xf3,0x28,0x06,0x43,0x48,0x9e,0x8b,0x42, + 0x08,0x15,0x89,0x09,0x43,0x00,0x0e,0x8d,0x42,0xe0,0x9c,0x0a,0x43,0xc1,0x8b,0x98,0x42,0xa6,0xc1,0x0a, + 0x43,0x02,0xa5,0xaa,0x42,0x07,0xa6,0xc1,0x0a,0x43,0xf9,0xf6,0xb0,0x42,0x08,0xa6,0xc1,0x0a,0x43,0x47, + 0x8e,0xb4,0x42,0x42,0xaf,0x0a,0x43,0x1f,0x6f,0xb8,0x42,0xe0,0x9c,0x0a,0x43,0xba,0x74,0xbc,0x42,0x08, + 0xa1,0xd2,0x09,0x43,0x40,0x47,0xd0,0x42,0x0d,0xab,0x07,0x43,0x91,0xb5,0xd0,0x42,0x3b,0xb9,0x04,0x43, + 0xec,0x71,0xba,0x42,0x08,0xe5,0x5b,0x03,0x43,0xe3,0x33,0xa8,0x42,0x63,0xd8,0x00,0x43,0xce,0x70,0x9f, + 0x42,0x1b,0x66,0xe6,0x42,0xae,0x2f,0xa5,0x42,0x07,0x1b,0x66,0xe6,0x42,0xa2,0x4a,0x9e,0x42,0x08,0xed, + 0x6f,0xed,0x42,0x73,0x24,0x9d,0x42,0xd8,0x0c,0xf5,0x42,0x99,0x6c,0x9c,0x42,0x27,0xab,0xfd,0x42,0xea, + 0xda,0x9c,0x42,0x08,0x36,0xca,0x03,0x43,0x2b,0x94,0x9e,0x42,0x68,0xc7,0x01,0x43,0x8f,0xbe,0xa2,0x42, + 0xfa,0x06,0x08,0x43,0x73,0xb4,0xb5,0x42,0x08,0x8e,0x2e,0x0a,0x43,0x1f,0x6f,0xb8,0x42,0x9d,0xe3,0x08, + 0x43,0xd7,0x1e,0x99,0x42,0x28,0x15,0x05,0x43,0x32,0x3b,0x93,0x42,0x08,0x63,0xf0,0x04,0x43,0x70,0xed, + 0x8f,0x42,0x71,0x0b,0xf4,0x42,0x32,0x3b,0x93,0x42,0x1b,0x66,0xe6,0x42,0x73,0xf4,0x94,0x42,0x07,0x1b, + 0x66,0xe6,0x42,0xe3,0xa3,0x8f,0x42,0x09,0x06,0x5e,0x28,0xba,0x42,0x35,0xe2,0x87,0x42,0x08,0x8e,0x55, + 0xc0,0x42,0xb8,0x4d,0x86,0x42,0x60,0xbf,0xd7,0x42,0x3e,0xf0,0x91,0x42,0x63,0xf6,0xe4,0x42,0x70,0xed, + 0x8f,0x42,0x08,0x7a,0x89,0xe5,0x42,0xac,0xc8,0x8f,0x42,0xcc,0xf7,0xe5,0x42,0xac,0xc8,0x8f,0x42,0x1b, + 0x66,0xe6,0x42,0xe3,0xa3,0x8f,0x42,0x07,0x1b,0x66,0xe6,0x42,0x73,0xf4,0x94,0x42,0x08,0x63,0xf6,0xe4, + 0x42,0x3b,0x19,0x95,0x42,0xe6,0x61,0xe3,0x42,0x00,0x3e,0x95,0x42,0xf4,0x16,0xe2,0x42,0xc4,0x62,0x95, + 0x42,0x08,0x6e,0x74,0xd6,0x42,0x15,0xd1,0x95,0x42,0x97,0x63,0xca,0x42,0xaf,0xcf,0x94,0x42,0xfb,0x2d, + 0xbe,0x42,0x86,0x80,0x90,0x42,0x08,0x97,0x03,0xba,0x42,0xce,0x10,0x8f,0x42,0x5e,0x28,0xba,0x42,0x3e, + 0xf0,0x91,0x42,0xf2,0x4f,0xbc,0x42,0x45,0xf7,0x96,0x42,0x08,0x27,0x54,0xbf,0x42,0x73,0x24,0x9d,0x42, + 0xa5,0xe8,0xc0,0x42,0x86,0xe0,0xa0,0x42,0xe4,0xca,0xc5,0x42,0xed,0x11,0xaa,0x42,0x08,0x54,0xaa,0xc8, + 0x42,0x86,0x40,0xb1,0x42,0x59,0x81,0xc5,0x42,0xa1,0x11,0xc4,0x42,0x3e,0xe7,0xbf,0x42,0xfb,0x8d,0xce, + 0x42,0x08,0xb4,0x6d,0xb7,0x42,0x30,0xc2,0xd9,0x42,0x46,0xf5,0xc9,0x42,0xdf,0x53,0xd9,0x42,0x38,0x40, + 0xcb,0x42,0x62,0x8f,0xcf,0x42,0x08,0x7d,0xf9,0xcc,0x42,0xec,0xa1,0xc2,0x42,0x07,0x43,0xcd,0x42,0x6c, + 0xdd,0xb8,0x42,0x2b,0x8b,0xcc,0x42,0x92,0xf5,0xaf,0x42,0x08,0xf9,0x8d,0xce,0x42,0x41,0x57,0xa7,0x42, + 0x5b,0xb8,0xd2,0x42,0xae,0x2f,0xa5,0x42,0x18,0x2f,0xd9,0x42,0x13,0x2a,0xa1,0x42,0x08,0x41,0x7e,0xdd, + 0x42,0xe3,0x03,0xa0,0x42,0x2e,0xf2,0xe1,0x42,0x7c,0x02,0x9f,0x42,0x1b,0x66,0xe6,0x42,0xa2,0x4a,0x9e, + 0x42,0x07,0x1b,0x66,0xe6,0x42,0xae,0x2f,0xa5,0x42,0x08,0x4d,0x63,0xe4,0x42,0x00,0x9e,0xa5,0x42,0xf4, + 0x16,0xe2,0x42,0x15,0x31,0xa6,0x42,0x99,0xca,0xdf,0x42,0x2b,0xc4,0xa6,0x42,0x08,0xc0,0x82,0xc6,0x42, + 0xc4,0xc2,0xa5,0x42,0x57,0xe1,0xd5,0x42,0x91,0xb5,0xd0,0x42,0x54,0xda,0xd0,0x42,0x97,0x93,0xd2,0x42, + 0x08,0x9c,0x3a,0xc7,0x42,0x17,0x58,0xdc,0x42,0x9c,0x0a,0xbf,0x42,0x6e,0xa4,0xde,0x42,0x90,0x25,0xb8, + 0x42,0xdf,0x53,0xd9,0x42,0x08,0x59,0x21,0xb5,0x42,0xf2,0xdf,0xd4,0x42,0x51,0x43,0xb3,0x42,0x91,0xb5, + 0xd0,0x42,0xc5,0x29,0xbb,0x42,0x0e,0x1a,0xca,0x42,0x08,0x65,0x36,0xc4,0x42,0xd0,0x07,0xbd,0x42,0x3e, + 0xe7,0xbf,0x42,0x37,0x09,0xbe,0x42,0x0c,0xea,0xc1,0x42,0xcd,0xd0,0xaf,0x42,0x08,0x2b,0x5b,0xc4,0x42, + 0x18,0x08,0xa3,0x42,0x67,0xa6,0xab,0x42,0x99,0x3c,0x94,0x42,0x5e,0x28,0xba,0x42,0x35,0xe2,0x87,0x42, + 0x09,]; + +private struct ThePath { +public: + enum Command { + Bounds, // always first, has 4 args (x0, y0, x1, y1) + StrokeMode, + FillMode, + StrokeFillMode, + NormalStroke, + ThinStroke, + MoveTo, + LineTo, + CubicTo, // cubic bezier + EndPath, + } + +public: + const(ubyte)[] path; + uint ppos; + +public: + this (const(void)[] apath) pure nothrow @trusted @nogc { + path = cast(const(ubyte)[])apath; + } + + @property bool empty () const pure nothrow @safe @nogc { pragma(inline, true); return (ppos >= path.length); } + + Command getCommand () nothrow @trusted @nogc { + pragma(inline, true); + if (ppos >= cast(uint)path.length) assert(0, "invalid path"); + return cast(Command)(path.ptr[ppos++]); + } + + // number of (x,y) pairs for this command + static int argCount (in Command cmd) nothrow @safe @nogc { + version(aliced) pragma(inline, true); + if (cmd == Command.Bounds) return 2; + else if (cmd == Command.MoveTo || cmd == Command.LineTo) return 1; + else if (cmd == Command.CubicTo) return 3; + else return 0; + } + + void skipArgs (int argc) nothrow @trusted @nogc { + pragma(inline, true); + ppos += cast(uint)(float.sizeof*2*argc); + } + + float getFloat () nothrow @trusted @nogc { + pragma(inline, true); + if (ppos >= cast(uint)path.length || cast(uint)path.length-ppos < float.sizeof) assert(0, "invalid path"); + version(LittleEndian) { + float res = *cast(const(float)*)(&path.ptr[ppos]); + ppos += cast(uint)float.sizeof; + return res; + } else { + static assert(float.sizeof == 4); + uint xp = path.ptr[ppos]|(path.ptr[ppos+1]<<8)|(path.ptr[ppos+2]<<16)|(path.ptr[ppos+3]<<24); + ppos += cast(uint)float.sizeof; + return *cast(const(float)*)(&xp); + } + } +} + +// this will add baphomet's background path to the current NanoVega path, so you can fill it. +public void addBaphometBack (NVGContext nvg, float ofsx=0, float ofsy=0, float scalex=1, float scaley=1) nothrow @trusted @nogc { + if (nvg is null) return; + + auto path = ThePath(baphometPath); + + float getScaledX () nothrow @trusted @nogc { pragma(inline, true); return (ofsx+path.getFloat()*scalex); } + float getScaledY () nothrow @trusted @nogc { pragma(inline, true); return (ofsy+path.getFloat()*scaley); } + + bool inPath = false; + while (!path.empty) { + auto cmd = path.getCommand(); + switch (cmd) { + case ThePath.Command.MoveTo: + inPath = true; + immutable float ex = getScaledX(); + immutable float ey = getScaledY(); + nvg.moveTo(ex, ey); + break; + case ThePath.Command.LineTo: + inPath = true; + immutable float ex = getScaledX(); + immutable float ey = getScaledY(); + nvg.lineTo(ex, ey); + break; + case ThePath.Command.CubicTo: // cubic bezier + inPath = true; + immutable float x1 = getScaledX(); + immutable float y1 = getScaledY(); + immutable float x2 = getScaledX(); + immutable float y2 = getScaledY(); + immutable float ex = getScaledX(); + immutable float ey = getScaledY(); + nvg.bezierTo(x1, y1, x2, y2, ex, ey); + break; + case ThePath.Command.EndPath: + if (inPath) return; + break; + default: + path.skipArgs(path.argCount(cmd)); + break; + } + } +} + +// this will add baphomet's pupil paths to the current NanoVega path, so you can fill it. +public void addBaphometPupils(bool left=true, bool right=true) (NVGContext nvg, float ofsx=0, float ofsy=0, float scalex=1, float scaley=1) nothrow @trusted @nogc { + // pupils starts with "fill-and-stroke" mode + if (nvg is null) return; + + auto path = ThePath(baphometPath); + + float getScaledX () nothrow @trusted @nogc { pragma(inline, true); return (ofsx+path.getFloat()*scalex); } + float getScaledY () nothrow @trusted @nogc { pragma(inline, true); return (ofsy+path.getFloat()*scaley); } + + bool inPath = false; + bool pupLeft = true; + while (!path.empty) { + auto cmd = path.getCommand(); + switch (cmd) { + case ThePath.Command.StrokeFillMode: inPath = true; break; + case ThePath.Command.MoveTo: + if (!inPath) goto default; + static if (!left) { if (pupLeft) goto default; } + static if (!right) { if (!pupLeft) goto default; } + immutable float ex = getScaledX(); + immutable float ey = getScaledY(); + nvg.moveTo(ex, ey); + break; + case ThePath.Command.LineTo: + if (!inPath) goto default; + static if (!left) { if (pupLeft) goto default; } + static if (!right) { if (!pupLeft) goto default; } + immutable float ex = getScaledX(); + immutable float ey = getScaledY(); + nvg.lineTo(ex, ey); + break; + case ThePath.Command.CubicTo: // cubic bezier + if (!inPath) goto default; + static if (!left) { if (pupLeft) goto default; } + static if (!right) { if (!pupLeft) goto default; } + immutable float x1 = getScaledX(); + immutable float y1 = getScaledY(); + immutable float x2 = getScaledX(); + immutable float y2 = getScaledY(); + immutable float ex = getScaledX(); + immutable float ey = getScaledY(); + nvg.bezierTo(x1, y1, x2, y2, ex, ey); + break; + case ThePath.Command.EndPath: + if (inPath) { + if (pupLeft) pupLeft = false; else return; + } + break; + default: + path.skipArgs(path.argCount(cmd)); + break; + } + } +} + +// mode: 'f' to allow fills; 's' to allow strokes; 'w' to allow stroke widths; 'c' to replace fills with strokes +public void renderBaphomet(string mode="fs") (NVGContext nvg, float ofsx=0, float ofsy=0, float scalex=1, float scaley=1) nothrow @trusted @nogc { + template hasChar(char ch, string s) { + static if (s.length == 0) enum hasChar = false; + else static if (s[0] == ch) enum hasChar = true; + else enum hasChar = hasChar!(ch, s[1..$]); + } + enum AllowStroke = hasChar!('s', mode); + enum AllowFill = hasChar!('f', mode); + enum AllowWidth = hasChar!('w', mode); + enum Contour = hasChar!('c', mode); + //static assert(AllowWidth || AllowFill); + + if (nvg is null) return; + + auto path = ThePath(baphometPath); + + float getScaledX () nothrow @trusted @nogc { pragma(inline, true); return (ofsx+path.getFloat()*scalex); } + float getScaledY () nothrow @trusted @nogc { pragma(inline, true); return (ofsy+path.getFloat()*scaley); } + + int mode = 0; + int sw = ThePath.Command.NormalStroke; + nvg.beginPath(); + while (!path.empty) { + auto cmd = path.getCommand(); + switch (cmd) { + case ThePath.Command.StrokeMode: mode = ThePath.Command.StrokeMode; break; + case ThePath.Command.FillMode: mode = ThePath.Command.FillMode; break; + case ThePath.Command.StrokeFillMode: mode = ThePath.Command.StrokeFillMode; break; + case ThePath.Command.NormalStroke: sw = ThePath.Command.NormalStroke; break; + case ThePath.Command.ThinStroke: sw = ThePath.Command.ThinStroke; break; + case ThePath.Command.MoveTo: + immutable float ex = getScaledX(); + immutable float ey = getScaledY(); + nvg.moveTo(ex, ey); + break; + case ThePath.Command.LineTo: + immutable float ex = getScaledX(); + immutable float ey = getScaledY(); + nvg.lineTo(ex, ey); + break; + case ThePath.Command.CubicTo: // cubic bezier + immutable float x1 = getScaledX(); + immutable float y1 = getScaledY(); + immutable float x2 = getScaledX(); + immutable float y2 = getScaledY(); + immutable float ex = getScaledX(); + immutable float ey = getScaledY(); + nvg.bezierTo(x1, y1, x2, y2, ex, ey); + break; + case ThePath.Command.EndPath: + if (mode == ThePath.Command.FillMode || mode == ThePath.Command.StrokeFillMode) { + static if (AllowFill || Contour) { + static if (Contour) { + if (mode == ThePath.Command.FillMode) { nvg.strokeWidth = 1; nvg.stroke(); } + } else { + nvg.fill(); + } + } + } + if (mode == ThePath.Command.StrokeMode || mode == ThePath.Command.StrokeFillMode) { + static if (AllowStroke || Contour) { + static if (AllowWidth) { + if (sw == ThePath.Command.NormalStroke) nvg.strokeWidth = 1; + else if (sw == ThePath.Command.ThinStroke) nvg.strokeWidth = 0.5; + else assert(0, "wtf?!"); + } + nvg.stroke(); + } + } + nvg.newPath(); + break; + default: + path.skipArgs(path.argCount(cmd)); + break; + } + } + nvg.newPath(); +} diff --git a/svg.d b/svg.d new file mode 100644 index 0000000..6fefc51 --- /dev/null +++ b/svg.d @@ -0,0 +1,5218 @@ +/* + * Copyright (c) 2013-14 Mikko Mononen memon@inside.org + * + * This software is provided 'as-is', without any express or implied + * warranty. In no event will the authors be held liable for any damages + * arising from the use of this software. + * + * Permission is granted to anyone to use this software for any purpose, + * including commercial applications, and to alter it and redistribute it + * freely, subject to the following restrictions: + * + * 1. The origin of this software must not be misrepresented; you must not + * claim that you wrote the original software. If you use this software + * in a product, an acknowledgment in the product documentation would be + * appreciated but is not required. + * 2. Altered source versions must be plainly marked as such, and must not be + * misrepresented as being the original software. + * 3. This notice may not be removed or altered from any source distribution. + * + * The SVG parser is based on Anti-Grain Geometry 2.4 SVG example + * Copyright (C) 2002-2004 Maxim Shemanarev (McSeem) (http://www.antigrain.com/) + * + * Arc calculation code based on canvg (https://code.google.com/p/canvg/) + * + * Bounding box calculation based on http://blog.hackers-cafe.net/2009/06/how-to-calculate-bezier-curves-bounding.html + * + * Fork developement, feature integration and new bugs: + * Ketmar // Invisible Vector + * Contains code from various contributors. + */ +/** + NanoVega.SVG is a simple stupid SVG parser. The output of the parser is a list of drawing commands. + + The library suits well for anything from rendering scalable icons in your editor application to prototyping a game. + + NanoVega.SVG supports a wide range of SVG features, but something may be missing. Your's Captain Obvious. + + + The shapes in the SVG images are transformed by the viewBox and converted to specified units. + That is, you should get the same looking data as your designed in your favorite app. + + NanoVega.SVG can return the paths in few different units. For example if you want to render an image, you may choose + to get the paths in pixels, or if you are feeding the data into a CNC-cutter, you may want to use millimeters. + + The units passed to NanoVega.SVG should be one of: 'px', 'pt', 'pc', 'mm', 'cm', 'in'. + DPI (dots-per-inch) controls how the unit conversion is done. + + If you don't know or care about the units stuff, "px" and 96 should get you going. + + Example Usage: + + --- + // Load + NSVG* image = nsvgParseFromFile("test.svg", "px", 96); + printf("size: %f x %f\n", image.width, image.height); + // Use... + image.forEachShape((in ref NSVG.Shape shape) { + if (!shape.visible) return; + shape.forEachPath((in ref NSVG.Path path) { + // this will issue final `LineTo` for closed pathes + path.forEachCommand!true(delegate (NSVG.Command cmd, const(float)[] args) nothrow @trusted @nogc { + final switch (cmd) { + case NSVG.Command.MoveTo: nvg.moveTo(args); break; + case NSVG.Command.LineTo: nvg.lineTo(args); break; + case NSVG.Command.QuadTo: nvg.quadTo(args); break; + case NSVG.Command.BezierTo: nvg.bezierTo(args); break; + } + }); + }); + }); + + NSVGrasterizer rast = nsvgCreateRasterizer(); + // Allocate memory for image + ubyte* img = malloc(w*h*4); + // Rasterize + nsvgRasterize(rast, image, 0, 0, 1, img, w, h, w*4); + + // Delete + image.kill(); + --- + */ +module arsd.svg; + +private import core.stdc.math : fabs, fabsf, atan2f, acosf, cosf, sinf, tanf, sqrt, sqrtf, floorf, ceilf, fmodf; +//private import iv.vfs; + +version(nanosvg_disable_vfs) { + enum NanoSVGHasIVVFS = false; +} else { + static if (is(typeof((){import iv.vfs;}))) { + enum NanoSVGHasIVVFS = true; + import iv.vfs; + } else { + enum NanoSVGHasIVVFS = false; + } +} + +version(aliced) {} else { + private alias usize = size_t; +} + +version = nanosvg_crappy_stylesheet_parser; +//version = nanosvg_debug_styles; +//version(rdmd) import iv.strex; + +//version = nanosvg_use_beziers; // convert everything to beziers +//version = nanosvg_only_cubic_beziers; // convert everything to cubic beziers + +/// +public enum NSVGDefaults { + CanvasWidth = 800, + CanvasHeight = 600, +} + + +// ////////////////////////////////////////////////////////////////////////// // +public alias NSVGrasterizer = NSVGrasterizerS*; /// +public alias NSVGRasterizer = NSVGrasterizer; /// + +/// +struct NSVG { + @disable this (this); + + /// + enum Command : int { + MoveTo, /// + LineTo, /// + QuadTo, /// + BezierTo, /// cubic bezier + } + + /// + enum PaintType : ubyte { + None, /// + Color, /// + LinearGradient, /// + RadialGradient, /// + } + + /// + enum SpreadType : ubyte { + Pad, /// + Reflect, /// + Repeat, /// + } + + /// + enum LineJoin : ubyte { + Miter, /// + Round, /// + Bevel, /// + } + + /// + enum LineCap : ubyte { + Butt, /// + Round, /// + Square, /// + } + + /// + enum FillRule : ubyte { + NonZero, /// + EvenOdd, /// + } + + alias Flags = ubyte; /// + enum : ubyte { + Visible = 0x01, /// + } + + /// + static struct GradientStop { + uint color; /// + float offset; /// + } + + /// + static struct Gradient { + float[6] xform; /// + SpreadType spread; /// + float fx, fy; /// + int nstops; /// + GradientStop[0] stops; /// + } + + /// + static struct Paint { + pure nothrow @safe @nogc: + @disable this (this); + PaintType type; /// + union { + uint color; /// + Gradient* gradient; /// + } + static uint rgb (ubyte r, ubyte g, ubyte b) { pragma(inline, true); return (r|(g<<8)|(b<<16)); } /// + @property const { + bool isNone () { pragma(inline, true); return (type == PaintType.None); } /// + bool isColor () { pragma(inline, true); return (type == PaintType.Color); } /// + // gradient types + bool isLinear () { pragma(inline, true); return (type == PaintType.LinearGradient); } /// + bool isRadial () { pragma(inline, true); return (type == PaintType.RadialGradient); } /// + // color + ubyte r () { pragma(inline, true); return color&0xff; } /// + ubyte g () { pragma(inline, true); return (color>>8)&0xff; } /// + ubyte b () { pragma(inline, true); return (color>>16)&0xff; } /// + ubyte a () { pragma(inline, true); return (color>>24)&0xff; } /// + } + } + + /// + static struct Path { + @disable this (this); + float* stream; /// Command, args...; Cubic bezier points: x0,y0, [cpx1,cpx1,cpx2,cpy2,x1,y1], ... + int nsflts; /// Total number of floats in stream. + bool closed; /// Flag indicating if shapes should be treated as closed. + float[4] bounds; /// Tight bounding box of the shape [minx,miny,maxx,maxy]. + NSVG.Path* next; /// Pointer to next path, or null if last element. + + /// + @property bool empty () const pure nothrow @safe @nogc { pragma(inline, true); return (nsflts == 0); } + + /// + float startX () const nothrow @trusted @nogc { + pragma(inline, true); + return (nsflts >= 3 && cast(Command)stream[0] == Command.MoveTo ? stream[1] : float.nan); + } + + /// + float startY () const nothrow @trusted @nogc { + pragma(inline, true); + return (nsflts >= 3 && cast(Command)stream[0] == Command.MoveTo ? stream[2] : float.nan); + } + + /// + bool startPoint (float* dx, float* dy) const nothrow @trusted @nogc { + if (nsflts >= 3 && cast(Command)stream[0] == Command.MoveTo) { + if (dx !is null) *dx = stream[1]; + if (dy !is null) *dy = stream[2]; + return true; + } else { + if (dx !is null) *dx = 0; + if (dy !is null) *dy = 0; + return false; + } + } + + /// + int countCubics () const nothrow @trusted @nogc { + if (nsflts < 3) return 0; + int res = 0, argc; + for (int pidx = 0; pidx+3 <= nsflts; ) { + final switch (cast(Command)stream[pidx++]) { + case Command.MoveTo: argc = 2; break; + case Command.LineTo: argc = 2; ++res; break; + case Command.QuadTo: argc = 4; ++res; break; + case Command.BezierTo: argc = 6; ++res; break; + } + if (pidx+argc > nsflts) break; // just in case + pidx += argc; + } + return res; + } + + /// + int countCommands(bool synthesizeCloseCommand=true) () const nothrow @trusted @nogc { + if (nsflts < 3) return 0; + int res = 0, argc; + for (int pidx = 0; pidx+3 <= nsflts; ) { + ++res; + final switch (cast(Command)stream[pidx++]) { + case Command.MoveTo: argc = 2; break; + case Command.LineTo: argc = 2; break; + case Command.QuadTo: argc = 4; break; + case Command.BezierTo: argc = 6; break; + } + if (pidx+argc > nsflts) break; // just in case + pidx += argc; + } + static if (synthesizeCloseCommand) { if (closed) ++res; } + return res; + } + + /// emits cubic beziers. + /// if `withMoveTo` is `false`, issue 8-arg commands for cubic beziers (i.e. include starting point). + /// if `withMoveTo` is `true`, issue 2-arg command for `moveTo`, and 6-arg command for cubic beziers. + void asCubics(bool withMoveTo=false, DG) (scope DG dg) inout if (__traits(compiles, (){ DG xdg; float[] f; xdg(f); })) { + if (dg is null) return; + if (nsflts < 3) return; + enum HasRes = __traits(compiles, (){ DG xdg; float[] f; bool res = xdg(f); }); + float cx = 0, cy = 0; + float[8] cubic = void; + + void synthLine (in float cx, in float cy, in float x, in float y) nothrow @trusted @nogc { + immutable float dx = x-cx; + immutable float dy = y-cy; + cubic.ptr[0] = cx; + cubic.ptr[1] = cy; + cubic.ptr[2] = cx+dx/3.0f; + cubic.ptr[3] = cy+dy/3.0f; + cubic.ptr[4] = x-dx/3.0f; + cubic.ptr[5] = y-dy/3.0f; + cubic.ptr[6] = x; + cubic.ptr[7] = y; + } + + void synthQuad (in float cx, in float cy, in float x1, in float y1, in float x2, in float y2) nothrow @trusted @nogc { + immutable float cx1 = x1+2.0f/3.0f*(cx-x1); + immutable float cy1 = y1+2.0f/3.0f*(cy-y1); + immutable float cx2 = x2+2.0f/3.0f*(cx-x2); + immutable float cy2 = y2+2.0f/3.0f*(cy-y2); + cubic.ptr[0] = cx; + cubic.ptr[1] = cy; + cubic.ptr[2] = cx1; + cubic.ptr[3] = cy2; + cubic.ptr[4] = cx2; + cubic.ptr[5] = cy2; + cubic.ptr[6] = x2; + cubic.ptr[7] = y2; + } + + for (int pidx = 0; pidx+3 <= nsflts; ) { + final switch (cast(Command)stream[pidx++]) { + case Command.MoveTo: + static if (withMoveTo) { + static if (HasRes) { if (dg(stream[pidx+0..pidx+2])) return; } else { dg(stream[pidx+0..pidx+2]); } + } + cx = stream[pidx++]; + cy = stream[pidx++]; + continue; + case Command.LineTo: + synthLine(cx, cy, stream[pidx+0], stream[pidx+1]); + pidx += 2; + break; + case Command.QuadTo: + synthQuad(cx, cy, stream[pidx+0], stream[pidx+1], stream[pidx+2], stream[pidx+3]); + pidx += 4; + break; + case Command.BezierTo: + cubic.ptr[0] = cx; + cubic.ptr[1] = cy; + cubic.ptr[2..8] = stream[pidx..pidx+6]; + pidx += 6; + break; + } + cx = cubic.ptr[6]; + cy = cubic.ptr[7]; + static if (withMoveTo) { + static if (HasRes) { if (dg(cubic[2..8])) return; } else { dg(cubic[2..8]); } + } else { + static if (HasRes) { if (dg(cubic[])) return; } else { dg(cubic[]); } + } + } + } + + /// if `synthesizeCloseCommand` is true, and the path is closed, this emits line to the first point. + void forEachCommand(bool synthesizeCloseCommand=true, DG) (scope DG dg) inout + if (__traits(compiles, (){ DG xdg; Command c; const(float)[] f; xdg(c, f); })) + { + if (dg is null) return; + if (nsflts < 3) return; + enum HasRes = __traits(compiles, (){ DG xdg; Command c; const(float)[] f; bool res = xdg(c, f); }); + int argc; + Command cmd; + for (int pidx = 0; pidx+3 <= nsflts; ) { + cmd = cast(Command)stream[pidx++]; + final switch (cmd) { + case Command.MoveTo: argc = 2; break; + case Command.LineTo: argc = 2; break; + case Command.QuadTo: argc = 4; break; + case Command.BezierTo: argc = 6; break; + } + if (pidx+argc > nsflts) break; // just in case + static if (HasRes) { if (dg(cmd, stream[pidx..pidx+argc])) return; } else { dg(cmd, stream[pidx..pidx+argc]); } + pidx += argc; + } + static if (synthesizeCloseCommand) { + if (closed && cast(Command)stream[0] == Command.MoveTo) { + static if (HasRes) { if (dg(Command.LineTo, stream[1..3])) return; } else { dg(Command.LineTo, stream[1..3]); } + } + } + } + } + + /// + static struct Shape { + @disable this (this); + char[64] id = 0; /// Optional 'id' attr of the shape or its group + NSVG.Paint fill; /// Fill paint + NSVG.Paint stroke; /// Stroke paint + float opacity; /// Opacity of the shape. + float strokeWidth; /// Stroke width (scaled). + float strokeDashOffset; /// Stroke dash offset (scaled). + float[8] strokeDashArray; /// Stroke dash array (scaled). + byte strokeDashCount; /// Number of dash values in dash array. + LineJoin strokeLineJoin; /// Stroke join type. + LineCap strokeLineCap; /// Stroke cap type. + float miterLimit; /// Miter limit + FillRule fillRule; /// Fill rule, see FillRule. + /*Flags*/ubyte flags; /// Logical or of NSVG_FLAGS_* flags + float[4] bounds; /// Tight bounding box of the shape [minx,miny,maxx,maxy]. + NSVG.Path* paths; /// Linked list of paths in the image. + NSVG.Shape* next; /// Pointer to next shape, or null if last element. + + @property bool visible () const pure nothrow @safe @nogc { pragma(inline, true); return ((flags&Visible) != 0); } /// + + /// delegate can accept: + /// NSVG.Path* + /// const(NSVG.Path)* + /// ref NSVG.Path + /// in ref NSVG.Path + /// delegate can return: + /// void + /// bool (true means `stop`) + void forEachPath(DG) (scope DG dg) inout + if (__traits(compiles, (){ DG xdg; NSVG.Path s; xdg(&s); }) || + __traits(compiles, (){ DG xdg; NSVG.Path s; xdg(s); })) + { + if (dg is null) return; + enum WantPtr = __traits(compiles, (){ DG xdg; NSVG.Path s; xdg(&s); }); + static if (WantPtr) { + enum HasRes = __traits(compiles, (){ DG xdg; NSVG.Path s; bool res = xdg(&s); }); + } else { + enum HasRes = __traits(compiles, (){ DG xdg; NSVG.Path s; bool res = xdg(s); }); + } + static if (__traits(compiles, (){ NSVG.Path* s = this.paths; })) { + alias TP = NSVG.Path*; + } else { + alias TP = const(NSVG.Path)*; + } + for (TP path = paths; path !is null; path = path.next) { + static if (HasRes) { + static if (WantPtr) { + if (dg(path)) return; + } else { + if (dg(*path)) return; + } + } else { + static if (WantPtr) dg(path); else dg(*path); + } + } + } + } + + float width; /// Width of the image. + float height; /// Height of the image. + NSVG.Shape* shapes; /// Linked list of shapes in the image. + + /// delegate can accept: + /// NSVG.Shape* + /// const(NSVG.Shape)* + /// ref NSVG.Shape + /// in ref NSVG.Shape + /// delegate can return: + /// void + /// bool (true means `stop`) + void forEachShape(DG) (scope DG dg) inout + if (__traits(compiles, (){ DG xdg; NSVG.Shape s; xdg(&s); }) || + __traits(compiles, (){ DG xdg; NSVG.Shape s; xdg(s); })) + { + if (dg is null) return; + enum WantPtr = __traits(compiles, (){ DG xdg; NSVG.Shape s; xdg(&s); }); + static if (WantPtr) { + enum HasRes = __traits(compiles, (){ DG xdg; NSVG.Shape s; bool res = xdg(&s); }); + } else { + enum HasRes = __traits(compiles, (){ DG xdg; NSVG.Shape s; bool res = xdg(s); }); + } + static if (__traits(compiles, (){ NSVG.Shape* s = this.shapes; })) { + alias TP = NSVG.Shape*; + } else { + alias TP = const(NSVG.Shape)*; + } + for (TP shape = shapes; shape !is null; shape = shape.next) { + static if (HasRes) { + static if (WantPtr) { + if (dg(shape)) return; + } else { + if (dg(*shape)) return; + } + } else { + static if (WantPtr) dg(shape); else dg(*shape); + } + } + } +} + + +// ////////////////////////////////////////////////////////////////////////// // +private: +nothrow @trusted @nogc { + +// ////////////////////////////////////////////////////////////////////////// // +// sscanf replacement: just enough to replace all our cases +int xsscanf(A...) (const(char)[] str, const(char)[] fmt, ref A args) { + int spos; + while (spos < str.length && str.ptr[spos] <= ' ') ++spos; + + static int hexdigit() (char c) { + pragma(inline, true); + return + (c >= '0' && c <= '9' ? c-'0' : + c >= 'A' && c <= 'F' ? c-'A'+10 : + c >= 'a' && c <= 'f' ? c-'a'+10 : + -1); + } + + bool parseInt(T : ulong) (ref T res) { + res = 0; + debug(xsscanf_int) { import std.stdio; writeln("parseInt00: str=", str[spos..$].quote); } + bool neg = false; + if (spos < str.length && str.ptr[spos] == '+') ++spos; + else if (spos < str.length && str.ptr[spos] == '-') { neg = true; ++spos; } + if (spos >= str.length || str.ptr[spos] < '0' || str.ptr[spos] > '9') return false; + while (spos < str.length && str.ptr[spos] >= '0' && str.ptr[spos] <= '9') res = res*10+str.ptr[spos++]-'0'; + debug(xsscanf_int) { import std.stdio; writeln("parseInt10: str=", str[spos..$].quote); } + if (neg) res = -res; + return true; + } + + bool parseHex(T : ulong) (ref T res) { + res = 0; + debug(xsscanf_int) { import std.stdio; writeln("parseHex00: str=", str[spos..$].quote); } + if (spos >= str.length || hexdigit(str.ptr[spos]) < 0) return false; + while (spos < str.length) { + auto d = hexdigit(str.ptr[spos]); + if (d < 0) break; + res = res*16+d; + ++spos; + } + debug(xsscanf_int) { import std.stdio; writeln("parseHex10: str=", str[spos..$].quote); } + return true; + } + + bool parseFloat(T : real) (ref T res) { + res = 0.0; + debug(xsscanf_float) { import std.stdio; writeln("parseFloat00: str=", str[spos..$].quote); } + bool neg = false; + if (spos < str.length && str.ptr[spos] == '+') ++spos; + else if (spos < str.length && str.ptr[spos] == '-') { neg = true; ++spos; } + bool wasChar = false; + // integer part + debug(xsscanf_float) { import std.stdio; writeln("parseFloat01: str=", str[spos..$].quote); } + if (spos < str.length && str.ptr[spos] >= '0' && str.ptr[spos] <= '9') wasChar = true; + while (spos < str.length && str.ptr[spos] >= '0' && str.ptr[spos] <= '9') res = res*10+str.ptr[spos++]-'0'; + // fractional part + if (spos < str.length && str.ptr[spos] == '.') { + debug(xsscanf_float) { import std.stdio; writeln("parseFloat02: str=", str[spos..$].quote); } + T div = 1.0/10; + ++spos; + if (spos < str.length && str.ptr[spos] >= '0' && str.ptr[spos] <= '9') wasChar = true; + debug(xsscanf_float) { import std.stdio; writeln("parseFloat03: str=", str[spos..$].quote); } + while (spos < str.length && str.ptr[spos] >= '0' && str.ptr[spos] <= '9') { + res += div*(str.ptr[spos++]-'0'); + div /= 10.0; + } + debug(xsscanf_float) { import std.stdio; writeln("parseFloat04: str=", str[spos..$].quote); } + debug(xsscanf_float) { import std.stdio; writeln("div=", div, "; res=", res, "; str=", str[spos..$].quote); } + } + // '[Ee][+-]num' part + if (wasChar && spos < str.length && (str.ptr[spos] == 'E' || str.ptr[spos] == 'e')) { + debug(xsscanf_float) { import std.stdio; writeln("parseFloat05: str=", str[spos..$].quote); } + ++spos; + bool xneg = false; + if (spos < str.length && str.ptr[spos] == '+') ++spos; + else if (spos < str.length && str.ptr[spos] == '-') { xneg = true; ++spos; } + int n = 0; + if (spos >= str.length || str.ptr[spos] < '0' || str.ptr[spos] > '9') return false; // number expected + debug(xsscanf_float) { import std.stdio; writeln("parseFloat06: str=", str[spos..$].quote); } + while (spos < str.length && str.ptr[spos] >= '0' && str.ptr[spos] <= '9') n = n*10+str.ptr[spos++]-'0'; + if (xneg) { + while (n-- > 0) res /= 10; + } else { + while (n-- > 0) res *= 10; + } + debug(xsscanf_float) { import std.stdio; writeln("parseFloat07: str=", str[spos..$].quote); } + } + if (!wasChar) return false; + debug(xsscanf_float) { import std.stdio; writeln("parseFloat10: str=", str[spos..$].quote); } + if (neg) res = -res; + return true; + } + + int fpos; + + void skipXSpaces () { + if (fpos < fmt.length && fmt.ptr[fpos] <= ' ') { + while (fpos < fmt.length && fmt.ptr[fpos] <= ' ') ++fpos; + while (spos < str.length && str.ptr[spos] <= ' ') ++spos; + } + } + + bool parseImpl(T/*, usize dummy*/) (ref T res) { + while (fpos < fmt.length) { + //{ import std.stdio; writeln("spos=", spos, "; fpos=", fpos, "\nfmt=", fmt[fpos..$].quote, "\nstr=", str[spos..$].quote); } + if (fmt.ptr[fpos] <= ' ') { + skipXSpaces(); + continue; + } + if (fmt.ptr[fpos] != '%') { + if (spos >= str.length || str.ptr[spos] != fmt.ptr[spos]) return false; + ++spos; + ++fpos; + continue; + } + if (fmt.length-fpos < 2) return false; // stray percent + fpos += 2; + bool skipAss = false; + if (fmt.ptr[fpos-1] == '*') { + ++fpos; + if (fpos >= fmt.length) return false; // stray star + skipAss = true; + } + switch (fmt.ptr[fpos-1]) { + case '%': + if (spos >= str.length || str.ptr[spos] != '%') return false; + ++spos; + break; + case 'd': + static if (is(T : ulong)) { + if (skipAss) { + long v; + if (!parseInt!long(v)) return false; + } else { + return parseInt!T(res); + } + } else { + if (!skipAss) assert(0, "invalid type"); + long v; + if (!parseInt!long(v)) return false; + } + break; + case 'x': + static if (is(T : ulong)) { + if (skipAss) { + long v; + if (!parseHex!long(v)) return false; + } else { + return parseHex!T(res); + } + } else { + if (!skipAss) assert(0, "invalid type"); + ulong v; + if (!parseHex!ulong(v)) return false; + } + break; + case 'f': + static if (is(T == float) || is(T == double) || is(T == real)) { + if (skipAss) { + double v; + if (!parseFloat!double(v)) return false; + } else { + return parseFloat!T(res); + } + } else { + if (!skipAss) assert(0, "invalid type"); + double v; + if (!parseFloat!double(v)) return false; + } + break; + case '[': + if (fmt.length-fpos < 1) return false; + auto stp = spos; + while (spos < str.length) { + bool ok = false; + foreach (immutable cidx, char c; fmt[fpos..$]) { + if (cidx != 0) { + if (c == '-') assert(0, "not yet"); + if (c == ']') break; + } + if (c == ' ') { + if (str.ptr[spos] <= ' ') { ok = true; break; } + } else { + if (str.ptr[spos] == c) { ok = true; break; } + } + } + //{ import std.stdio; writeln("** spos=", spos, "; fpos=", fpos, "\nfmt=", fmt[fpos..$].quote, "\nstr=", str[spos..$].quote, "\nok: ", ok); } + if (!ok) break; // not a match + ++spos; // skip match + } + ++fpos; + while (fpos < fmt.length && fmt[fpos] != ']') ++fpos; + if (fpos < fmt.length) ++fpos; + static if (is(T == const(char)[])) { + if (!skipAss) { + res = str[stp..spos]; + return true; + } + } else { + if (!skipAss) assert(0, "invalid type"); + } + break; + case 's': + auto stp = spos; + while (spos < str.length && str.ptr[spos] > ' ') ++spos; + static if (is(T == const(char)[])) { + if (!skipAss) { + res = str[stp..spos]; + return true; + } + } else { + // skip non-spaces + if (!skipAss) assert(0, "invalid type"); + } + break; + default: assert(0, "unknown format specifier"); + } + } + return false; + } + + foreach (usize aidx, immutable T; A) { + //pragma(msg, "aidx=", aidx, "; T=", T); + if (!parseImpl!(T)(args[aidx])) return -(spos+1); + //{ import std.stdio; writeln("@@@ aidx=", aidx+3, "; spos=", spos, "; fpos=", fpos, "\nfmt=", fmt[fpos..$].quote, "\nstr=", str[spos..$].quote); } + } + skipXSpaces(); + return (fpos < fmt.length ? -(spos+1) : spos); +} + + +// ////////////////////////////////////////////////////////////////////////// // +T* xalloc(T) (usize addmem=0) if (!is(T == class)) { + import core.stdc.stdlib : malloc; + if (T.sizeof == 0 && addmem == 0) addmem = 1; + auto res = cast(ubyte*)malloc(T.sizeof+addmem+256); + if (res is null) assert(0, "NanoVega.SVG: out of memory"); + res[0..T.sizeof+addmem] = 0; + return cast(T*)res; +} + +T* xcalloc(T) (usize count) if (!is(T == class) && !is(T == struct)) { + import core.stdc.stdlib : malloc; + usize sz = T.sizeof*count; + if (sz == 0) sz = 1; + auto res = cast(ubyte*)malloc(sz+256); + if (res is null) assert(0, "NanoVega.SVG: out of memory"); + res[0..sz] = 0; + return cast(T*)res; +} + +void xfree(T) (ref T* p) { + if (p !is null) { + import core.stdc.stdlib : free; + free(p); + p = null; + } +} + + +alias AttrList = const(const(char)[])[]; + +public enum NSVG_PI = 3.14159265358979323846264338327f; /// +enum NSVG_KAPPA90 = 0.5522847493f; // Lenght proportional to radius of a cubic bezier handle for 90deg arcs. + +enum NSVG_ALIGN_MIN = 0; +enum NSVG_ALIGN_MID = 1; +enum NSVG_ALIGN_MAX = 2; +enum NSVG_ALIGN_NONE = 0; +enum NSVG_ALIGN_MEET = 1; +enum NSVG_ALIGN_SLICE = 2; + + +int nsvg__isspace() (char c) { pragma(inline, true); return (c && c <= ' '); } // because +int nsvg__isdigit() (char c) { pragma(inline, true); return (c >= '0' && c <= '9'); } +int nsvg__isnum() (char c) { pragma(inline, true); return ((c >= '0' && c <= '9') || c == '+' || c == '-' || c == '.' || c == 'e' || c == 'E'); } + +int nsvg__hexdigit() (char c) { + pragma(inline, true); + return + (c >= '0' && c <= '9' ? c-'0' : + c >= 'A' && c <= 'F' ? c-'A'+10 : + c >= 'a' && c <= 'f' ? c-'a'+10 : + -1); +} + +float nsvg__minf() (float a, float b) { pragma(inline, true); return (a < b ? a : b); } +float nsvg__maxf() (float a, float b) { pragma(inline, true); return (a > b ? a : b); } + + +// Simple XML parser +enum NSVG_XML_TAG = 1; +enum NSVG_XML_CONTENT = 2; +enum NSVG_XML_MAX_ATTRIBS = 256; + +void nsvg__parseContent (const(char)[] s, scope void function (void* ud, const(char)[] s) nothrow @nogc contentCb, void* ud) { + // Trim start white spaces + while (s.length && nsvg__isspace(s[0])) s = s[1..$]; + if (s.length == 0) return; + //{ import std.stdio; writeln("s=", s.quote); } + if (contentCb !is null) contentCb(ud, s); +} + +static void nsvg__parseElement (const(char)[] s, + scope void function (void* ud, const(char)[] el, AttrList attr) nothrow @nogc startelCb, + scope void function (void* ud, const(char)[] el) nothrow @nogc endelCb, + void* ud) +{ + const(char)[][NSVG_XML_MAX_ATTRIBS] attr; + int nattr = 0; + const(char)[] name; + int start = 0; + int end = 0; + char quote; + + // Skip white space after the '<' + while (s.length && nsvg__isspace(s[0])) s = s[1..$]; + + // Check if the tag is end tag + if (s.length && s[0] == '/') { + s = s[1..$]; + end = 1; + } else { + start = 1; + } + + // Skip comments, data and preprocessor stuff. + if (s.length == 0 || s[0] == '?' || s[0] == '!') return; + + // Get tag name + //{ import std.stdio; writeln("bs=", s.quote); } + { + usize pos = 0; + while (pos < s.length && !nsvg__isspace(s[pos])) ++pos; + name = s[0..pos]; + s = s[pos..$]; + } + //{ import std.stdio; writeln("name=", name.quote); } + //{ import std.stdio; writeln("as=", s.quote); } + + // Get attribs + while (!end && s.length && attr.length-nattr >= 2) { + // skip white space before the attrib name + while (s.length && nsvg__isspace(s[0])) s = s[1..$]; + if (s.length == 0) break; + if (s[0] == '/') { end = 1; break; } + // find end of the attrib name + { + usize pos = 0; + while (pos < s.length && !nsvg__isspace(s[pos]) && s[pos] != '=') ++pos; + attr[nattr++] = s[0..pos]; + s = s[pos..$]; + } + // skip until the beginning of the value + while (s.length && s[0] != '\"' && s[0] != '\'') s = s[1..$]; + if (s.length == 0) break; + // store value and find the end of it + quote = s[0]; + s = s[1..$]; + { + usize pos = 0; + while (pos < s.length && s[pos] != quote) ++pos; + attr[nattr++] = s[0..pos]; + s = s[pos+(pos < s.length ? 1 : 0)..$]; + } + //{ import std.stdio; writeln("n=", attr[nattr-2].quote, "\nv=", attr[nattr-1].quote, "\n"); } + } + + debug(nanosvg) { + import std.stdio; + writeln("==========================="); + foreach (immutable idx, const(char)[] v; attr[0..nattr]) writeln(" #", idx, ": ", v.quote); + } + + // Call callbacks. + if (start && startelCb !is null) startelCb(ud, name, attr[0..nattr]); + if (end && endelCb !is null) endelCb(ud, name); +} + +void nsvg__parseXML (const(char)[] input, + scope void function (void* ud, const(char)[] el, AttrList attr) nothrow @nogc startelCb, + scope void function (void* ud, const(char)[] el) nothrow @nogc endelCb, + scope void function (void* ud, const(char)[] s) nothrow @nogc contentCb, + void* ud) +{ + usize cpos = 0; + int state = NSVG_XML_CONTENT; + while (cpos < input.length) { + if (state == NSVG_XML_CONTENT && input[cpos] == '<') { + if (input.length-cpos >= 9 && input[cpos..cpos+9] == " 1 && input.ptr[cpos] == ']' && input.ptr[cpos+1] == ']') { + cpos += 2; + while (cpos < input.length && input.ptr[cpos] <= ' ') ++cpos; + if (cpos < input.length && input.ptr[cpos] == '>') { ++cpos; break; } + } else { + ++cpos; + } + } + continue; + } + // start of a tag + //{ import std.stdio; writeln("ctx: ", input[0..cpos].quote); } + ////version(nanosvg_debug_styles) { import std.stdio; writeln("ctx: ", input[0..cpos].quote); } + nsvg__parseContent(input[0..cpos], contentCb, ud); + input = input[cpos+1..$]; + if (input.length > 2 && input.ptr[0] == '!' && input.ptr[1] == '-' && input.ptr[2] == '-') { + //{ import std.stdio; writeln("ctx0: ", input.quote); } + // skip comments + cpos = 3; + while (cpos < input.length) { + if (input.length-cpos > 2 && input.ptr[cpos] == '-' && input.ptr[cpos+1] == '-' && input.ptr[cpos+2] == '>') { + cpos += 3; + break; + } + ++cpos; + } + input = input[cpos..$]; + //{ import std.stdio; writeln("ctx1: ", input.quote); } + } else { + state = NSVG_XML_TAG; + } + cpos = 0; + } else if (state == NSVG_XML_TAG && input[cpos] == '>') { + // start of a content or new tag + //{ import std.stdio; writeln("tag: ", input[0..cpos].quote); } + nsvg__parseElement(input[0..cpos], startelCb, endelCb, ud); + input = input[cpos+1..$]; + cpos = 0; + state = NSVG_XML_CONTENT; + } else { + ++cpos; + } + } +} + + +/* Simple SVG parser. */ + +enum NSVG_MAX_ATTR = 128; + +enum GradientUnits : ubyte { + User, + Object, +} + +enum NSVG_MAX_DASHES = 8; + +enum Units : ubyte { + user, + px, + pt, + pc, + mm, + cm, + in_, + percent, + em, + ex, +} + +struct Coordinate { + float value; + Units units; +} + +struct LinearData { + Coordinate x1, y1, x2, y2; +} + +struct RadialData { + Coordinate cx, cy, r, fx, fy; +} + +struct GradientData { + char[64] id = 0; + char[64] ref_ = 0; + NSVG.PaintType type; + union { + LinearData linear; + RadialData radial; + } + NSVG.SpreadType spread; + GradientUnits units; + float[6] xform; + int nstops; + NSVG.GradientStop* stops; + GradientData* next; +} + +struct Attrib { + char[64] id = 0; + float[6] xform; + uint fillColor; + uint strokeColor; + float opacity; + float fillOpacity; + float strokeOpacity; + char[64] fillGradient = 0; + char[64] strokeGradient = 0; + float strokeWidth; + float strokeDashOffset; + float[NSVG_MAX_DASHES] strokeDashArray; + int strokeDashCount; + NSVG.LineJoin strokeLineJoin; + NSVG.LineCap strokeLineCap; + float miterLimit; + NSVG.FillRule fillRule; + float fontSize; + uint stopColor; + float stopOpacity; + float stopOffset; + ubyte hasFill; + ubyte hasStroke; + ubyte visible; +} + +version(nanosvg_crappy_stylesheet_parser) { +struct Style { + const(char)[] name; + const(char)[] value; +} +} + +struct Parser { + Attrib[NSVG_MAX_ATTR] attr; + int attrHead; + float* stream; + int nsflts; + int csflts; + NSVG.Path* plist; + NSVG* image; + GradientData* gradients; + NSVG.Shape* shapesTail; + float viewMinx, viewMiny, viewWidth, viewHeight; + int alignX, alignY, alignType; + float dpi; + bool pathFlag; + bool defsFlag; + int canvaswdt = -1; + int canvashgt = -1; + version(nanosvg_crappy_stylesheet_parser) { + Style* styles; + uint styleCount; + bool inStyle; + } +} + +const(char)[] fromAsciiz (const(char)[] s) { + //foreach (immutable idx, char ch; s) if (!ch) return s[0..idx]; + //return s; + if (s.length) { + import core.stdc.string : memchr; + if (auto zp = cast(const(char)*)memchr(s.ptr, 0, s.length)) return s[0..cast(usize)(zp-s.ptr)]; + } + return s; +} + +// ////////////////////////////////////////////////////////////////////////// // +// matrix operations made public for the sake of... something. + +/// +public void nsvg__xformIdentity (float* t) { + t[0] = 1.0f; t[1] = 0.0f; + t[2] = 0.0f; t[3] = 1.0f; + t[4] = 0.0f; t[5] = 0.0f; +} + +/// +public void nsvg__xformSetTranslation (float* t, in float tx, in float ty) { + t[0] = 1.0f; t[1] = 0.0f; + t[2] = 0.0f; t[3] = 1.0f; + t[4] = tx; t[5] = ty; +} + +/// +public void nsvg__xformSetScale (float* t, in float sx, in float sy) { + t[0] = sx; t[1] = 0.0f; + t[2] = 0.0f; t[3] = sy; + t[4] = 0.0f; t[5] = 0.0f; +} + +/// +public void nsvg__xformSetSkewX (float* t, in float a) { + t[0] = 1.0f; t[1] = 0.0f; + t[2] = tanf(a); t[3] = 1.0f; + t[4] = 0.0f; t[5] = 0.0f; +} + +/// +public void nsvg__xformSetSkewY (float* t, in float a) { + t[0] = 1.0f; t[1] = tanf(a); + t[2] = 0.0f; t[3] = 1.0f; + t[4] = 0.0f; t[5] = 0.0f; +} + +/// +public void nsvg__xformSetRotation (float* t, in float a) { + immutable cs = cosf(a), sn = sinf(a); + t[0] = cs; t[1] = sn; + t[2] = -sn; t[3] = cs; + t[4] = 0.0f; t[5] = 0.0f; +} + +/// +public void nsvg__xformMultiply (float* t, const(float)* s) { + immutable t0 = t[0]*s[0]+t[1]*s[2]; + immutable t2 = t[2]*s[0]+t[3]*s[2]; + immutable t4 = t[4]*s[0]+t[5]*s[2]+s[4]; + t[1] = t[0]*s[1]+t[1]*s[3]; + t[3] = t[2]*s[1]+t[3]*s[3]; + t[5] = t[4]*s[1]+t[5]*s[3]+s[5]; + t[0] = t0; + t[2] = t2; + t[4] = t4; +} + +/// +public void nsvg__xformInverse (float* inv, const(float)* t) { + immutable double det = cast(double)t[0]*t[3]-cast(double)t[2]*t[1]; + if (det > -1e-6 && det < 1e-6) { + nsvg__xformIdentity(inv); + return; + } + immutable double invdet = 1.0/det; + inv[0] = cast(float)(t[3]*invdet); + inv[2] = cast(float)(-t[2]*invdet); + inv[4] = cast(float)((cast(double)t[2]*t[5]-cast(double)t[3]*t[4])*invdet); + inv[1] = cast(float)(-t[1]*invdet); + inv[3] = cast(float)(t[0]*invdet); + inv[5] = cast(float)((cast(double)t[1]*t[4]-cast(double)t[0]*t[5])*invdet); +} + +/// +public void nsvg__xformPremultiply (float* t, const(float)* s) { + float[6] s2 = s[0..6]; + //memcpy(s2.ptr, s, float.sizeof*6); + nsvg__xformMultiply(s2.ptr, t); + //memcpy(t, s2.ptr, float.sizeof*6); + t[0..6] = s2[]; +} + +/// +public void nsvg__xformPoint (float* dx, float* dy, in float x, in float y, const(float)* t) { + if (dx !is null) *dx = x*t[0]+y*t[2]+t[4]; + if (dy !is null) *dy = x*t[1]+y*t[3]+t[5]; +} + +/// +public void nsvg__xformVec (float* dx, float* dy, in float x, in float y, const(float)* t) { + if (dx !is null) *dx = x*t[0]+y*t[2]; + if (dy !is null) *dy = x*t[1]+y*t[3]; +} + +/// +public enum NSVG_EPSILON = (1e-12); + +/// +public int nsvg__ptInBounds (const(float)* pt, const(float)* bounds) { + pragma(inline, true); + return pt[0] >= bounds[0] && pt[0] <= bounds[2] && pt[1] >= bounds[1] && pt[1] <= bounds[3]; +} + +/// +public double nsvg__evalBezier (double t, double p0, double p1, double p2, double p3) { + pragma(inline, true); + double it = 1.0-t; + return it*it*it*p0+3.0*it*it*t*p1+3.0*it*t*t*p2+t*t*t*p3; +} + +/// +public void nsvg__curveBounds (float* bounds, const(float)* curve) { + const float* v0 = &curve[0]; + const float* v1 = &curve[2]; + const float* v2 = &curve[4]; + const float* v3 = &curve[6]; + + // Start the bounding box by end points + bounds[0] = nsvg__minf(v0[0], v3[0]); + bounds[1] = nsvg__minf(v0[1], v3[1]); + bounds[2] = nsvg__maxf(v0[0], v3[0]); + bounds[3] = nsvg__maxf(v0[1], v3[1]); + + // Bezier curve fits inside the convex hull of it's control points. + // If control points are inside the bounds, we're done. + if (nsvg__ptInBounds(v1, bounds) && nsvg__ptInBounds(v2, bounds)) return; + + // Add bezier curve inflection points in X and Y. + double[2] roots = void; + foreach (int i; 0..2) { + immutable double a = -3.0*v0[i]+9.0*v1[i]-9.0*v2[i]+3.0*v3[i]; + immutable double b = 6.0*v0[i]-12.0*v1[i]+6.0*v2[i]; + immutable double c = 3.0*v1[i]-3.0*v0[i]; + int count = 0; + if (fabs(a) < NSVG_EPSILON) { + if (fabs(b) > NSVG_EPSILON) { + immutable double t = -c/b; + if (t > NSVG_EPSILON && t < 1.0-NSVG_EPSILON) roots.ptr[count++] = t; + } + } else { + immutable double b2ac = b*b-4.0*c*a; + if (b2ac > NSVG_EPSILON) { + double t = (-b+sqrt(b2ac))/(2.0*a); + if (t > NSVG_EPSILON && t < 1.0-NSVG_EPSILON) roots.ptr[count++] = t; + t = (-b-sqrt(b2ac))/(2.0*a); + if (t > NSVG_EPSILON && t < 1.0-NSVG_EPSILON) roots.ptr[count++] = t; + } + } + foreach (int j; 0..count) { + immutable double v = nsvg__evalBezier(roots.ptr[j], v0[i], v1[i], v2[i], v3[i]); + bounds[0+i] = nsvg__minf(bounds[0+i], cast(float)v); + bounds[2+i] = nsvg__maxf(bounds[2+i], cast(float)v); + } + } +} + + +// ////////////////////////////////////////////////////////////////////////// // +Parser* nsvg__createParser () { + Parser* p = xalloc!Parser; + if (p is null) goto error; + + p.image = xalloc!NSVG; + if (p.image is null) goto error; + + // Init style + nsvg__xformIdentity(p.attr[0].xform.ptr); + p.attr[0].id[] = 0; + p.attr[0].fillColor = NSVG.Paint.rgb(0, 0, 0); + p.attr[0].strokeColor = NSVG.Paint.rgb(0, 0, 0); + p.attr[0].opacity = 1; + p.attr[0].fillOpacity = 1; + p.attr[0].strokeOpacity = 1; + p.attr[0].stopOpacity = 1; + p.attr[0].strokeWidth = 1; + p.attr[0].strokeLineJoin = NSVG.LineJoin.Miter; + p.attr[0].strokeLineCap = NSVG.LineCap.Butt; + p.attr[0].miterLimit = 4; + p.attr[0].fillRule = NSVG.FillRule.EvenOdd; + p.attr[0].hasFill = 1; + p.attr[0].visible = 1; + + return p; + +error: + if (p !is null) { + xfree(p.image); + xfree(p); + } + return null; +} + +void nsvg__deletePaths (NSVG.Path* path) { + while (path !is null) { + NSVG.Path* next = path.next; + xfree(path.stream); + xfree(path); + path = next; + } +} + +void nsvg__deletePaint (NSVG.Paint* paint) { + if (paint.type == NSVG.PaintType.LinearGradient || paint.type == NSVG.PaintType.RadialGradient) xfree(paint.gradient); +} + +void nsvg__deleteGradientData (GradientData* grad) { + GradientData* next; + while (grad !is null) { + next = grad.next; + xfree(grad.stops); + xfree(grad); + grad = next; + } +} + +void nsvg__deleteParser (Parser* p) { + if (p !is null) { + nsvg__deletePaths(p.plist); + nsvg__deleteGradientData(p.gradients); + kill(p.image); + xfree(p.stream); + version(nanosvg_crappy_stylesheet_parser) xfree(p.styles); + xfree(p); + } +} + +void nsvg__resetPath (Parser* p) { + p.nsflts = 0; +} + +void nsvg__addToStream (Parser* p, in float v) { + if (p.nsflts+1 > p.csflts) { + import core.stdc.stdlib : realloc; + p.csflts = (p.csflts == 0 ? 32 : p.csflts < 16384 ? p.csflts*2 : p.csflts+4096); //k8: arbitrary + p.stream = cast(float*)realloc(p.stream, p.csflts*float.sizeof); + if (p.stream is null) assert(0, "nanosvg: out of memory"); + } + p.stream[p.nsflts++] = v; +} + +void nsvg__addCommand (Parser* p, NSVG.Command c) { + nsvg__addToStream(p, cast(float)c); +} + +void nsvg__addPoint (Parser* p, in float x, in float y) { + nsvg__addToStream(p, x); + nsvg__addToStream(p, y); +} + +void nsvg__moveTo (Parser* p, in float x, in float y) { + // this is always called right after `nsvg__resetPath()` + if (p.nsflts != 0) assert(0, "internal error in NanoVega.SVG"); + nsvg__addCommand(p, NSVG.Command.MoveTo); + nsvg__addPoint(p, x, y); + /* + if (p.npts > 0) { + p.pts[(p.npts-1)*2+0] = x; + p.pts[(p.npts-1)*2+1] = y; + } else { + nsvg__addPoint(p, x, y); + } + */ +} + +void nsvg__lineTo (Parser* p, in float x, in float y) { + if (p.nsflts > 0) { + version(nanosvg_use_beziers) { + immutable float px = p.pts[(p.npts-1)*2+0]; + immutable float py = p.pts[(p.npts-1)*2+1]; + immutable float dx = x-px; + immutable float dy = y-py; + nsvg__addCommand(NSVG.Command.BezierTo); + nsvg__addPoint(p, px+dx/3.0f, py+dy/3.0f); + nsvg__addPoint(p, x-dx/3.0f, y-dy/3.0f); + nsvg__addPoint(p, x, y); + } else { + nsvg__addCommand(p, NSVG.Command.LineTo); + nsvg__addPoint(p, x, y); + } + } +} + +void nsvg__cubicBezTo (Parser* p, in float cpx1, in float cpy1, in float cpx2, in float cpy2, in float x, in float y) { + nsvg__addCommand(p, NSVG.Command.BezierTo); + nsvg__addPoint(p, cpx1, cpy1); + nsvg__addPoint(p, cpx2, cpy2); + nsvg__addPoint(p, x, y); +} + +void nsvg__quadBezTo (Parser* p, in float cpx1, in float cpy1, in float x, in float y) { + nsvg__addCommand(p, NSVG.Command.QuadTo); + nsvg__addPoint(p, cpx1, cpy1); + nsvg__addPoint(p, x, y); +} + +Attrib* nsvg__getAttr (Parser* p) { + return p.attr.ptr+p.attrHead; +} + +void nsvg__pushAttr (Parser* p) { + if (p.attrHead < NSVG_MAX_ATTR-1) { + import core.stdc.string : memmove; + ++p.attrHead; + memmove(p.attr.ptr+p.attrHead, p.attr.ptr+(p.attrHead-1), Attrib.sizeof); + } +} + +void nsvg__popAttr (Parser* p) { + if (p.attrHead > 0) --p.attrHead; +} + +float nsvg__actualOrigX (Parser* p) { pragma(inline, true); return p.viewMinx; } +float nsvg__actualOrigY (Parser* p) { pragma(inline, true); return p.viewMiny; } +float nsvg__actualWidth (Parser* p) { pragma(inline, true); return p.viewWidth; } +float nsvg__actualHeight (Parser* p) { pragma(inline, true); return p.viewHeight; } + +float nsvg__actualLength (Parser* p) { + immutable float w = nsvg__actualWidth(p); + immutable float h = nsvg__actualHeight(p); + return sqrtf(w*w+h*h)/sqrtf(2.0f); +} + +float nsvg__convertToPixels (Parser* p, Coordinate c, float orig, float length) { + Attrib* attr = nsvg__getAttr(p); + switch (c.units) { + case Units.user: return c.value; + case Units.px: return c.value; + case Units.pt: return c.value/72.0f*p.dpi; + case Units.pc: return c.value/6.0f*p.dpi; + case Units.mm: return c.value/25.4f*p.dpi; + case Units.cm: return c.value/2.54f*p.dpi; + case Units.in_: return c.value*p.dpi; + case Units.em: return c.value*attr.fontSize; + case Units.ex: return c.value*attr.fontSize*0.52f; // x-height of Helvetica. + case Units.percent: return orig+c.value/100.0f*length; + default: return c.value; + } + assert(0); + //return c.value; +} + +GradientData* nsvg__findGradientData (Parser* p, const(char)[] id) { + GradientData* grad = p.gradients; + id = id.fromAsciiz; + while (grad !is null) { + if (grad.id.fromAsciiz == id) return grad; + grad = grad.next; + } + return null; +} + +NSVG.Gradient* nsvg__createGradient (Parser* p, const(char)[] id, const(float)* localBounds, NSVG.PaintType* paintType) { + Attrib* attr = nsvg__getAttr(p); + GradientData* data = null; + GradientData* ref_ = null; + NSVG.GradientStop* stops = null; + NSVG.Gradient* grad; + float ox = void, oy = void, sw = void, sh = void; + int nstops = 0; + + id = id.fromAsciiz; + data = nsvg__findGradientData(p, id); + if (data is null) return null; + + // TODO: use ref_ to fill in all unset values too. + ref_ = data; + while (ref_ !is null) { + if (stops is null && ref_.stops !is null) { + stops = ref_.stops; + nstops = ref_.nstops; + break; + } + ref_ = nsvg__findGradientData(p, ref_.ref_[]); + } + if (stops is null) return null; + + grad = xalloc!(NSVG.Gradient)(NSVG.GradientStop.sizeof*nstops); + if (grad is null) return null; + + // The shape width and height. + if (data.units == GradientUnits.Object) { + ox = localBounds[0]; + oy = localBounds[1]; + sw = localBounds[2]-localBounds[0]; + sh = localBounds[3]-localBounds[1]; + } else { + ox = nsvg__actualOrigX(p); + oy = nsvg__actualOrigY(p); + sw = nsvg__actualWidth(p); + sh = nsvg__actualHeight(p); + } + immutable float sl = sqrtf(sw*sw+sh*sh)/sqrtf(2.0f); + + if (data.type == NSVG.PaintType.LinearGradient) { + immutable float x1 = nsvg__convertToPixels(p, data.linear.x1, ox, sw); + immutable float y1 = nsvg__convertToPixels(p, data.linear.y1, oy, sh); + immutable float x2 = nsvg__convertToPixels(p, data.linear.x2, ox, sw); + immutable float y2 = nsvg__convertToPixels(p, data.linear.y2, oy, sh); + // Calculate transform aligned to the line + immutable float dx = x2-x1; + immutable float dy = y2-y1; + grad.xform[0] = dy; grad.xform[1] = -dx; + grad.xform[2] = dx; grad.xform[3] = dy; + grad.xform[4] = x1; grad.xform[5] = y1; + } else { + immutable float cx = nsvg__convertToPixels(p, data.radial.cx, ox, sw); + immutable float cy = nsvg__convertToPixels(p, data.radial.cy, oy, sh); + immutable float fx = nsvg__convertToPixels(p, data.radial.fx, ox, sw); + immutable float fy = nsvg__convertToPixels(p, data.radial.fy, oy, sh); + immutable float r = nsvg__convertToPixels(p, data.radial.r, 0, sl); + // Calculate transform aligned to the circle + grad.xform[0] = r; grad.xform[1] = 0; + grad.xform[2] = 0; grad.xform[3] = r; + grad.xform[4] = cx; grad.xform[5] = cy; + // fix from https://github.com/memononen/nanosvg/issues/26#issuecomment-278713651 + grad.fx = (fx-cx)/r; // was fx/r; + grad.fy = (fy-cy)/r; // was fy/r; + } + + nsvg__xformMultiply(grad.xform.ptr, data.xform.ptr); + nsvg__xformMultiply(grad.xform.ptr, attr.xform.ptr); + + grad.spread = data.spread; + //memcpy(grad.stops.ptr, stops, nstops*NSVG.GradientStop.sizeof); + grad.stops.ptr[0..nstops] = stops[0..nstops]; + grad.nstops = nstops; + + *paintType = data.type; + + return grad; +} + +float nsvg__getAverageScale (float* t) { + float sx = sqrtf(t[0]*t[0]+t[2]*t[2]); + float sy = sqrtf(t[1]*t[1]+t[3]*t[3]); + return (sx+sy)*0.5f; +} + +void nsvg__quadBounds (float* bounds, const(float)* curve) nothrow @trusted @nogc { + // cheat: convert quadratic bezier to cubic bezier + immutable float cx = curve[0]; + immutable float cy = curve[1]; + immutable float x1 = curve[2]; + immutable float y1 = curve[3]; + immutable float x2 = curve[4]; + immutable float y2 = curve[5]; + immutable float cx1 = x1+2.0f/3.0f*(cx-x1); + immutable float cy1 = y1+2.0f/3.0f*(cy-y1); + immutable float cx2 = x2+2.0f/3.0f*(cx-x2); + immutable float cy2 = y2+2.0f/3.0f*(cy-y2); + float[8] cubic = void; + cubic.ptr[0] = cx; + cubic.ptr[1] = cy; + cubic.ptr[2] = cx1; + cubic.ptr[3] = cy1; + cubic.ptr[4] = cx2; + cubic.ptr[5] = cy2; + cubic.ptr[6] = x2; + cubic.ptr[7] = y2; + nsvg__curveBounds(bounds, cubic.ptr); +} + +void nsvg__getLocalBounds (float* bounds, NSVG.Shape* shape, const(float)* xform) { + bool first = true; + + void addPoint (in float x, in float y) nothrow @trusted @nogc { + if (!first) { + bounds[0] = nsvg__minf(bounds[0], x); + bounds[1] = nsvg__minf(bounds[1], y); + bounds[2] = nsvg__maxf(bounds[2], x); + bounds[3] = nsvg__maxf(bounds[3], y); + } else { + bounds[0] = bounds[2] = x; + bounds[1] = bounds[3] = y; + first = false; + } + } + + void addRect (in float x0, in float y0, in float x1, in float y1) nothrow @trusted @nogc { + addPoint(x0, y0); + addPoint(x1, y0); + addPoint(x1, y1); + addPoint(x0, y1); + } + + float cx = 0, cy = 0; + for (NSVG.Path* path = shape.paths; path !is null; path = path.next) { + path.forEachCommand!false(delegate (NSVG.Command cmd, const(float)[] args) nothrow @trusted @nogc { + import core.stdc.string : memmove; + assert(args.length <= 6); + float[8] xpt = void; + // transform points + foreach (immutable n; 0..args.length/2) { + nsvg__xformPoint(&xpt.ptr[n*2+0], &xpt.ptr[n*2+1], args.ptr[n*2+0], args.ptr[n*2+1], xform); + } + // add to bounds + final switch (cmd) { + case NSVG.Command.MoveTo: + cx = xpt.ptr[0]; + cy = xpt.ptr[1]; + break; + case NSVG.Command.LineTo: + addPoint(cx, cy); + addPoint(xpt.ptr[0], xpt.ptr[1]); + cx = xpt.ptr[0]; + cy = xpt.ptr[1]; + break; + case NSVG.Command.QuadTo: + memmove(xpt.ptr+2, xpt.ptr, 4); // make room for starting point + xpt.ptr[0] = cx; + xpt.ptr[1] = cy; + float[4] curveBounds = void; + nsvg__quadBounds(curveBounds.ptr, xpt.ptr); + addRect(curveBounds.ptr[0], curveBounds.ptr[1], curveBounds.ptr[2], curveBounds.ptr[3]); + cx = xpt.ptr[4]; + cy = xpt.ptr[5]; + break; + case NSVG.Command.BezierTo: + memmove(xpt.ptr+2, xpt.ptr, 6); // make room for starting point + xpt.ptr[0] = cx; + xpt.ptr[1] = cy; + float[4] curveBounds = void; + nsvg__curveBounds(curveBounds.ptr, xpt.ptr); + addRect(curveBounds.ptr[0], curveBounds.ptr[1], curveBounds.ptr[2], curveBounds.ptr[3]); + cx = xpt.ptr[6]; + cy = xpt.ptr[7]; + break; + } + }); + /* + nsvg__xformPoint(&curve.ptr[0], &curve.ptr[1], path.pts[0], path.pts[1], xform); + for (int i = 0; i < path.npts-1; i += 3) { + nsvg__xformPoint(&curve.ptr[2], &curve.ptr[3], path.pts[(i+1)*2], path.pts[(i+1)*2+1], xform); + nsvg__xformPoint(&curve.ptr[4], &curve.ptr[5], path.pts[(i+2)*2], path.pts[(i+2)*2+1], xform); + nsvg__xformPoint(&curve.ptr[6], &curve.ptr[7], path.pts[(i+3)*2], path.pts[(i+3)*2+1], xform); + nsvg__curveBounds(curveBounds.ptr, curve.ptr); + if (first) { + bounds[0] = curveBounds.ptr[0]; + bounds[1] = curveBounds.ptr[1]; + bounds[2] = curveBounds.ptr[2]; + bounds[3] = curveBounds.ptr[3]; + first = false; + } else { + bounds[0] = nsvg__minf(bounds[0], curveBounds.ptr[0]); + bounds[1] = nsvg__minf(bounds[1], curveBounds.ptr[1]); + bounds[2] = nsvg__maxf(bounds[2], curveBounds.ptr[2]); + bounds[3] = nsvg__maxf(bounds[3], curveBounds.ptr[3]); + } + curve.ptr[0] = curve.ptr[6]; + curve.ptr[1] = curve.ptr[7]; + } + */ + } +} + +void nsvg__addShape (Parser* p) { + Attrib* attr = nsvg__getAttr(p); + float scale = 1.0f; + NSVG.Shape* shape; + NSVG.Path* path; + int i; + + if (p.plist is null) return; + + shape = xalloc!(NSVG.Shape); + if (shape is null) goto error; + //memset(shape, 0, NSVG.Shape.sizeof); + + shape.id[] = attr.id[]; + scale = nsvg__getAverageScale(attr.xform.ptr); + shape.strokeWidth = attr.strokeWidth*scale; + shape.strokeDashOffset = attr.strokeDashOffset*scale; + shape.strokeDashCount = cast(char)attr.strokeDashCount; + for (i = 0; i < attr.strokeDashCount; i++) shape.strokeDashArray[i] = attr.strokeDashArray[i]*scale; + shape.strokeLineJoin = attr.strokeLineJoin; + shape.strokeLineCap = attr.strokeLineCap; + shape.miterLimit = attr.miterLimit; + shape.fillRule = attr.fillRule; + shape.opacity = attr.opacity; + + shape.paths = p.plist; + p.plist = null; + + // Calculate shape bounds + shape.bounds.ptr[0] = shape.paths.bounds.ptr[0]; + shape.bounds.ptr[1] = shape.paths.bounds.ptr[1]; + shape.bounds.ptr[2] = shape.paths.bounds.ptr[2]; + shape.bounds.ptr[3] = shape.paths.bounds.ptr[3]; + for (path = shape.paths.next; path !is null; path = path.next) { + shape.bounds.ptr[0] = nsvg__minf(shape.bounds.ptr[0], path.bounds[0]); + shape.bounds.ptr[1] = nsvg__minf(shape.bounds.ptr[1], path.bounds[1]); + shape.bounds.ptr[2] = nsvg__maxf(shape.bounds.ptr[2], path.bounds[2]); + shape.bounds.ptr[3] = nsvg__maxf(shape.bounds.ptr[3], path.bounds[3]); + } + + // Set fill + if (attr.hasFill == 0) { + shape.fill.type = NSVG.PaintType.None; + } else if (attr.hasFill == 1) { + shape.fill.type = NSVG.PaintType.Color; + shape.fill.color = attr.fillColor; + shape.fill.color |= cast(uint)(attr.fillOpacity*255)<<24; + } else if (attr.hasFill == 2) { + float[6] inv; + float[4] localBounds; + nsvg__xformInverse(inv.ptr, attr.xform.ptr); + nsvg__getLocalBounds(localBounds.ptr, shape, inv.ptr); + shape.fill.gradient = nsvg__createGradient(p, attr.fillGradient[], localBounds.ptr, &shape.fill.type); + if (shape.fill.gradient is null) shape.fill.type = NSVG.PaintType.None; + } + + // Set stroke + if (attr.hasStroke == 0) { + shape.stroke.type = NSVG.PaintType.None; + } else if (attr.hasStroke == 1) { + shape.stroke.type = NSVG.PaintType.Color; + shape.stroke.color = attr.strokeColor; + shape.stroke.color |= cast(uint)(attr.strokeOpacity*255)<<24; + } else if (attr.hasStroke == 2) { + float[6] inv; + float[4] localBounds; + nsvg__xformInverse(inv.ptr, attr.xform.ptr); + nsvg__getLocalBounds(localBounds.ptr, shape, inv.ptr); + shape.stroke.gradient = nsvg__createGradient(p, attr.strokeGradient[], localBounds.ptr, &shape.stroke.type); + if (shape.stroke.gradient is null) shape.stroke.type = NSVG.PaintType.None; + } + + // Set flags + shape.flags = (attr.visible ? NSVG.Visible : 0x00); + + // Add to tail + if (p.image.shapes is null) + p.image.shapes = shape; + else + p.shapesTail.next = shape; + + p.shapesTail = shape; + + return; + +error: + if (shape) xfree(shape); +} + +void nsvg__addPath (Parser* p, bool closed) { + Attrib* attr = nsvg__getAttr(p); + + if (p.nsflts < 4) return; + + if (closed) { + auto cmd = cast(NSVG.Command)p.stream[0]; + if (cmd != NSVG.Command.MoveTo) assert(0, "NanoVega.SVG: invalid path"); + nsvg__lineTo(p, p.stream[1], p.stream[2]); + } + + float cx = 0, cy = 0; + float[4] bounds = void; + bool first = true; + + NSVG.Path* path = xalloc!(NSVG.Path); + if (path is null) goto error; + //memset(path, 0, NSVG.Path.sizeof); + + path.stream = xcalloc!float(p.nsflts); + if (path.stream is null) goto error; + path.closed = closed; + path.nsflts = p.nsflts; + + // transform path and calculate bounds + void addPoint (in float x, in float y) nothrow @trusted @nogc { + if (!first) { + bounds[0] = nsvg__minf(bounds[0], x); + bounds[1] = nsvg__minf(bounds[1], y); + bounds[2] = nsvg__maxf(bounds[2], x); + bounds[3] = nsvg__maxf(bounds[3], y); + } else { + bounds[0] = bounds[2] = x; + bounds[1] = bounds[3] = y; + first = false; + } + } + + void addRect (in float x0, in float y0, in float x1, in float y1) nothrow @trusted @nogc { + addPoint(x0, y0); + addPoint(x1, y0); + addPoint(x1, y1); + addPoint(x0, y1); + } + + version(none) { + foreach (immutable idx, float f; p.stream[0..p.nsflts]) { + import core.stdc.stdio; + printf("idx=%u; f=%g\n", cast(uint)idx, cast(double)f); + } + } + + for (int i = 0; i+3 <= p.nsflts; ) { + int argc = 0; // pair of coords + NSVG.Command cmd = cast(NSVG.Command)p.stream[i]; + final switch (cmd) { + case NSVG.Command.MoveTo: argc = 1; break; + case NSVG.Command.LineTo: argc = 1; break; + case NSVG.Command.QuadTo: argc = 2; break; + case NSVG.Command.BezierTo: argc = 3; break; + } + // copy command + path.stream[i] = p.stream[i]; + ++i; + auto starti = i; + // transform points + while (argc-- > 0) { + nsvg__xformPoint(&path.stream[i+0], &path.stream[i+1], p.stream[i+0], p.stream[i+1], attr.xform.ptr); + i += 2; + } + // do bounds + final switch (cmd) { + case NSVG.Command.MoveTo: + cx = path.stream[starti+0]; + cy = path.stream[starti+1]; + break; + case NSVG.Command.LineTo: + addPoint(cx, cy); + cx = path.stream[starti+0]; + cy = path.stream[starti+1]; + addPoint(cx, cy); + break; + case NSVG.Command.QuadTo: + float[6] curve = void; + curve.ptr[0] = cx; + curve.ptr[1] = cy; + curve.ptr[2..6] = path.stream[starti+0..starti+4]; + cx = path.stream[starti+2]; + cy = path.stream[starti+3]; + float[4] curveBounds = void; + nsvg__quadBounds(curveBounds.ptr, curve.ptr); + addRect(curveBounds.ptr[0], curveBounds.ptr[1], curveBounds.ptr[2], curveBounds.ptr[3]); + break; + case NSVG.Command.BezierTo: + float[8] curve = void; + curve.ptr[0] = cx; + curve.ptr[1] = cy; + curve.ptr[2..8] = path.stream[starti+0..starti+6]; + cx = path.stream[starti+4]; + cy = path.stream[starti+5]; + float[4] curveBounds = void; + nsvg__curveBounds(curveBounds.ptr, curve.ptr); + addRect(curveBounds.ptr[0], curveBounds.ptr[1], curveBounds.ptr[2], curveBounds.ptr[3]); + break; + } + } + path.bounds[0..4] = bounds[0..4]; + + path.next = p.plist; + p.plist = path; + + return; + +error: + if (path !is null) { + if (path.stream !is null) xfree(path.stream); + xfree(path); + } +} + +// We roll our own string to float because the std library one uses locale and messes things up. +// special hack: stop at '\0' (actually, it stops on any non-digit, so no special code is required) +float nsvg__atof (const(char)[] s) nothrow @trusted @nogc { + if (s.length == 0) return 0; // oops + + const(char)* cur = s.ptr; + auto left = s.length; + double res = 0.0, sign = 1.0; + bool hasIntPart = false, hasFracPart = false; + + char peekChar () nothrow @trusted @nogc { pragma(inline, true); return (left > 0 ? *cur : '\x00'); } + char getChar () nothrow @trusted @nogc { if (left > 0) { --left; return *cur++; } else return '\x00'; } + + // Parse optional sign + switch (peekChar) { + case '-': sign = -1; goto case; + case '+': getChar(); break; + default: break; + } + + // Parse integer part + if (nsvg__isdigit(peekChar)) { + // Parse digit sequence + hasIntPart = true; + while (nsvg__isdigit(peekChar)) res = res*10.0+(getChar()-'0'); + } + + // Parse fractional part. + if (peekChar == '.') { + getChar(); // Skip '.' + if (nsvg__isdigit(peekChar)) { + // Parse digit sequence + hasFracPart = true; + int divisor = 1; + long num = 0; + while (nsvg__isdigit(peekChar)) { + divisor *= 10; + num = num*10+(getChar()-'0'); + } + res += cast(double)num/divisor; + } + } + + // A valid number should have integer or fractional part. + if (!hasIntPart && !hasFracPart) return 0; + + // Parse optional exponent + if (peekChar == 'e' || peekChar == 'E') { + getChar(); // skip 'E' + // parse optional sign + bool epositive = true; + switch (peekChar) { + case '-': epositive = false; goto case; + case '+': getChar(); break; + default: break; + } + int expPart = 0; + while (nsvg__isdigit(peekChar)) expPart = expPart*10+(getChar()-'0'); + if (epositive) { + foreach (immutable _; 0..expPart) res *= 10.0; + } else { + foreach (immutable _; 0..expPart) res /= 10.0; + } + } + + return cast(float)(res*sign); +} + +// `it` should be big enough +// returns number of chars eaten +int nsvg__parseNumber (const(char)[] s, char[] it) { + int i = 0; + it[] = 0; + + const(char)[] os = s; + + // sign + if (s.length && (s[0] == '-' || s[0] == '+')) { + if (it.length-i > 1) it[i++] = s[0]; + s = s[1..$]; + } + // integer part + while (s.length && nsvg__isdigit(s[0])) { + if (it.length-i > 1) it[i++] = s[0]; + s = s[1..$]; + } + if (s.length && s[0] == '.') { + // decimal point + if (it.length-i > 1) it[i++] = s[0]; + s = s[1..$]; + // fraction part + while (s.length && nsvg__isdigit(s[0])) { + if (it.length-i > 1) it[i++] = s[0]; + s = s[1..$]; + } + } + // exponent + if (s.length && (s[0] == 'e' || s[0] == 'E')) { + if (it.length-i > 1) it[i++] = s[0]; + s = s[1..$]; + if (s.length && (s[0] == '-' || s[0] == '+')) { + if (it.length-i > 1) it[i++] = s[0]; + s = s[1..$]; + } + while (s.length && nsvg__isdigit(s[0])) { + if (it.length-i > 1) it[i++] = s[0]; + s = s[1..$]; + } + } + + return cast(int)(s.ptr-os.ptr); +} + +// `it` should be big enough +int nsvg__getNextPathItem (const(char)[] s, char[] it) { + int res = 0; + it[] = '\0'; + // skip white spaces and commas + while (res < s.length && (nsvg__isspace(s[res]) || s[res] == ',')) ++res; + if (res >= s.length) return cast(int)s.length; + if (s[res] == '-' || s[res] == '+' || s[res] == '.' || nsvg__isdigit(s[res])) { + res += nsvg__parseNumber(s[res..$], it); + } else if (s.length) { + // Parse command + it[0] = s[res++]; + } + return res; +} + +uint nsvg__parseColorHex (const(char)[] str) { + char[12] tmp = 0; + uint c = 0; + ubyte r = 0, g = 0, b = 0; + int n = 0; + if (str.length) str = str[1..$]; // skip # + // calculate number of characters + while (n < str.length && !nsvg__isspace(str[n])) ++n; + if (n == 3 || n == 6) { + foreach (char ch; str[0..n]) { + auto d0 = nsvg__hexdigit(ch); + if (d0 < 0) break; + c = c*16+d0; + } + if (n == 3) { + c = (c&0xf)|((c&0xf0)<<4)|((c&0xf00)<<8); + c |= c<<4; + } + } + r = (c>>16)&0xff; + g = (c>>8)&0xff; + b = c&0xff; + return NSVG.Paint.rgb(r, g, b); +} + +uint nsvg__parseColorRGB (const(char)[] str) { + int r = -1, g = -1, b = -1; + const(char)[] s1, s2; + assert(str.length > 4); + xsscanf(str[4..$], "%d%[%%, \t]%d%[%%, \t]%d", r, s1, g, s2, b); + if (s1[].xindexOf('%') >= 0) { + return NSVG.Paint.rgb(cast(ubyte)((r*255)/100), cast(ubyte)((g*255)/100), cast(ubyte)((b*255)/100)); + } else { + return NSVG.Paint.rgb(cast(ubyte)r, cast(ubyte)g, cast(ubyte)b); + } +} + +struct NSVGNamedColor { + string name; + uint color; +} + +static immutable NSVGNamedColor[147] nsvg__colors = [ + NSVGNamedColor("aliceblue", NSVG.Paint.rgb(240, 248, 255)), + NSVGNamedColor("antiquewhite", NSVG.Paint.rgb(250, 235, 215)), + NSVGNamedColor("aqua", NSVG.Paint.rgb( 0, 255, 255)), + NSVGNamedColor("aquamarine", NSVG.Paint.rgb(127, 255, 212)), + NSVGNamedColor("azure", NSVG.Paint.rgb(240, 255, 255)), + NSVGNamedColor("beige", NSVG.Paint.rgb(245, 245, 220)), + NSVGNamedColor("bisque", NSVG.Paint.rgb(255, 228, 196)), + NSVGNamedColor("black", NSVG.Paint.rgb( 0, 0, 0)), // basic color + NSVGNamedColor("blanchedalmond", NSVG.Paint.rgb(255, 235, 205)), + NSVGNamedColor("blue", NSVG.Paint.rgb( 0, 0, 255)), // basic color + NSVGNamedColor("blueviolet", NSVG.Paint.rgb(138, 43, 226)), + NSVGNamedColor("brown", NSVG.Paint.rgb(165, 42, 42)), + NSVGNamedColor("burlywood", NSVG.Paint.rgb(222, 184, 135)), + NSVGNamedColor("cadetblue", NSVG.Paint.rgb( 95, 158, 160)), + NSVGNamedColor("chartreuse", NSVG.Paint.rgb(127, 255, 0)), + NSVGNamedColor("chocolate", NSVG.Paint.rgb(210, 105, 30)), + NSVGNamedColor("coral", NSVG.Paint.rgb(255, 127, 80)), + NSVGNamedColor("cornflowerblue", NSVG.Paint.rgb(100, 149, 237)), + NSVGNamedColor("cornsilk", NSVG.Paint.rgb(255, 248, 220)), + NSVGNamedColor("crimson", NSVG.Paint.rgb(220, 20, 60)), + NSVGNamedColor("cyan", NSVG.Paint.rgb( 0, 255, 255)), // basic color + NSVGNamedColor("darkblue", NSVG.Paint.rgb( 0, 0, 139)), + NSVGNamedColor("darkcyan", NSVG.Paint.rgb( 0, 139, 139)), + NSVGNamedColor("darkgoldenrod", NSVG.Paint.rgb(184, 134, 11)), + NSVGNamedColor("darkgray", NSVG.Paint.rgb(169, 169, 169)), + NSVGNamedColor("darkgreen", NSVG.Paint.rgb( 0, 100, 0)), + NSVGNamedColor("darkgrey", NSVG.Paint.rgb(169, 169, 169)), + NSVGNamedColor("darkkhaki", NSVG.Paint.rgb(189, 183, 107)), + NSVGNamedColor("darkmagenta", NSVG.Paint.rgb(139, 0, 139)), + NSVGNamedColor("darkolivegreen", NSVG.Paint.rgb( 85, 107, 47)), + NSVGNamedColor("darkorange", NSVG.Paint.rgb(255, 140, 0)), + NSVGNamedColor("darkorchid", NSVG.Paint.rgb(153, 50, 204)), + NSVGNamedColor("darkred", NSVG.Paint.rgb(139, 0, 0)), + NSVGNamedColor("darksalmon", NSVG.Paint.rgb(233, 150, 122)), + NSVGNamedColor("darkseagreen", NSVG.Paint.rgb(143, 188, 143)), + NSVGNamedColor("darkslateblue", NSVG.Paint.rgb( 72, 61, 139)), + NSVGNamedColor("darkslategray", NSVG.Paint.rgb( 47, 79, 79)), + NSVGNamedColor("darkslategrey", NSVG.Paint.rgb( 47, 79, 79)), + NSVGNamedColor("darkturquoise", NSVG.Paint.rgb( 0, 206, 209)), + NSVGNamedColor("darkviolet", NSVG.Paint.rgb(148, 0, 211)), + NSVGNamedColor("deeppink", NSVG.Paint.rgb(255, 20, 147)), + NSVGNamedColor("deepskyblue", NSVG.Paint.rgb( 0, 191, 255)), + NSVGNamedColor("dimgray", NSVG.Paint.rgb(105, 105, 105)), + NSVGNamedColor("dimgrey", NSVG.Paint.rgb(105, 105, 105)), + NSVGNamedColor("dodgerblue", NSVG.Paint.rgb( 30, 144, 255)), + NSVGNamedColor("firebrick", NSVG.Paint.rgb(178, 34, 34)), + NSVGNamedColor("floralwhite", NSVG.Paint.rgb(255, 250, 240)), + NSVGNamedColor("forestgreen", NSVG.Paint.rgb( 34, 139, 34)), + NSVGNamedColor("fuchsia", NSVG.Paint.rgb(255, 0, 255)), + NSVGNamedColor("gainsboro", NSVG.Paint.rgb(220, 220, 220)), + NSVGNamedColor("ghostwhite", NSVG.Paint.rgb(248, 248, 255)), + NSVGNamedColor("gold", NSVG.Paint.rgb(255, 215, 0)), + NSVGNamedColor("goldenrod", NSVG.Paint.rgb(218, 165, 32)), + NSVGNamedColor("gray", NSVG.Paint.rgb(128, 128, 128)), // basic color + NSVGNamedColor("green", NSVG.Paint.rgb( 0, 128, 0)), // basic color + NSVGNamedColor("greenyellow", NSVG.Paint.rgb(173, 255, 47)), + NSVGNamedColor("grey", NSVG.Paint.rgb(128, 128, 128)), // basic color + NSVGNamedColor("honeydew", NSVG.Paint.rgb(240, 255, 240)), + NSVGNamedColor("hotpink", NSVG.Paint.rgb(255, 105, 180)), + NSVGNamedColor("indianred", NSVG.Paint.rgb(205, 92, 92)), + NSVGNamedColor("indigo", NSVG.Paint.rgb( 75, 0, 130)), + NSVGNamedColor("ivory", NSVG.Paint.rgb(255, 255, 240)), + NSVGNamedColor("khaki", NSVG.Paint.rgb(240, 230, 140)), + NSVGNamedColor("lavender", NSVG.Paint.rgb(230, 230, 250)), + NSVGNamedColor("lavenderblush", NSVG.Paint.rgb(255, 240, 245)), + NSVGNamedColor("lawngreen", NSVG.Paint.rgb(124, 252, 0)), + NSVGNamedColor("lemonchiffon", NSVG.Paint.rgb(255, 250, 205)), + NSVGNamedColor("lightblue", NSVG.Paint.rgb(173, 216, 230)), + NSVGNamedColor("lightcoral", NSVG.Paint.rgb(240, 128, 128)), + NSVGNamedColor("lightcyan", NSVG.Paint.rgb(224, 255, 255)), + NSVGNamedColor("lightgoldenrodyellow", NSVG.Paint.rgb(250, 250, 210)), + NSVGNamedColor("lightgray", NSVG.Paint.rgb(211, 211, 211)), + NSVGNamedColor("lightgreen", NSVG.Paint.rgb(144, 238, 144)), + NSVGNamedColor("lightgrey", NSVG.Paint.rgb(211, 211, 211)), + NSVGNamedColor("lightpink", NSVG.Paint.rgb(255, 182, 193)), + NSVGNamedColor("lightsalmon", NSVG.Paint.rgb(255, 160, 122)), + NSVGNamedColor("lightseagreen", NSVG.Paint.rgb( 32, 178, 170)), + NSVGNamedColor("lightskyblue", NSVG.Paint.rgb(135, 206, 250)), + NSVGNamedColor("lightslategray", NSVG.Paint.rgb(119, 136, 153)), + NSVGNamedColor("lightslategrey", NSVG.Paint.rgb(119, 136, 153)), + NSVGNamedColor("lightsteelblue", NSVG.Paint.rgb(176, 196, 222)), + NSVGNamedColor("lightyellow", NSVG.Paint.rgb(255, 255, 224)), + NSVGNamedColor("lime", NSVG.Paint.rgb( 0, 255, 0)), + NSVGNamedColor("limegreen", NSVG.Paint.rgb( 50, 205, 50)), + NSVGNamedColor("linen", NSVG.Paint.rgb(250, 240, 230)), + NSVGNamedColor("magenta", NSVG.Paint.rgb(255, 0, 255)), // basic color + NSVGNamedColor("maroon", NSVG.Paint.rgb(128, 0, 0)), + NSVGNamedColor("mediumaquamarine", NSVG.Paint.rgb(102, 205, 170)), + NSVGNamedColor("mediumblue", NSVG.Paint.rgb( 0, 0, 205)), + NSVGNamedColor("mediumorchid", NSVG.Paint.rgb(186, 85, 211)), + NSVGNamedColor("mediumpurple", NSVG.Paint.rgb(147, 112, 219)), + NSVGNamedColor("mediumseagreen", NSVG.Paint.rgb( 60, 179, 113)), + NSVGNamedColor("mediumslateblue", NSVG.Paint.rgb(123, 104, 238)), + NSVGNamedColor("mediumspringgreen", NSVG.Paint.rgb( 0, 250, 154)), + NSVGNamedColor("mediumturquoise", NSVG.Paint.rgb( 72, 209, 204)), + NSVGNamedColor("mediumvioletred", NSVG.Paint.rgb(199, 21, 133)), + NSVGNamedColor("midnightblue", NSVG.Paint.rgb( 25, 25, 112)), + NSVGNamedColor("mintcream", NSVG.Paint.rgb(245, 255, 250)), + NSVGNamedColor("mistyrose", NSVG.Paint.rgb(255, 228, 225)), + NSVGNamedColor("moccasin", NSVG.Paint.rgb(255, 228, 181)), + NSVGNamedColor("navajowhite", NSVG.Paint.rgb(255, 222, 173)), + NSVGNamedColor("navy", NSVG.Paint.rgb( 0, 0, 128)), + NSVGNamedColor("oldlace", NSVG.Paint.rgb(253, 245, 230)), + NSVGNamedColor("olive", NSVG.Paint.rgb(128, 128, 0)), + NSVGNamedColor("olivedrab", NSVG.Paint.rgb(107, 142, 35)), + NSVGNamedColor("orange", NSVG.Paint.rgb(255, 165, 0)), + NSVGNamedColor("orangered", NSVG.Paint.rgb(255, 69, 0)), + NSVGNamedColor("orchid", NSVG.Paint.rgb(218, 112, 214)), + NSVGNamedColor("palegoldenrod", NSVG.Paint.rgb(238, 232, 170)), + NSVGNamedColor("palegreen", NSVG.Paint.rgb(152, 251, 152)), + NSVGNamedColor("paleturquoise", NSVG.Paint.rgb(175, 238, 238)), + NSVGNamedColor("palevioletred", NSVG.Paint.rgb(219, 112, 147)), + NSVGNamedColor("papayawhip", NSVG.Paint.rgb(255, 239, 213)), + NSVGNamedColor("peachpuff", NSVG.Paint.rgb(255, 218, 185)), + NSVGNamedColor("peru", NSVG.Paint.rgb(205, 133, 63)), + NSVGNamedColor("pink", NSVG.Paint.rgb(255, 192, 203)), + NSVGNamedColor("plum", NSVG.Paint.rgb(221, 160, 221)), + NSVGNamedColor("powderblue", NSVG.Paint.rgb(176, 224, 230)), + NSVGNamedColor("purple", NSVG.Paint.rgb(128, 0, 128)), + NSVGNamedColor("red", NSVG.Paint.rgb(255, 0, 0)), // basic color + NSVGNamedColor("rosybrown", NSVG.Paint.rgb(188, 143, 143)), + NSVGNamedColor("royalblue", NSVG.Paint.rgb( 65, 105, 225)), + NSVGNamedColor("saddlebrown", NSVG.Paint.rgb(139, 69, 19)), + NSVGNamedColor("salmon", NSVG.Paint.rgb(250, 128, 114)), + NSVGNamedColor("sandybrown", NSVG.Paint.rgb(244, 164, 96)), + NSVGNamedColor("seagreen", NSVG.Paint.rgb( 46, 139, 87)), + NSVGNamedColor("seashell", NSVG.Paint.rgb(255, 245, 238)), + NSVGNamedColor("sienna", NSVG.Paint.rgb(160, 82, 45)), + NSVGNamedColor("silver", NSVG.Paint.rgb(192, 192, 192)), + NSVGNamedColor("skyblue", NSVG.Paint.rgb(135, 206, 235)), + NSVGNamedColor("slateblue", NSVG.Paint.rgb(106, 90, 205)), + NSVGNamedColor("slategray", NSVG.Paint.rgb(112, 128, 144)), + NSVGNamedColor("slategrey", NSVG.Paint.rgb(112, 128, 144)), + NSVGNamedColor("snow", NSVG.Paint.rgb(255, 250, 250)), + NSVGNamedColor("springgreen", NSVG.Paint.rgb( 0, 255, 127)), + NSVGNamedColor("steelblue", NSVG.Paint.rgb( 70, 130, 180)), + NSVGNamedColor("tan", NSVG.Paint.rgb(210, 180, 140)), + NSVGNamedColor("teal", NSVG.Paint.rgb( 0, 128, 128)), + NSVGNamedColor("thistle", NSVG.Paint.rgb(216, 191, 216)), + NSVGNamedColor("tomato", NSVG.Paint.rgb(255, 99, 71)), + NSVGNamedColor("turquoise", NSVG.Paint.rgb( 64, 224, 208)), + NSVGNamedColor("violet", NSVG.Paint.rgb(238, 130, 238)), + NSVGNamedColor("wheat", NSVG.Paint.rgb(245, 222, 179)), + NSVGNamedColor("white", NSVG.Paint.rgb(255, 255, 255)), // basic color + NSVGNamedColor("whitesmoke", NSVG.Paint.rgb(245, 245, 245)), + NSVGNamedColor("yellow", NSVG.Paint.rgb(255, 255, 0)), // basic color + NSVGNamedColor("yellowgreen", NSVG.Paint.rgb(154, 205, 50)), +]; + +enum nsvg__color_name_maxlen = () { + int res = 0; + foreach (const ref known; nsvg__colors) if (res < known.name.length) res = cast(int)known.name.length; + return res; +}(); + + +// `s0` and `s1` are never empty here +// `s0` is always lowercased +int xstrcmp (const(char)[] s0, const(char)[] s1) { + /* + const(char)* sp0 = s0.ptr; + const(char)* sp1 = s1.ptr; + foreach (; 0..(s0.length < s1.length ? s0.length : s1.length)) { + int c1 = cast(int)(*sp1++); + if (c1 >= 'A' && c1 <= 'Z') c1 += 32; // poor man's tolower + if (auto diff = cast(int)(*sp0++)-c1) return diff; + } + // equals so far + if (s0.length < s1.length) return -1; + if (s0.length > s1.length) return 1; + return 0; + */ + import core.stdc.string : memcmp; + if (auto diff = memcmp(s0.ptr, s1.ptr, (s0.length < s1.length ? s0.length : s1.length))) return diff; + // equals so far + if (s0.length < s1.length) return -1; + if (s0.length > s1.length) return 1; + return 0; +} + + +uint nsvg__parseColorName (const(char)[] str) { + if (str.length == 0 || str.length > nsvg__color_name_maxlen) return NSVG.Paint.rgb(128, 128, 128); + // check if `str` contains only letters, and convert it to lowercase + char[nsvg__color_name_maxlen] slow = void; + foreach (immutable cidx, char ch; str) { + if (ch >= 'A' && ch <= 'Z') ch += 32; // poor man's tolower + if (ch < 'a' || ch > 'z') return NSVG.Paint.rgb(128, 128, 128); // alas + slow.ptr[cidx] = ch; + } + int low = 0; + int high = cast(int)nsvg__colors.length-1; + while (low <= high) { + int med = (low+high)/2; + assert(med >= 0 && med < nsvg__colors.length); + int res = xstrcmp(nsvg__colors.ptr[med].name, str); + if (res < 0) low = med+1; + else if (res > 0) high = med-1; + else return nsvg__colors.ptr[med].color; + } + return NSVG.Paint.rgb(128, 128, 128); +} + +uint nsvg__parseColor (const(char)[] str) { + while (str.length && str[0] <= ' ') str = str[1..$]; + if (str.length >= 1 && str[0] == '#') return nsvg__parseColorHex(str); + if (str.length >= 4 && str[0] == 'r' && str[1] == 'g' && str[2] == 'b' && str[3] == '(') return nsvg__parseColorRGB(str); + return nsvg__parseColorName(str); +} + +float nsvg__parseOpacity (const(char)[] str) { + float val = 0; + xsscanf(str, "%f", val); + if (val < 0.0f) val = 0.0f; + if (val > 1.0f) val = 1.0f; + return val; +} + +float nsvg__parseMiterLimit (const(char)[] str) { + float val = 0; + xsscanf(str, "%f", val); + if (val < 0.0f) val = 0.0f; + return val; +} + +Units nsvg__parseUnits (const(char)[] units) { + if (units.length && units.ptr[0] == '%') return Units.percent; + if (units.length == 2) { + if (units.ptr[0] == 'p' && units.ptr[1] == 'x') return Units.px; + if (units.ptr[0] == 'p' && units.ptr[1] == 't') return Units.pt; + if (units.ptr[0] == 'p' && units.ptr[1] == 'c') return Units.pc; + if (units.ptr[0] == 'm' && units.ptr[1] == 'm') return Units.mm; + if (units.ptr[0] == 'c' && units.ptr[1] == 'm') return Units.cm; + if (units.ptr[0] == 'i' && units.ptr[1] == 'n') return Units.in_; + if (units.ptr[0] == 'e' && units.ptr[1] == 'm') return Units.em; + if (units.ptr[0] == 'e' && units.ptr[1] == 'x') return Units.ex; + } + return Units.user; +} + +Coordinate nsvg__parseCoordinateRaw (const(char)[] str) { + Coordinate coord = Coordinate(0, Units.user); + const(char)[] units; + xsscanf(str, "%f%s", coord.value, units); + coord.units = nsvg__parseUnits(units); + return coord; +} + +Coordinate nsvg__coord (float v, Units units) { + Coordinate coord = Coordinate(v, units); + return coord; +} + +float nsvg__parseCoordinate (Parser* p, const(char)[] str, float orig, float length) { + Coordinate coord = nsvg__parseCoordinateRaw(str); + return nsvg__convertToPixels(p, coord, orig, length); +} + +int nsvg__parseTransformArgs (const(char)[] str, float* args, int maxNa, int* na) { + usize end, ptr; + char[65] it = void; + + assert(str.length); + *na = 0; + + ptr = 0; + while (ptr < str.length && str[ptr] != '(') ++ptr; + if (ptr >= str.length) return 1; + + end = ptr; + while (end < str.length && str[end] != ')') ++end; + if (end >= str.length) return 1; + + while (ptr < end) { + if (str[ptr] == '-' || str[ptr] == '+' || str[ptr] == '.' || nsvg__isdigit(str[ptr])) { + if (*na >= maxNa) return 0; + ptr += nsvg__parseNumber(str[ptr..end], it[]); + args[(*na)++] = nsvg__atof(it[]); // `it` is guaranteed to be asciiz, and `nsvg__atof()` will stop + } else { + ++ptr; + } + } + return cast(int)end; // fuck off, 64bit +} + + +int nsvg__parseMatrix (float* xform, const(char)[] str) { + float[6] t = void; + int na = 0; + int len = nsvg__parseTransformArgs(str, t.ptr, 6, &na); + if (na != 6) return len; + xform[0..6] = t[]; + return len; +} + +int nsvg__parseTranslate (float* xform, const(char)[] str) { + float[2] args = void; + float[6] t = void; + int na = 0; + int len = nsvg__parseTransformArgs(str, args.ptr, 2, &na); + if (na == 1) args[1] = 0.0; + nsvg__xformSetTranslation(t.ptr, args.ptr[0], args.ptr[1]); + xform[0..6] = t[]; + return len; +} + +int nsvg__parseScale (float* xform, const(char)[] str) { + float[2] args = void; + int na = 0; + float[6] t = void; + int len = nsvg__parseTransformArgs(str, args.ptr, 2, &na); + if (na == 1) args.ptr[1] = args.ptr[0]; + nsvg__xformSetScale(t.ptr, args.ptr[0], args.ptr[1]); + xform[0..6] = t[]; + return len; +} + +int nsvg__parseSkewX (float* xform, const(char)[] str) { + float[1] args = void; + int na = 0; + float[6] t = void; + int len = nsvg__parseTransformArgs(str, args.ptr, 1, &na); + nsvg__xformSetSkewX(t.ptr, args.ptr[0]/180.0f*NSVG_PI); + xform[0..6] = t[]; + return len; +} + +int nsvg__parseSkewY (float* xform, const(char)[] str) { + float[1] args = void; + int na = 0; + float[6] t = void; + int len = nsvg__parseTransformArgs(str, args.ptr, 1, &na); + nsvg__xformSetSkewY(t.ptr, args.ptr[0]/180.0f*NSVG_PI); + xform[0..6] = t[]; + return len; +} + +int nsvg__parseRotate (float* xform, const(char)[] str) { + float[3] args = void; + int na = 0; + float[6] m = void; + float[6] t = void; + int len = nsvg__parseTransformArgs(str, args.ptr, 3, &na); + if (na == 1) args.ptr[1] = args.ptr[2] = 0.0f; + nsvg__xformIdentity(m.ptr); + + if (na > 1) { + nsvg__xformSetTranslation(t.ptr, -args.ptr[1], -args.ptr[2]); + nsvg__xformMultiply(m.ptr, t.ptr); + } + + nsvg__xformSetRotation(t.ptr, args.ptr[0]/180.0f*NSVG_PI); + nsvg__xformMultiply(m.ptr, t.ptr); + + if (na > 1) { + nsvg__xformSetTranslation(t.ptr, args.ptr[1], args.ptr[2]); + nsvg__xformMultiply(m.ptr, t.ptr); + } + + xform[0..6] = m[]; + + return len; +} + +bool startsWith (const(char)[] str, const(char)[] sw) { + pragma(inline, true); + return (sw.length <= str.length && str[0..sw.length] == sw[]); +} + +void nsvg__parseTransform (float* xform, const(char)[] str) { + float[6] t = void; + nsvg__xformIdentity(xform); + while (str.length) { + int len; + if (startsWith(str, "matrix")) len = nsvg__parseMatrix(t.ptr, str); + else if (startsWith(str, "translate")) len = nsvg__parseTranslate(t.ptr, str); + else if (startsWith(str, "scale")) len = nsvg__parseScale(t.ptr, str); + else if (startsWith(str, "rotate")) len = nsvg__parseRotate(t.ptr, str); + else if (startsWith(str, "skewX")) len = nsvg__parseSkewX(t.ptr, str); + else if (startsWith(str, "skewY")) len = nsvg__parseSkewY(t.ptr, str); + else { str = str[1..$]; continue; } + str = str[len..$]; + nsvg__xformPremultiply(xform, t.ptr); + } +} + +// `id` should be prealloced +void nsvg__parseUrl (char[] id, const(char)[] str) { + int i = 0; + if (str.length >= 4) { + str = str[4..$]; // "url("; + if (str.length && str[0] == '#') str = str[1..$]; + while (str.length && str[0] != ')') { + if (id.length-i > 1) id[i++] = str[0]; + str = str[1..$]; + } + } + if (id.length-i > 0) id[i] = '\0'; +} + +NSVG.LineCap nsvg__parseLineCap (const(char)[] str) { + if (str == "butt") return NSVG.LineCap.Butt; + if (str == "round") return NSVG.LineCap.Round; + if (str == "square") return NSVG.LineCap.Square; + // TODO: handle inherit. + return NSVG.LineCap.Butt; +} + +NSVG.LineJoin nsvg__parseLineJoin (const(char)[] str) { + if (str == "miter") return NSVG.LineJoin.Miter; + if (str == "round") return NSVG.LineJoin.Round; + if (str == "bevel") return NSVG.LineJoin.Bevel; + // TODO: handle inherit. + return NSVG.LineJoin.Miter; +} + +NSVG.FillRule nsvg__parseFillRule (const(char)[] str) { + if (str == "nonzero") return NSVG.FillRule.NonZero; + if (str == "evenodd") return NSVG.FillRule.EvenOdd; + // TODO: handle inherit. + return NSVG.FillRule.EvenOdd; +} + + +int nsvg__parseStrokeDashArray (Parser* p, const(char)[] str, float* strokeDashArray) { + char[65] item = 0; + int count = 0; + float sum = 0.0f; + + int nsvg__getNextDashItem () { + int n = 0; + item[] = '\0'; + // skip white spaces and commas + while (str.length && (nsvg__isspace(str[0]) || str[0] == ',')) str = str[1..$]; + // advance until whitespace, comma or end + while (str.length && (!nsvg__isspace(str[0]) && str[0] != ',')) { + if (item.length-n > 1) item[n++] = str[0]; + str = str[1..$]; + } + return n; + } + + // Handle "none" + if (!str.length || str[0] == 'n') return 0; + + // Parse dashes + while (str.length) { + auto len = nsvg__getNextDashItem(); + if (len < 1) break; + if (count < NSVG_MAX_DASHES) strokeDashArray[count++] = fabsf(nsvg__parseCoordinate(p, item[0..len], 0.0f, nsvg__actualLength(p))); + } + + foreach (int i; 0..count) sum += strokeDashArray[i]; + if (sum <= 1e-6f) count = 0; + + return count; +} + +const(char)[] trimLeft (const(char)[] s, char ech=0) { + usize pos = 0; + while (pos < s.length) { + if (s.ptr[pos] <= ' ') { ++pos; continue; } + if (ech && s.ptr[pos] == ech) { ++pos; continue; } + if (s.ptr[pos] == '/' && s.length-pos > 1 && s.ptr[pos+1] == '*') { + pos += 2; + while (s.length-pos > 1 && !(s.ptr[pos] == '*' && s.ptr[pos+1] == '/')) ++pos; + if ((pos += 2) > s.length) pos = s.length; + continue; + } + break; + } + return s[pos..$]; +} + +static const(char)[] trimRight (const(char)[] s, char ech=0) { + usize pos = 0; + while (pos < s.length) { + if (s.ptr[pos] <= ' ' || (ech && s.ptr[pos] == ech)) { + if (s[pos..$].trimLeft(ech).length == 0) return s[0..pos]; + } else if (s.ptr[pos] == '/' && s.length-pos > 1 && s.ptr[pos+1] == '*') { + if (s[pos..$].trimLeft(ech).length == 0) return s[0..pos]; + } + ++pos; + } + return s; +} + +version(nanosvg_crappy_stylesheet_parser) { +Style* findStyle (Parser* p, char fch, const(char)[] name) { + if (name.length == 0) return null; + foreach (ref st; p.styles[0..p.styleCount]) { + if (st.name.length < 2 || st.name.ptr[0] != fch) continue; + if (st.name[1..$] == name) return &st; + } + return null; +} + +void nsvg__parseClassOrId (Parser* p, char lch, const(char)[] str) { + while (str.length) { + while (str.length && str.ptr[0] <= ' ') str = str[1..$]; + if (str.length == 0) break; + usize pos = 1; + while (pos < str.length && str.ptr[pos] > ' ') ++pos; + version(nanosvg_debug_styles) { import std.stdio; writeln("class to find: ", lch, str[0..pos].quote); } + if (auto st = p.findStyle(lch, str[0..pos])) { + version(nanosvg_debug_styles) { import std.stdio; writeln("class: [", str[0..pos], "]; value: ", st.value.quote); } + nsvg__parseStyle(p, st.value); + } + str = str[pos..$]; + } +} +} + +bool nsvg__parseAttr (Parser* p, const(char)[] name, const(char)[] value) { + float[6] xform = void; + Attrib* attr = nsvg__getAttr(p); + if (attr is null) return false; //??? + + if (name == "style") { + nsvg__parseStyle(p, value); + } else if (name == "display") { + if (value == "none") attr.visible = 0; + // Don't reset .visible on display:inline, one display:none hides the whole subtree + } else if (name == "fill") { + if (value == "none") { + attr.hasFill = 0; + } else if (startsWith(value, "url(")) { + attr.hasFill = 2; + nsvg__parseUrl(attr.fillGradient[], value); + } else { + attr.hasFill = 1; + attr.fillColor = nsvg__parseColor(value); + } + } else if (name == "opacity") { + attr.opacity = nsvg__parseOpacity(value); + } else if (name == "fill-opacity") { + attr.fillOpacity = nsvg__parseOpacity(value); + } else if (name == "stroke") { + if (value == "none") { + attr.hasStroke = 0; + } else if (startsWith(value, "url(")) { + attr.hasStroke = 2; + nsvg__parseUrl(attr.strokeGradient[], value); + } else { + attr.hasStroke = 1; + attr.strokeColor = nsvg__parseColor(value); + } + } else if (name == "stroke-width") { + attr.strokeWidth = nsvg__parseCoordinate(p, value, 0.0f, nsvg__actualLength(p)); + } else if (name == "stroke-dasharray") { + attr.strokeDashCount = nsvg__parseStrokeDashArray(p, value, attr.strokeDashArray.ptr); + } else if (name == "stroke-dashoffset") { + attr.strokeDashOffset = nsvg__parseCoordinate(p, value, 0.0f, nsvg__actualLength(p)); + } else if (name == "stroke-opacity") { + attr.strokeOpacity = nsvg__parseOpacity(value); + } else if (name == "stroke-linecap") { + attr.strokeLineCap = nsvg__parseLineCap(value); + } else if (name == "stroke-linejoin") { + attr.strokeLineJoin = nsvg__parseLineJoin(value); + } else if (name == "stroke-miterlimit") { + attr.miterLimit = nsvg__parseMiterLimit(value); + } else if (name == "fill-rule") { + attr.fillRule = nsvg__parseFillRule(value); + } else if (name == "font-size") { + attr.fontSize = nsvg__parseCoordinate(p, value, 0.0f, nsvg__actualLength(p)); + } else if (name == "transform") { + nsvg__parseTransform(xform.ptr, value); + nsvg__xformPremultiply(attr.xform.ptr, xform.ptr); + } else if (name == "stop-color") { + attr.stopColor = nsvg__parseColor(value); + } else if (name == "stop-opacity") { + attr.stopOpacity = nsvg__parseOpacity(value); + } else if (name == "offset") { + attr.stopOffset = nsvg__parseCoordinate(p, value, 0.0f, 1.0f); + } else if (name == "class") { + version(nanosvg_crappy_stylesheet_parser) nsvg__parseClassOrId(p, '.', value); + } else if (name == "id") { + // apply classes here too + version(nanosvg_crappy_stylesheet_parser) nsvg__parseClassOrId(p, '#', value); + attr.id[] = 0; + if (value.length > attr.id.length-1) value = value[0..attr.id.length-1]; + attr.id[0..value.length] = value[]; + } else { + return false; + } + return true; +} + +bool nsvg__parseNameValue (Parser* p, const(char)[] str) { + const(char)[] name; + + str = str.trimLeft; + usize pos = 0; + while (pos < str.length && str.ptr[pos] != ':') { + if (str.length-pos > 1 && str.ptr[pos] == '/' && str.ptr[pos+1] == '*') { + pos += 2; + while (str.length-pos > 1 && !(str.ptr[pos] == '*' && str.ptr[pos+1] == '/')) ++pos; + if ((pos += 2) > str.length) pos = str.length; + } else { + ++pos; + } + } + + name = str[0..pos].trimLeft.trimRight; + if (name.length == 0) return false; + + str = str[pos+(pos < str.length ? 1 : 0)..$].trimLeft.trimRight(';'); + + version(nanosvg_debug_styles) { import std.stdio; writeln("** name=", name.quote, "; value=", str.quote); } + + return nsvg__parseAttr(p, name, str); +} + +void nsvg__parseStyle (Parser* p, const(char)[] str) { + while (str.length) { + str = str.trimLeft; + usize pos = 0; + while (pos < str.length && str[pos] != ';') { + if (str.length-pos > 1 && str.ptr[pos] == '/' && str.ptr[pos+1] == '*') { + pos += 2; + while (str.length-pos > 1 && !(str.ptr[pos] == '*' && str.ptr[pos+1] == '/')) ++pos; + if ((pos += 2) > str.length) pos = str.length; + } else { + ++pos; + } + } + const(char)[] val = trimRight(str[0..pos]); + version(nanosvg_debug_styles) { import std.stdio; writeln("style: ", val.quote); } + str = str[pos+(pos < str.length ? 1 : 0)..$]; + if (val.length > 0) nsvg__parseNameValue(p, val); + } +} + +void nsvg__parseAttribs (Parser* p, AttrList attr) { + for (usize i = 0; attr.length-i >= 2; i += 2) { + if (attr[i] == "style") nsvg__parseStyle(p, attr[i+1]); + else if (attr[i] == "class") { version(nanosvg_crappy_stylesheet_parser) nsvg__parseClassOrId(p, '.', attr[i+1]); } + else nsvg__parseAttr(p, attr[i], attr[i+1]); + } +} + +int nsvg__getArgsPerElement (char cmd) { + switch (cmd) { + case 'v': case 'V': + case 'h': case 'H': + return 1; + case 'm': case 'M': + case 'l': case 'L': + case 't': case 'T': + return 2; + case 'q': case 'Q': + case 's': case 'S': + return 4; + case 'c': case 'C': + return 6; + case 'a': case 'A': + return 7; + default: + } + return 0; +} + +void nsvg__pathMoveTo (Parser* p, float* cpx, float* cpy, const(float)* args, bool rel) { + debug(nanosvg) { import std.stdio; writeln("nsvg__pathMoveTo: args=", args[0..2]); } + if (rel) { *cpx += args[0]; *cpy += args[1]; } else { *cpx = args[0]; *cpy = args[1]; } + nsvg__moveTo(p, *cpx, *cpy); +} + +void nsvg__pathLineTo (Parser* p, float* cpx, float* cpy, const(float)* args, bool rel) { + debug(nanosvg) { import std.stdio; writeln("nsvg__pathLineTo: args=", args[0..2]); } + if (rel) { *cpx += args[0]; *cpy += args[1]; } else { *cpx = args[0]; *cpy = args[1]; } + nsvg__lineTo(p, *cpx, *cpy); +} + +void nsvg__pathHLineTo (Parser* p, float* cpx, float* cpy, const(float)* args, bool rel) { + debug(nanosvg) { import std.stdio; writeln("nsvg__pathHLineTo: args=", args[0..1]); } + if (rel) *cpx += args[0]; else *cpx = args[0]; + nsvg__lineTo(p, *cpx, *cpy); +} + +void nsvg__pathVLineTo (Parser* p, float* cpx, float* cpy, const(float)* args, bool rel) { + debug(nanosvg) { import std.stdio; writeln("nsvg__pathVLineTo: args=", args[0..1]); } + if (rel) *cpy += args[0]; else *cpy = args[0]; + nsvg__lineTo(p, *cpx, *cpy); +} + +void nsvg__pathCubicBezTo (Parser* p, float* cpx, float* cpy, float* cpx2, float* cpy2, const(float)* args, bool rel) { + debug(nanosvg) { import std.stdio; writeln("nsvg__pathCubicBezTo: args=", args[0..6]); } + float cx1 = args[0]; + float cy1 = args[1]; + float cx2 = args[2]; + float cy2 = args[3]; + float x2 = args[4]; + float y2 = args[5]; + + if (rel) { + cx1 += *cpx; + cy1 += *cpy; + cx2 += *cpx; + cy2 += *cpy; + x2 += *cpx; + y2 += *cpy; + } + + nsvg__cubicBezTo(p, cx1, cy1, cx2, cy2, x2, y2); + + *cpx2 = cx2; + *cpy2 = cy2; + *cpx = x2; + *cpy = y2; +} + +void nsvg__pathCubicBezShortTo (Parser* p, float* cpx, float* cpy, float* cpx2, float* cpy2, const(float)* args, bool rel) { + debug(nanosvg) { import std.stdio; writeln("nsvg__pathCubicBezShortTo: args=", args[0..4]); } + + float cx2 = args[0]; + float cy2 = args[1]; + float x2 = args[2]; + float y2 = args[3]; + immutable float x1 = *cpx; + immutable float y1 = *cpy; + + if (rel) { + cx2 += *cpx; + cy2 += *cpy; + x2 += *cpx; + y2 += *cpy; + } + + immutable float cx1 = 2*x1-*cpx2; + immutable float cy1 = 2*y1-*cpy2; + + nsvg__cubicBezTo(p, cx1, cy1, cx2, cy2, x2, y2); + + *cpx2 = cx2; + *cpy2 = cy2; + *cpx = x2; + *cpy = y2; +} + +void nsvg__pathQuadBezTo (Parser* p, float* cpx, float* cpy, float* cpx2, float* cpy2, const(float)* args, bool rel) { + debug(nanosvg) { import std.stdio; writeln("nsvg__pathQuadBezTo: args=", args[0..4]); } + + float cx = args[0]; + float cy = args[1]; + float x2 = args[2]; + float y2 = args[3]; + immutable float x1 = *cpx; + immutable float y1 = *cpy; + + if (rel) { + cx += *cpx; + cy += *cpy; + x2 += *cpx; + y2 += *cpy; + } + + version(nanosvg_only_cubic_beziers) { + // convert to cubic bezier + immutable float cx1 = x1+2.0f/3.0f*(cx-x1); + immutable float cy1 = y1+2.0f/3.0f*(cy-y1); + immutable float cx2 = x2+2.0f/3.0f*(cx-x2); + immutable float cy2 = y2+2.0f/3.0f*(cy-y2); + nsvg__cubicBezTo(p, cx1, cy1, cx2, cy2, x2, y2); + } else { + nsvg__quadBezTo(p, cx, cy, x2, y2); + } + + *cpx2 = cx; + *cpy2 = cy; + *cpx = x2; + *cpy = y2; +} + +void nsvg__pathQuadBezShortTo (Parser* p, float* cpx, float* cpy, float* cpx2, float* cpy2, const(float)* args, bool rel) { + debug(nanosvg) { import std.stdio; writeln("nsvg__pathQuadBezShortTo: args=", args[0..2]); } + + float x2 = args[0]; + float y2 = args[1]; + immutable float x1 = *cpx; + immutable float y1 = *cpy; + + if (rel) { + x2 += *cpx; + y2 += *cpy; + } + + immutable float cx = 2*x1-*cpx2; + immutable float cy = 2*y1-*cpy2; + + version(nanosvg_only_cubic_beziers) { + // convert to cubic bezier + immutable float cx1 = x1+2.0f/3.0f*(cx-x1); + immutable float cy1 = y1+2.0f/3.0f*(cy-y1); + immutable float cx2 = x2+2.0f/3.0f*(cx-x2); + immutable float cy2 = y2+2.0f/3.0f*(cy-y2); + nsvg__cubicBezTo(p, cx1, cy1, cx2, cy2, x2, y2); + } else { + nsvg__quadBezTo(p, cx, cy, x2, y2); + } + + *cpx2 = cx; + *cpy2 = cy; + *cpx = x2; + *cpy = y2; +} + +float nsvg__sqr (in float x) pure nothrow @safe @nogc { pragma(inline, true); return x*x; } +float nsvg__vmag (in float x, float y) nothrow @safe @nogc { pragma(inline, true); return sqrtf(x*x+y*y); } + +float nsvg__vecrat (float ux, float uy, float vx, float vy) nothrow @safe @nogc { + pragma(inline, true); + return (ux*vx+uy*vy)/(nsvg__vmag(ux, uy)*nsvg__vmag(vx, vy)); +} + +float nsvg__vecang (float ux, float uy, float vx, float vy) nothrow @safe @nogc { + float r = nsvg__vecrat(ux, uy, vx, vy); + if (r < -1.0f) r = -1.0f; + if (r > 1.0f) r = 1.0f; + return (ux*vy < uy*vx ? -1.0f : 1.0f)*acosf(r); +} + +void nsvg__pathArcTo (Parser* p, float* cpx, float* cpy, const(float)* args, bool rel) { + // ported from canvg (https://code.google.com/p/canvg/) + float rx = fabsf(args[0]); // y radius + float ry = fabsf(args[1]); // x radius + immutable float rotx = args[2]/180.0f*NSVG_PI; // x rotation engle + immutable float fa = (fabsf(args[3]) > 1e-6 ? 1 : 0); // large arc + immutable float fs = (fabsf(args[4]) > 1e-6 ? 1 : 0); // sweep direction + immutable float x1 = *cpx; // start point + immutable float y1 = *cpy; + + // end point + float x2 = args[5]; + float y2 = args[6]; + + if (rel) { x2 += *cpx; y2 += *cpy; } + + float dx = x1-x2; + float dy = y1-y2; + immutable float d0 = sqrtf(dx*dx+dy*dy); + if (d0 < 1e-6f || rx < 1e-6f || ry < 1e-6f) { + // the arc degenerates to a line + nsvg__lineTo(p, x2, y2); + *cpx = x2; + *cpy = y2; + return; + } + + immutable float sinrx = sinf(rotx); + immutable float cosrx = cosf(rotx); + + // convert to center point parameterization + // http://www.w3.org/TR/SVG11/implnote.html#ArcImplementationNotes + // 1) Compute x1', y1' + immutable float x1p = cosrx*dx/2.0f+sinrx*dy/2.0f; + immutable float y1p = -sinrx*dx/2.0f+cosrx*dy/2.0f; + immutable float d1 = nsvg__sqr(x1p)/nsvg__sqr(rx)+nsvg__sqr(y1p)/nsvg__sqr(ry); + if (d1 > 1) { + immutable float d2 = sqrtf(d1); + rx *= d2; + ry *= d2; + } + // 2) Compute cx', cy' + float s = 0.0f; + float sa = nsvg__sqr(rx)*nsvg__sqr(ry)-nsvg__sqr(rx)*nsvg__sqr(y1p)-nsvg__sqr(ry)*nsvg__sqr(x1p); + immutable float sb = nsvg__sqr(rx)*nsvg__sqr(y1p)+nsvg__sqr(ry)*nsvg__sqr(x1p); + if (sa < 0.0f) sa = 0.0f; + if (sb > 0.0f) s = sqrtf(sa/sb); + if (fa == fs) s = -s; + immutable float cxp = s*rx*y1p/ry; + immutable float cyp = s*-ry*x1p/rx; + + // 3) Compute cx,cy from cx',cy' + immutable float cx = (x1+x2)/2.0f+cosrx*cxp-sinrx*cyp; + immutable float cy = (y1+y2)/2.0f+sinrx*cxp+cosrx*cyp; + + // 4) Calculate theta1, and delta theta. + immutable float ux = (x1p-cxp)/rx; + immutable float uy = (y1p-cyp)/ry; + immutable float vx = (-x1p-cxp)/rx; + immutable float vy = (-y1p-cyp)/ry; + immutable float a1 = nsvg__vecang(1.0f, 0.0f, ux, uy); // Initial angle + float da = nsvg__vecang(ux, uy, vx, vy); // Delta angle + + if (fs == 0 && da > 0) da -= 2*NSVG_PI; + else if (fs == 1 && da < 0) da += 2*NSVG_PI; + + float[6] t = void; + // approximate the arc using cubic spline segments + t.ptr[0] = cosrx; t.ptr[1] = sinrx; + t.ptr[2] = -sinrx; t.ptr[3] = cosrx; + t.ptr[4] = cx; t.ptr[5] = cy; + + // split arc into max 90 degree segments + // the loop assumes an iteration per end point (including start and end), this +1 + immutable ndivs = cast(int)(fabsf(da)/(NSVG_PI*0.5f)+1.0f); + immutable float hda = (da/cast(float)ndivs)/2.0f; + float kappa = fabsf(4.0f/3.0f*(1.0f-cosf(hda))/sinf(hda)); + if (da < 0.0f) kappa = -kappa; + + immutable float ndivsf = cast(float)ndivs; + float px = 0, py = 0, ptanx = 0, ptany = 0; + foreach (int i; 0..ndivs+1) { + float x = void, y = void, tanx = void, tany = void; + immutable float a = a1+da*(i/ndivsf); + immutable float loopdx = cosf(a); + immutable float loopdy = sinf(a); + nsvg__xformPoint(&x, &y, loopdx*rx, loopdy*ry, t.ptr); // position + nsvg__xformVec(&tanx, &tany, -loopdy*rx*kappa, loopdx*ry*kappa, t.ptr); // tangent + if (i > 0) nsvg__cubicBezTo(p, px+ptanx, py+ptany, x-tanx, y-tany, x, y); + px = x; + py = y; + ptanx = tanx; + ptany = tany; + } + + *cpx = x2; + *cpy = y2; +} + +void nsvg__parsePath (Parser* p, AttrList attr) { + const(char)[] s = null; + char cmd = '\0'; + float[10] args = void; + int nargs; + int rargs = 0; + float cpx = void, cpy = void, cpx2 = void, cpy2 = void; + bool closedFlag = false; + char[65] item = void; + + for (usize i = 0; attr.length-i >= 2; i += 2) { + if (attr[i] == "d") { + s = attr[i+1]; + } else { + const(char)[][2] tmp; + tmp[0] = attr[i]; + tmp[1] = attr[i+1]; + nsvg__parseAttribs(p, tmp[]); + } + } + + if (s.length) { + nsvg__resetPath(p); + cpx = 0; + cpy = 0; + cpx2 = 0; + cpy2 = 0; + closedFlag = false; + nargs = 0; + + while (s.length) { + auto skl = nsvg__getNextPathItem(s, item[]); + if (skl < s.length) s = s[skl..$]; else s = s[$..$]; + debug(nanosvg) { import std.stdio; writeln(":: ", item.fromAsciiz.quote, " : ", s.quote); } + if (!item[0]) break; + if (nsvg__isnum(item[0])) { + if (nargs < 10) { + args[nargs++] = nsvg__atof(item[]); + } + if (nargs >= rargs) { + switch (cmd) { + case 'm': case 'M': // move to + nsvg__pathMoveTo(p, &cpx, &cpy, args.ptr, (cmd == 'm' ? 1 : 0)); + // Moveto can be followed by multiple coordinate pairs, + // which should be treated as linetos. + cmd = (cmd == 'm' ? 'l' : 'L'); + rargs = nsvg__getArgsPerElement(cmd); + cpx2 = cpx; cpy2 = cpy; + break; + case 'l': case 'L': // line to + nsvg__pathLineTo(p, &cpx, &cpy, args.ptr, (cmd == 'l' ? 1 : 0)); + cpx2 = cpx; cpy2 = cpy; + break; + case 'H': case 'h': // horizontal line to + nsvg__pathHLineTo(p, &cpx, &cpy, args.ptr, (cmd == 'h' ? 1 : 0)); + cpx2 = cpx; cpy2 = cpy; + break; + case 'V': case 'v': // vertical line to + nsvg__pathVLineTo(p, &cpx, &cpy, args.ptr, (cmd == 'v' ? 1 : 0)); + cpx2 = cpx; cpy2 = cpy; + break; + case 'C': case 'c': // cubic bezier + nsvg__pathCubicBezTo(p, &cpx, &cpy, &cpx2, &cpy2, args.ptr, (cmd == 'c' ? 1 : 0)); + break; + case 'S': case 's': // "short" cubic bezier + nsvg__pathCubicBezShortTo(p, &cpx, &cpy, &cpx2, &cpy2, args.ptr, (cmd == 's' ? 1 : 0)); + break; + case 'Q': case 'q': // quadratic bezier + nsvg__pathQuadBezTo(p, &cpx, &cpy, &cpx2, &cpy2, args.ptr, (cmd == 'q' ? 1 : 0)); + break; + case 'T': case 't': // "short" quadratic bezier + nsvg__pathQuadBezShortTo(p, &cpx, &cpy, &cpx2, &cpy2, args.ptr, cmd == 't' ? 1 : 0); + break; + case 'A': case 'a': // arc + nsvg__pathArcTo(p, &cpx, &cpy, args.ptr, cmd == 'a' ? 1 : 0); + cpx2 = cpx; cpy2 = cpy; + break; + default: + if (nargs >= 2) { + cpx = args[nargs-2]; + cpy = args[nargs-1]; + cpx2 = cpx; + cpy2 = cpy; + } + break; + } + nargs = 0; + } + } else { + cmd = item[0]; + rargs = nsvg__getArgsPerElement(cmd); + if (cmd == 'M' || cmd == 'm') { + // commit path + if (p.nsflts > 0) nsvg__addPath(p, closedFlag); + // start new subpath + nsvg__resetPath(p); + closedFlag = false; + nargs = 0; + } else if (cmd == 'Z' || cmd == 'z') { + closedFlag = true; + // commit path + if (p.nsflts > 0) { + // move current point to first point + if ((cast(NSVG.Command)p.stream[0]) != NSVG.Command.MoveTo) assert(0, "NanoVega.SVG: invalid path"); + cpx = p.stream[1]; + cpy = p.stream[2]; + cpx2 = cpx; + cpy2 = cpy; + nsvg__addPath(p, closedFlag); + } + // start new subpath + nsvg__resetPath(p); + nsvg__moveTo(p, cpx, cpy); + closedFlag = false; + nargs = 0; + } + } + } + // commit path + if (p.nsflts) nsvg__addPath(p, closedFlag); + } + + nsvg__addShape(p); +} + +void nsvg__parseRect (Parser* p, AttrList attr) { + float x = 0.0f; + float y = 0.0f; + float w = 0.0f; + float h = 0.0f; + float rx = -1.0f; // marks not set + float ry = -1.0f; + + for (usize i = 0; attr.length-i >= 2; i += 2) { + if (!nsvg__parseAttr(p, attr[i], attr[i+1])) { + if (attr[i] == "x") x = nsvg__parseCoordinate(p, attr[i+1], nsvg__actualOrigX(p), nsvg__actualWidth(p)); + else if (attr[i] == "y") y = nsvg__parseCoordinate(p, attr[i+1], nsvg__actualOrigY(p), nsvg__actualHeight(p)); + else if (attr[i] == "width") w = nsvg__parseCoordinate(p, attr[i+1], 0.0f, nsvg__actualWidth(p)); + else if (attr[i] == "height") h = nsvg__parseCoordinate(p, attr[i+1], 0.0f, nsvg__actualHeight(p)); + else if (attr[i] == "rx") rx = fabsf(nsvg__parseCoordinate(p, attr[i+1], 0.0f, nsvg__actualWidth(p))); + else if (attr[i] == "ry") ry = fabsf(nsvg__parseCoordinate(p, attr[i+1], 0.0f, nsvg__actualHeight(p))); + } + } + + if (rx < 0.0f && ry > 0.0f) rx = ry; + if (ry < 0.0f && rx > 0.0f) ry = rx; + if (rx < 0.0f) rx = 0.0f; + if (ry < 0.0f) ry = 0.0f; + if (rx > w/2.0f) rx = w/2.0f; + if (ry > h/2.0f) ry = h/2.0f; + + if (w != 0.0f && h != 0.0f) { + nsvg__resetPath(p); + + if (rx < 0.00001f || ry < 0.0001f) { + nsvg__moveTo(p, x, y); + nsvg__lineTo(p, x+w, y); + nsvg__lineTo(p, x+w, y+h); + nsvg__lineTo(p, x, y+h); + } else { + // Rounded rectangle + nsvg__moveTo(p, x+rx, y); + nsvg__lineTo(p, x+w-rx, y); + nsvg__cubicBezTo(p, x+w-rx*(1-NSVG_KAPPA90), y, x+w, y+ry*(1-NSVG_KAPPA90), x+w, y+ry); + nsvg__lineTo(p, x+w, y+h-ry); + nsvg__cubicBezTo(p, x+w, y+h-ry*(1-NSVG_KAPPA90), x+w-rx*(1-NSVG_KAPPA90), y+h, x+w-rx, y+h); + nsvg__lineTo(p, x+rx, y+h); + nsvg__cubicBezTo(p, x+rx*(1-NSVG_KAPPA90), y+h, x, y+h-ry*(1-NSVG_KAPPA90), x, y+h-ry); + nsvg__lineTo(p, x, y+ry); + nsvg__cubicBezTo(p, x, y+ry*(1-NSVG_KAPPA90), x+rx*(1-NSVG_KAPPA90), y, x+rx, y); + } + + nsvg__addPath(p, 1); + + nsvg__addShape(p); + } +} + +void nsvg__parseCircle (Parser* p, AttrList attr) { + float cx = 0.0f; + float cy = 0.0f; + float r = 0.0f; + + for (usize i = 0; attr.length-i >= 2; i += 2) { + if (!nsvg__parseAttr(p, attr[i], attr[i+1])) { + if (attr[i] == "cx") cx = nsvg__parseCoordinate(p, attr[i+1], nsvg__actualOrigX(p), nsvg__actualWidth(p)); + else if (attr[i] == "cy") cy = nsvg__parseCoordinate(p, attr[i+1], nsvg__actualOrigY(p), nsvg__actualHeight(p)); + else if (attr[i] == "r") r = fabsf(nsvg__parseCoordinate(p, attr[i+1], 0.0f, nsvg__actualLength(p))); + } + } + + if (r > 0.0f) { + nsvg__resetPath(p); + + nsvg__moveTo(p, cx+r, cy); + nsvg__cubicBezTo(p, cx+r, cy+r*NSVG_KAPPA90, cx+r*NSVG_KAPPA90, cy+r, cx, cy+r); + nsvg__cubicBezTo(p, cx-r*NSVG_KAPPA90, cy+r, cx-r, cy+r*NSVG_KAPPA90, cx-r, cy); + nsvg__cubicBezTo(p, cx-r, cy-r*NSVG_KAPPA90, cx-r*NSVG_KAPPA90, cy-r, cx, cy-r); + nsvg__cubicBezTo(p, cx+r*NSVG_KAPPA90, cy-r, cx+r, cy-r*NSVG_KAPPA90, cx+r, cy); + + nsvg__addPath(p, 1); + + nsvg__addShape(p); + } +} + +void nsvg__parseEllipse (Parser* p, AttrList attr) { + float cx = 0.0f; + float cy = 0.0f; + float rx = 0.0f; + float ry = 0.0f; + + for (usize i = 0; attr.length-i >= 2; i += 2) { + if (!nsvg__parseAttr(p, attr[i], attr[i+1])) { + if (attr[i] == "cx") cx = nsvg__parseCoordinate(p, attr[i+1], nsvg__actualOrigX(p), nsvg__actualWidth(p)); + else if (attr[i] == "cy") cy = nsvg__parseCoordinate(p, attr[i+1], nsvg__actualOrigY(p), nsvg__actualHeight(p)); + else if (attr[i] == "rx") rx = fabsf(nsvg__parseCoordinate(p, attr[i+1], 0.0f, nsvg__actualWidth(p))); + else if (attr[i] == "ry") ry = fabsf(nsvg__parseCoordinate(p, attr[i+1], 0.0f, nsvg__actualHeight(p))); + } + } + + if (rx > 0.0f && ry > 0.0f) { + nsvg__resetPath(p); + + nsvg__moveTo(p, cx+rx, cy); + nsvg__cubicBezTo(p, cx+rx, cy+ry*NSVG_KAPPA90, cx+rx*NSVG_KAPPA90, cy+ry, cx, cy+ry); + nsvg__cubicBezTo(p, cx-rx*NSVG_KAPPA90, cy+ry, cx-rx, cy+ry*NSVG_KAPPA90, cx-rx, cy); + nsvg__cubicBezTo(p, cx-rx, cy-ry*NSVG_KAPPA90, cx-rx*NSVG_KAPPA90, cy-ry, cx, cy-ry); + nsvg__cubicBezTo(p, cx+rx*NSVG_KAPPA90, cy-ry, cx+rx, cy-ry*NSVG_KAPPA90, cx+rx, cy); + + nsvg__addPath(p, 1); + + nsvg__addShape(p); + } +} + +void nsvg__parseLine (Parser* p, AttrList attr) { + float x1 = 0.0; + float y1 = 0.0; + float x2 = 0.0; + float y2 = 0.0; + + for (usize i = 0; attr.length-i >= 2; i += 2) { + if (!nsvg__parseAttr(p, attr[i], attr[i+1])) { + if (attr[i] == "x1") x1 = nsvg__parseCoordinate(p, attr[i+1], nsvg__actualOrigX(p), nsvg__actualWidth(p)); + else if (attr[i] == "y1") y1 = nsvg__parseCoordinate(p, attr[i+1], nsvg__actualOrigY(p), nsvg__actualHeight(p)); + else if (attr[i] == "x2") x2 = nsvg__parseCoordinate(p, attr[i+1], nsvg__actualOrigX(p), nsvg__actualWidth(p)); + else if (attr[i] == "y2") y2 = nsvg__parseCoordinate(p, attr[i+1], nsvg__actualOrigY(p), nsvg__actualHeight(p)); + } + } + + nsvg__resetPath(p); + + nsvg__moveTo(p, x1, y1); + nsvg__lineTo(p, x2, y2); + + nsvg__addPath(p, 0); + + nsvg__addShape(p); +} + +void nsvg__parsePoly (Parser* p, AttrList attr, bool closeFlag) { + float[2] args = void; + int nargs, npts = 0; + char[65] item = 0; + + nsvg__resetPath(p); + + for (usize i = 0; attr.length-i >= 2; i += 2) { + if (!nsvg__parseAttr(p, attr[i], attr[i+1])) { + if (attr[i] == "points") { + const(char)[]s = attr[i+1]; + nargs = 0; + while (s.length) { + auto skl = nsvg__getNextPathItem(s, item[]); + if (skl < s.length) s = s[skl..$]; else s = s[$..$]; + args[nargs++] = nsvg__atof(item[]); + if (nargs >= 2) { + if (npts == 0) nsvg__moveTo(p, args[0], args[1]); else nsvg__lineTo(p, args[0], args[1]); + nargs = 0; + ++npts; + } + } + } + } + } + + nsvg__addPath(p, closeFlag); + + nsvg__addShape(p); +} + +void nsvg__parseSVG (Parser* p, AttrList attr) { + for (usize i = 0; attr.length-i >= 2; i += 2) { + if (!nsvg__parseAttr(p, attr[i], attr[i+1])) { + if (attr[i] == "width") { + p.image.width = nsvg__parseCoordinate(p, attr[i+1], 0.0f, p.canvaswdt); + //{ import core.stdc.stdio; printf("(%d) w=%d [%.*s]\n", p.canvaswdt, cast(int)p.image.width, cast(uint)attr[i+1].length, attr[i+1].ptr); } + } else if (attr[i] == "height") { + p.image.height = nsvg__parseCoordinate(p, attr[i+1], 0.0f, p.canvashgt); + } else if (attr[i] == "viewBox") { + xsscanf(attr[i+1], "%f%*[%%, \t]%f%*[%%, \t]%f%*[%%, \t]%f", p.viewMinx, p.viewMiny, p.viewWidth, p.viewHeight); + } else if (attr[i] == "preserveAspectRatio") { + if (attr[i+1].xindexOf("none") >= 0) { + // No uniform scaling + p.alignType = NSVG_ALIGN_NONE; + } else { + // Parse X align + if (attr[i+1].xindexOf("xMin") >= 0) p.alignX = NSVG_ALIGN_MIN; + else if (attr[i+1].xindexOf("xMid") >= 0) p.alignX = NSVG_ALIGN_MID; + else if (attr[i+1].xindexOf("xMax") >= 0) p.alignX = NSVG_ALIGN_MAX; + // Parse X align + if (attr[i+1].xindexOf("yMin") >= 0) p.alignY = NSVG_ALIGN_MIN; + else if (attr[i+1].xindexOf("yMid") >= 0) p.alignY = NSVG_ALIGN_MID; + else if (attr[i+1].xindexOf("yMax") >= 0) p.alignY = NSVG_ALIGN_MAX; + // Parse meet/slice + p.alignType = NSVG_ALIGN_MEET; + if (attr[i+1].xindexOf("slice") >= 0) p.alignType = NSVG_ALIGN_SLICE; + } + } + } + } +} + +void nsvg__parseGradient (Parser* p, AttrList attr, NSVG.PaintType type) { + GradientData* grad = xalloc!GradientData; + if (grad is null) return; + //memset(grad, 0, GradientData.sizeof); + grad.units = GradientUnits.Object; + grad.type = type; + if (grad.type == NSVG.PaintType.LinearGradient) { + grad.linear.x1 = nsvg__coord(0.0f, Units.percent); + grad.linear.y1 = nsvg__coord(0.0f, Units.percent); + grad.linear.x2 = nsvg__coord(100.0f, Units.percent); + grad.linear.y2 = nsvg__coord(0.0f, Units.percent); + } else if (grad.type == NSVG.PaintType.RadialGradient) { + grad.radial.cx = nsvg__coord(50.0f, Units.percent); + grad.radial.cy = nsvg__coord(50.0f, Units.percent); + grad.radial.r = nsvg__coord(50.0f, Units.percent); + } + + nsvg__xformIdentity(grad.xform.ptr); + + for (usize i = 0; attr.length-i >= 2; i += 2) { + if (attr[i] == "id") { + grad.id[] = 0; + const(char)[] s = attr[i+1]; + if (s.length > grad.id.length-1) s = s[0..grad.id.length-1]; + grad.id[0..s.length] = s[]; + } else if (!nsvg__parseAttr(p, attr[i], attr[i+1])) { + if (attr[i] == "gradientUnits") { if (attr[i+1] == "objectBoundingBox") grad.units = GradientUnits.Object; else grad.units = GradientUnits.User; } + else if (attr[i] == "gradientTransform") { nsvg__parseTransform(grad.xform.ptr, attr[i+1]); } + else if (attr[i] == "cx") { grad.radial.cx = nsvg__parseCoordinateRaw(attr[i+1]); } + else if (attr[i] == "cy") { grad.radial.cy = nsvg__parseCoordinateRaw(attr[i+1]); } + else if (attr[i] == "r") { grad.radial.r = nsvg__parseCoordinateRaw(attr[i+1]); } + else if (attr[i] == "fx") { grad.radial.fx = nsvg__parseCoordinateRaw(attr[i+1]); } + else if (attr[i] == "fy") { grad.radial.fy = nsvg__parseCoordinateRaw(attr[i+1]); } + else if (attr[i] == "x1") { grad.linear.x1 = nsvg__parseCoordinateRaw(attr[i+1]); } + else if (attr[i] == "y1") { grad.linear.y1 = nsvg__parseCoordinateRaw(attr[i+1]); } + else if (attr[i] == "x2") { grad.linear.x2 = nsvg__parseCoordinateRaw(attr[i+1]); } + else if (attr[i] == "y2") { grad.linear.y2 = nsvg__parseCoordinateRaw(attr[i+1]); } + else if (attr[i] == "spreadMethod") { + if (attr[i+1] == "pad") grad.spread = NSVG.SpreadType.Pad; + else if (attr[i+1] == "reflect") grad.spread = NSVG.SpreadType.Reflect; + else if (attr[i+1] == "repeat") grad.spread = NSVG.SpreadType.Repeat; + } else if (attr[i] == "xlink:href") { + grad.ref_[] = 0; + const(char)[] s = attr[i+1]; + if (s.length > 0 && s.ptr[0] == '#') s = s[1..$]; // remove '#' + if (s.length > grad.ref_.length-1) s = s[0..grad.ref_.length-1]; + grad.ref_[0..s.length] = s[]; + } + } + } + + grad.next = p.gradients; + p.gradients = grad; +} + +void nsvg__parseGradientStop (Parser* p, AttrList attr) { + import core.stdc.stdlib : realloc; + + Attrib* curAttr = nsvg__getAttr(p); + GradientData* grad; + NSVG.GradientStop* stop; + int idx; + + curAttr.stopOffset = 0; + curAttr.stopColor = 0; + curAttr.stopOpacity = 1.0f; + + for (usize i = 0; attr.length-i >= 2; i += 2) nsvg__parseAttr(p, attr[i], attr[i+1]); + + // Add stop to the last gradient. + grad = p.gradients; + if (grad is null) return; + + ++grad.nstops; + grad.stops = cast(NSVG.GradientStop*)realloc(grad.stops, NSVG.GradientStop.sizeof*grad.nstops+256); + if (grad.stops is null) assert(0, "nanosvg: out of memory"); + + // Insert + idx = grad.nstops-1; + foreach (int i; 0..grad.nstops-1) { + if (curAttr.stopOffset < grad.stops[i].offset) { + idx = i; + break; + } + } + if (idx != grad.nstops-1) { + for (int i = grad.nstops-1; i > idx; --i) grad.stops[i] = grad.stops[i-1]; + } + + stop = grad.stops+idx; + stop.color = curAttr.stopColor; + stop.color |= cast(uint)(curAttr.stopOpacity*255)<<24; + stop.offset = curAttr.stopOffset; +} + +void nsvg__startElement (void* ud, const(char)[] el, AttrList attr) { + Parser* p = cast(Parser*)ud; + + version(nanosvg_debug_styles) { import std.stdio; writeln("tagB: ", el.quote); } + version(nanosvg_crappy_stylesheet_parser) { p.inStyle = (el == "style"); } + + if (p.defsFlag) { + // Skip everything but gradients in defs + if (el == "linearGradient") { + nsvg__parseGradient(p, attr, NSVG.PaintType.LinearGradient); + } else if (el == "radialGradient") { + nsvg__parseGradient(p, attr, NSVG.PaintType.RadialGradient); + } else if (el == "stop") { + nsvg__parseGradientStop(p, attr); + } + return; + } + + if (el == "g") { + nsvg__pushAttr(p); + nsvg__parseAttribs(p, attr); + } else if (el == "path") { + if (p.pathFlag) return; // do not allow nested paths + p.pathFlag = true; + nsvg__pushAttr(p); + nsvg__parsePath(p, attr); + nsvg__popAttr(p); + } else if (el == "rect") { + nsvg__pushAttr(p); + nsvg__parseRect(p, attr); + nsvg__popAttr(p); + } else if (el == "circle") { + nsvg__pushAttr(p); + nsvg__parseCircle(p, attr); + nsvg__popAttr(p); + } else if (el == "ellipse") { + nsvg__pushAttr(p); + nsvg__parseEllipse(p, attr); + nsvg__popAttr(p); + } else if (el == "line") { + nsvg__pushAttr(p); + nsvg__parseLine(p, attr); + nsvg__popAttr(p); + } else if (el == "polyline") { + nsvg__pushAttr(p); + nsvg__parsePoly(p, attr, 0); + nsvg__popAttr(p); + } else if (el == "polygon") { + nsvg__pushAttr(p); + nsvg__parsePoly(p, attr, 1); + nsvg__popAttr(p); + } else if (el == "linearGradient") { + nsvg__parseGradient(p, attr, NSVG.PaintType.LinearGradient); + } else if (el == "radialGradient") { + nsvg__parseGradient(p, attr, NSVG.PaintType.RadialGradient); + } else if (el == "stop") { + nsvg__parseGradientStop(p, attr); + } else if (el == "defs") { + p.defsFlag = true; + } else if (el == "svg") { + nsvg__parseSVG(p, attr); + } +} + +void nsvg__endElement (void* ud, const(char)[] el) { + version(nanosvg_debug_styles) { import std.stdio; writeln("tagE: ", el.quote); } + Parser* p = cast(Parser*)ud; + if (el == "g") nsvg__popAttr(p); + else if (el == "path") p.pathFlag = false; + else if (el == "defs") p.defsFlag = false; + else if (el == "style") { version(nanosvg_crappy_stylesheet_parser) p.inStyle = false; } +} + +void nsvg__content (void* ud, const(char)[] s) { + version(nanosvg_crappy_stylesheet_parser) { + Parser* p = cast(Parser*)ud; + if (!p.inStyle) { + return; + } + // cheap hack + for (;;) { + while (s.length && s.ptr[0] <= ' ') s = s[1..$]; + if (!s.startsWith("')) s = s[0..$-1]; + if (s.length > 1 && s[$-2..$] == "]]") s = s[0..$-2]; else break; + } + version(nanosvg_debug_styles) { import std.stdio; writeln("ctx: ", s.quote); } + uint tokensAdded = 0; + while (s.length) { + if (s.length > 1 && s.ptr[0] == '/' && s.ptr[1] == '*') { + // comment + s = s[2..$]; + while (s.length > 1 && !(s.ptr[0] == '*' && s.ptr[1] == '/')) s = s[1..$]; + if (s.length <= 2) break; + s = s[2..$]; + continue; + } else if (s.ptr[0] <= ' ') { + while (s.length && s.ptr[0] <= ' ') s = s[1..$]; + continue; + } + //version(nanosvg_debug_styles) { import std.stdio; writeln("::: ", s.quote); } + if (s.ptr[0] == '{') { + usize pos = 1; + while (pos < s.length && s.ptr[pos] != '}') { + if (s.length-pos > 1 && s.ptr[pos] == '/' && s.ptr[pos+1] == '*') { + // skip comment + pos += 2; + while (s.length-pos > 1 && !(s.ptr[pos] == '*' && s.ptr[pos+1] == '/')) ++pos; + if (s.length-pos <= 2) { pos = cast(uint)s.length; break; } + pos += 2; + } else { + ++pos; + } + } + version(nanosvg_debug_styles) { import std.stdio; writeln("*** style: ", s[1..pos].quote); } + if (tokensAdded > 0) { + foreach (immutable idx; p.styleCount-tokensAdded..p.styleCount) p.styles[idx].value = s[1..pos]; + } + tokensAdded = 0; + if (s.length-pos < 1) break; + s = s[pos+1..$]; + } else { + usize pos = 0; + while (pos < s.length && s.ptr[pos] > ' ' && s.ptr[pos] != '{' && s.ptr[pos] != '/') ++pos; + const(char)[] tk = s[0..pos]; + version(nanosvg_debug_styles) { import std.stdio; writeln("token: ", tk.quote); } + s = s[pos..$]; + { + import core.stdc.stdlib : realloc; + import core.stdc.string : memset; + p.styles = cast(typeof(p.styles))realloc(p.styles, p.styles[0].sizeof*(p.styleCount+1)); + memset(p.styles+p.styleCount, 0, p.styles[0].sizeof); + ++p.styleCount; + } + p.styles[p.styleCount-1].name = tk; + ++tokensAdded; + } + } + version(nanosvg_debug_styles) foreach (const ref st; p.styles[0..p.styleCount]) { import std.stdio; writeln("name: ", st.name.quote, "; value: ", st.value.quote); } + } +} + +void nsvg__imageBounds (Parser* p, float* bounds) { + NSVG.Shape* shape; + shape = p.image.shapes; + if (shape is null) { + bounds[0..4] = 0.0; + return; + } + bounds[0] = shape.bounds.ptr[0]; + bounds[1] = shape.bounds.ptr[1]; + bounds[2] = shape.bounds.ptr[2]; + bounds[3] = shape.bounds.ptr[3]; + for (shape = shape.next; shape !is null; shape = shape.next) { + bounds[0] = nsvg__minf(bounds[0], shape.bounds.ptr[0]); + bounds[1] = nsvg__minf(bounds[1], shape.bounds.ptr[1]); + bounds[2] = nsvg__maxf(bounds[2], shape.bounds.ptr[2]); + bounds[3] = nsvg__maxf(bounds[3], shape.bounds.ptr[3]); + } +} + +float nsvg__viewAlign (float content, float container, int type) { + if (type == NSVG_ALIGN_MIN) return 0; + if (type == NSVG_ALIGN_MAX) return container-content; + // mid + return (container-content)*0.5f; +} + +void nsvg__scaleGradient (NSVG.Gradient* grad, float tx, float ty, float sx, float sy) { + float[6] t = void; + nsvg__xformSetTranslation(t.ptr, tx, ty); + nsvg__xformMultiply(grad.xform.ptr, t.ptr); + + nsvg__xformSetScale(t.ptr, sx, sy); + nsvg__xformMultiply(grad.xform.ptr, t.ptr); +} + +void nsvg__scaleToViewbox (Parser* p, const(char)[] units) { + NSVG.Shape* shape; + NSVG.Path* path; + float tx = void, ty = void, sx = void, sy = void, us = void, avgs = void; + float[4] bounds = void; + float[6] t = void; + float* pt; + + // Guess image size if not set completely. + nsvg__imageBounds(p, bounds.ptr); + + if (p.viewWidth == 0) { + if (p.image.width > 0) { + p.viewWidth = p.image.width; + } else { + p.viewMinx = bounds[0]; + p.viewWidth = bounds[2]-bounds[0]; + } + } + if (p.viewHeight == 0) { + if (p.image.height > 0) { + p.viewHeight = p.image.height; + } else { + p.viewMiny = bounds[1]; + p.viewHeight = bounds[3]-bounds[1]; + } + } + if (p.image.width == 0) + p.image.width = p.viewWidth; + if (p.image.height == 0) + p.image.height = p.viewHeight; + + tx = -p.viewMinx; + ty = -p.viewMiny; + sx = p.viewWidth > 0 ? p.image.width/p.viewWidth : 0; + sy = p.viewHeight > 0 ? p.image.height/p.viewHeight : 0; + // Unit scaling + us = 1.0f/nsvg__convertToPixels(p, nsvg__coord(1.0f, nsvg__parseUnits(units)), 0.0f, 1.0f); + + // Fix aspect ratio + if (p.alignType == NSVG_ALIGN_MEET) { + // fit whole image into viewbox + sx = sy = nsvg__minf(sx, sy); + tx += nsvg__viewAlign(p.viewWidth*sx, p.image.width, p.alignX)/sx; + ty += nsvg__viewAlign(p.viewHeight*sy, p.image.height, p.alignY)/sy; + } else if (p.alignType == NSVG_ALIGN_SLICE) { + // fill whole viewbox with image + sx = sy = nsvg__maxf(sx, sy); + tx += nsvg__viewAlign(p.viewWidth*sx, p.image.width, p.alignX)/sx; + ty += nsvg__viewAlign(p.viewHeight*sy, p.image.height, p.alignY)/sy; + } + + // Transform + sx *= us; + sy *= us; + avgs = (sx+sy)/2.0f; + for (shape = p.image.shapes; shape !is null; shape = shape.next) { + shape.bounds.ptr[0] = (shape.bounds.ptr[0]+tx)*sx; + shape.bounds.ptr[1] = (shape.bounds.ptr[1]+ty)*sy; + shape.bounds.ptr[2] = (shape.bounds.ptr[2]+tx)*sx; + shape.bounds.ptr[3] = (shape.bounds.ptr[3]+ty)*sy; + for (path = shape.paths; path !is null; path = path.next) { + path.bounds[0] = (path.bounds[0]+tx)*sx; + path.bounds[1] = (path.bounds[1]+ty)*sy; + path.bounds[2] = (path.bounds[2]+tx)*sx; + path.bounds[3] = (path.bounds[3]+ty)*sy; + for (int i = 0; i+3 <= path.nsflts; ) { + int argc = 0; // pair of coords + NSVG.Command cmd = cast(NSVG.Command)path.stream[i++]; + final switch (cmd) { + case NSVG.Command.MoveTo: argc = 1; break; + case NSVG.Command.LineTo: argc = 1; break; + case NSVG.Command.QuadTo: argc = 2; break; + case NSVG.Command.BezierTo: argc = 3; break; + } + // scale points + while (argc-- > 0) { + path.stream[i+0] = (path.stream[i+0]+tx)*sx; + path.stream[i+1] = (path.stream[i+1]+ty)*sy; + i += 2; + } + } + } + + if (shape.fill.type == NSVG.PaintType.LinearGradient || shape.fill.type == NSVG.PaintType.RadialGradient) { + nsvg__scaleGradient(shape.fill.gradient, tx, ty, sx, sy); + //memcpy(t.ptr, shape.fill.gradient.xform.ptr, float.sizeof*6); + t.ptr[0..6] = shape.fill.gradient.xform[0..6]; + nsvg__xformInverse(shape.fill.gradient.xform.ptr, t.ptr); + } + if (shape.stroke.type == NSVG.PaintType.LinearGradient || shape.stroke.type == NSVG.PaintType.RadialGradient) { + nsvg__scaleGradient(shape.stroke.gradient, tx, ty, sx, sy); + //memcpy(t.ptr, shape.stroke.gradient.xform.ptr, float.sizeof*6); + t.ptr[0..6] = shape.stroke.gradient.xform[0..6]; + nsvg__xformInverse(shape.stroke.gradient.xform.ptr, t.ptr); + } + + shape.strokeWidth *= avgs; + shape.strokeDashOffset *= avgs; + foreach (immutable int i; 0..shape.strokeDashCount) shape.strokeDashArray[i] *= avgs; + } +} + +/// +public NSVG* nsvgParse (const(char)[] input, const(char)[] units="px", float dpi=96, int canvaswdt=-1, int canvashgt=-1) { + Parser* p; + NSVG* ret = null; + + /* + static if (NanoSVGHasVFS) { + if (input.length > 4 && input[0..5] == "NSVG\x00" && units == "px" && dpi == 96) { + return nsvgUnserialize(wrapStream(MemoryStreamRO(input))); + } + } + */ + + p = nsvg__createParser(); + if (p is null) return null; + p.dpi = dpi; + p.canvaswdt = (canvaswdt < 1 ? NSVGDefaults.CanvasWidth : canvaswdt); + p.canvashgt = (canvashgt < 1 ? NSVGDefaults.CanvasHeight : canvashgt); + + nsvg__parseXML(input, &nsvg__startElement, &nsvg__endElement, &nsvg__content, p); + + // Scale to viewBox + nsvg__scaleToViewbox(p, units); + + ret = p.image; + p.image = null; + + nsvg__deleteParser(p); + + return ret; +} + +/// +public void kill (NSVG* image) { + import core.stdc.string : memset; + NSVG.Shape* snext, shape; + if (image is null) return; + shape = image.shapes; + while (shape !is null) { + snext = shape.next; + nsvg__deletePaths(shape.paths); + nsvg__deletePaint(&shape.fill); + nsvg__deletePaint(&shape.stroke); + xfree(shape); + shape = snext; + } + memset(image, 0, (*image).sizeof); + xfree(image); +} + +} // nothrow @trusted @nogc + + +/// +public NSVG* nsvgParseFromFile (const(char)[] filename, const(char)[] units="px", float dpi=96, int canvaswdt=-1, int canvashgt=-1) nothrow { + import core.stdc.stdlib : malloc, free; + enum AddedBytes = 8; + + char* data = null; + scope(exit) if (data !is null) free(data); + + if (filename.length == 0) return null; + + try { + static if (NanoSVGHasIVVFS) { + auto fl = VFile(filename); + auto size = fl.size; + if (size > int.max/8 || size < 1) return null; + data = cast(char*)malloc(cast(uint)size+AddedBytes); + if (data is null) return null; + data[0..cast(uint)size+AddedBytes] = 0; + fl.rawReadExact(data[0..cast(uint)size]); + fl.close(); + } else { + import core.stdc.stdio : FILE, fopen, fclose, fread, ftell, fseek; + import std.internal.cstring : tempCString; + auto fl = fopen(filename.tempCString, "rb"); + if (fl is null) return null; + scope(exit) fclose(fl); + if (fseek(fl, 0, 2/*SEEK_END*/) != 0) return null; + auto size = ftell(fl); + if (fseek(fl, 0, 0/*SEEK_SET*/) != 0) return null; + if (size < 16 || size > int.max/32) return null; + data = cast(char*)malloc(cast(uint)size+AddedBytes); + if (data is null) assert(0, "out of memory in NanoVega fontstash"); + data[0..cast(uint)size+AddedBytes] = 0; + char* dptr = data; + auto left = cast(uint)size; + while (left > 0) { + auto rd = fread(dptr, 1, left, fl); + if (rd == 0) return null; // unexpected EOF or reading error, it doesn't matter + dptr += rd; + left -= rd; + } + } + return nsvgParse(data[0..cast(uint)size], units, dpi, canvaswdt, canvashgt); + } catch (Exception e) { + return null; + } +} + + +static if (NanoSVGHasIVVFS) { +/// +public NSVG* nsvgParseFromFile(ST) (auto ref ST fi, const(char)[] units="px", float dpi=96, int canvaswdt=-1, int canvashgt=-1) nothrow +if (isReadableStream!ST && isSeekableStream!ST && streamHasSize!ST) +{ + import core.stdc.stdlib : malloc, free; + + enum AddedBytes = 8; + usize size; + char* data = null; + scope(exit) if (data is null) free(data); + + try { + auto sz = fi.size; + auto pp = fi.tell; + if (pp >= sz) return null; + sz -= pp; + if (sz > 0x3ff_ffff) return null; + size = cast(usize)sz; + data = cast(char*)malloc(size+AddedBytes); + if (data is null) return null; + scope(exit) free(data); + data[0..size+AddedBytes] = 0; + fi.rawReadExact(data[0..size]); + return nsvgParse(data[0..size], units, dpi, canvaswdt, canvashgt); + } catch (Exception e) { + return null; + } +} +} + + +// ////////////////////////////////////////////////////////////////////////// // +// rasterizer +private: +nothrow @trusted @nogc { + +enum NSVG__SUBSAMPLES = 5; +enum NSVG__FIXSHIFT = 10; +enum NSVG__FIX = 1< NSVG__MEMPAGE_SIZE) return null; + if (r.curpage is null || r.curpage.size+size > NSVG__MEMPAGE_SIZE) { + r.curpage = nsvg__nextPage(r, r.curpage); + } + buf = &r.curpage.mem[r.curpage.size]; + r.curpage.size += size; + return buf; +} + +int nsvg__ptEquals (float x1, float y1, float x2, float y2, float tol) { + immutable float dx = x2-x1; + immutable float dy = y2-y1; + return dx*dx+dy*dy < tol*tol; +} + +void nsvg__addPathPoint (NSVGrasterizer r, float x, float y, int flags) { + import core.stdc.stdlib : realloc; + + NSVGpoint* pt; + + if (r.npoints > 0) { + pt = r.points+(r.npoints-1); + if (nsvg__ptEquals(pt.x, pt.y, x, y, r.distTol)) { + pt.flags |= flags; + return; + } + } + + if (r.npoints+1 > r.cpoints) { + r.cpoints = (r.cpoints > 0 ? r.cpoints*2 : 64); + r.points = cast(NSVGpoint*)realloc(r.points, NSVGpoint.sizeof*r.cpoints+256); + if (r.points is null) assert(0, "nanosvg: out of memory"); + } + + pt = r.points+r.npoints; + pt.x = x; + pt.y = y; + pt.flags = cast(ubyte)flags; + ++r.npoints; +} + +void nsvg__appendPathPoint (NSVGrasterizer r, NSVGpoint pt) { + import core.stdc.stdlib : realloc; + if (r.npoints+1 > r.cpoints) { + r.cpoints = (r.cpoints > 0 ? r.cpoints*2 : 64); + r.points = cast(NSVGpoint*)realloc(r.points, NSVGpoint.sizeof*r.cpoints+256); + if (r.points is null) assert(0, "nanosvg: out of memory"); + } + r.points[r.npoints] = pt; + ++r.npoints; +} + +void nsvg__duplicatePoints (NSVGrasterizer r) { + import core.stdc.stdlib : realloc; + import core.stdc.string : memmove; + if (r.npoints > r.cpoints2) { + r.cpoints2 = r.npoints; + r.points2 = cast(NSVGpoint*)realloc(r.points2, NSVGpoint.sizeof*r.cpoints2+256); + if (r.points2 is null) assert(0, "nanosvg: out of memory"); + } + memmove(r.points2, r.points, NSVGpoint.sizeof*r.npoints); + r.npoints2 = r.npoints; +} + +void nsvg__addEdge (NSVGrasterizer r, float x0, float y0, float x1, float y1) { + NSVGedge* e; + + // Skip horizontal edges + if (y0 == y1) return; + + if (r.nedges+1 > r.cedges) { + import core.stdc.stdlib : realloc; + r.cedges = (r.cedges > 0 ? r.cedges*2 : 64); + r.edges = cast(NSVGedge*)realloc(r.edges, NSVGedge.sizeof*r.cedges+256); + if (r.edges is null) assert(0, "nanosvg: out of memory"); + } + + e = &r.edges[r.nedges]; + ++r.nedges; + + if (y0 < y1) { + e.x0 = x0; + e.y0 = y0; + e.x1 = x1; + e.y1 = y1; + e.dir = 1; + } else { + e.x0 = x1; + e.y0 = y1; + e.x1 = x0; + e.y1 = y0; + e.dir = -1; + } +} + +float nsvg__normalize (float *x, float* y) { + immutable float d = sqrtf((*x)*(*x)+(*y)*(*y)); + if (d > 1e-6f) { + float id = 1.0f/d; + *x *= id; + *y *= id; + } + return d; +} + +void nsvg__flattenCubicBez (NSVGrasterizer r, in float x1, in float y1, in float x2, in float y2, in float x3, in float y3, in float x4, in float y4, in int level, in int type) { + if (level > 10) return; + + // check for collinear points, and use AFD tesselator on such curves (it is WAY faster for this case) + version(none) { + if (level == 0) { + static bool collinear (in float v0x, in float v0y, in float v1x, in float v1y, in float v2x, in float v2y) nothrow @trusted @nogc { + immutable float cz = (v1x-v0x)*(v2y-v0y)-(v2x-v0x)*(v1y-v0y); + return (fabsf(cz*cz) <= 0.01f); + } + if (collinear(x1, y1, x2, y2, x3, y3) && collinear(x2, y2, x3, y3, x3, y4)) { + //{ import core.stdc.stdio; printf("AFD fallback!\n"); } + nsvg__flattenCubicBezAFD(r, x1, y1, x2, y2, x3, y3, x4, y4, type); + return; + } + } + } + + immutable x12 = (x1+x2)*0.5f; + immutable y12 = (y1+y2)*0.5f; + immutable x23 = (x2+x3)*0.5f; + immutable y23 = (y2+y3)*0.5f; + immutable x34 = (x3+x4)*0.5f; + immutable y34 = (y3+y4)*0.5f; + immutable x123 = (x12+x23)*0.5f; + immutable y123 = (y12+y23)*0.5f; + + immutable dx = x4-x1; + immutable dy = y4-y1; + immutable d2 = fabsf(((x2-x4)*dy-(y2-y4)*dx)); + immutable d3 = fabsf(((x3-x4)*dy-(y3-y4)*dx)); + + if ((d2+d3)*(d2+d3) < r.tessTol*(dx*dx+dy*dy)) { + nsvg__addPathPoint(r, x4, y4, type); + return; + } + + immutable x234 = (x23+x34)*0.5f; + immutable y234 = (y23+y34)*0.5f; + immutable x1234 = (x123+x234)*0.5f; + immutable y1234 = (y123+y234)*0.5f; + + 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); +} + +// Adaptive forward differencing for bezier tesselation. +// See Lien, Sheue-Ling, Michael Shantz, and Vaughan Pratt. +// "Adaptive forward differencing for rendering curves and surfaces." +// ACM SIGGRAPH Computer Graphics. Vol. 21. No. 4. ACM, 1987. +// original code by Taylor Holliday +void nsvg__flattenCubicBezAFD (NSVGrasterizer r, in float x1, in float y1, in float x2, in float y2, in float x3, in float y3, in float x4, in float y4, in int type) nothrow @trusted @nogc { + enum AFD_ONE = (1<<10); + + // power basis + immutable float ax = -x1+3*x2-3*x3+x4; + immutable float ay = -y1+3*y2-3*y3+y4; + immutable float bx = 3*x1-6*x2+3*x3; + immutable float by = 3*y1-6*y2+3*y3; + immutable float cx = -3*x1+3*x2; + immutable float cy = -3*y1+3*y2; + + // Transform to forward difference basis (stepsize 1) + float px = x1; + float py = y1; + float dx = ax+bx+cx; + float dy = ay+by+cy; + float ddx = 6*ax+2*bx; + float ddy = 6*ay+2*by; + float dddx = 6*ax; + float dddy = 6*ay; + + //printf("dx: %f, dy: %f\n", dx, dy); + //printf("ddx: %f, ddy: %f\n", ddx, ddy); + //printf("dddx: %f, dddy: %f\n", dddx, dddy); + + int t = 0; + int dt = AFD_ONE; + + immutable float tol = r.tessTol*4; + + while (t < AFD_ONE) { + // Flatness measure. + float d = ddx*ddx+ddy*ddy+dddx*dddx+dddy*dddy; + + // printf("d: %f, th: %f\n", d, th); + + // Go to higher resolution if we're moving a lot or overshooting the end. + while ((d > tol && dt > 1) || (t+dt > AFD_ONE)) { + // printf("up\n"); + + // Apply L to the curve. Increase curve resolution. + dx = 0.5f*dx-(1.0f/8.0f)*ddx+(1.0f/16.0f)*dddx; + dy = 0.5f*dy-(1.0f/8.0f)*ddy+(1.0f/16.0f)*dddy; + ddx = (1.0f/4.0f)*ddx-(1.0f/8.0f)*dddx; + ddy = (1.0f/4.0f)*ddy-(1.0f/8.0f)*dddy; + dddx = (1.0f/8.0f)*dddx; + dddy = (1.0f/8.0f)*dddy; + + // Half the stepsize. + dt >>= 1; + + // Recompute d + d = ddx*ddx+ddy*ddy+dddx*dddx+dddy*dddy; + } + + // Go to lower resolution if we're really flat + // and we aren't going to overshoot the end. + // XXX: tol/32 is just a guess for when we are too flat. + while ((d > 0 && d < tol/32.0f && dt < AFD_ONE) && (t+2*dt <= AFD_ONE)) { + // printf("down\n"); + + // Apply L^(-1) to the curve. Decrease curve resolution. + dx = 2*dx+ddx; + dy = 2*dy+ddy; + ddx = 4*ddx+4*dddx; + ddy = 4*ddy+4*dddy; + dddx = 8*dddx; + dddy = 8*dddy; + + // Double the stepsize. + dt <<= 1; + + // Recompute d + d = ddx*ddx+ddy*ddy+dddx*dddx+dddy*dddy; + } + + // Forward differencing. + px += dx; + py += dy; + dx += ddx; + dy += ddy; + ddx += dddx; + ddy += dddy; + + // Output a point. + nsvg__addPathPoint(r, px, py, (t > 0 ? type : 0)); + + // Advance along the curve. + t += dt; + + // Ensure we don't overshoot. + assert(t <= AFD_ONE); + } +} + +void nsvg__flattenShape (NSVGrasterizer r, const(NSVG.Shape)* shape, float scale) { + for (const(NSVG.Path)* path = shape.paths; path !is null; path = path.next) { + r.npoints = 0; + if (path.empty) continue; + // first point + float x0, y0; + path.startPoint(&x0, &y0); + nsvg__addPathPoint(r, x0*scale, y0*scale, 0); + // cubic beziers + path.asCubics(delegate (const(float)[] cubic) { + assert(cubic.length >= 8); + nsvg__flattenCubicBez(r, + cubic.ptr[0]*scale, cubic.ptr[1]*scale, + cubic.ptr[2]*scale, cubic.ptr[3]*scale, + cubic.ptr[4]*scale, cubic.ptr[5]*scale, + cubic.ptr[6]*scale, cubic.ptr[7]*scale, + 0, 0); + }); + // close path + nsvg__addPathPoint(r, x0*scale, y0*scale, 0); + // Flatten path + /+ + nsvg__addPathPoint(r, path.pts[0]*scale, path.pts[1]*scale, 0); + for (int i = 0; i < path.npts-1; i += 3) { + const(float)* p = path.pts+(i*2); + nsvg__flattenCubicBez(r, p[0]*scale, p[1]*scale, p[2]*scale, p[3]*scale, p[4]*scale, p[5]*scale, p[6]*scale, p[7]*scale, 0, 0); + } + // Close path + nsvg__addPathPoint(r, path.pts[0]*scale, path.pts[1]*scale, 0); + +/ + // Build edges + for (int i = 0, j = r.npoints-1; i < r.npoints; j = i++) { + nsvg__addEdge(r, r.points[j].x, r.points[j].y, r.points[i].x, r.points[i].y); + } + } +} + +alias PtFlags = ubyte; +enum : ubyte { + PtFlagsCorner = 0x01, + PtFlagsBevel = 0x02, + PtFlagsLeft = 0x04, +} + +void nsvg__initClosed (NSVGpoint* left, NSVGpoint* right, const(NSVGpoint)* p0, const(NSVGpoint)* p1, float lineWidth) { + immutable float w = lineWidth*0.5f; + float dx = p1.x-p0.x; + float dy = p1.y-p0.y; + immutable float len = nsvg__normalize(&dx, &dy); + immutable float px = p0.x+dx*len*0.5f, py = p0.y+dy*len*0.5f; + immutable float dlx = dy, dly = -dx; + immutable float lx = px-dlx*w, ly = py-dly*w; + immutable float rx = px+dlx*w, ry = py+dly*w; + left.x = lx; left.y = ly; + right.x = rx; right.y = ry; +} + +void nsvg__buttCap (NSVGrasterizer r, NSVGpoint* left, NSVGpoint* right, const(NSVGpoint)* p, float dx, float dy, float lineWidth, int connect) { + immutable float w = lineWidth*0.5f; + immutable float px = p.x, py = p.y; + immutable float dlx = dy, dly = -dx; + immutable float lx = px-dlx*w, ly = py-dly*w; + immutable float rx = px+dlx*w, ry = py+dly*w; + + nsvg__addEdge(r, lx, ly, rx, ry); + + if (connect) { + nsvg__addEdge(r, left.x, left.y, lx, ly); + nsvg__addEdge(r, rx, ry, right.x, right.y); + } + left.x = lx; left.y = ly; + right.x = rx; right.y = ry; +} + +void nsvg__squareCap (NSVGrasterizer r, NSVGpoint* left, NSVGpoint* right, const(NSVGpoint)* p, float dx, float dy, float lineWidth, int connect) { + immutable float w = lineWidth*0.5f; + immutable float px = p.x-dx*w, py = p.y-dy*w; + immutable float dlx = dy, dly = -dx; + immutable float lx = px-dlx*w, ly = py-dly*w; + immutable float rx = px+dlx*w, ry = py+dly*w; + + nsvg__addEdge(r, lx, ly, rx, ry); + + if (connect) { + nsvg__addEdge(r, left.x, left.y, lx, ly); + nsvg__addEdge(r, rx, ry, right.x, right.y); + } + left.x = lx; left.y = ly; + right.x = rx; right.y = ry; +} + +void nsvg__roundCap (NSVGrasterizer r, NSVGpoint* left, NSVGpoint* right, const(NSVGpoint)* p, float dx, float dy, float lineWidth, int ncap, int connect) { + immutable float w = lineWidth*0.5f; + immutable float px = p.x, py = p.y; + immutable float dlx = dy, dly = -dx; + float lx = 0, ly = 0, rx = 0, ry = 0, prevx = 0, prevy = 0; + + foreach (int i; 0..ncap) { + immutable float a = i/cast(float)(ncap-1)*NSVG_PI; + immutable float ax = cosf(a)*w, ay = sinf(a)*w; + immutable float x = px-dlx*ax-dx*ay; + immutable float y = py-dly*ax-dy*ay; + + if (i > 0) nsvg__addEdge(r, prevx, prevy, x, y); + + prevx = x; + prevy = y; + + if (i == 0) { + lx = x; + ly = y; + } else if (i == ncap-1) { + rx = x; + ry = y; + } + } + + if (connect) { + nsvg__addEdge(r, left.x, left.y, lx, ly); + nsvg__addEdge(r, rx, ry, right.x, right.y); + } + + left.x = lx; left.y = ly; + right.x = rx; right.y = ry; +} + +void nsvg__bevelJoin (NSVGrasterizer r, NSVGpoint* left, NSVGpoint* right, const(NSVGpoint)* p0, const(NSVGpoint)* p1, float lineWidth) { + immutable float w = lineWidth*0.5f; + immutable float dlx0 = p0.dy, dly0 = -p0.dx; + immutable float dlx1 = p1.dy, dly1 = -p1.dx; + immutable float lx0 = p1.x-(dlx0*w), ly0 = p1.y-(dly0*w); + immutable float rx0 = p1.x+(dlx0*w), ry0 = p1.y+(dly0*w); + immutable float lx1 = p1.x-(dlx1*w), ly1 = p1.y-(dly1*w); + immutable float rx1 = p1.x+(dlx1*w), ry1 = p1.y+(dly1*w); + + nsvg__addEdge(r, lx0, ly0, left.x, left.y); + nsvg__addEdge(r, lx1, ly1, lx0, ly0); + + nsvg__addEdge(r, right.x, right.y, rx0, ry0); + nsvg__addEdge(r, rx0, ry0, rx1, ry1); + + left.x = lx1; left.y = ly1; + right.x = rx1; right.y = ry1; +} + +void nsvg__miterJoin (NSVGrasterizer r, NSVGpoint* left, NSVGpoint* right, const(NSVGpoint)* p0, const(NSVGpoint)* p1, float lineWidth) { + immutable float w = lineWidth*0.5f; + immutable float dlx0 = p0.dy, dly0 = -p0.dx; + immutable float dlx1 = p1.dy, dly1 = -p1.dx; + float lx0 = void, rx0 = void, lx1 = void, rx1 = void; + float ly0 = void, ry0 = void, ly1 = void, ry1 = void; + + if (p1.flags&PtFlagsLeft) { + lx0 = lx1 = p1.x-p1.dmx*w; + ly0 = ly1 = p1.y-p1.dmy*w; + nsvg__addEdge(r, lx1, ly1, left.x, left.y); + + rx0 = p1.x+(dlx0*w); + ry0 = p1.y+(dly0*w); + rx1 = p1.x+(dlx1*w); + ry1 = p1.y+(dly1*w); + nsvg__addEdge(r, right.x, right.y, rx0, ry0); + nsvg__addEdge(r, rx0, ry0, rx1, ry1); + } else { + lx0 = p1.x-(dlx0*w); + ly0 = p1.y-(dly0*w); + lx1 = p1.x-(dlx1*w); + ly1 = p1.y-(dly1*w); + nsvg__addEdge(r, lx0, ly0, left.x, left.y); + nsvg__addEdge(r, lx1, ly1, lx0, ly0); + + rx0 = rx1 = p1.x+p1.dmx*w; + ry0 = ry1 = p1.y+p1.dmy*w; + nsvg__addEdge(r, right.x, right.y, rx1, ry1); + } + + left.x = lx1; left.y = ly1; + right.x = rx1; right.y = ry1; +} + +void nsvg__roundJoin (NSVGrasterizer r, NSVGpoint* left, NSVGpoint* right, const(NSVGpoint)* p0, const(NSVGpoint)* p1, float lineWidth, int ncap) { + int i, n; + float w = lineWidth*0.5f; + float dlx0 = p0.dy, dly0 = -p0.dx; + float dlx1 = p1.dy, dly1 = -p1.dx; + float a0 = atan2f(dly0, dlx0); + float a1 = atan2f(dly1, dlx1); + float da = a1-a0; + float lx, ly, rx, ry; + + if (da < NSVG_PI) da += NSVG_PI*2; + if (da > NSVG_PI) da -= NSVG_PI*2; + + n = cast(int)ceilf((fabsf(da)/NSVG_PI)*ncap); + if (n < 2) n = 2; + if (n > ncap) n = ncap; + + lx = left.x; + ly = left.y; + rx = right.x; + ry = right.y; + + for (i = 0; i < n; i++) { + float u = i/cast(float)(n-1); + float a = a0+u*da; + float ax = cosf(a)*w, ay = sinf(a)*w; + float lx1 = p1.x-ax, ly1 = p1.y-ay; + float rx1 = p1.x+ax, ry1 = p1.y+ay; + + nsvg__addEdge(r, lx1, ly1, lx, ly); + nsvg__addEdge(r, rx, ry, rx1, ry1); + + lx = lx1; ly = ly1; + rx = rx1; ry = ry1; + } + + left.x = lx; left.y = ly; + right.x = rx; right.y = ry; +} + +void nsvg__straightJoin (NSVGrasterizer r, NSVGpoint* left, NSVGpoint* right, const(NSVGpoint)* p1, float lineWidth) { + float w = lineWidth*0.5f; + float lx = p1.x-(p1.dmx*w), ly = p1.y-(p1.dmy*w); + float rx = p1.x+(p1.dmx*w), ry = p1.y+(p1.dmy*w); + + nsvg__addEdge(r, lx, ly, left.x, left.y); + nsvg__addEdge(r, right.x, right.y, rx, ry); + + left.x = lx; left.y = ly; + right.x = rx; right.y = ry; +} + +int nsvg__curveDivs (float r, float arc, float tol) { + float da = acosf(r/(r+tol))*2.0f; + int divs = cast(int)ceilf(arc/da); + if (divs < 2) divs = 2; + return divs; +} + +void nsvg__expandStroke (NSVGrasterizer r, const(NSVGpoint)* points, int npoints, int closed, int lineJoin, int lineCap, float lineWidth) { + int ncap = nsvg__curveDivs(lineWidth*0.5f, NSVG_PI, r.tessTol); // Calculate divisions per half circle. + //NSVGpoint left = {0, 0, 0, 0, 0, 0, 0, 0}, right = {0, 0, 0, 0, 0, 0, 0, 0}, firstLeft = {0, 0, 0, 0, 0, 0, 0, 0}, firstRight = {0, 0, 0, 0, 0, 0, 0, 0}; + NSVGpoint left, right, firstLeft, firstRight; + const(NSVGpoint)* p0, p1; + int j, s, e; + + // Build stroke edges + if (closed) { + // Looping + p0 = &points[npoints-1]; + p1 = &points[0]; + s = 0; + e = npoints; + } else { + // Add cap + p0 = &points[0]; + p1 = &points[1]; + s = 1; + e = npoints-1; + } + + if (closed) { + nsvg__initClosed(&left, &right, p0, p1, lineWidth); + firstLeft = left; + firstRight = right; + } else { + // Add cap + float dx = p1.x-p0.x; + float dy = p1.y-p0.y; + nsvg__normalize(&dx, &dy); + if (lineCap == NSVG.LineCap.Butt) + nsvg__buttCap(r, &left, &right, p0, dx, dy, lineWidth, 0); + else if (lineCap == NSVG.LineCap.Square) + nsvg__squareCap(r, &left, &right, p0, dx, dy, lineWidth, 0); + else if (lineCap == NSVG.LineCap.Round) + nsvg__roundCap(r, &left, &right, p0, dx, dy, lineWidth, ncap, 0); + } + + for (j = s; j < e; ++j) { + if (p1.flags&PtFlagsCorner) { + if (lineJoin == NSVG.LineJoin.Round) + nsvg__roundJoin(r, &left, &right, p0, p1, lineWidth, ncap); + else if (lineJoin == NSVG.LineJoin.Bevel || (p1.flags&PtFlagsBevel)) + nsvg__bevelJoin(r, &left, &right, p0, p1, lineWidth); + else + nsvg__miterJoin(r, &left, &right, p0, p1, lineWidth); + } else { + nsvg__straightJoin(r, &left, &right, p1, lineWidth); + } + p0 = p1++; + } + + if (closed) { + // Loop it + nsvg__addEdge(r, firstLeft.x, firstLeft.y, left.x, left.y); + nsvg__addEdge(r, right.x, right.y, firstRight.x, firstRight.y); + } else { + // Add cap + float dx = p1.x-p0.x; + float dy = p1.y-p0.y; + nsvg__normalize(&dx, &dy); + if (lineCap == NSVG.LineCap.Butt) + nsvg__buttCap(r, &right, &left, p1, -dx, -dy, lineWidth, 1); + else if (lineCap == NSVG.LineCap.Square) + nsvg__squareCap(r, &right, &left, p1, -dx, -dy, lineWidth, 1); + else if (lineCap == NSVG.LineCap.Round) + nsvg__roundCap(r, &right, &left, p1, -dx, -dy, lineWidth, ncap, 1); + } +} + +void nsvg__prepareStroke (NSVGrasterizer r, float miterLimit, int lineJoin) { + int i, j; + NSVGpoint* p0, p1; + + p0 = r.points+(r.npoints-1); + p1 = r.points; + for (i = 0; i < r.npoints; i++) { + // Calculate segment direction and length + p0.dx = p1.x-p0.x; + p0.dy = p1.y-p0.y; + p0.len = nsvg__normalize(&p0.dx, &p0.dy); + // Advance + p0 = p1++; + } + + // calculate joins + p0 = r.points+(r.npoints-1); + p1 = r.points; + for (j = 0; j < r.npoints; j++) { + float dlx0, dly0, dlx1, dly1, dmr2, cross; + dlx0 = p0.dy; + dly0 = -p0.dx; + dlx1 = p1.dy; + dly1 = -p1.dx; + // Calculate extrusions + p1.dmx = (dlx0+dlx1)*0.5f; + p1.dmy = (dly0+dly1)*0.5f; + dmr2 = p1.dmx*p1.dmx+p1.dmy*p1.dmy; + if (dmr2 > 0.000001f) { + float s2 = 1.0f/dmr2; + if (s2 > 600.0f) { + s2 = 600.0f; + } + p1.dmx *= s2; + p1.dmy *= s2; + } + + // Clear flags, but keep the corner. + p1.flags = (p1.flags&PtFlagsCorner) ? PtFlagsCorner : 0; + + // Keep track of left turns. + cross = p1.dx*p0.dy-p0.dx*p1.dy; + if (cross > 0.0f) + p1.flags |= PtFlagsLeft; + + // Check to see if the corner needs to be beveled. + if (p1.flags&PtFlagsCorner) { + if ((dmr2*miterLimit*miterLimit) < 1.0f || lineJoin == NSVG.LineJoin.Bevel || lineJoin == NSVG.LineJoin.Round) { + p1.flags |= PtFlagsBevel; + } + } + + p0 = p1++; + } +} + +void nsvg__flattenShapeStroke (NSVGrasterizer r, const(NSVG.Shape)* shape, float scale) { + int i, j, closed; + const(NSVG.Path)* path; + const(NSVGpoint)* p0, p1; + float miterLimit = shape.miterLimit; + int lineJoin = shape.strokeLineJoin; + int lineCap = shape.strokeLineCap; + float lineWidth = shape.strokeWidth*scale; + + for (path = shape.paths; path !is null; path = path.next) { + // Flatten path + r.npoints = 0; + if (!path.empty) { + // first point + { + float x0, y0; + path.startPoint(&x0, &y0); + nsvg__addPathPoint(r, x0*scale, y0*scale, PtFlagsCorner); + } + // cubic beziers + path.asCubics(delegate (const(float)[] cubic) { + assert(cubic.length >= 8); + nsvg__flattenCubicBez(r, + cubic.ptr[0]*scale, cubic.ptr[1]*scale, + cubic.ptr[2]*scale, cubic.ptr[3]*scale, + cubic.ptr[4]*scale, cubic.ptr[5]*scale, + cubic.ptr[6]*scale, cubic.ptr[7]*scale, + 0, PtFlagsCorner); + }); + } + /+ + nsvg__addPathPoint(r, path.pts[0]*scale, path.pts[1]*scale, PtFlagsCorner); + for (i = 0; i < path.npts-1; i += 3) { + const(float)* p = &path.pts[i*2]; + nsvg__flattenCubicBez(r, p[0]*scale, p[1]*scale, p[2]*scale, p[3]*scale, p[4]*scale, p[5]*scale, p[6]*scale, p[7]*scale, 0, PtFlagsCorner); + } + +/ + if (r.npoints < 2) continue; + + closed = path.closed; + + // If the first and last points are the same, remove the last, mark as closed path. + p0 = &r.points[r.npoints-1]; + p1 = &r.points[0]; + if (nsvg__ptEquals(p0.x, p0.y, p1.x, p1.y, r.distTol)) { + r.npoints--; + p0 = &r.points[r.npoints-1]; + closed = 1; + } + + if (shape.strokeDashCount > 0) { + int idash = 0, dashState = 1; + float totalDist = 0, dashLen, allDashLen, dashOffset; + NSVGpoint cur; + + if (closed) nsvg__appendPathPoint(r, r.points[0]); + + // Duplicate points . points2. + nsvg__duplicatePoints(r); + + r.npoints = 0; + cur = r.points2[0]; + nsvg__appendPathPoint(r, cur); + + // Figure out dash offset. + allDashLen = 0; + for (j = 0; j < shape.strokeDashCount; j++) allDashLen += shape.strokeDashArray[j]; + if (shape.strokeDashCount&1) allDashLen *= 2.0f; + // Find location inside pattern + dashOffset = fmodf(shape.strokeDashOffset, allDashLen); + if (dashOffset < 0.0f) dashOffset += allDashLen; + + while (dashOffset > shape.strokeDashArray[idash]) { + dashOffset -= shape.strokeDashArray[idash]; + idash = (idash+1)%shape.strokeDashCount; + } + dashLen = (shape.strokeDashArray[idash]-dashOffset)*scale; + + for (j = 1; j < r.npoints2; ) { + float dx = r.points2[j].x-cur.x; + float dy = r.points2[j].y-cur.y; + float dist = sqrtf(dx*dx+dy*dy); + + if (totalDist+dist > dashLen) { + // Calculate intermediate point + float d = (dashLen-totalDist)/dist; + float x = cur.x+dx*d; + float y = cur.y+dy*d; + nsvg__addPathPoint(r, x, y, PtFlagsCorner); + + // Stroke + if (r.npoints > 1 && dashState) { + nsvg__prepareStroke(r, miterLimit, lineJoin); + nsvg__expandStroke(r, r.points, r.npoints, 0, lineJoin, lineCap, lineWidth); + } + // Advance dash pattern + dashState = !dashState; + idash = (idash+1)%shape.strokeDashCount; + dashLen = shape.strokeDashArray[idash]*scale; + // Restart + cur.x = x; + cur.y = y; + cur.flags = PtFlagsCorner; + totalDist = 0.0f; + r.npoints = 0; + nsvg__appendPathPoint(r, cur); + } else { + totalDist += dist; + cur = r.points2[j]; + nsvg__appendPathPoint(r, cur); + j++; + } + } + // Stroke any leftover path + if (r.npoints > 1 && dashState) nsvg__expandStroke(r, r.points, r.npoints, 0, lineJoin, lineCap, lineWidth); + } else { + nsvg__prepareStroke(r, miterLimit, lineJoin); + nsvg__expandStroke(r, r.points, r.npoints, closed, lineJoin, lineCap, lineWidth); + } + } +} + +extern(C) int nsvg__cmpEdge (in void *p, in void *q) nothrow @trusted @nogc { + NSVGedge* a = cast(NSVGedge*)p; + NSVGedge* b = cast(NSVGedge*)q; + if (a.y0 < b.y0) return -1; + if (a.y0 > b.y0) return 1; + return 0; +} + + +static NSVGactiveEdge* nsvg__addActive (NSVGrasterizer r, const(NSVGedge)* e, float startPoint) { + NSVGactiveEdge* z; + + if (r.freelist !is null) { + // Restore from freelist. + z = r.freelist; + r.freelist = z.next; + } else { + // Alloc new edge. + z = cast(NSVGactiveEdge*)nsvg__alloc(r, NSVGactiveEdge.sizeof); + if (z is null) return null; + } + + immutable float dxdy = (e.x1-e.x0)/(e.y1-e.y0); + //STBTT_assert(e.y0 <= start_point); + // round dx down to avoid going too far + if (dxdy < 0) + z.dx = cast(int)(-floorf(NSVG__FIX*-dxdy)); + else + z.dx = cast(int)floorf(NSVG__FIX*dxdy); + z.x = cast(int)floorf(NSVG__FIX*(e.x0+dxdy*(startPoint-e.y0))); + //z.x -= off_x*FIX; + z.ey = e.y1; + z.next = null; + z.dir = e.dir; + + return z; +} + +void nsvg__freeActive (NSVGrasterizer r, NSVGactiveEdge* z) { + z.next = r.freelist; + r.freelist = z; +} + +void nsvg__fillScanline (ubyte* scanline, int len, int x0, int x1, int maxWeight, int* xmin, int* xmax) { + int i = x0>>NSVG__FIXSHIFT; + int j = x1>>NSVG__FIXSHIFT; + if (i < *xmin) *xmin = i; + if (j > *xmax) *xmax = j; + if (i < len && j >= 0) { + if (i == j) { + // x0, x1 are the same pixel, so compute combined coverage + scanline[i] += cast(ubyte)((x1-x0)*maxWeight>>NSVG__FIXSHIFT); + } else { + if (i >= 0) // add antialiasing for x0 + scanline[i] += cast(ubyte)(((NSVG__FIX-(x0&NSVG__FIXMASK))*maxWeight)>>NSVG__FIXSHIFT); + else + i = -1; // clip + + if (j < len) // add antialiasing for x1 + scanline[j] += cast(ubyte)(((x1&NSVG__FIXMASK)*maxWeight)>>NSVG__FIXSHIFT); + else + j = len; // clip + + for (++i; i < j; ++i) // fill pixels between x0 and x1 + scanline[i] += cast(ubyte)maxWeight; + } + } +} + +// note: this routine clips fills that extend off the edges... ideally this +// wouldn't happen, but it could happen if the truetype glyph bounding boxes +// are wrong, or if the user supplies a too-small bitmap +void nsvg__fillActiveEdges (ubyte* scanline, int len, const(NSVGactiveEdge)* e, int maxWeight, int* xmin, int* xmax, char fillRule) { + // non-zero winding fill + int x0 = 0, w = 0; + if (fillRule == NSVG.FillRule.NonZero) { + // Non-zero + while (e !is null) { + if (w == 0) { + // if we're currently at zero, we need to record the edge start point + x0 = e.x; w += e.dir; + } else { + int x1 = e.x; w += e.dir; + // if we went to zero, we need to draw + if (w == 0) nsvg__fillScanline(scanline, len, x0, x1, maxWeight, xmin, xmax); + } + e = e.next; + } + } else if (fillRule == NSVG.FillRule.EvenOdd) { + // Even-odd + while (e !is null) { + if (w == 0) { + // if we're currently at zero, we need to record the edge start point + x0 = e.x; w = 1; + } else { + int x1 = e.x; w = 0; + nsvg__fillScanline(scanline, len, x0, x1, maxWeight, xmin, xmax); + } + e = e.next; + } + } +} + +float nsvg__clampf() (float a, float mn, float mx) { pragma(inline, true); return (a < mn ? mn : (a > mx ? mx : a)); } + +uint nsvg__RGBA() (ubyte r, ubyte g, ubyte b, ubyte a) { pragma(inline, true); return (r)|(g<<8)|(b<<16)|(a<<24); } + +uint nsvg__lerpRGBA (uint c0, uint c1, float u) { + int iu = cast(int)(nsvg__clampf(u, 0.0f, 1.0f)*256.0f); + int r = (((c0)&0xff)*(256-iu)+(((c1)&0xff)*iu))>>8; + int g = (((c0>>8)&0xff)*(256-iu)+(((c1>>8)&0xff)*iu))>>8; + int b = (((c0>>16)&0xff)*(256-iu)+(((c1>>16)&0xff)*iu))>>8; + int a = (((c0>>24)&0xff)*(256-iu)+(((c1>>24)&0xff)*iu))>>8; + return nsvg__RGBA(cast(ubyte)r, cast(ubyte)g, cast(ubyte)b, cast(ubyte)a); +} + +uint nsvg__applyOpacity (uint c, float u) { + int iu = cast(int)(nsvg__clampf(u, 0.0f, 1.0f)*256.0f); + int r = (c)&0xff; + int g = (c>>8)&0xff; + int b = (c>>16)&0xff; + int a = (((c>>24)&0xff)*iu)>>8; + return nsvg__RGBA(cast(ubyte)r, cast(ubyte)g, cast(ubyte)b, cast(ubyte)a); +} + +int nsvg__div255() (int x) { pragma(inline, true); return ((x+1)*257)>>16; } + +void nsvg__scanlineSolid (ubyte* dst, int count, ubyte* cover, int x, int y, float tx, float ty, float scale, const(NSVGcachedPaint)* cache) { + if (cache.type == NSVG.PaintType.Color) { + int cr = cache.colors[0]&0xff; + int cg = (cache.colors[0]>>8)&0xff; + int cb = (cache.colors[0]>>16)&0xff; + int ca = (cache.colors[0]>>24)&0xff; + + foreach (int i; 0..count) { + int r, g, b; + int a = nsvg__div255(cast(int)cover[0]*ca); + int ia = 255-a; + // Premultiply + r = nsvg__div255(cr*a); + g = nsvg__div255(cg*a); + b = nsvg__div255(cb*a); + + // Blend over + r += nsvg__div255(ia*cast(int)dst[0]); + g += nsvg__div255(ia*cast(int)dst[1]); + b += nsvg__div255(ia*cast(int)dst[2]); + a += nsvg__div255(ia*cast(int)dst[3]); + + dst[0] = cast(ubyte)r; + dst[1] = cast(ubyte)g; + dst[2] = cast(ubyte)b; + dst[3] = cast(ubyte)a; + + ++cover; + dst += 4; + } + } else if (cache.type == NSVG.PaintType.LinearGradient) { + // TODO: spread modes. + // TODO: plenty of opportunities to optimize. + const(float)* t = cache.xform.ptr; + //int i, cr, cg, cb, ca; + //uint c; + + float fx = (x-tx)/scale; + float fy = (y-ty)/scale; + float dx = 1.0f/scale; + + foreach (int i; 0..count) { + //int r, g, b, a, ia; + float gy = fx*t[1]+fy*t[3]+t[5]; + uint c = cache.colors[cast(int)nsvg__clampf(gy*255.0f, 0, 255.0f)]; + int cr = (c)&0xff; + int cg = (c>>8)&0xff; + int cb = (c>>16)&0xff; + int ca = (c>>24)&0xff; + + int a = nsvg__div255(cast(int)cover[0]*ca); + int ia = 255-a; + + // Premultiply + int r = nsvg__div255(cr*a); + int g = nsvg__div255(cg*a); + int b = nsvg__div255(cb*a); + + // Blend over + r += nsvg__div255(ia*cast(int)dst[0]); + g += nsvg__div255(ia*cast(int)dst[1]); + b += nsvg__div255(ia*cast(int)dst[2]); + a += nsvg__div255(ia*cast(int)dst[3]); + + dst[0] = cast(ubyte)r; + dst[1] = cast(ubyte)g; + dst[2] = cast(ubyte)b; + dst[3] = cast(ubyte)a; + + ++cover; + dst += 4; + fx += dx; + } + } else if (cache.type == NSVG.PaintType.RadialGradient) { + // TODO: spread modes. + // TODO: plenty of opportunities to optimize. + // TODO: focus (fx, fy) + //float fx, fy, dx, gx, gy, gd; + const(float)* t = cache.xform.ptr; + //int i, cr, cg, cb, ca; + //uint c; + + float fx = (x-tx)/scale; + float fy = (y-ty)/scale; + float dx = 1.0f/scale; + + foreach (int i; 0..count) { + //int r, g, b, a, ia; + float gx = fx*t[0]+fy*t[2]+t[4]; + float gy = fx*t[1]+fy*t[3]+t[5]; + float gd = sqrtf(gx*gx+gy*gy); + uint c = cache.colors[cast(int)nsvg__clampf(gd*255.0f, 0, 255.0f)]; + int cr = (c)&0xff; + int cg = (c>>8)&0xff; + int cb = (c>>16)&0xff; + int ca = (c>>24)&0xff; + + int a = nsvg__div255(cast(int)cover[0]*ca); + int ia = 255-a; + + // Premultiply + int r = nsvg__div255(cr*a); + int g = nsvg__div255(cg*a); + int b = nsvg__div255(cb*a); + + // Blend over + r += nsvg__div255(ia*cast(int)dst[0]); + g += nsvg__div255(ia*cast(int)dst[1]); + b += nsvg__div255(ia*cast(int)dst[2]); + a += nsvg__div255(ia*cast(int)dst[3]); + + dst[0] = cast(ubyte)r; + dst[1] = cast(ubyte)g; + dst[2] = cast(ubyte)b; + dst[3] = cast(ubyte)a; + + ++cover; + dst += 4; + fx += dx; + } + } +} + +void nsvg__rasterizeSortedEdges (NSVGrasterizer r, float tx, float ty, float scale, const(NSVGcachedPaint)* cache, char fillRule) { + NSVGactiveEdge* active = null; + int s; + int e = 0; + int maxWeight = (255/NSVG__SUBSAMPLES); // weight per vertical scanline + int xmin, xmax; + + foreach (int y; 0..r.height) { + import core.stdc.string : memset; + memset(r.scanline, 0, r.width); + xmin = r.width; + xmax = 0; + for (s = 0; s < NSVG__SUBSAMPLES; ++s) { + // find center of pixel for this scanline + float scany = y*NSVG__SUBSAMPLES+s+0.5f; + NSVGactiveEdge** step = &active; + + // update all active edges; + // remove all active edges that terminate before the center of this scanline + while (*step) { + NSVGactiveEdge* z = *step; + if (z.ey <= scany) { + *step = z.next; // delete from list + //NSVG__assert(z.valid); + nsvg__freeActive(r, z); + } else { + z.x += z.dx; // advance to position for current scanline + step = &((*step).next); // advance through list + } + } + + // resort the list if needed + for (;;) { + int changed = 0; + step = &active; + while (*step && (*step).next) { + if ((*step).x > (*step).next.x) { + NSVGactiveEdge* t = *step; + NSVGactiveEdge* q = t.next; + t.next = q.next; + q.next = t; + *step = q; + changed = 1; + } + step = &(*step).next; + } + if (!changed) break; + } + + // insert all edges that start before the center of this scanline -- omit ones that also end on this scanline + while (e < r.nedges && r.edges[e].y0 <= scany) { + if (r.edges[e].y1 > scany) { + NSVGactiveEdge* z = nsvg__addActive(r, &r.edges[e], scany); + if (z is null) break; + // find insertion point + if (active is null) { + active = z; + } else if (z.x < active.x) { + // insert at front + z.next = active; + active = z; + } else { + // find thing to insert AFTER + NSVGactiveEdge* p = active; + while (p.next && p.next.x < z.x) + p = p.next; + // at this point, p.next.x is NOT < z.x + z.next = p.next; + p.next = z; + } + } + e++; + } + + // now process all active edges in non-zero fashion + if (active !is null) + nsvg__fillActiveEdges(r.scanline, r.width, active, maxWeight, &xmin, &xmax, fillRule); + } + // Blit + if (xmin < 0) xmin = 0; + if (xmax > r.width-1) xmax = r.width-1; + if (xmin <= xmax) { + nsvg__scanlineSolid(&r.bitmap[y*r.stride]+xmin*4, xmax-xmin+1, &r.scanline[xmin], xmin, y, tx, ty, scale, cache); + } + } + +} + +void nsvg__unpremultiplyAlpha (ubyte* image, int w, int h, int stride) { + // Unpremultiply + foreach (int y; 0..h) { + ubyte *row = &image[y*stride]; + foreach (int x; 0..w) { + int r = row[0], g = row[1], b = row[2], a = row[3]; + if (a != 0) { + row[0] = cast(ubyte)(r*255/a); + row[1] = cast(ubyte)(g*255/a); + row[2] = cast(ubyte)(b*255/a); + } + row += 4; + } + } + + // Defringe + foreach (int y; 0..h) { + ubyte *row = &image[y*stride]; + foreach (int x; 0..w) { + int r = 0, g = 0, b = 0, a = row[3], n = 0; + if (a == 0) { + if (x-1 > 0 && row[-1] != 0) { + r += row[-4]; + g += row[-3]; + b += row[-2]; + n++; + } + if (x+1 < w && row[7] != 0) { + r += row[4]; + g += row[5]; + b += row[6]; + n++; + } + if (y-1 > 0 && row[-stride+3] != 0) { + r += row[-stride]; + g += row[-stride+1]; + b += row[-stride+2]; + n++; + } + if (y+1 < h && row[stride+3] != 0) { + r += row[stride]; + g += row[stride+1]; + b += row[stride+2]; + n++; + } + if (n > 0) { + row[0] = cast(ubyte)(r/n); + row[1] = cast(ubyte)(g/n); + row[2] = cast(ubyte)(b/n); + } + } + row += 4; + } + } +} + + +void nsvg__initPaint (NSVGcachedPaint* cache, const(NSVG.Paint)* paint, float opacity) { + const(NSVG.Gradient)* grad; + + cache.type = paint.type; + + if (paint.type == NSVG.PaintType.Color) { + cache.colors[0] = nsvg__applyOpacity(paint.color, opacity); + return; + } + + grad = paint.gradient; + + cache.spread = grad.spread; + //memcpy(cache.xform.ptr, grad.xform.ptr, float.sizeof*6); + cache.xform[0..6] = grad.xform[0..6]; + + if (grad.nstops == 0) { + //for (i = 0; i < 256; i++) cache.colors[i] = 0; + cache.colors[0..256] = 0; + } if (grad.nstops == 1) { + foreach (int i; 0..256) cache.colors[i] = nsvg__applyOpacity(grad.stops.ptr[i].color, opacity); + } else { + uint cb = 0; + //float ua, ub, du, u; + int ia, ib, count; + + uint ca = nsvg__applyOpacity(grad.stops.ptr[0].color, opacity); + float ua = nsvg__clampf(grad.stops.ptr[0].offset, 0, 1); + float ub = nsvg__clampf(grad.stops.ptr[grad.nstops-1].offset, ua, 1); + ia = cast(int)(ua*255.0f); + ib = cast(int)(ub*255.0f); + //for (i = 0; i < ia; i++) cache.colors[i] = ca; + cache.colors[0..ia] = ca; + + foreach (int i; 0..grad.nstops-1) { + ca = nsvg__applyOpacity(grad.stops.ptr[i].color, opacity); + cb = nsvg__applyOpacity(grad.stops.ptr[i+1].color, opacity); + ua = nsvg__clampf(grad.stops.ptr[i].offset, 0, 1); + ub = nsvg__clampf(grad.stops.ptr[i+1].offset, 0, 1); + ia = cast(int)(ua*255.0f); + ib = cast(int)(ub*255.0f); + count = ib-ia; + if (count <= 0) continue; + float u = 0; + immutable float du = 1.0f/cast(float)count; + foreach (int j; 0..count) { + cache.colors[ia+j] = nsvg__lerpRGBA(ca, cb, u); + u += du; + } + } + + //for (i = ib; i < 256; i++) cache.colors[i] = cb; + cache.colors[ib..256] = cb; + } + +} + +extern(C) { + private alias _compare_fp_t = int function (const void*, const void*) nothrow @nogc; + private extern(C) void qsort (scope void* base, size_t nmemb, size_t size, _compare_fp_t compar) nothrow @nogc; +} + +/** + * Rasterizes SVG image, returns RGBA image (non-premultiplied alpha). + * + * Params: + * r = pointer to rasterizer context + * image = pointer to SVG image to rasterize + * tx, ty = image offset (applied after scaling) + * scale = image scale + * dst = pointer to destination image data, 4 bytes per pixel (RGBA) + * w = width of the image to render + * h = height of the image to render + * stride = number of bytes per scaleline in the destination buffer + */ +public void rasterize (NSVGrasterizer r, const(NSVG)* image, float tx, float ty, float scale, ubyte* dst, int w, int h, int stride=-1) { + const(NSVG.Shape)* shape = null; + NSVGedge* e = null; + NSVGcachedPaint cache; + int i; + + if (stride <= 0) stride = w*4; + r.bitmap = dst; + r.width = w; + r.height = h; + r.stride = stride; + + if (w > r.cscanline) { + import core.stdc.stdlib : realloc; + r.cscanline = w; + r.scanline = cast(ubyte*)realloc(r.scanline, w+256); + if (r.scanline is null) assert(0, "nanosvg: out of memory"); + } + + for (i = 0; i < h; i++) { + import core.stdc.string : memset; + memset(&dst[i*stride], 0, w*4); + } + + for (shape = image.shapes; shape !is null; shape = shape.next) { + if (!(shape.flags&NSVG.Visible)) continue; + + if (shape.fill.type != NSVG.PaintType.None) { + //import core.stdc.stdlib : qsort; // not @nogc + + nsvg__resetPool(r); + r.freelist = null; + r.nedges = 0; + + nsvg__flattenShape(r, shape, scale); + + // Scale and translate edges + for (i = 0; i < r.nedges; i++) { + e = &r.edges[i]; + e.x0 = tx+e.x0; + e.y0 = (ty+e.y0)*NSVG__SUBSAMPLES; + e.x1 = tx+e.x1; + e.y1 = (ty+e.y1)*NSVG__SUBSAMPLES; + } + + // Rasterize edges + qsort(r.edges, r.nedges, NSVGedge.sizeof, &nsvg__cmpEdge); + + // now, traverse the scanlines and find the intersections on each scanline, use non-zero rule + nsvg__initPaint(&cache, &shape.fill, shape.opacity); + + nsvg__rasterizeSortedEdges(r, tx, ty, scale, &cache, shape.fillRule); + } + if (shape.stroke.type != NSVG.PaintType.None && (shape.strokeWidth*scale) > 0.01f) { + //import core.stdc.stdlib : qsort; // not @nogc + + nsvg__resetPool(r); + r.freelist = null; + r.nedges = 0; + + nsvg__flattenShapeStroke(r, shape, scale); + + //dumpEdges(r, "edge.svg"); + + // Scale and translate edges + for (i = 0; i < r.nedges; i++) { + e = &r.edges[i]; + e.x0 = tx+e.x0; + e.y0 = (ty+e.y0)*NSVG__SUBSAMPLES; + e.x1 = tx+e.x1; + e.y1 = (ty+e.y1)*NSVG__SUBSAMPLES; + } + + // Rasterize edges + qsort(r.edges, r.nedges, NSVGedge.sizeof, &nsvg__cmpEdge); + + // now, traverse the scanlines and find the intersections on each scanline, use non-zero rule + nsvg__initPaint(&cache, &shape.stroke, shape.opacity); + + nsvg__rasterizeSortedEdges(r, tx, ty, scale, &cache, NSVG.FillRule.NonZero); + } + } + + nsvg__unpremultiplyAlpha(dst, w, h, stride); + + r.bitmap = null; + r.width = 0; + r.height = 0; + r.stride = 0; +} + +} // nothrow @trusted @nogc + + +// ////////////////////////////////////////////////////////////////////////// // +ptrdiff_t xindexOf (const(void)[] hay, const(void)[] need, usize stIdx=0) pure @trusted nothrow @nogc { + if (hay.length <= stIdx || need.length == 0 || need.length > hay.length-stIdx) { + return -1; + } else { + //import iv.strex : memmem; + auto res = memmem(hay.ptr+stIdx, hay.length-stIdx, need.ptr, need.length); + return (res !is null ? cast(ptrdiff_t)(res-hay.ptr) : -1); + } +} + +ptrdiff_t xindexOf (const(void)[] hay, ubyte ch, usize stIdx=0) pure @trusted nothrow @nogc { + return xindexOf(hay, (&ch)[0..1], stIdx); +} + +pure nothrow @trusted @nogc: +version(linux) { + extern(C) inout(void)* memmem (inout(void)* haystack, usize haystacklen, inout(void)* needle, usize needlelen); +} else { + inout(void)* memmem (inout(void)* haystack, usize haystacklen, inout(void)* needle, usize needlelen) { + auto h = cast(const(ubyte)*)haystack; + auto n = cast(const(ubyte)*)needle; + // usize is unsigned + if (needlelen > haystacklen) return null; + foreach (immutable i; 0..haystacklen-needlelen+1) { + import core.stdc.string : memcmp; + if (memcmp(h+i, n, needlelen) == 0) return cast(void*)(h+i); + } + return null; + } +}