From 3fd0540a8392fafbf5df4f5a14530f83c103b888 Mon Sep 17 00:00:00 2001 From: Michel Pollet Date: Sat, 20 Jan 2024 12:05:26 +0000 Subject: [PATCH] Version 1.5 Changes are just too long to list... Signed-off-by: Michel Pollet --- .gitignore | 2 + Makefile | 69 +- README.md | 63 +- contrib/libsofd.c | 2413 - contrib/libsofd.h | 194 - contrib/mii-icon-64.h | 580 + contrib/mii-icon-64.png | Bin 0 -> 2755 bytes contrib/stb_image_write.h | 1724 - disks/{Games1MB.po => GamesWithFirmware.po} | Bin docs/screen_config.png | Bin 0 -> 65970 bytes docs/video_main.webm | Bin 0 -> 761920 bytes fonts/DroidSans.ttf | Bin 190044 -> 0 bytes fonts/FreeLicense.txt | 20 - fonts/PRNumber3.ttf | Bin 501900 -> 0 bytes fonts/PrintChar21.ttf | Bin 501992 -> 0 bytes fonts/ProggyClean.ttf | Bin 41208 -> 0 bytes libmish/Makefile | 7 +- libmish/src/mish_capture_select.c | 33 + libmish/src/mish_cmd.c | 5 +- libmish/src/mish_input.c | 2 +- libmish/src/mish_priv.h | 3 + libmish/src/mish_session.c | 34 +- libmish/src/mish_telnet.c | 3 +- libmish/tests/mish_argv_make_test.c | 5 +- libmish/tests/mish_cmd_test.c | 2 +- libmui/Makefile | 101 + libmui/README.md | 132 + libmui/fonts/Cairo.ttf | Bin 0 -> 66044 bytes libmui/fonts/Charcoal_mui.sfd | 31701 +++++++++ libmui/fonts/Charcoal_mui.ttf | Bin 0 -> 55172 bytes libmui/fonts/Charcoal_mui_italic.sfd | 31334 +++++++++ libmui/fonts/Dingbat.ttf | Bin 0 -> 21452 bytes ...ple2e_original_font_MotterTekturaRetro.otf | Bin 0 -> 23032 bytes libmui/fonts/typicon.ttf | Bin 0 -> 99920 bytes libmui/fonts/typicons.sfd | 53788 ++++++++++++++++ libmui/mui/c2_arrays.c | 46 + libmui/mui/c2_arrays.h | 33 + libmui/mui/c2_geometry.c | 819 + libmui/mui/c2_geometry.h | 148 + libmui/mui/c2_geometry_inline.h | 244 + libmui/mui/c2_geometry_poly.h | 119 + libmui/mui/c_array.h | 161 + libmui/mui/cg.c | 2512 + libmui/mui/cg.h | 314 + libmui/mui/mui.c | 285 + libmui/mui/mui.h | 984 + libmui/mui/mui_alert.c | 90 + libmui/mui/mui_cdef_boxes.c | 188 + libmui/mui/mui_cdef_buttons.c | 267 + libmui/mui/mui_cdef_listbox.c | 307 + libmui/mui/mui_cdef_scrollbar.c | 366 + libmui/mui/mui_controls.c | 318 + libmui/mui/mui_drawable.c | 280 + libmui/mui/mui_font.c | 351 + libmui/mui/mui_menus.c | 916 + libmui/mui/mui_menus_draw.c | 240 + libmui/mui/mui_priv.h | 87 + libmui/mui/mui_stdfile.c | 409 + libmui/mui/mui_utils.c | 40 + libmui/mui/mui_window.c | 503 + libmui/mui/stb_truetype.h | 5077 ++ libmui/mui/stb_ttc.h | 660 + libmui/mui/xft.c | 2874 + libmui/mui/xft.h | 141 + libmui/tests/mui_playground.c | 616 + libmui/tests/mui_plugin.h | 33 + libmui/tests/ui_tests.c | 236 + nuklear/mii_emu.c | 323 - nuklear/mii_fonts.c | 9 - nuklear/mii_nuklear.c | 500 - nuklear/mii_stb_implement.c | 22 - nuklear/nuklear.h | 30034 --------- nuklear/nuklear_xlib_gl3.h | 758 - src/drivers/mii_disk2.c | 18 +- src/drivers/mii_epromcard.c | 86 +- src/drivers/mii_mouse.c | 111 +- src/drivers/mii_noslotclock.c | 4 +- src/drivers/mii_smartport.c | 44 +- src/drivers/mii_ssc.c | 143 + src/drivers/mii_titan_iie.c | 2 +- src/format/dsk.c | 2 +- src/format/mii_dd.c | 7 + src/format/mii_dd.h | 27 +- src/mii.c | 245 +- src/mii.h | 58 +- src/mii_65c02.c | 126 +- src/mii_65c02.h | 62 +- src/mii_65c02_asm.c | 12 +- src/mii_65c02_disasm.c | 4 + src/mii_65c02_ops.h | 2 +- src/mii_analog.c | 37 +- src/mii_analog.h | 7 +- src/mii_argv.c | 4 +- src/mii_bank.c | 44 +- src/mii_bank.h | 3 + {nuklear => src}/mii_mish.c | 54 +- {nuklear => src}/mii_mish_dd.c | 3 +- src/mii_slot.h | 29 +- src/mii_speaker.c | 143 +- src/mii_speaker.h | 13 +- src/mii_sw.h | 41 +- src/mii_types.h | 14 - src/mii_video.c | 77 +- src/mii_video.h | 32 +- src/roms/mii_rom_scc_3410065a.bin | Bin 0 -> 2048 bytes test/mii_cpu_test.c | 66 +- ui_gl/mii_emu_gl.c | 1008 + ui_gl/mii_mui.h | 53 + ui_gl/mii_mui_1mb.c | 336 + ui_gl/mii_mui_2dsk.c | 255 + ui_gl/mii_mui_loadbin.c | 213 + ui_gl/mii_mui_menus.c | 403 + ui_gl/mii_mui_menus.h | 132 + ui_gl/mii_mui_settings.c | 381 + ui_gl/mii_mui_settings.h | 190 + ui_gl/mii_mui_slots.c | 439 + {nuklear => ui_gl}/mii_thread.c | 112 +- {nuklear => ui_gl}/mii_thread.h | 6 +- utils/icon-convert-tcc.c | 80 + 119 files changed, 142127 insertions(+), 36526 deletions(-) delete mode 100644 contrib/libsofd.c delete mode 100644 contrib/libsofd.h create mode 100644 contrib/mii-icon-64.h create mode 100644 contrib/mii-icon-64.png delete mode 100644 contrib/stb_image_write.h rename disks/{Games1MB.po => GamesWithFirmware.po} (100%) create mode 100644 docs/screen_config.png create mode 100644 docs/video_main.webm delete mode 100644 fonts/DroidSans.ttf delete mode 100644 fonts/FreeLicense.txt delete mode 100644 fonts/PRNumber3.ttf delete mode 100644 fonts/PrintChar21.ttf delete mode 100644 fonts/ProggyClean.ttf create mode 100644 libmui/Makefile create mode 100644 libmui/README.md create mode 100644 libmui/fonts/Cairo.ttf create mode 100644 libmui/fonts/Charcoal_mui.sfd create mode 100644 libmui/fonts/Charcoal_mui.ttf create mode 100644 libmui/fonts/Charcoal_mui_italic.sfd create mode 100644 libmui/fonts/Dingbat.ttf create mode 100644 libmui/fonts/apple2e_original_font_MotterTekturaRetro.otf create mode 100644 libmui/fonts/typicon.ttf create mode 100644 libmui/fonts/typicons.sfd create mode 100644 libmui/mui/c2_arrays.c create mode 100644 libmui/mui/c2_arrays.h create mode 100644 libmui/mui/c2_geometry.c create mode 100644 libmui/mui/c2_geometry.h create mode 100644 libmui/mui/c2_geometry_inline.h create mode 100644 libmui/mui/c2_geometry_poly.h create mode 100644 libmui/mui/c_array.h create mode 100644 libmui/mui/cg.c create mode 100644 libmui/mui/cg.h create mode 100644 libmui/mui/mui.c create mode 100644 libmui/mui/mui.h create mode 100644 libmui/mui/mui_alert.c create mode 100644 libmui/mui/mui_cdef_boxes.c create mode 100644 libmui/mui/mui_cdef_buttons.c create mode 100644 libmui/mui/mui_cdef_listbox.c create mode 100644 libmui/mui/mui_cdef_scrollbar.c create mode 100644 libmui/mui/mui_controls.c create mode 100644 libmui/mui/mui_drawable.c create mode 100644 libmui/mui/mui_font.c create mode 100644 libmui/mui/mui_menus.c create mode 100644 libmui/mui/mui_menus_draw.c create mode 100644 libmui/mui/mui_priv.h create mode 100644 libmui/mui/mui_stdfile.c create mode 100644 libmui/mui/mui_utils.c create mode 100644 libmui/mui/mui_window.c create mode 100644 libmui/mui/stb_truetype.h create mode 100644 libmui/mui/stb_ttc.h create mode 100644 libmui/mui/xft.c create mode 100644 libmui/mui/xft.h create mode 100644 libmui/tests/mui_playground.c create mode 100644 libmui/tests/mui_plugin.h create mode 100644 libmui/tests/ui_tests.c delete mode 100644 nuklear/mii_emu.c delete mode 100644 nuklear/mii_fonts.c delete mode 100644 nuklear/mii_nuklear.c delete mode 100644 nuklear/mii_stb_implement.c delete mode 100644 nuklear/nuklear.h delete mode 100644 nuklear/nuklear_xlib_gl3.h create mode 100644 src/drivers/mii_ssc.c rename {nuklear => src}/mii_mish.c (90%) rename {nuklear => src}/mii_mish_dd.c (90%) delete mode 100644 src/mii_types.h create mode 100644 src/roms/mii_rom_scc_3410065a.bin create mode 100644 ui_gl/mii_emu_gl.c create mode 100644 ui_gl/mii_mui.h create mode 100644 ui_gl/mii_mui_1mb.c create mode 100644 ui_gl/mii_mui_2dsk.c create mode 100644 ui_gl/mii_mui_loadbin.c create mode 100644 ui_gl/mii_mui_menus.c create mode 100644 ui_gl/mii_mui_menus.h create mode 100644 ui_gl/mii_mui_settings.c create mode 100644 ui_gl/mii_mui_settings.h create mode 100644 ui_gl/mii_mui_slots.c rename {nuklear => ui_gl}/mii_thread.c (67%) rename {nuklear => ui_gl}/mii_thread.h (89%) create mode 100644 utils/icon-convert-tcc.c diff --git a/.gitignore b/.gitignore index 70d02f0..848adb6 100644 --- a/.gitignore +++ b/.gitignore @@ -4,3 +4,5 @@ build-* compile_commands.json cachegrind.out.* callgrind.out.* +.cache +*.miov diff --git a/Makefile b/Makefile index 5af3dc3..4582b4f 100644 --- a/Makefile +++ b/Makefile @@ -3,15 +3,24 @@ CC = gcc SHELL = /bin/bash # This is where (g)make looks for the source files for implicit rules -VPATH := src src/format src/drivers nuklear contrib +VPATH := src src/format src/drivers contrib +VPATH += ui_gl CPPFLAGS += -Isrc -Isrc/format -Isrc/roms -Isrc/drivers -CPPFLAGS += -Icontrib -Inuklear +CPPFLAGS += -Icontrib CPPFLAGS += -Ilibmish/src -CFLAGS += --std=gnu99 -Wall -Wextra -O2 -g +CPPFLAGS += -Ilibmui/mui + +OPTIMIZE ?= -O2 -march=native +CFLAGS += --std=gnu99 -Wall -Wextra -g +CFLAGS += -fno-omit-frame-pointer +CFLAGS += $(OPTIMIZE) CFLAGS += -Wno-unused-parameter -Wno-unused-function -LDLIBS += -lX11 -lm -lGL -lGLU -LDLIBS += -lpthread -lutil +LDLIBS += -lX11 -lGL -lGLU +LDLIBS += -lpthread -lutil -lm + +VERSION := ${shell git log -1 --date=short --pretty="%h %cd"} +CPPFLAGS += -DMII_VERSION="\"$(VERSION)\"" HAS_ALSA := $(shell pkg-config --exists alsa && echo 1) ifeq ($(HAS_ALSA),1) @@ -26,21 +35,35 @@ BIN := $(O)/bin LIB := $(O)/lib OBJ := $(O)/obj -all : $(BIN)/mii_emu +all : $(BIN)/mii_emu_gl MII_SRC := $(wildcard src/*.c src/format/*.c \ src/drivers/*.c contrib/*.c) -UI_SRC := $(wildcard nuklear/*.c) +UI_SRC := $(wildcard ui_gl/*.c) + SRC := $(MII_SRC) $(UI_SRC) ALL_OBJ := ${patsubst %, ${OBJ}/%, ${notdir ${SRC:.c=.o}}} -$(BIN)/mii_emu : $(ALL_OBJ) -$(BIN)/mii_emu : $(LIB)/libmish.a +CPPFLAGS += ${shell pkg-config --cflags pixman-1} +LDLIBS += ${shell pkg-config --libs pixman-1} -libmish : $(LIB)/libmish.a +$(BIN)/mii_emu_gl : $(ALL_OBJ) | mui mish +$(BIN)/mii_emu_gl : $(LIB)/libmish.a +$(BIN)/mii_emu_gl : $(LIB)/libmui.a + +.PHONY : mish mui +mish : $(LIB)/libmish.a LDLIBS += $(LIB)/libmish.a -$(LIB)/libmish.a : | $(LIB) $(OBJ) $(BIN) - make -j -C libmish O="../" CC="$(CC)" V="$(V)" +$(LIB)/libmish.a : ${wildcard libmish/src/*} | $(LIB) $(OBJ) $(BIN) + mkdir -p $(OBJ)/libmish && \ + make -j -C libmish O="../" CC="$(CC)" V="$(V)" static + +LDLIBS += $(LIB)/libmui.a +mui : $(LIB)/libmui.a +$(LIB)/libmui.a : ${wildcard libmui/mui/*} | $(LIB) $(OBJ) $(BIN) + mkdir -p $(OBJ)/libmui && \ + make -j -C libmui BUILD_DIR="../" CC="$(CC)" \ + V="$(V)" OPTIMIZE="$(OPTIMIZE)" static # Smartport firmware needs the assembler first test/asm/%.bin : test/asm/%.asm | $(BIN)/mii_asm @@ -48,17 +71,16 @@ test/asm/%.bin : test/asm/%.asm | $(BIN)/mii_asm # And it also INCBIN the firmware driver $(OBJ)/mii_smarport.o : test/asm/mii_smartport_driver.bin -$(OBJ)/libsofd.o : CPPFLAGS += -DHAVE_X11 - clean : - rm -rf $(O) + rm -rf $(O); make -C libmui clean; make -C libmish clean # This is for development purpose. This will recompile the project # everytime a file is modified. watch : while true; do \ clear; $(MAKE) -j all tests; \ - inotifywait -qre close_write src src/format nuklear test; \ + inotifywait -qre close_write src src/format ui_gl test \ + libmui libmui/mui; \ done tests : $(BIN)/mii_test $(BIN)/mii_cpu_test $(BIN)/mii_asm @@ -82,8 +104,8 @@ else Q := @ endif -$(OBJ)/%.o : %.c | $(OBJ) - @echo " CC $<" +$(OBJ)/%.o : %.c | $(OBJ) + @echo " CC " ${filter -O%, $(CPPFLAGS) $(CFLAGS)} " $<" $(Q)$(CC) -MMD $(CPPFLAGS) $(CFLAGS) -c -o $@ $< $(BIN)/% : | $(BIN) @@ -95,12 +117,19 @@ $(OBJ) $(BIN) $(LIB) : # Generates the necessary file to help clangd index the files properly. # This currently has to be done manually, but helps a lot if you use 'kate' -# editor or anthing else that is compatible with the LSP protocol +# editor or anthing else that is compatible with the LSP protocol (vscode too) compile_commands.json: lsp lsp: { $$(which gmake) CC=gcc V=1 --always-make --dry-run all tests; \ - $$(which gmake) CC=gcc V=1 --always-make --dry-run -C libmish ; } | \ + $$(which gmake) CC=gcc V=1 --always-make --dry-run -C libmish ; \ + $$(which gmake) CC=gcc V=1 --always-make --dry-run -C libmui ; } | \ sh utils/clangd_gen.sh >compile_commands.json -include $(O)/*.d -include $(O)/obj/*.d + + +install: + mkdir -p $(DESTDIR)/bin + mkdir -p $(DESTDIR)/share/games/mii/ + cp $(BIN)/mii_emu_gl $(DESTDIR)/bin/ diff --git a/README.md b/README.md index 3059e4c..81a866b 100644 --- a/README.md +++ b/README.md @@ -1,3 +1,21 @@ +# MII Version Changelog +## 1.5 + * BIG update, loads of changes, fixes, improvements. + * New super UI, using home-made libmui, channeling both GS/OS and MacOS 7.x! + * New emulation fixes, way more accurate. Video redone, audio redone. + * New front-end program using XLib and OpenGL 'low level'. + * New Icon. +## 1.0 + * Fixed a few graphics rendering bugs/color swapped + * Fixed a few Makefile issues involving pathnamed with 'spaces' in them. + * More tweaks to the emulation, added a few cycles here and there. + ## 0.9 + * Added a 'debugger' shell, accessible via telnet. + * Added a mini-assembler, used to compile the drivers and the CPU unit tests. + * Added a 'Titan Accelerator IIe' simulation, to turn on/off fast mode. +## 0.5 + * Initial release + # MII Apple //e Emulator I know there are many out there, but none of them were ticking my fancy, so I decide to write my own. To start with it was "How hard can it be really?" then it snowballed as more and more things were fixed & added. It's been shelved for a while because well, it lacked documentation, headers, licence and stuff, so I spent some time cleaning it up for release. @@ -5,8 +23,8 @@ I know there are many out there, but none of them were ticking my fancy, so I de One primary reason for this project was that linapple (or -pie) codebase is really horrible. It dates back from 2000's or before, with loads of Windows crud leftover, some SDL crud added, the audio just doesn't really work, and overall if you want to hack around the codebase, it's pretty dreadful. -![Monochrome Double-Hi res](docs/screen_main.png) -*Double hires in monochrome* +![Quick how to load and boot](docs/video_main.webm) +*Quick Howto Load & Boot* I wanted something: @@ -32,12 +50,14 @@ I wanted something: * Adds a small 'attack' filter when playing back to soften the often annoying 'click' of typical audio effects from the apple II. * Mouse Card -- mouse isn't captured like in some other emulators. * No Slot Clock - * Joystick (in a limited way...) + * Joystick Support * Smartport DMA 'hard drive' card * "Titan Accelerator //e" simulation, to turn on/off fast mode. * Terence's J Boldt [1MB ROM card](https://github.com/tjboldt/ProDOS-ROM-Drive), also because I own a couple! * Floppy Drive [more on that later] - * No dependencies (X11) OpenGL rendering, using Nuklear backend for UI + * No dependencies (X11) OpenGL rendering + * Built-in debugger (using telnet access) + * Super cool looking UI! ![Phosphorescent Green](docs/screen_green.png) *Good old green monitor style. Theres Amber too.* @@ -48,13 +68,24 @@ I wanted something: * libgl-dev * libglu-dev * libx11-dev + * libpixman-1-dev * Many of them will probably be installed already. * For more details on development, see [Compiling](docs/Compiling.md) * Then just type `make` and it should compile. - * To run it, just type `build-x86_64-linux-gnu/bin/bin/mii_emu` and it should start. - * `mii_emu --help` will display: + * To run it, just type `build-x86_64-linux-gnu/bin/mii_emu_gl` and it should start. - Usage: ./build-x86_64-linux-gnu/bin/mii_emu [options] +## Command line options +If you run it with no options, and there are no config file, it will present +you with a dialog to select the ROMs and the drives. + +![Config dialog](docs/screen_config.png) +*Main slot configuration dialog* + +You can also use the command line to specify them, and other options. + + * `mii_emu_gl --help` will display: + + Usage: ./build-x86_64-linux-gnu/bin/mii_emu_gl [options] Options: -h, --help This help -v, --verbose Verbose output @@ -89,16 +120,17 @@ I wanted something: disk2 - Apple Disk ][ ## Key Bindings -There are just a few keys that are mapped for anything useful. +There are just a few keys that are mapped for anything useful. List is not exausive, but here are the main ones: * **Control-F12** is Control-Reset on the IIe. (**Shift-Control-F12** is **Open Apple-Reset**) * **'Super'** left and **'Super'** right are **Open** and **Close Apple** keys. These keys are mapped to the left and right 'Windows' keys on a PC keyboard, and they might want to open the start menu (I know it's the case with Cinnamon), so you might want to disable that. * **F5** sets the CPU speed to 1MHz * **F6** to 4MHz. * These keys control the built-in debugger: - * **Control-F11** Stops the emulator; see the command prompt/telnet for how to proceed, dump state, disassembly etc. - * **Control-F10** 'steps' the emulator, ie one instruction at a time. - * **Control-F9** is 'continue' -- resumes the emulator. + * **Control-F4** Stops the emulator; see the command prompt/telnet for how to proceed, dump state, disassembly etc. + * **Control-F5** is 'continue' -- resumes the emulator. + * **Control-F6** 'steps' the emulator, ie one instruction at a time. + * **Control-F7** 'next' instruction, ie step over a JSR instruction. ![Telnet into mii_emu](docs/screen_mish.png) @@ -124,20 +156,15 @@ There are just a few keys that are mapped for anything useful. + This allows you to make sure your disk images aren't corrupted when 'hard rebooting' the emulator, if you are in the process of testing/developing a program for example. ## What it cannot do - * MouseCalc crashes (VBL IRQ, or a mouse mode I don't support yet) * A2Desktop PT3 player doesn't see keypresses. - * Sometimes the emulator goes in 'slow mode', ie 0.2MHz. Likely the frame scheduler playing up. * Thats' about it really, all the other things I tried work - * Joystick support is a bit limited, no 'mapping' I used a (USB) 8bitdo NES30 Pro, and it works, but it's not perfect. But, I can play choplifter with it, so it's good enough for now... + * Joystick support is a bit limited, no 'mapping' I used a (USB) 8bitdo NES30 Pro, and it works, but it's not perfect. But, I can play choplifter with it, so it's good enough for now... *NOTE* Soon will have it's own config dialog to do mapping. ## What it could do with - * Not sure about keeping Nuklear, it does a lot bit it's hard work customizing anything - * Add a memory extension 'card' -- not sure why, but hey, why not. - * Joystick support. As soon as I find a USB joystick that vaguely looks retro, I'll get one. * The main window is 1280x720 on purpose, so it could do Full Screen. * Port it to Raspbery Pi. I don't expect compiling issues, just video issues with GLes * Make a tool to 'flatten' overlay files back into the primary image. - * Some sort of UI to select/eject disks. + * Make a UI for the debugger, instead of telnet. ![Total Replay](docs/screen_total.png) diff --git a/contrib/libsofd.c b/contrib/libsofd.c deleted file mode 100644 index f475b67..0000000 --- a/contrib/libsofd.c +++ /dev/null @@ -1,2413 +0,0 @@ -/* libSOFD - Simple Open File Dialog [for X11 without toolkit] - * - * Copyright (C) 2014 Robin Gareus - * - * 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. - */ - -/* Test and example: - * gcc -Wall -D SOFD_TEST -g -o sofd libsofd.c -lX11 - * - * public API documentation and example code at the bottom of this file - * - * This small lib may one day include openGL rendering and - * wayland window support, but not today. Today we celebrate - * 30 years of X11. - */ - -#ifdef SOFD_TEST -#define HAVE_X11 -#include "libsofd.h" -#endif - -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include - -// shared 'recently used' implementation -// sadly, xbel does not qualify as simple. -// hence we use a simple format alike the -// gtk-bookmark list (one file per line) - -#define MAX_RECENT_ENTRIES 24 -#define MAX_RECENT_AGE (15552000) // 180 days (in sec) - -typedef struct { - char path[1024]; - time_t atime; -} FibRecentFile; - -static FibRecentFile *_recentlist = NULL; -static unsigned int _recentcnt = 0; -static uint8_t _recentlock = 0; - -static int fib_isxdigit (const char x) { - if ( - (x >= '0' && x <= '9') - || - (x >= 'a' && x <= 'f') - || - (x >= 'A' && x <= 'F') - ) return 1; - return 0; -} - -static void decode_3986 (char *str) { - int len = strlen (str); - int idx = 0; - while (idx + 2 < len) { - char *in = &str[idx]; - if (('%' == *in) && fib_isxdigit (in[1]) && fib_isxdigit (in[2])) { - char hexstr[3]; - hexstr[0] = in[1]; - hexstr[1] = in[2]; - hexstr[2] = 0; - long hex = strtol (hexstr, NULL, 16); - *in = hex; - memmove (&str[idx+1], &str[idx + 3], len - idx - 2); - len -= 2; - } - ++idx; - } -} - -static char *encode_3986 (const char *str) { - size_t alloc, newlen; - char *ns = NULL; - unsigned char in; - size_t i = 0; - size_t length; - - if (!str) return strdup (""); - - alloc = strlen (str) + 1; - newlen = alloc; - - ns = (char*) malloc (alloc); - - length = alloc; - while (--length) { - in = *str; - - switch (in) { - case '0': case '1': case '2': case '3': case '4': - case '5': case '6': case '7': case '8': case '9': - case 'a': case 'b': case 'c': case 'd': case 'e': - case 'f': case 'g': case 'h': case 'i': case 'j': - case 'k': case 'l': case 'm': case 'n': case 'o': - case 'p': case 'q': case 'r': case 's': case 't': - case 'u': case 'v': case 'w': case 'x': case 'y': case 'z': - case 'A': case 'B': case 'C': case 'D': case 'E': - case 'F': case 'G': case 'H': case 'I': case 'J': - case 'K': case 'L': case 'M': case 'N': case 'O': - case 'P': case 'Q': case 'R': case 'S': case 'T': - case 'U': case 'V': case 'W': case 'X': case 'Y': case 'Z': - case '_': case '~': case '.': case '-': - case '/': case ',': // XXX not in RFC3986 - ns[i++] = in; - break; - default: - newlen += 2; /* this'll become a %XX */ - if (newlen > alloc) { - alloc *= 2; - ns = (char*) realloc (ns, alloc); - } - snprintf (&ns[i], 4, "%%%02X", in); - i += 3; - break; - } - ++str; - } - ns[i] = 0; - return ns; -} - -void x_fib_free_recent () { - free (_recentlist); - _recentlist = NULL; - _recentcnt = 0; -} - -static int cmp_recent (const void *p1, const void *p2) { - FibRecentFile *a = (FibRecentFile*) p1; - FibRecentFile *b = (FibRecentFile*) p2; - if (a->atime == b->atime) return 0; - return a->atime < b->atime; -} - -int x_fib_add_recent (const char *path, time_t atime) { - unsigned int i; - struct stat fs; - if (_recentlock) { return -1; } - if (access (path, R_OK)) { - return -1; - } - if (stat (path, &fs)) { - return -1; - } - if (!S_ISREG (fs.st_mode)) { - return -1; - } - if (atime == 0) atime = time (NULL); - if (MAX_RECENT_AGE > 0 && atime + MAX_RECENT_AGE < time (NULL)) { - return -1; - } - - for (i = 0; i < _recentcnt; ++i) { - if (!strcmp (_recentlist[i].path, path)) { - if (_recentlist[i].atime < atime) { - _recentlist[i].atime = atime; - } - qsort (_recentlist, _recentcnt, sizeof(FibRecentFile), cmp_recent); - return _recentcnt; - } - } - _recentlist = (FibRecentFile*)realloc (_recentlist, (_recentcnt + 1) * sizeof(FibRecentFile)); - _recentlist[_recentcnt].atime = atime; - strcpy (_recentlist[_recentcnt].path, path); - qsort (_recentlist, _recentcnt + 1, sizeof(FibRecentFile), cmp_recent); - - if (_recentcnt >= MAX_RECENT_ENTRIES) { - return (_recentcnt); - } - return (++_recentcnt); -} - -#ifdef PATHSEP -#undef PATHSEP -#endif - -#ifdef PLATFORM_WINDOWS -#define DIRSEP '\\' -#else -#define DIRSEP '/' -#endif - -static void mkpath(const char *dir) { - char tmp[1024]; - char *p; - size_t len; - - snprintf (tmp, sizeof(tmp), "%s", dir); - len = strlen(tmp); - if (tmp[len - 1] == '/') - tmp[len - 1] = 0; - for (p = tmp + 1; *p; ++p) - if(*p == DIRSEP) { - *p = 0; -#ifdef PLATFORM_WINDOWS - mkdir(tmp); -#else - mkdir(tmp, 0755); -#endif - *p = DIRSEP; - } -#ifdef PLATFORM_WINDOWS - mkdir(tmp); -#else - mkdir(tmp, 0755); -#endif -} - -int x_fib_save_recent (const char *fn) { - if (_recentlock) { return -1; } - if (!fn) { return -1; } - if (_recentcnt < 1 || !_recentlist) { return -1; } - unsigned int i; - char *dn = strdup (fn); - mkpath (dirname (dn)); - free (dn); - - FILE *rf = fopen (fn, "w"); - if (!rf) return -1; - - qsort (_recentlist, _recentcnt, sizeof(FibRecentFile), cmp_recent); - for (i = 0; i < _recentcnt; ++i) { - char *n = encode_3986 (_recentlist[i].path); - fprintf (rf, "%s %lu\n", n, _recentlist[i].atime); - free (n); - } - fclose (rf); - return 0; -} - -int x_fib_load_recent (const char *fn) { - char tmp[1024]; - if (_recentlock) { return -1; } - if (!fn) { return -1; } - x_fib_free_recent (); - if (access (fn, R_OK)) { - return -1; - } - FILE *rf = fopen (fn, "r"); - if (!rf) return -1; - while (fgets (tmp, sizeof(tmp), rf) - && strlen (tmp) > 1 - && strlen (tmp) < sizeof(tmp)) - { - char *s; - tmp[strlen (tmp) - 1] = '\0'; // strip newline - if (!(s = strchr (tmp, ' '))) { // find name <> atime sep - continue; - } - *s = '\0'; - time_t t = atol (++s); - decode_3986 (tmp); - x_fib_add_recent (tmp, t); - } - fclose (rf); - return 0; -} - -unsigned int x_fib_recent_count () { - return _recentcnt; -} - -const char *x_fib_recent_at (unsigned int i) { - if (i >= _recentcnt) - return NULL; - return _recentlist[i].path; -} - -#ifdef PLATFORM_WINDOWS -#define PATHSEP "\\" -#else -#define PATHSEP "/" -#endif - -const char *x_fib_recent_file(const char *appname) { - static char recent_file[1024]; - assert(!strchr(appname, '/')); - const char *xdg = getenv("XDG_DATA_HOME"); - if (xdg && (strlen(xdg) + strlen(appname) + 10) < sizeof(recent_file)) { - sprintf(recent_file, "%s" PATHSEP "%s" PATHSEP "recent", xdg, appname); - return recent_file; - } -#ifdef PLATFORM_WINDOWS - const char * homedrive = getenv("HOMEDRIVE"); - const char * homepath = getenv("HOMEPATH"); - if (homedrive && homepath && (strlen(homedrive) + strlen(homepath) + strlen(appname) + 29) < PATH_MAX) { - sprintf(recent_file, "%s%s" PATHSEP "Application Data" PATHSEP "%s" PATHSEP "recent.txt", homedrive, homepath, appname); - return recent_file; - } -#elif defined PLATFORM_OSX - const char *home = getenv("HOME"); - if (home && (strlen(home) + strlen(appname) + 29) < sizeof(recent_file)) { - sprintf(recent_file, "%s/Library/Preferences/%s/recent", home, appname); - return recent_file; - } -#else - const char *home = getenv("HOME"); - if (home && (strlen(home) + strlen(appname) + 22) < sizeof(recent_file)) { - sprintf(recent_file, "%s/.local/share/%s/recent", home, appname); - return recent_file; - } -#endif - return NULL; -} - -#ifdef HAVE_X11 -#include -#include - -#include -#include -#include -#include -#include - -#ifndef MIN -#define MIN(A,B) ( (A) < (B) ? (A) : (B) ) -#endif - -#ifndef MAX -#define MAX(A,B) ( (A) < (B) ? (B) : (A) ) -#endif - -static Window _fib_win = 0; -static GC _fib_gc = 0; -static XColor _c_gray0, _c_gray1, _c_gray2, _c_gray3, _c_gray4, _c_gray5, _c_gray6; -static Font _fibfont = 0; -static Pixmap _pixbuffer = None; - -static int _fib_width = 100; -static int _fib_height = 100; -static int _btn_w = 0; -static int _btn_span = 0; - -static int _fib_font_height = 0; -static int _fib_dir_indent = 0; -static int _fib_spc_norm = 0; -static int _fib_font_ascent = 0; -static int _fib_font_vsep = 0; -static int _fib_font_size_width = 0; -static int _fib_font_time_width = 0; -static int _fib_place_width = 0; - -static int _scrl_f = 0; -static int _scrl_y0 = -1; -static int _scrl_y1 = -1; -static int _scrl_my = -1; -static int _scrl_mf = -1; -static int _view_p = -1; - -static int _fsel = -1; -static int _hov_b = -1; -static int _hov_f = -1; -static int _hov_p = -1; -static int _hov_h = -1; -static int _hov_l = -1; -static int _hov_s = -1; -static int _sort = 0; -static int _columns = 0; -static int _fib_filter_fn = 1; -static int _fib_hidden_fn = 0; -static int _fib_show_places = 0; - -static uint8_t _fib_mapped = 0; -static uint8_t _fib_resized = 0; -static unsigned long _dblclk = 0; - -static int _status = -2; -static char _rv_open[1024] = ""; - -static char _fib_cfg_custom_places[1024] = ""; -static char _fib_cfg_custom_font[256] = ""; -static char _fib_cfg_title[128] = "xjadeo - Open Video File"; - -typedef struct { - char name[256]; - int x0; - int xw; -} FibPathButton; - -typedef struct { - char name[256]; - char strtime[32]; - char strsize[32]; - int ssizew; - off_t size; - time_t mtime; - uint8_t flags; // 2: selected, 4: isdir 8: recent-entry - FibRecentFile *rfp; -} FibFileEntry; - -typedef struct { - char text[24]; - uint8_t flags; // 2: selected, 4: toggle, 8 disable - int x0; - int tw; - int xw; - void (*callback)(Display*); -} FibButton; - -typedef struct { - char name[256]; - char path[1024]; - uint8_t flags; // 1: hover, 2: selected, 4:add sep -} FibPlace; - -static char _cur_path[1024] = ""; -static FibFileEntry *_dirlist = NULL; -static FibPathButton *_pathbtn = NULL; -static FibPlace *_placelist = NULL; -static int _dircount = 0; -static int _pathparts = 0; -static int _placecnt = 0; - -static FibButton _btn_ok; -static FibButton _btn_cancel; -static FibButton _btn_filter; -static FibButton _btn_places; -static FibButton _btn_hidden; -static FibButton *_btns[] = {&_btn_places, &_btn_filter, &_btn_hidden, &_btn_cancel, &_btn_ok}; - -static int (*_fib_filter_function)(const char *filename); - -/* hardcoded layout */ -#define DSEP 6 // px; horiz space beween elements, also l+r margin for file-list -#define PSEP 4 // px; horiz space beween paths -#define FILECOLUMN (17 * _fib_dir_indent) //px; min width of file-column -#define LISTTOP 2.7 //em; top of the file-browser list -#define LISTBOT 4.75 //em; bottom of the file-browers list -#define BTNBTMMARGIN 0.75 //em; height/margin of the button row -#define BTNPADDING 2 // px - only used for open/cancel buttons -#define SCROLLBARW (3 + (_fib_spc_norm&~1)) //px; - should be SCROLLBARW = (N * 2 + 3) -#define SCROLLBOXH 10 //px; arrow box top+bottom -#define PLACESW _fib_place_width //px; -#define PLACESWMAX (15 *_fib_spc_norm) //px; -#define PATHBTNTOP _fib_font_vsep //px; offset by (_fib_font_ascent); -#define FAREAMRGB 3 //px; base L+R margin -#define FAREAMRGR (FAREAMRGB + 1) //px; right margin of file-area + 1 (line width) -#define FAREAMRGL (_fib_show_places ? PLACESW + FAREAMRGB : FAREAMRGB) //px; left margin of file-area -#define TEXTSEP 4 //px; -#define FAREATEXTL (FAREAMRGL + TEXTSEP) //px; filename text-left FAREAMRGL + TEXTSEP -#define SORTBTNOFF -10 //px; - -#ifndef DBLCLKTME -#define DBLCLKTME 200 //msec; double click time -#endif - -#define DRAW_OUTLINE -#define DOUBLE_BUFFER - -static int query_font_geometry (Display *dpy, GC gc, const char *txt, int *w, int *h, int *a, int *d) { - XCharStruct text_structure; - int font_direction, font_ascent, font_descent; - XFontStruct *fontinfo = XQueryFont (dpy, XGContextFromGC (gc)); - - if (!fontinfo) { return -1; } - XTextExtents (fontinfo, txt, strlen (txt), &font_direction, &font_ascent, &font_descent, &text_structure); - if (w) *w = XTextWidth (fontinfo, txt, strlen (txt)); - if (h) *h = text_structure.ascent + text_structure.descent; - if (a) *a = text_structure.ascent; - if (d) *d = text_structure.descent; - XFreeFontInfo (NULL, fontinfo, 1); - return 0; -} - -static void VDrawRectangle (Display *dpy, Drawable d, GC gc, int x, int y, unsigned int w, unsigned int h) { - const unsigned long blackColor = BlackPixel (dpy, DefaultScreen (dpy)); -#ifdef DRAW_OUTLINE - XSetForeground (dpy, gc, _c_gray5.pixel); - XDrawLine (dpy, d, gc, x + 1, y + h, x + w, y + h); - XDrawLine (dpy, d, gc, x + w, y + 1, x + w, y + h); - - XSetForeground (dpy, gc, blackColor); - XDrawLine (dpy, d, gc, x + 1, y, x + w, y); - XDrawLine (dpy, d, gc, x, y + 1, x, y + h); -#else - XSetForeground (dpy, _fib_gc, blackColor); - XDrawRectangle (dpy, d, gc, x, y, w, h); -#endif -} - -static void fib_expose (Display *dpy, Window realwin) { - int i; - XID win; - const unsigned long whiteColor = WhitePixel (dpy, DefaultScreen (dpy)); - const unsigned long blackColor = BlackPixel (dpy, DefaultScreen (dpy)); - if (!_fib_mapped) return; - - if (_fib_resized -#ifdef DOUBLE_BUFFER - || !_pixbuffer -#endif - ) - { -#ifdef DOUBLE_BUFFER - unsigned int w = 0, h = 0; - if (_pixbuffer != None) { - Window ignored_w; - int ignored_i; - unsigned int ignored_u; - XGetGeometry(dpy, _pixbuffer, &ignored_w, &ignored_i, &ignored_i, &w, &h, &ignored_u, &ignored_u); - if (_fib_width != (int)w || _fib_height != (int)h) { - XFreePixmap (dpy, _pixbuffer); - _pixbuffer = None; - } - } - if (_pixbuffer == None) { - XWindowAttributes wa; - XGetWindowAttributes (dpy, realwin, &wa); - _pixbuffer = XCreatePixmap (dpy, realwin, _fib_width, _fib_height, wa.depth); - } -#endif - if (_pixbuffer != None) { - XSetForeground (dpy, _fib_gc, _c_gray1.pixel); - XFillRectangle (dpy, _pixbuffer, _fib_gc, 0, 0, _fib_width, _fib_height); - } else { - XSetForeground (dpy, _fib_gc, _c_gray1.pixel); - XFillRectangle (dpy, realwin, _fib_gc, 0, 0, _fib_width, _fib_height); - } - _fib_resized = 0; - } - - if (_pixbuffer == None) { - win = realwin; - } else { - win = _pixbuffer; - } - - // Top Row: dirs and up navigation - - int ppw = 0; - int ppx = FAREAMRGB; - - for (i = _pathparts - 1; i >= 0; --i) { - ppw += _pathbtn[i].xw + PSEP; - if (ppw >= _fib_width - PSEP - _pathbtn[0].xw - FAREAMRGB) break; // XXX, first change is from "/" to "<", NOOP - } - ++i; - // border-less "<" parent/up, IFF space is limited - if (i > 0) { - if (0 == _hov_p || (_hov_p > 0 && _hov_p < _pathparts - 1)) { - XSetForeground (dpy, _fib_gc, _c_gray4.pixel); - } else { - XSetForeground (dpy, _fib_gc, blackColor); - } - XDrawString (dpy, win, _fib_gc, ppx, PATHBTNTOP, "<", 1); - ppx += _pathbtn[0].xw + PSEP; - if (i == _pathparts) --i; - } - - _view_p = i; - - while (i < _pathparts) { - if (i == _hov_p) { - XSetForeground (dpy, _fib_gc, _c_gray0.pixel); - } else { - XSetForeground (dpy, _fib_gc, _c_gray2.pixel); - } - XFillRectangle (dpy, win, _fib_gc, - ppx + 1, PATHBTNTOP - _fib_font_ascent, - _pathbtn[i].xw - 1, _fib_font_height); - VDrawRectangle (dpy, win, _fib_gc, - ppx, PATHBTNTOP - _fib_font_ascent, - _pathbtn[i].xw, _fib_font_height); - XDrawString (dpy, win, _fib_gc, ppx + 1 + BTNPADDING, PATHBTNTOP, - _pathbtn[i].name, strlen (_pathbtn[i].name)); - _pathbtn[i].x0 = ppx; // current position - ppx += _pathbtn[i].xw + PSEP; - ++i; - } - - // middle, scroll list of file names - const int ltop = LISTTOP * _fib_font_vsep; - const int llen = (_fib_height - LISTBOT * _fib_font_vsep) / _fib_font_vsep; - const int fsel_height = 4 + llen * _fib_font_vsep; - const int fsel_width = _fib_width - FAREAMRGL - FAREAMRGR - (llen < _dircount ? SCROLLBARW : 0); - const int t_x = FAREATEXTL; - int t_s = FAREATEXTL + fsel_width; - int t_t = FAREATEXTL + fsel_width; - - // check which colums can be visible - // depending on available width of window. - _columns = 0; - if (fsel_width > FILECOLUMN + _fib_font_size_width + _fib_font_time_width) { - _columns |= 2; - t_s = FAREAMRGL + fsel_width - _fib_font_time_width - TEXTSEP; - } - if (fsel_width > FILECOLUMN + _fib_font_size_width) { - _columns |= 1; - t_t = t_s - _fib_font_size_width - TEXTSEP; - } - - int fstop = _scrl_f; // first entry in scroll position - const int ttop = ltop - _fib_font_height + _fib_font_ascent; - - if (fstop > 0 && fstop + llen > _dircount) { - fstop = MAX (0, _dircount - llen); - _scrl_f = fstop; - } - - // list header - XSetForeground (dpy, _fib_gc, _c_gray3.pixel); - XFillRectangle (dpy, win, _fib_gc, FAREAMRGL, ltop - _fib_font_vsep, fsel_width, _fib_font_vsep); - - // draw background of file list - XSetForeground (dpy, _fib_gc, _c_gray2.pixel); - XFillRectangle (dpy, win, _fib_gc, FAREAMRGL, ltop, fsel_width, fsel_height); - -#ifdef DRAW_OUTLINE - VDrawRectangle (dpy, win, _fib_gc, FAREAMRGL, ltop - _fib_font_vsep -1, _fib_width - FAREAMRGL - FAREAMRGR, fsel_height + _fib_font_vsep + 1); -#endif - - switch (_hov_h) { - case 1: - XSetForeground (dpy, _fib_gc, _c_gray0.pixel); - XFillRectangle (dpy, win, _fib_gc, t_x + _fib_dir_indent - TEXTSEP + 1, ltop - _fib_font_vsep, t_t - t_x - _fib_dir_indent - 1, _fib_font_vsep); - break; - case 2: - XSetForeground (dpy, _fib_gc, _c_gray0.pixel); - XFillRectangle (dpy, win, _fib_gc, t_t - TEXTSEP + 1, ltop - _fib_font_vsep, _fib_font_size_width + TEXTSEP - 1, _fib_font_vsep); - break; - case 3: - XSetForeground (dpy, _fib_gc, _c_gray0.pixel); - XFillRectangle (dpy, win, _fib_gc, t_s - TEXTSEP + 1, ltop - _fib_font_vsep, TEXTSEP + TEXTSEP + _fib_font_time_width - 1, _fib_font_vsep); - break; - default: - break; - } - - // column headings and sort order - int arp = MAX (2, _fib_font_height / 5); // arrow scale - const int trioff = _fib_font_height - _fib_font_ascent - arp + 1; - XPoint ptri[4] = { {0, ttop - trioff }, {arp, -arp - arp - 1}, {-arp - arp, 0}, {arp, arp + arp + 1}}; - if (_sort & 1) { - ptri[0].y = ttop -arp - arp - 1; - ptri[1].y *= -1; - ptri[3].y *= -1; - } - switch (_sort) { - case 0: - case 1: - ptri[0].x = t_t + SORTBTNOFF + 2 - arp; - XSetForeground (dpy, _fib_gc, _c_gray6.pixel); - XFillPolygon (dpy, win, _fib_gc, ptri, 3, Convex, CoordModePrevious); - XDrawLines (dpy, win, _fib_gc, ptri, 4, CoordModePrevious); - break; - case 2: - case 3: - if (_columns & 1) { - ptri[0].x = t_s + SORTBTNOFF + 2 - arp; - XSetForeground (dpy, _fib_gc, _c_gray6.pixel); - XFillPolygon (dpy, win, _fib_gc, ptri, 3, Convex, CoordModePrevious); - XDrawLines (dpy, win, _fib_gc, ptri, 4, CoordModePrevious); - } - break; - case 4: - case 5: - if (_columns & 2) { - ptri[0].x = FAREATEXTL + fsel_width + SORTBTNOFF + 2 - arp; - XSetForeground (dpy, _fib_gc, _c_gray6.pixel); - XFillPolygon (dpy, win, _fib_gc, ptri, 3, Convex, CoordModePrevious); - XDrawLines (dpy, win, _fib_gc, ptri, 4, CoordModePrevious); - } - break; - } - -#if 0 // bottom header bottom border - XSetForeground (dpy, _fib_gc, _c_gray5.pixel); - XSetLineAttributes (dpy, _fib_gc, 1, LineOnOffDash, CapButt, JoinMiter); - XDrawLine (dpy, win, _fib_gc, - FAREAMRGL + 1, ltop, - FAREAMRGL + fsel_width, ltop); - XSetLineAttributes (dpy, _fib_gc, 1, LineSolid, CapButt, JoinMiter); -#endif - - XSetForeground (dpy, _fib_gc, _c_gray4.pixel); - XDrawLine (dpy, win, _fib_gc, - t_x + _fib_dir_indent - TEXTSEP, ltop - _fib_font_vsep + 3, - t_x + _fib_dir_indent - TEXTSEP, ltop - 3); - - XSetForeground (dpy, _fib_gc, blackColor); - XDrawString (dpy, win, _fib_gc, t_x + _fib_dir_indent, ttop, "Name", 4); - - if (_columns & 1) { - XSetForeground (dpy, _fib_gc, _c_gray4.pixel); - XDrawLine (dpy, win, _fib_gc, - t_t - TEXTSEP, ltop - _fib_font_vsep + 3, - t_t - TEXTSEP, ltop - 3); - XSetForeground (dpy, _fib_gc, blackColor); - XDrawString (dpy, win, _fib_gc, t_t, ttop, "Size", 4); - } - - if (_columns & 2) { - XSetForeground (dpy, _fib_gc, _c_gray4.pixel); - XDrawLine (dpy, win, _fib_gc, - t_s - TEXTSEP, ltop - _fib_font_vsep + 3, - t_s - TEXTSEP, ltop - 3); - XSetForeground (dpy, _fib_gc, blackColor); - if (_pathparts > 0) - XDrawString (dpy, win, _fib_gc, t_s, ttop, "Last Modified", 13); - else - XDrawString (dpy, win, _fib_gc, t_s, ttop, "Last Used", 9); - } - - // scrollbar sep - if (llen < _dircount) { - const int sx0 = _fib_width - SCROLLBARW - FAREAMRGR; - XSetForeground (dpy, _fib_gc, _c_gray4.pixel); - XDrawLine (dpy, win, _fib_gc, - sx0 - 1, ltop - _fib_font_vsep, -#ifdef DRAW_OUTLINE - sx0 - 1, ltop + fsel_height -#else - sx0 - 1, ltop - 1 -#endif - ); - } - - // clip area for file-name - XRectangle clp = {FAREAMRGL + 1, ltop, t_t - FAREAMRGL - TEXTSEP - TEXTSEP - 1, fsel_height}; - - // list files in view - for (i = 0; i < llen; ++i) { - const int j = i + fstop; - if (j >= _dircount) break; - - const int t_y = ltop + (i+1) * _fib_font_vsep - 4; - - XSetForeground (dpy, _fib_gc, blackColor); - if (_dirlist[j].flags & 2) { - XSetForeground (dpy, _fib_gc, blackColor); - XFillRectangle (dpy, win, _fib_gc, - FAREAMRGL, t_y - _fib_font_ascent, fsel_width, _fib_font_height); - XSetForeground (dpy, _fib_gc, whiteColor); - } - if (_hov_f == j && !(_dirlist[j].flags & 2)) { - XSetForeground (dpy, _fib_gc, _c_gray4.pixel); - } - if (_dirlist[j].flags & 4) { - XDrawString (dpy, win, _fib_gc, t_x, t_y, "D", 1); - } - XSetClipRectangles (dpy, _fib_gc, 0, 0, &clp, 1, Unsorted); - XDrawString (dpy, win, _fib_gc, - t_x + _fib_dir_indent, t_y, - _dirlist[j].name, strlen (_dirlist[j].name)); - XSetClipMask (dpy, _fib_gc, None); - - if (_columns & 1) // right-aligned 'size' - XDrawString (dpy, win, _fib_gc, - t_s - TEXTSEP - 2 - _dirlist[j].ssizew, t_y, - _dirlist[j].strsize, strlen (_dirlist[j].strsize)); - if (_columns & 2) - XDrawString (dpy, win, _fib_gc, - t_s, t_y, - _dirlist[j].strtime, strlen (_dirlist[j].strtime)); - } - - // scrollbar - if (llen < _dircount) { - float sl = (fsel_height + _fib_font_vsep - (SCROLLBOXH + SCROLLBOXH)) / (float) _dircount; - sl = MAX ((8. / llen), sl); // 8px min height of scroller - const int sy1 = llen * sl; - const float mx = (fsel_height + _fib_font_vsep - (SCROLLBOXH + SCROLLBOXH) - sy1) / (float)(_dircount - llen); - const int sy0 = fstop * mx; - const int sx0 = _fib_width - SCROLLBARW - FAREAMRGR; - const int stop = ltop - _fib_font_vsep; - - _scrl_y0 = stop + SCROLLBOXH + sy0; - _scrl_y1 = _scrl_y0 + sy1; - - assert (fstop + llen <= _dircount); - // scroll-bar background - XSetForeground (dpy, _fib_gc, _c_gray3.pixel); - XFillRectangle (dpy, win, _fib_gc, sx0, stop, SCROLLBARW, fsel_height + _fib_font_vsep); - - // scroller - if (_hov_s == 0) { - XSetForeground (dpy, _fib_gc, _c_gray0.pixel); - } else { - XSetForeground (dpy, _fib_gc, _c_gray1.pixel); - } - XFillRectangle (dpy, win, _fib_gc, sx0 + 1, stop + SCROLLBOXH + sy0, SCROLLBARW - 2, sy1); - - int scrw = (SCROLLBARW -3) / 2; - // arrows top and bottom - if (_hov_s == 1) { - XSetForeground (dpy, _fib_gc, _c_gray0.pixel); - } else { - XSetForeground (dpy, _fib_gc, _c_gray1.pixel); - } - XPoint ptst[4] = { {sx0 + 1, stop + 8}, {scrw, -7}, {scrw, 7}, {-2 * scrw, 0}}; - XFillPolygon (dpy, win, _fib_gc, ptst, 3, Convex, CoordModePrevious); - XDrawLines (dpy, win, _fib_gc, ptst, 4, CoordModePrevious); - - if (_hov_s == 2) { - XSetForeground (dpy, _fib_gc, _c_gray0.pixel); - } else { - XSetForeground (dpy, _fib_gc, _c_gray1.pixel); - } - XPoint ptsb[4] = { {sx0 + 1, ltop + fsel_height - 9}, {2*scrw, 0}, {-scrw, 7}, {-scrw, -7}}; - XFillPolygon (dpy, win, _fib_gc, ptsb, 3, Convex, CoordModePrevious); - XDrawLines (dpy, win, _fib_gc, ptsb, 4, CoordModePrevious); - } else { - _scrl_y0 = _scrl_y1 = -1; - } - - if (_fib_show_places) { - assert (_placecnt > 0); - - // heading - XSetForeground (dpy, _fib_gc, _c_gray3.pixel); - XFillRectangle (dpy, win, _fib_gc, FAREAMRGB, ltop - _fib_font_vsep, PLACESW - TEXTSEP, _fib_font_vsep); - - // body - XSetForeground (dpy, _fib_gc, _c_gray2.pixel); - XFillRectangle (dpy, win, _fib_gc, FAREAMRGB, ltop, PLACESW - TEXTSEP, fsel_height); - -#ifdef DRAW_OUTLINE - VDrawRectangle (dpy, win, _fib_gc, FAREAMRGB, ltop - _fib_font_vsep -1, PLACESW - TEXTSEP, fsel_height + _fib_font_vsep + 1); -#endif - - XSetForeground (dpy, _fib_gc, blackColor); - XDrawString (dpy, win, _fib_gc, FAREAMRGB + TEXTSEP, ttop, "Places", 6); - - XRectangle pclip = {FAREAMRGB + 1, ltop, PLACESW - TEXTSEP -1, fsel_height}; - XSetClipRectangles (dpy, _fib_gc, 0, 0, &pclip, 1, Unsorted); - const int plx = FAREAMRGB + TEXTSEP; - for (i = 0; i < llen && i < _placecnt; ++i) { - const int ply = ltop + (i+1) * _fib_font_vsep - 4; - if (i == _hov_l) { - XSetForeground (dpy, _fib_gc, _c_gray4.pixel); - } else { - XSetForeground (dpy, _fib_gc, blackColor); - } - XDrawString (dpy, win, _fib_gc, - plx, ply, - _placelist[i].name, strlen (_placelist[i].name)); - if (_placelist[i].flags & 4) { - XSetForeground (dpy, _fib_gc, _c_gray3.pixel); - const int plly = ply - _fib_font_ascent + _fib_font_height; - const int pllx0 = FAREAMRGB; - const int pllx1 = FAREAMRGB + (PLACESW - TEXTSEP); - XDrawLine (dpy, win, _fib_gc, pllx0, plly, pllx1, plly); - } - } - XSetClipMask (dpy, _fib_gc, None); - - if (_placecnt > llen) { - const int plly = ltop + fsel_height - _fib_font_height + _fib_font_ascent; - const int pllx0 = FAREAMRGB + (PLACESW - TEXTSEP) * .75; - const int pllx1 = FAREAMRGB + (PLACESW - TEXTSEP - TEXTSEP); - - XSetForeground (dpy, _fib_gc, blackColor); - XSetLineAttributes (dpy, _fib_gc, 1, LineOnOffDash, CapButt, JoinMiter); - XDrawLine (dpy, win, _fib_gc, pllx0, plly, pllx1, plly); - XSetLineAttributes (dpy, _fib_gc, 1, LineSolid, CapButt, JoinMiter); - } - } - - // Bottom Buttons - const int numb = sizeof(_btns) / sizeof(FibButton*); - int xtra = _fib_width - _btn_span; - const int cbox = _fib_font_ascent - 2; - const int bbase = _fib_height - BTNBTMMARGIN * _fib_font_vsep - BTNPADDING; - const int cblw = cbox > 20 ? 5 : ( cbox > 9 ? 3 : 1); - - int bx = FAREAMRGB; - for (i = 0; i < numb; ++i) { - if (_btns[i]->flags & 8) { continue; } - if (_btns[i]->flags & 4) { - // checkbutton - const int cby0 = bbase - cbox + 1 + BTNPADDING; - if (i == _hov_b) { - XSetForeground (dpy, _fib_gc, _c_gray4.pixel); - } else { - XSetForeground (dpy, _fib_gc, blackColor); - } - XDrawRectangle (dpy, win, _fib_gc, - bx, cby0 - 1, cbox + 1, cbox + 1); - - if (i == _hov_b) { - XSetForeground (dpy, _fib_gc, _c_gray5.pixel); - } else { - XSetForeground (dpy, _fib_gc, blackColor); - } - XDrawString (dpy, win, _fib_gc, BTNPADDING + bx + _fib_font_ascent, 1 + bbase + BTNPADDING, - _btns[i]->text, strlen (_btns[i]->text)); - - if (i == _hov_b) { - XSetForeground (dpy, _fib_gc, _c_gray0.pixel); - } else { - if (_btns[i]->flags & 2) { - XSetForeground (dpy, _fib_gc, _c_gray1.pixel); - } else { - XSetForeground (dpy, _fib_gc, _c_gray2.pixel); - } - } - XFillRectangle (dpy, win, _fib_gc, - bx+1, cby0, cbox, cbox); - - if (_btns[i]->flags & 2) { - XSetLineAttributes (dpy, _fib_gc, cblw, LineSolid, CapRound, JoinMiter); - XSetForeground (dpy, _fib_gc, _c_gray6.pixel); - XDrawLine (dpy, win, _fib_gc, - bx + 2, cby0 + 1, - bx + cbox - 1, cby0 + cbox - 2); - XDrawLine (dpy, win, _fib_gc, - bx + cbox - 1, cby0 + 1, - bx + 2, cby0 + cbox - 2); - XSetLineAttributes (dpy, _fib_gc, 1, LineSolid, CapButt, JoinMiter); - } - } else { - if (xtra > 0) { - bx += xtra; - xtra = 0; - } - // pushbutton - - uint8_t can_hover = 1; // special case - if (_btns[i] == &_btn_ok) { - if (_fsel < 0 || _fsel >= _dircount) { - can_hover = 0; - } - } - - if (can_hover && i == _hov_b) { - XSetForeground (dpy, _fib_gc, _c_gray0.pixel); - } else { - XSetForeground (dpy, _fib_gc, _c_gray2.pixel); - } - XFillRectangle (dpy, win, _fib_gc, - bx + 1, bbase - _fib_font_ascent, - _btn_w - 1, _fib_font_height + BTNPADDING + BTNPADDING); - VDrawRectangle (dpy, win, _fib_gc, - bx, bbase - _fib_font_ascent, - _btn_w, _fib_font_height + BTNPADDING + BTNPADDING); - XDrawString (dpy, win, _fib_gc, bx + (_btn_w - _btns[i]->tw) * .5, 1 + bbase + BTNPADDING, - _btns[i]->text, strlen (_btns[i]->text)); - } - _btns[i]->x0 = bx; - bx += _btns[i]->xw + DSEP; - } - - if (_pixbuffer != None) { - XCopyArea(dpy, _pixbuffer, realwin, _fib_gc, 0, 0, _fib_width, _fib_height, 0, 0); - } - XFlush (dpy); -} - -static void fib_reset () { - _hov_p = _hov_f = _hov_h = _hov_l = -1; - _scrl_f = 0; - _fib_resized = 1; -} - -static int cmp_n_up (const void *p1, const void *p2) { - FibFileEntry *a = (FibFileEntry*) p1; - FibFileEntry *b = (FibFileEntry*) p2; - if ((a->flags & 4) && !(b->flags & 4)) return -1; - if (!(a->flags & 4) && (b->flags & 4)) return 1; - return strcmp (a->name, b->name); -} - -static int cmp_n_down (const void *p1, const void *p2) { - FibFileEntry *a = (FibFileEntry*) p1; - FibFileEntry *b = (FibFileEntry*) p2; - if ((a->flags & 4) && !(b->flags & 4)) return -1; - if (!(a->flags & 4) && (b->flags & 4)) return 1; - return strcmp (b->name, a->name); -} - -static int cmp_t_up (const void *p1, const void *p2) { - FibFileEntry *a = (FibFileEntry*) p1; - FibFileEntry *b = (FibFileEntry*) p2; - if ((a->flags & 4) && !(b->flags & 4)) return -1; - if (!(a->flags & 4) && (b->flags & 4)) return 1; - if (a->mtime == b->mtime) return 0; - return a->mtime > b->mtime ? -1 : 1; -} - -static int cmp_t_down (const void *p1, const void *p2) { - FibFileEntry *a = (FibFileEntry*) p1; - FibFileEntry *b = (FibFileEntry*) p2; - if ((a->flags & 4) && !(b->flags & 4)) return -1; - if (!(a->flags & 4) && (b->flags & 4)) return 1; - if (a->mtime == b->mtime) return 0; - return a->mtime > b->mtime ? 1 : -1; -} - -static int cmp_s_up (const void *p1, const void *p2) { - FibFileEntry *a = (FibFileEntry*) p1; - FibFileEntry *b = (FibFileEntry*) p2; - if ((a->flags & 4) && (b->flags & 4)) return 0; // dir, no size, retain order - if ((a->flags & 4) && !(b->flags & 4)) return -1; - if (!(a->flags & 4) && (b->flags & 4)) return 1; - if (a->size == b->size) return 0; - return a->size > b->size ? -1 : 1; -} - -static int cmp_s_down (const void *p1, const void *p2) { - FibFileEntry *a = (FibFileEntry*) p1; - FibFileEntry *b = (FibFileEntry*) p2; - if ((a->flags & 4) && (b->flags & 4)) return 0; // dir, no size, retain order - if ((a->flags & 4) && !(b->flags & 4)) return -1; - if (!(a->flags & 4) && (b->flags & 4)) return 1; - if (a->size == b->size) return 0; - return a->size > b->size ? 1 : -1; -} - -static void fmt_size (Display *dpy, FibFileEntry *f) { - if (f->size > 10995116277760) { - sprintf (f->strsize, "%.0f TB", f->size / 1099511627776.f); - } - if (f->size > 1099511627776) { - sprintf (f->strsize, "%.1f TB", f->size / 1099511627776.f); - } - else if (f->size > 10737418240) { - sprintf (f->strsize, "%.0f GB", f->size / 1073741824.f); - } - else if (f->size > 1073741824) { - sprintf (f->strsize, "%.1f GB", f->size / 1073741824.f); - } - else if (f->size > 10485760) { - sprintf (f->strsize, "%.0f MB", f->size / 1048576.f); - } - else if (f->size > 1048576) { - sprintf (f->strsize, "%.1f MB", f->size / 1048576.f); - } - else if (f->size > 10240) { - sprintf (f->strsize, "%.0f KB", f->size / 1024.f); - } - else if (f->size >= 1000) { - sprintf (f->strsize, "%.1f KB", f->size / 1024.f); - } - else { - sprintf (f->strsize, "%.0f B", f->size / 1.f); - } - int sw = 0; - query_font_geometry (dpy, _fib_gc, f->strsize, &sw, NULL, NULL, NULL); - if (sw > _fib_font_size_width) { - _fib_font_size_width = sw; - } - f->ssizew = sw; -} - -static void fmt_time (Display *dpy, FibFileEntry *f) { - struct tm *tmp; - tmp = localtime (&f->mtime); - if (!tmp) { - return; - } - strftime (f->strtime, sizeof(f->strtime), "%F %H:%M", tmp); - - int tw = 0; - query_font_geometry (dpy, _fib_gc, f->strtime, &tw, NULL, NULL, NULL); - if (tw > _fib_font_time_width) { - _fib_font_time_width = tw; - } -} - -static void fib_resort (const char * sel) { - if (_dircount < 1) { return; } - int (*sortfn)(const void *p1, const void *p2); - switch (_sort) { - case 1: sortfn = &cmp_n_down; break; - case 2: sortfn = &cmp_s_down; break; - case 3: sortfn = &cmp_s_up; break; - case 4: sortfn = &cmp_t_down; break; - case 5: sortfn = &cmp_t_up; break; - default: - sortfn = &cmp_n_up; - break; - } - qsort (_dirlist, _dircount, sizeof(_dirlist[0]), sortfn); - int i; - for (i = 0; i < _dircount && sel; ++i) { - if (!strcmp (_dirlist[i].name, sel)) { - _fsel = i; - break; - } - } -} - -static void fib_select (Display *dpy, int item) { - if (_fsel >= 0) { - _dirlist[_fsel].flags &= ~2; - } - _fsel = item; - if (_fsel >= 0 && _fsel < _dircount) { - _dirlist[_fsel].flags |= 2; - const int llen = (_fib_height - LISTBOT * _fib_font_vsep) / _fib_font_vsep; - if (_fsel < _scrl_f) { - _scrl_f = _fsel; - } - else if (_fsel >= _scrl_f + llen) { - _scrl_f = 1 + _fsel - llen; - } - } else { - _fsel = -1; - } - - fib_expose (dpy, _fib_win); -} - -static inline int fib_filter (const char *name) { - if (_fib_filter_function) { - return _fib_filter_function (name); - } else { - return 1; - } -} - -static void fib_pre_opendir (Display *dpy) { - if (_dirlist) free (_dirlist); - if (_pathbtn) free (_pathbtn); - _dirlist = NULL; - _pathbtn = NULL; - _dircount = 0; - _pathparts = 0; - query_font_geometry (dpy, _fib_gc, "Size ", &_fib_font_size_width, NULL, NULL, NULL); - fib_reset (); - _fsel = -1; -} - -static void fib_post_opendir (Display *dpy, const char *sel) { - if (_dircount > 0) - _fsel = 0; // select first - else - _fsel = -1; - fib_resort (sel); - - if (_dircount > 0 && _fsel >= 0) { - fib_select (dpy, _fsel); - } else { - fib_expose (dpy, _fib_win); - } -} - -static int fib_dirlistadd (Display *dpy, const int i, const char* path, const char *name, time_t mtime) { - char tp[1024]; - struct stat fs; - if (!_fib_hidden_fn && name[0] == '.') return -1; - if (!strcmp (name, ".")) return -1; - if (!strcmp (name, "..")) return -1; - strcpy (tp, path); - strcat (tp, name); - if (access (tp, R_OK)) { - return -1; - } - if (stat (tp, &fs)) { - return -1; - } - assert (i < _dircount); // could happen if dir changes while we're reading. - if (i >= _dircount) return -1; - if (S_ISDIR (fs.st_mode)) { - _dirlist[i].flags |= 4; - } - else if (S_ISREG (fs.st_mode)) { - if (!fib_filter (name)) return -1; - } -#if 0 // only needed with lstat() - else if (S_ISLNK (fs.st_mode)) { - if (!fib_filter (name)) return -1; - } -#endif - else { - return -1; - } - strcpy (_dirlist[i].name, name); - _dirlist[i].mtime = mtime > 0 ? mtime : fs.st_mtime; - _dirlist[i].size = fs.st_size; - if (!(_dirlist[i].flags & 4)) - fmt_size (dpy, &_dirlist[i]); - fmt_time (dpy, &_dirlist[i]); - return 0; -} - -static int fib_openrecent (Display *dpy, const char *sel) { - int i; - unsigned int j; - assert (_recentcnt > 0); - fib_pre_opendir (dpy); - query_font_geometry (dpy, _fib_gc, "Last Used", &_fib_font_time_width, NULL, NULL, NULL); - _dirlist = (FibFileEntry*) calloc (_recentcnt, sizeof(FibFileEntry)); - _dircount = _recentcnt; - for (j = 0, i = 0; j < _recentcnt; ++j) { - char base[1024]; - char *s = strrchr (_recentlist[j].path, '/'); - if (!s || !*++s) continue; - size_t len = (s - _recentlist[j].path); - strncpy (base, _recentlist[j].path, len); - base[len] = '\0'; - if (!fib_dirlistadd (dpy, i, base, s, _recentlist[j].atime)) { - _dirlist[i].rfp = &_recentlist[j]; - _dirlist[i].flags |= 8; - ++i; - } - } - _dircount = i; - fib_post_opendir (dpy, sel); - return _dircount; -} - -static int fib_opendir (Display *dpy, const char* path, const char *sel) { - char *t0, *t1; - int i; - - assert (path); - - if (strlen (path) == 0 && _recentcnt > 0) { // XXX we should use a better indication for this - strcpy (_cur_path, ""); - return fib_openrecent (dpy, sel); - } - - assert (strlen (path) < sizeof(_cur_path) -1); - assert (strlen (path) > 0); - assert (strstr (path, "//") == NULL); - assert (path[0] == '/'); - - fib_pre_opendir (dpy); - - query_font_geometry (dpy, _fib_gc, "Last Modified", &_fib_font_time_width, NULL, NULL, NULL); - DIR *dir = opendir (path); - if (!dir) { - strcpy (_cur_path, "/"); - } else { - int i; - struct dirent *de; - if (path != _cur_path) - strcpy (_cur_path, path); - - if (_cur_path[strlen (_cur_path) -1] != '/') - strcat (_cur_path, "/"); - - while ((de = readdir (dir))) { - if (!_fib_hidden_fn && de->d_name[0] == '.') continue; - ++_dircount; - } - - if (_dircount > 0) - _dirlist = (FibFileEntry*) calloc (_dircount, sizeof(FibFileEntry)); - - rewinddir (dir); - - i = 0; - while ((de = readdir (dir))) { - if (!fib_dirlistadd (dpy, i, _cur_path, de->d_name, 0)) - ++i; - } - _dircount = i; - closedir (dir); - } - - t0 = _cur_path; - while (*t0 && (t0 = strchr (t0, '/'))) { - ++_pathparts; - ++t0; - } - assert (_pathparts > 0); - _pathbtn = (FibPathButton*) calloc (_pathparts + 1, sizeof(FibPathButton)); - - t1 = _cur_path; - i = 0; - while (*t1 && (t0 = strchr (t1, '/'))) { - if (i == 0) { - strcpy (_pathbtn[i].name, "/"); - } else { - *t0 = 0; - strcpy (_pathbtn[i].name, t1); - } - query_font_geometry (dpy, _fib_gc, _pathbtn[i].name, &_pathbtn[i].xw, NULL, NULL, NULL); - _pathbtn[i].xw += BTNPADDING + BTNPADDING; - *t0 = '/'; - t1 = t0 + 1; - ++i; - } - fib_post_opendir (dpy, sel); - return _dircount; -} - -static int fib_open (Display *dpy, int item) { - char tp[1024]; - if (_dirlist[item].flags & 8) { - assert (_dirlist[item].rfp); - strcpy (_rv_open, _dirlist[item].rfp->path); - _status = 1; - return 0; - } - strcpy (tp, _cur_path); - strcat (tp, _dirlist[item].name); - if (_dirlist[item].flags & 4) { - fib_opendir (dpy, tp, NULL); - return 0; - } else { - _status = 1; - strcpy (_rv_open, tp); - } - return 0; -} - -static void cb_cancel (Display *dpy) { - _status = -1; -} - -static void cb_open (Display *dpy) { - if (_fsel >= 0 && _fsel < _dircount) { - fib_open (dpy, _fsel); - } -} - -static void sync_button_states () { - if (_fib_show_places) - _btn_places.flags |= 2; - else - _btn_places.flags &= ~2; - if (_fib_filter_fn) // inverse -> show all - _btn_filter.flags &= ~2; - else - _btn_filter.flags |= 2; - if (_fib_hidden_fn) - _btn_hidden.flags |= 2; - else - _btn_hidden.flags &= ~2; -} - -static void cb_places (Display *dpy) { - _fib_show_places = ! _fib_show_places; - if (_placecnt < 1) - _fib_show_places = 0; - sync_button_states (); - _fib_resized = 1; - fib_expose (dpy, _fib_win); -} - -static void cb_filter (Display *dpy) { - _fib_filter_fn = ! _fib_filter_fn; - sync_button_states (); - char *sel = _fsel >= 0 ? strdup (_dirlist[_fsel].name) : NULL; - fib_opendir (dpy, _cur_path, sel); - free (sel); -} - -static void cb_hidden (Display *dpy) { - _fib_hidden_fn = ! _fib_hidden_fn; - sync_button_states (); - char *sel = _fsel >= 0 ? strdup (_dirlist[_fsel].name) : NULL; - fib_opendir (dpy, _cur_path, sel); - free (sel); -} - -static int fib_widget_at_pos (Display *dpy, int x, int y, int *it) { - const int btop = _fib_height - BTNBTMMARGIN * _fib_font_vsep - _fib_font_ascent - BTNPADDING; - const int bbot = btop + _fib_font_height + BTNPADDING + BTNPADDING; - const int llen = (_fib_height - LISTBOT * _fib_font_vsep) / _fib_font_vsep; - const int ltop = LISTTOP * _fib_font_vsep; - const int fbot = ltop + 4 + llen * _fib_font_vsep; - const int ptop = PATHBTNTOP - _fib_font_ascent; - assert (it); - - // paths at top - if (y > ptop && y < ptop + _fib_font_height && _view_p >= 0 && _pathparts > 0) { - int i = _view_p; - *it = -1; - if (i > 0) { // special case '<' - if (x > FAREAMRGB && x <= FAREAMRGB + _pathbtn[0].xw) { - *it = _view_p - 1; - i = _pathparts; - } - } - while (i < _pathparts) { - if (x >= _pathbtn[i].x0 && x <= _pathbtn[i].x0 + _pathbtn[i].xw) { - *it = i; - break; - } - ++i; - } - assert (*it < _pathparts); - if (*it >= 0) return 1; - else return 0; - } - - // buttons at bottom - if (y > btop && y < bbot) { - size_t i; - *it = -1; - for (i = 0; i < sizeof(_btns) / sizeof(FibButton*); ++i) { - const int bx = _btns[i]->x0; - if (_btns[i]->flags & 8) { continue; } - if (x > bx && x < bx + _btns[i]->xw) { - *it = i; - } - } - if (*it >= 0) return 3; - else return 0; - } - - // main file area - if (y >= ltop - _fib_font_vsep && y < fbot && x > FAREAMRGL && x < _fib_width - FAREAMRGR) { - // scrollbar - if (_scrl_y0 > 0 && x >= _fib_width - (FAREAMRGR + SCROLLBARW) && x <= _fib_width - FAREAMRGR) { - if (y >= _scrl_y0 && y < _scrl_y1) { - *it = 0; - } else if (y >= _scrl_y1) { - *it = 2; - } else { - *it = 1; - } - return 4; - } - // file-list - else if (y >= ltop) { - const int item = (y - ltop) / _fib_font_vsep + _scrl_f; - *it = -1; - if (item >= 0 && item < _dircount) { - *it = item; - } - if (*it >= 0) return 2; - else return 0; - } - else { - *it = -1; - const int fsel_width = _fib_width - FAREAMRGL - FAREAMRGR - (llen < _dircount ? SCROLLBARW : 0); - const int t_s = FAREAMRGL + fsel_width - _fib_font_time_width - TEXTSEP - TEXTSEP; - const int t_t = FAREAMRGL + fsel_width - TEXTSEP - _fib_font_size_width - ((_columns & 2) ? ( _fib_font_time_width + TEXTSEP + TEXTSEP) : 0); - if (x >= fsel_width + FAREAMRGL) ; - else if ((_columns & 2) && x >= t_s) *it = 3; - else if ((_columns & 1) && x >= t_t) *it = 2; - else if (x >= FAREATEXTL + _fib_dir_indent - TEXTSEP) *it = 1; - if (*it >= 0) return 5; - else return 0; - } - } - - // places list - if (_fib_show_places && y >= ltop && y < fbot && x > FAREAMRGB && x < FAREAMRGL - FAREAMRGB) { - const int item = (y - ltop) / _fib_font_vsep; - *it = -1; - if (item >= 0 && item < _placecnt) { - *it = item; - } - if (*it >= 0) return 6; - else return 0; - } - - return 0; -} - -static void fib_update_hover (Display *dpy, int need_expose, const int type, const int item) { - int hov_p = -1; - int hov_b = -1; - int hov_h = -1; - int hov_s = -1; -#ifdef LIST_ENTRY_HOVER - int hov_f = -1; - int hov_l = -1; -#endif - - switch (type) { - case 1: hov_p = item; break; - case 3: hov_b = item; break; - case 4: hov_s = item; break; - case 5: hov_h = item; break; -#ifdef LIST_ENTRY_HOVER - case 6: hov_l = item; break; - case 2: hov_f = item; break; -#endif - default: break; - } -#ifdef LIST_ENTRY_HOVER - if (hov_f != _hov_f) { _hov_f = hov_f; need_expose = 1; } - if (hov_l != _hov_l) { _hov_l = hov_l; need_expose = 1; } -#endif - if (hov_b != _hov_b) { _hov_b = hov_b; need_expose = 1; } - if (hov_p != _hov_p) { _hov_p = hov_p; need_expose = 1; } - if (hov_h != _hov_h) { _hov_h = hov_h; need_expose = 1; } - if (hov_s != _hov_s) { _hov_s = hov_s; need_expose = 1; } - - if (need_expose) { - fib_expose (dpy, _fib_win); - } -} - -static void fib_motion (Display *dpy, int x, int y) { - int it = -1; - - if (_scrl_my >= 0) { - const int sdiff = y - _scrl_my; - const int llen = (_fib_height - LISTBOT * _fib_font_vsep) / _fib_font_vsep; - const int fsel_height = 4 + llen * _fib_font_vsep; - const float sl = (fsel_height + _fib_font_vsep - (SCROLLBOXH + SCROLLBOXH)) / (float) _dircount; - - int news = _scrl_mf + sdiff / sl; - if (news < 0) news = 0; - if (news >= (_dircount - llen)) news = _dircount - llen; - if (news != _scrl_f) { - _scrl_f = news; - fib_expose (dpy, _fib_win); - } - return; - } - - const int type = fib_widget_at_pos (dpy, x, y, &it); - fib_update_hover (dpy, 0, type, it); -} - -static void fib_mousedown (Display *dpy, int x, int y, int btn, unsigned long time) { - int it; - switch (fib_widget_at_pos (dpy, x, y, &it)) { - case 4: // scrollbar - if (btn == 1) { - _dblclk = 0; - if (it == 0) { - _scrl_my = y; - _scrl_mf = _scrl_f; - } else { - int llen = (_fib_height - LISTBOT * _fib_font_vsep) / _fib_font_vsep; - if (llen < 2) llen = 2; - int news = _scrl_f; - if (it == 1) { - news -= llen - 1; - } else { - news += llen - 1; - } - if (news < 0) news = 0; - if (news >= (_dircount - llen)) news = _dircount - llen; - if (news != _scrl_f && _scrl_y0 >= 0) { - assert (news >=0); - _scrl_f = news; - fib_update_hover (dpy, 1, 4, it); - } - } - } - break; - case 2: // file-list - if (btn == 4 || btn == 5) { - const int llen = (_fib_height - LISTBOT * _fib_font_vsep) / _fib_font_vsep; - int news = _scrl_f + ((btn == 4) ? - 1 : 1); - if (news < 0) news = 0; - if (news >= (_dircount - llen)) news = _dircount - llen; - if (news != _scrl_f && _scrl_y0 >= 0) { - assert (news >=0); - _scrl_f = news; - fib_update_hover (dpy, 1, 0, 0); - } - _dblclk = 0; - } - else if (btn == 1 && it >= 0 && it < _dircount) { - if (_fsel == it) { - if (time - _dblclk < DBLCLKTME) { - fib_open (dpy, it); - _dblclk = 0; - } - _dblclk = time; - } else { - fib_select (dpy, it); - _dblclk = time; - } - if (_fsel >= 0) { - if (!(_dirlist[_fsel].flags & 4)) { - } - } - } - break; - case 1: // paths - assert (_fsel < _dircount); - assert (it >= 0 && it < _pathparts); - { - int i = 0; - char path[1024] = "/"; - while (++i <= it) { - strcat (path, _pathbtn[i].name); - strcat (path, "/"); - } - char *sel = NULL; - if (i < _pathparts) - sel = strdup (_pathbtn[i].name); - else if (i == _pathparts && _fsel >= 0) - sel = strdup (_dirlist[_fsel].name); - fib_opendir (dpy, path, sel); - free (sel); - } - break; - case 3: // btn - if (btn == 1 && _btns[it]->callback) { - _btns[it]->callback (dpy); - } - break; - case 5: // sort - if (btn == 1) { - switch (it) { - case 1: if (_sort == 0) _sort = 1; else _sort = 0; break; - case 2: if (_sort == 2) _sort = 3; else _sort = 2; break; - case 3: if (_sort == 4) _sort = 5; else _sort = 4; break; - } - if (_fsel >= 0) { - assert (_dirlist && _dircount >= _fsel); - _dirlist[_fsel].flags &= ~2; - char *sel = strdup (_dirlist[_fsel].name); - fib_resort (sel); - free (sel); - } else { - fib_resort (NULL); - _fsel = -1; - } - fib_reset (); - _hov_h = it; - fib_select (dpy, _fsel); - } - break; - case 6: - if (btn == 1 && it >= 0 && it < _placecnt) { - fib_opendir (dpy, _placelist[it].path, NULL); - } - break; - default: - break; - } -} - -static void fib_mouseup (Display *dpy, int x, int y, int btn, unsigned long time) { - _scrl_my = -1; -} - -static void add_place_raw (Display *dpy, const char *name, const char *path) { - _placelist = (FibPlace*) realloc (_placelist, (_placecnt + 1) * sizeof(FibPlace)); - strcpy (_placelist[_placecnt].path, path); - strcpy (_placelist[_placecnt].name, name); - _placelist[_placecnt].flags = 0; - - int sw; - query_font_geometry (dpy, _fib_gc, name, &sw, NULL, NULL, NULL); - if (sw > _fib_place_width) { - _fib_place_width = sw; - } - ++_placecnt; -} - -static int add_place_places (Display *dpy, const char *name, const char *url) { - char const * path; - struct stat fs; - int i; - if (!url || strlen (url) < 1) return -1; - if (!name || strlen (name) < 1) return -1; - if (url[0] == '/') { - path = url; - } - else if (!strncmp (url, "file:///", 8)) { - path = &url[7]; - } - else { - return -1; - } - - if (access (path, R_OK)) { - return -1; - } - if (stat (path, &fs)) { - return -1; - } - if (!S_ISDIR (fs.st_mode)) { - return -1; - } - - for (i = 0; i < _placecnt; ++i) { - if (!strcmp (path, _placelist[i].path)) { - return -1; - } - } - add_place_raw (dpy, name, path); - return 0; -} - -static int parse_gtk_bookmarks (Display *dpy, const char *fn) { - char tmp[1024]; - if (access (fn, R_OK)) { - return -1; - } - FILE *bm = fopen (fn, "r"); - if (!bm) return -1; - int found = 0; - while (fgets (tmp, sizeof(tmp), bm) - && strlen (tmp) > 1 - && strlen (tmp) < sizeof(tmp)) - { - char *s, *n; - tmp[strlen (tmp) - 1] = '\0'; // strip newline - if ((s = strchr (tmp, ' '))) { - *s = '\0'; - n = strdup (++s); - decode_3986 (tmp); - if (!add_place_places (dpy, n, tmp)) { - ++found; - } - free (n); - } else if ((s = strrchr (tmp, '/'))) { - n = strdup (++s); - decode_3986 (tmp); - if (!add_place_places (dpy, n, tmp)) { - ++found; - } - free (n); - } - } - fclose (bm); - return found; -} - -static const char *ignore_mountpoints[] = { - "/bin", "/boot", "/dev", "/etc", - "/lib", "/live", "/mnt", "/opt", - "/root", "/sbin", "/srv", "/tmp", - "/usr", "/var", "/proc", "/sbin", - "/net", "/sys" -}; - -static const char *ignore_fs[] = { - "auto", "autofs", - "debugfs", "devfs", - "devpts", "ecryptfs", - "fusectl", "kernfs", - "linprocfs", "proc", - "ptyfs", "rootfs", - "selinuxfs", "sysfs", - "tmpfs", "usbfs", - "nfsd", "rpc_pipefs", -}; - -static const char *ignore_devices[] = { - "binfmt_", "devpts", - "gvfs", "none", - "nfsd", "sunrpc", - "/dev/loop", "/dev/vn" -}; - -static int check_mount (const char *mountpoint, const char *fs, const char *device) { - size_t i; - if (!mountpoint || !fs || !device) return -1; - //printf("%s %s %s\n", mountpoint, fs, device); - for (i = 0 ; i < sizeof(ignore_mountpoints) / sizeof(char*); ++i) { - if (!strncmp (mountpoint, ignore_mountpoints[i], strlen (ignore_mountpoints[i]))) { - return 1; - } - } - if (!strncmp (mountpoint, "/home", 5)) { - return 1; - } - for (i = 0 ; i < sizeof(ignore_fs) / sizeof(char*); ++i) { - if (!strncmp (fs, ignore_fs[i], strlen (ignore_fs[i]))) { - return 1; - } - } - for (i = 0 ; i < sizeof(ignore_devices) / sizeof(char*); ++i) { - if (!strncmp (device, ignore_devices[i], strlen (ignore_devices[i]))) { - return 1; - } - } - return 0; -} - -static int read_mtab (Display *dpy, const char *mtab) { - FILE *mt = fopen (mtab, "r"); - if (!mt) return -1; - int found = 0; - struct mntent *mntent; - while ((mntent = getmntent (mt)) != NULL) { - char *s; - if (check_mount (mntent->mnt_dir, mntent->mnt_type, mntent->mnt_fsname)) - continue; - - if ((s = strrchr (mntent->mnt_dir, '/'))) { - ++s; - } else { - s = mntent->mnt_dir; - } - if (!add_place_places (dpy, s, mntent->mnt_dir)) { - ++found; - } - } - fclose (mt); - return found; -} - -static void populate_places (Display *dpy) { - char tmp[1024]; - int spacer = -1; - if (_placecnt > 0) return; - _fib_place_width = 0; - - if (_recentcnt > 0) { - add_place_raw (dpy, "Recently Used", ""); - _placelist[0].flags |= 4; - } - - add_place_places (dpy, "Home", getenv ("HOME")); - - if (getenv ("HOME")) { - strcpy (tmp, getenv ("HOME")); - strcat (tmp, "/Desktop"); - add_place_places (dpy, "Desktop", tmp); - } - - add_place_places (dpy, "Filesystem", "/"); - - if (_placecnt > 0) spacer = _placecnt -1; - - if (strlen (_fib_cfg_custom_places) > 0) { - parse_gtk_bookmarks (dpy, _fib_cfg_custom_places); - } - - if (read_mtab (dpy, "/proc/mounts") < 1) { - read_mtab (dpy, "/etc/mtab"); - } - - int parsed_bookmarks = 0; - if (!parsed_bookmarks && getenv ("HOME")) { - strcpy (tmp, getenv ("HOME")); - strcat (tmp, "/.gtk-bookmarks"); - if (parse_gtk_bookmarks (dpy, tmp) > 0) { - parsed_bookmarks = 1; - } - } - if (!parsed_bookmarks && getenv ("XDG_CONFIG_HOME")) { - strcpy (tmp, getenv ("XDG_CONFIG_HOME")); - strcat (tmp, "/gtk-3.0/bookmarks"); - if (parse_gtk_bookmarks (dpy, tmp) > 0) { - parsed_bookmarks = 1; - } - } - if (!parsed_bookmarks && getenv ("HOME")) { - strcpy (tmp, getenv ("HOME")); - strcat (tmp, "/.config/gtk-3.0/bookmarks"); - if (parse_gtk_bookmarks (dpy, tmp) > 0) { - parsed_bookmarks = 1; - } - } - if (_fib_place_width > 0) { - _fib_place_width = MIN (_fib_place_width + TEXTSEP + _fib_dir_indent /*extra*/ , PLACESWMAX); - } - if (spacer > 0 && spacer < _placecnt -1) { - _placelist[ spacer ].flags |= 4; - } -} - -static uint8_t font_err = 0; -static int x_error_handler (Display *d, XErrorEvent *e) { - font_err = 1; - return 0; -} - -int x_fib_show (Display *dpy, Window parent, int x, int y) { - if (_fib_win) { - XSetInputFocus (dpy, _fib_win, RevertToParent, CurrentTime); - return -1; - } - - _status = 0; - _rv_open[0] = '\0'; - - Colormap colormap = DefaultColormap (dpy, DefaultScreen (dpy)); - _c_gray1.flags= DoRed | DoGreen | DoBlue; - _c_gray0.red = _c_gray0.green = _c_gray0.blue = 61710; // 95% hover prelight - _c_gray1.red = _c_gray1.green = _c_gray1.blue = 60416; // 93% window bg, scrollbar-fg - _c_gray2.red = _c_gray2.green = _c_gray2.blue = 54016; // 83% button & list bg - _c_gray3.red = _c_gray3.green = _c_gray3.blue = 48640; // 75% heading + scrollbar-bg - _c_gray4.red = _c_gray4.green = _c_gray4.blue = 26112; // 40% prelight text, sep lines - _c_gray5.red = _c_gray5.green = _c_gray5.blue = 12800; // 20% 3D border - _c_gray6.red = _c_gray6.green = _c_gray6.blue = 6400; // 10% checkbox cross, sort triangles - - if (!XAllocColor (dpy, colormap, &_c_gray0)) return -1; - if (!XAllocColor (dpy, colormap, &_c_gray1)) return -1; - if (!XAllocColor (dpy, colormap, &_c_gray2)) return -1; - if (!XAllocColor (dpy, colormap, &_c_gray3)) return -1; - if (!XAllocColor (dpy, colormap, &_c_gray4)) return -1; - if (!XAllocColor (dpy, colormap, &_c_gray5)) return -1; - if (!XAllocColor (dpy, colormap, &_c_gray6)) return -1; - - XSetWindowAttributes attr; - memset (&attr, 0, sizeof(XSetWindowAttributes)); - attr.border_pixel = _c_gray2.pixel; - - attr.event_mask = ExposureMask | KeyPressMask - | ButtonPressMask | ButtonReleaseMask - | ConfigureNotify | StructureNotifyMask - | PointerMotionMask | LeaveWindowMask; - - _fib_win = XCreateWindow ( - dpy, DefaultRootWindow (dpy), - x, y, _fib_width, _fib_height, - 1, CopyFromParent, InputOutput, CopyFromParent, - CWEventMask | CWBorderPixel, &attr); - - if (!_fib_win) { return 1; } - - if (parent) - XSetTransientForHint (dpy, _fib_win, parent); - - XStoreName (dpy, _fib_win, "Select File"); - - Atom wmDelete = XInternAtom (dpy, "WM_DELETE_WINDOW", True); - XSetWMProtocols (dpy, _fib_win, &wmDelete, 1); - - _fib_gc = XCreateGC (dpy, _fib_win, 0, NULL); - XSetLineAttributes (dpy, _fib_gc, 1, LineSolid, CapButt, JoinMiter); - const char dl[1] = {1}; - XSetDashes (dpy, _fib_gc, 0, dl, 1); - - int (*handler)(Display *, XErrorEvent *) = XSetErrorHandler (&x_error_handler); - -#define _XTESTFONT(FN) \ - { \ - font_err = 0; \ - _fibfont = XLoadFont (dpy, FN); \ - XSetFont (dpy, _fib_gc, _fibfont); \ - XSync (dpy, False); \ - } - - font_err = 1; - if (getenv ("XJFONT")) _XTESTFONT (getenv ("XJFONT")); - if (font_err && strlen (_fib_cfg_custom_font) > 0) _XTESTFONT (_fib_cfg_custom_font); - if (font_err) _XTESTFONT ("-*-helvetica-medium-r-normal-*-24-*-*-*-*-*-*-*"); - if (font_err) _XTESTFONT ("-*-verdana-medium-r-normal-*-24-*-*-*-*-*-*-*"); - if (font_err) _XTESTFONT ("-misc-fixed-medium-r-normal-*-24-*-*-*-*-*-*-*"); - if (font_err) _XTESTFONT ("-misc-fixed-medium-r-normal-*-20-*-*-*-*-*-*-*"); - if (font_err) _fibfont = None; - XSync (dpy, False); - XSetErrorHandler (handler); - - if (_fib_font_height == 0) { // 1st time only - query_font_geometry (dpy, _fib_gc, "D ", &_fib_dir_indent, NULL, NULL, NULL); - query_font_geometry (dpy, _fib_gc, "_", &_fib_spc_norm, NULL, NULL, NULL); - if (query_font_geometry (dpy, _fib_gc, "|0Yy", NULL, &_fib_font_height, &_fib_font_ascent, NULL)) { - XFreeGC (dpy, _fib_gc); - XDestroyWindow (dpy, _fib_win); - _fib_win = 0; - return -1; - } - _fib_font_height += 3; - _fib_font_ascent += 2; - _fib_font_vsep = _fib_font_height + 2; - } - - populate_places (dpy); - - strcpy (_btn_ok.text, "Open"); - strcpy (_btn_cancel.text, "Cancel"); - strcpy (_btn_filter.text, "List All Files"); - strcpy (_btn_places.text, "Show Places"); - strcpy (_btn_hidden.text, "Show Hidden"); - - _btn_ok.callback = &cb_open; - _btn_cancel.callback = &cb_cancel; - _btn_filter.callback = &cb_filter; - _btn_places.callback = &cb_places; - _btn_hidden.callback = &cb_hidden; - _btn_filter.flags |= 4; - _btn_places.flags |= 4; - _btn_hidden.flags |= 4; - - if (!_fib_filter_function) { - _btn_filter.flags |= 8; - } - - size_t i; - int btncnt = 0; - _btn_w = 0; - _btn_span = 0; - for (i = 0; i < sizeof(_btns) / sizeof(FibButton*); ++i) { - if (_btns[i]->flags & 8) { continue; } - query_font_geometry (dpy, _fib_gc, _btns[i]->text, &_btns[i]->tw, NULL, NULL, NULL); - if (_btns[i]->flags & 4) { - _btn_span += _btns[i]->tw + _fib_font_ascent + TEXTSEP; - } else { - ++btncnt; - if (_btns[i]->tw > _btn_w) - _btn_w = _btns[i]->tw; - } - } - - _btn_w += BTNPADDING + BTNPADDING + TEXTSEP + TEXTSEP + TEXTSEP; - _btn_span += _btn_w * btncnt + DSEP * (i - 1) + FAREAMRGR + FAREAMRGB; - - for (i = 0; i < sizeof(_btns) / sizeof(FibButton*); ++i) { - if (_btns[i]->flags & 8) { continue; } - if (_btns[i]->flags & 4) { - _btns[i]->xw = _btns[i]->tw + _fib_font_ascent + TEXTSEP; - } else { - _btns[i]->xw = _btn_w; - } - } - - sync_button_states () ; - - _fib_height = _fib_font_vsep * (15.8); - _fib_width = MAX (_btn_span, 440); - - XResizeWindow (dpy, _fib_win, _fib_width, _fib_height); - - XTextProperty x_wname, x_iname; - XSizeHints hints; - XWMHints wmhints; - - hints.flags = PSize | PMinSize; - hints.min_width = _btn_span; - hints.min_height = 8 * _fib_font_vsep; - - char *w_name = & _fib_cfg_title[0]; - - wmhints.input = True; - wmhints.flags = InputHint; - if (XStringListToTextProperty (&w_name, 1, &x_wname) && - XStringListToTextProperty (&w_name, 1, &x_iname)) - { - XSetWMProperties (dpy, _fib_win, &x_wname, &x_iname, NULL, 0, &hints, &wmhints, NULL); - XFree (x_wname.value); - XFree (x_iname.value); - } - - XSetWindowBackground (dpy, _fib_win, _c_gray1.pixel); - - _fib_mapped = 0; - XMapRaised (dpy, _fib_win); - - if (!strlen (_cur_path) || !fib_opendir (dpy, _cur_path, NULL)) { - fib_opendir (dpy, getenv ("HOME") ? getenv ("HOME") : "/", NULL); - } - -#if 0 - XGrabPointer (dpy, _fib_win, True, - ButtonReleaseMask | ButtonPressMask | EnterWindowMask | LeaveWindowMask | PointerMotionMask | StructureNotifyMask, - GrabModeAsync, GrabModeAsync, None, None, CurrentTime); - XGrabKeyboard (dpy, _fib_win, True, GrabModeAsync, GrabModeAsync, CurrentTime); - //XSetInputFocus (dpy, parent, RevertToNone, CurrentTime); -#endif - _recentlock = 1; - return 0; -} - -void x_fib_close (Display *dpy) { - if (!_fib_win) return; - XFreeGC (dpy, _fib_gc); - XDestroyWindow (dpy, _fib_win); - _fib_win = 0; - free (_dirlist); - _dirlist = NULL; - free (_pathbtn); - _pathbtn = NULL; - if (_fibfont != None) XUnloadFont (dpy, _fibfont); - _fibfont = None; - free (_placelist); - _placelist = NULL; - _dircount = 0; - _pathparts = 0; - _placecnt = 0; - if (_pixbuffer != None) XFreePixmap (dpy, _pixbuffer); - _pixbuffer = None; - Colormap colormap = DefaultColormap (dpy, DefaultScreen (dpy)); - XFreeColors (dpy, colormap, &_c_gray0.pixel, 1, 0); - XFreeColors (dpy, colormap, &_c_gray1.pixel, 1, 0); - XFreeColors (dpy, colormap, &_c_gray2.pixel, 1, 0); - XFreeColors (dpy, colormap, &_c_gray3.pixel, 1, 0); - XFreeColors (dpy, colormap, &_c_gray4.pixel, 1, 0); - XFreeColors (dpy, colormap, &_c_gray5.pixel, 1, 0); - XFreeColors (dpy, colormap, &_c_gray6.pixel, 1, 0); - _recentlock = 0; -} - -int x_fib_handle_events (Display *dpy, XEvent *event) { - if (!_fib_win) return 0; - if (_status) return 0; - if (event->xany.window != _fib_win) { - return 0; - } - - switch (event->type) { - case MapNotify: - _fib_mapped = 1; - break; - case UnmapNotify: - _fib_mapped = 0; - break; - case LeaveNotify: - fib_update_hover (dpy, 1, 0, 0); - break; - case ClientMessage: - if (!strcmp (XGetAtomName (dpy, event->xclient.message_type), "WM_PROTOCOLS")) { - _status = -1; - } - break; - case ConfigureNotify: - if ( - (event->xconfigure.width > 1 && event->xconfigure.height > 1) - && - (event->xconfigure.width != _fib_width || event->xconfigure.height != _fib_height) - ) - { - _fib_width = event->xconfigure.width; - _fib_height = event->xconfigure.height; - _fib_resized = 1; - } - break; - case Expose: - if (event->xexpose.count == 0) { - fib_expose (dpy, event->xany.window); - } - break; - case MotionNotify: - fib_motion (dpy, event->xmotion.x, event->xmotion.y); - if (event->xmotion.is_hint == NotifyHint) { - XGetMotionEvents (dpy, event->xany.window, CurrentTime, CurrentTime, NULL); - } - break; - case ButtonPress: - fib_mousedown (dpy, event->xbutton.x, event->xbutton.y, event->xbutton.button, event->xbutton.time); - break; - case ButtonRelease: - fib_mouseup (dpy, event->xbutton.x, event->xbutton.y, event->xbutton.button, event->xbutton.time); - break; - case KeyRelease: - break; - case KeyPress: - { - KeySym key; - char buf[100]; - static XComposeStatus stat; - XLookupString (&event->xkey, buf, sizeof(buf), &key, &stat); - switch (key) { - case XK_Escape: - _status = -1; - break; - case XK_Up: - if (_fsel > 0) { - fib_select (dpy, _fsel - 1); - } - break; - case XK_Down: - if (_fsel < _dircount -1) { - fib_select ( dpy, _fsel + 1); - } - break; - case XK_Page_Up: - if (_fsel > 0) { - int llen = (_fib_height - LISTBOT * _fib_font_vsep) / _fib_font_vsep; - if (llen < 1) llen = 1; else --llen; - int fs = MAX (0, _fsel - llen); - fib_select ( dpy, fs); - } - break; - case XK_Page_Down: - if (_fsel < _dircount) { - int llen = (_fib_height - LISTBOT * _fib_font_vsep) / _fib_font_vsep; - if (llen < 1) llen = 1; else --llen; - int fs = MIN (_dircount - 1, _fsel + llen); - fib_select ( dpy, fs); - } - break; - case XK_Left: - if (_pathparts > 1) { - int i = 0; - char path[1024] = "/"; - while (++i < _pathparts - 1) { - strcat (path, _pathbtn[i].name); - strcat (path, "/"); - } - char *sel = strdup (_pathbtn[_pathparts-1].name); - fib_opendir (dpy, path, sel); - free (sel); - } - break; - case XK_Right: - if (_fsel >= 0 && _fsel < _dircount) { - if (_dirlist[_fsel].flags & 4) { - cb_open (dpy); - } - } - break; - case XK_Return: - cb_open (dpy); - break; - default: - if ((key >= XK_a && key <= XK_z) || (key >= XK_0 && key <= XK_9)) { - int i; - for (i = 0; i < _dircount; ++i) { - int j = (_fsel + i + 1) % _dircount; - char kcmp = _dirlist[j].name[0]; - if (kcmp > 0x40 && kcmp <= 0x5A) kcmp |= 0x20; - if (kcmp == (char)key) { - fib_select ( dpy, j); - break; - } - } - } - break; - } - } - break; - } - - if (_status) { - x_fib_close (dpy); - } - return _status; -} - -int x_fib_status () { - return _status; -} - -int x_fib_configure (int k, const char *v) { - if (_fib_win) { return -1; } - switch (k) { - case 0: - if (strlen (v) >= sizeof(_cur_path) -1) return -2; - if (strlen (v) < 1) return -2; - if (v[0] != '/') return -2; - if (strstr (v, "//")) return -2; - strncpy (_cur_path, v, sizeof(_cur_path)); - break; - case 1: - if (strlen (v) >= sizeof(_fib_cfg_title) -1) return -2; - strncpy (_fib_cfg_title, v, sizeof(_fib_cfg_title)); - break; - case 2: - if (strlen (v) >= sizeof(_fib_cfg_custom_font) -1) return -2; - strncpy (_fib_cfg_custom_font, v, sizeof(_fib_cfg_custom_font)); - break; - case 3: - if (strlen (v) >= sizeof(_fib_cfg_custom_places) -1) return -2; - strncpy (_fib_cfg_custom_places, v, sizeof(_fib_cfg_custom_places)); - break; - default: - return -2; - } - return 0; -} - -int x_fib_cfg_buttons (int k, int v) { - if (_fib_win) { return -1; } - switch (k) { - case 1: - if (v < 0) { - _btn_hidden.flags |= 8; - } else { - _btn_hidden.flags &= ~8; - } - if (v == 1) { - _btn_hidden.flags |= 2; - _fib_hidden_fn = 1; - } else if (v == 0) { - _btn_hidden.flags &= 2; - _fib_hidden_fn = 0; - } - break; - case 2: - if (v < 0) { - _btn_places.flags |= 8; - } else { - _btn_places.flags &= ~8; - } - if (v == 1) { - _btn_places.flags |= 2; - _fib_show_places = 1; - } else if (v == 0) { - _btn_places.flags &= ~2; - _fib_show_places = 0; - } - break; - case 3: - // NB. filter button is automatically hidden - // IFF the filter-function is NULL. - if (v < 0) { - _btn_filter.flags |= 8; - } else { - _btn_filter.flags &= ~8; - } - if (v == 1) { - _btn_filter.flags &= ~2; // inverse - 'show all' = !filter - _fib_filter_fn = 1; - } else if (v == 0) { - _btn_filter.flags |= 2; - _fib_filter_fn = 0; - } - break; - default: - return -2; - } - return 0; -} - -int x_fib_cfg_filter_callback (int (*cb)(const char*)) { - if (_fib_win) { return -1; } - _fib_filter_function = cb; - return 0; -} - -char *x_fib_filename () { - if (_status > 0 && !_fib_win) - return strdup (_rv_open); - else - return NULL; -} -#endif // HAVE_X11 - - -/* example usage */ -#ifdef SOFD_TEST - -static int fib_filter_movie_filename (const char *name) { - if (!_fib_filter_fn) return 1; - const int l3 = strlen (name) - 3; - const int l4 = l3 - 1; - const int l5 = l4 - 1; - const int l6 = l5 - 1; - const int l9 = l6 - 3; - if ( - (l4 > 0 && ( - !strcasecmp (&name[l4], ".avi") - || !strcasecmp (&name[l4], ".mov") - || !strcasecmp (&name[l4], ".ogg") - || !strcasecmp (&name[l4], ".ogv") - || !strcasecmp (&name[l4], ".mpg") - || !strcasecmp (&name[l4], ".mov") - || !strcasecmp (&name[l4], ".mp4") - || !strcasecmp (&name[l4], ".mkv") - || !strcasecmp (&name[l4], ".vob") - || !strcasecmp (&name[l4], ".asf") - || !strcasecmp (&name[l4], ".avs") - || !strcasecmp (&name[l4], ".dts") - || !strcasecmp (&name[l4], ".flv") - || !strcasecmp (&name[l4], ".m4v") - )) || - (l5 > 0 && ( - !strcasecmp (&name[l5], ".h264") - || !strcasecmp (&name[l5], ".webm") - )) || - (l6 > 0 && ( - !strcasecmp (&name[l6], ".dirac") - )) || - (l9 > 0 && ( - !strcasecmp (&name[l9], ".matroska") - )) || - (l3 > 0 && ( - !strcasecmp (&name[l3], ".dv") - || !strcasecmp (&name[l3], ".ts") - )) - ) - { - return 1; - } - return 0; -} - -int main (int argc, char **argv) { - Display* dpy = XOpenDisplay (0); - if (!dpy) return -1; - - x_fib_cfg_filter_callback (fib_filter_movie_filename); - x_fib_configure (1, "Open Movie File"); - x_fib_load_recent ("/tmp/sofd.recent"); - x_fib_show (dpy, 0, 300, 300); - - while (1) { - XEvent event; - while (XPending (dpy) > 0) { - XNextEvent (dpy, &event); - if (x_fib_handle_events (dpy, &event)) { - if (x_fib_status () > 0) { - char *fn = x_fib_filename (); - printf ("OPEN '%s'\n", fn); - x_fib_add_recent (fn, time (NULL)); - free (fn); - } - } - } - if (x_fib_status ()) { - break; - } - usleep (80000); - } - x_fib_close (dpy); - - x_fib_save_recent ("/tmp/sofd.recent"); - - x_fib_free_recent (); - XCloseDisplay (dpy); - return 0; -} -#endif diff --git a/contrib/libsofd.h b/contrib/libsofd.h deleted file mode 100644 index 2abf788..0000000 --- a/contrib/libsofd.h +++ /dev/null @@ -1,194 +0,0 @@ -/* libSOFD - Simple Open File Dialog [for X11 without toolkit] - * - * Copyright (C) 2014 Robin Gareus - * - * 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. - */ - -#ifndef LIBSOFD_H -#define LIBSOFD_H 1 - -#ifdef HAVE_X11 -#include - -#ifdef __cplusplus -extern "C" { -#endif - -/////////////////////////////////////////////////////////////////////////////// -/* public API */ - -/** open a file select dialog - * @param dpy X Display connection - * @param parent (optional) if not NULL, become transient for given window - * @param x if >0 set explict initial width of the window - * @param y if >0 set explict initial height of the window - * @return 0 on success - */ -int x_fib_show (Display *dpy, Window parent, int x, int y); - -/** force close the dialog. - * This is normally not needed, the dialog closes itself - * when a file is selected or the user cancels selection. - * @param dpy X Display connection - */ -void x_fib_close (Display *dpy); - -/** non-blocking X11 event handler. - * It is safe to run this function even if the dialog is - * closed or was not initialized. - * - * @param dpy X Display connection - * @param event the XEvent to process - * @return status - * 0: the event was not for this window, or file-dialog still - * active, or the dialog window is not displayed. - * >0: file was selected, dialog closed - * <0: file selection was cancelled. - */ -int x_fib_handle_events (Display *dpy, XEvent *event); - -/** last status of the dialog - * @return >0: file was selected, <0: canceled or inactive. 0: active - */ -int x_fib_status (); - -/** query the selected filename - * @return NULL if none set, or allocated string to be free()ed by the called - */ -char *x_fib_filename (); - -/** customize/configure the dialog before calling \ref x_fib_show - * changes only have any effect if the dialog is not visible. - * @param k key to change - * 0: set current dir to display (must end with slash) - * 1: set title of dialog window - * 2: specify a custom X11 font to use - * 3: specify a custom 'places' file to include - * (following gtk-bookmark convention) - * @param v value - * @return 0 on success. - */ -int x_fib_configure (int k, const char *v); - -/** customize/configure the dialog before calling \ref x_fib_show - * changes only have any effect if the dialog is not visible. - * - * @param k button to change: - * 1: show hidden files - * 2: show places - * 3: show filter/list all (automatically hidden if there is no - * filter function) - * @param v <0 to hide the button >=0 show button, - * 0: set button-state to not-checked - * 1: set button-state to checked - * >1: retain current state - * @return 0 on success. - */ -int x_fib_cfg_buttons (int k, int v); - -/** set custom callback to filter file-names. - * NULL will disable the filter and hide the 'show all' button. - * changes only have any effect if the dialog is not visible. - * - * @param cb callback function to check file - * the callback function is called with the file name (basename only) - * and is expected to return 1 if the file passes the filter - * and 0 if the file should not be listed by default. - * @return 0 on success. - */ -int x_fib_cfg_filter_callback (int (*cb)(const char*)); - -#ifdef __cplusplus -} -#endif - -#endif /* END X11 specific functions */ - -#ifdef __cplusplus -extern "C" { -#endif - -/* 'recently used' API. x-platform - * NOTE: all functions use a static cache and are not reentrant. - * It is expected that none of these functions are called in - * parallel from different threads. - */ - -/** release static resources of 'recently used files' - */ -void x_fib_free_recent (); - -/** add an entry to the recently used list - * - * The dialog does not add files automatically on open, - * if the application succeeds to open a selected file, - * this function should be called. - * - * @param path complete path to file - * @param atime time of last use, 0: NOW - * @return -1 on error, number of current entries otherwise - */ -int x_fib_add_recent (const char *path, time_t atime); - -/** get a platform specific path to a good location for - * saving the recently used file list. - * (follows XDG_DATA_HOME on Unix, and CSIDL_LOCAL_APPDATA spec) - * - * @param application-name to use to include in file - * @return pointer to static path or NULL - */ -const char *x_fib_recent_file(const char *appname); - -/** save the current list of recently used files to the given filename - * (the format is one file per line, filename URL encoded and space separated - * with last-used timestamp) - * - * This function tries to creates the containing directory if it does - * not exist. - * - * @param fn file to save the list to - * @return 0: on success - */ -int x_fib_save_recent (const char *fn); - -/** load a recently used file list. - * - * @param fn file to load the list from - * @return 0: on success - */ -int x_fib_load_recent (const char *fn); - -/** get number of entries in the current list - * @return number of entries in the recently used list - */ -unsigned int x_fib_recent_count (); - -/** get recently used entry at given position - * - * @param i entry to query - * @return pointer to static string - */ -const char *x_fib_recent_at (unsigned int i); - -#ifdef __cplusplus -} -#endif - -#endif // header guard diff --git a/contrib/mii-icon-64.h b/contrib/mii-icon-64.h new file mode 100644 index 0000000..7af920c --- /dev/null +++ b/contrib/mii-icon-64.h @@ -0,0 +1,580 @@ +/* this file is auto-generated by icon-convert-tcc.c */ +static const unsigned long mii_icon64[] = { + 64,64, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x01000000, + 0x01000000, 0x039e9ea1, 0x04bebfc1, 0x04c9cacd, 0x04c9cacd, 0x05ccced0, 0x05ccced0, 0x05cfcfd1, + 0x05d0d1d3, 0x06d2d3d6, 0x06d3d4d6, 0x06d2d3d6, 0x06d4d5d7, 0x0778a88e, 0x07499b67, 0x07499b67, + 0x07499b67, 0x07499b67, 0x07499b67, 0x074a9b67, 0x06499b65, 0x064a9c65, 0x064a9c64, 0x064a9c63, + 0x064a9b61, 0x064a9c5f, 0x064b9d5e, 0x054b9c5d, 0x054b9c5c, 0x054c9c55, 0x044d9a48, 0x01000000, + 0x01000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x2ff1f3f7, 0x7bf4f6fa, 0xaff4f7fb, 0xd0f4f7fb, 0xf2f5f7fb, + 0xfdf5f8fc, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f8fc, + 0xfff5f8fc, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f8fc, 0xffc2ddc8, 0xff5fba4f, 0xff60ba4e, + 0xff60ba4e, 0xff60ba4e, 0xff60ba4e, 0xff60ba4e, 0xff60ba4e, 0xff60ba4e, 0xff60ba4e, 0xff60ba4e, + 0xff60ba4e, 0xff60ba4e, 0xff60ba4e, 0xff60ba4e, 0xff60ba4e, 0xff60ba4e, 0xff60ba4f, 0xfe5fba52, + 0xf65fba53, 0xd35fba52, 0xb15eb955, 0x7c5db85c, 0x315ab663, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x51f3f5f9, 0xcef5f7fb, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f8fc, + 0xfff5f7fb, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f7fb, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f8fc, + 0xfff5f8fc, 0xfff5f7fb, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f7fb, 0xfff1f5f7, 0xff70be68, 0xff61bb46, + 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, + 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, + 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff60bb4b, 0xcd5eb954, 0x515bb75f, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x13edeff3, + 0xb4f4f7fb, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f8fc, + 0xfff5f8fc, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f8fc, + 0xfff5f8fc, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f8fc, 0xffbad9be, 0xff61bb47, + 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, + 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, + 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff60bb49, 0xb25db959, + 0x1358b361, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x20eef1f4, 0xdff5f7fb, + 0xfff5f8fc, 0xfff5f8fc, 0xfff5f7fb, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f7fb, 0xfff5f8fc, 0xfff5f8fc, + 0xfff5f7fb, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f7fb, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f7fb, 0xfff5f8fc, + 0xfff5f8fc, 0xfff5f7fb, 0xfff5f8fc, 0xfff5f8fc, 0xfff5f7fb, 0xfff5f8fc, 0xffecf3f2, 0xff61ba52, + 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, + 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, + 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, + 0xdd5eb953, 0x1f59b465, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x14eaecf0, 0xdff5f7fb, 0xfff5f7fb, + 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, + 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, + 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xfff5f8fb, 0xffa2cfa3, + 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, + 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, + 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, + 0xff61bb46, 0xdd5eb954, 0x1256b062, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x01000000, 0xb4f4f6fa, 0xfff5f7fb, 0xfff5f7fb, + 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, + 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, + 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xfff5f7fb, 0xffd6e6dc, + 0xff60bb48, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, + 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, + 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, + 0xff61bb46, 0xff61bb46, 0xb05db858, 0x01000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x4feff1f5, 0xfff5f7fa, 0xfff5f7fb, 0xfff4f7fa, + 0xfff5f7fa, 0xfff5f7fa, 0xfff5f7fb, 0xfff5f7fa, 0xfff5f7fa, 0xfff5f7fb, 0xfff5f7fa, 0xfff5f7fb, + 0xfff4f7fa, 0xfff5f7fa, 0xfff5f7fa, 0xfff5f7fb, 0xfff5f7fa, 0xfff5f7fa, 0xfff5f7fb, 0xfff5f7fa, + 0xfff5f7fb, 0xfff4f7fa, 0xfff5f7fa, 0xfff5f7fa, 0xfff5f7fb, 0xfff5f7fa, 0xfff5f7fa, 0xfff5f7fa, + 0xff75bf6e, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, + 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, + 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, + 0xff61bb46, 0xff61bb46, 0xff60bb4a, 0x4959b460, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x00000000, 0x01000000, 0xccf3f5f9, 0xfff5f7fa, 0xfff5f7fa, 0xfff5f7fa, + 0xfff5f7fa, 0xfff5f7fa, 0xfff5f7fa, 0xfff5f7fa, 0xfff5f7fa, 0xfff5f6fa, 0xfff5f7fa, 0xfff5f7fa, + 0xfff5f7fa, 0xfff5f7fa, 0xfff5f7fa, 0xfff5f7fa, 0xfff5f7fa, 0xfff5f7fa, 0xfff5f6fa, 0xfff5f7fa, + 0xfff5f7fa, 0xfff5f7fa, 0xfff5f7fa, 0xfff5f7fa, 0xfff5f7fa, 0xfff5f7fa, 0xfff5f7fa, 0xfff5f7fa, + 0xffb3d6b5, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, + 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, + 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xff61bb46, + 0xff61bb46, 0xff61bb46, 0xff61bb46, 0xc85db854, 0x01000000, 0x00000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x00000000, 0x2ce8e9ed, 0xfff5f6fa, 0xfff5f6fa, 0xfff5f6fa, 0xfff4f6fa, + 0xfff5f6fa, 0xfff5f6fa, 0xfff5f6fa, 0xfff5f6fa, 0xfff5f6fa, 0xfff5f6fa, 0xfff5f6fa, 0xfff5f6fa, + 0xfff4f6fa, 0xfff5f6fa, 0xfff5f6fa, 0xfff5f6fa, 0xfff5f6fa, 0xfff5f6fa, 0xfff5f6fa, 0xfff5f6fa, + 0xfff5f6fa, 0xfff4f6fa, 0xfff5f6fa, 0xfff5f6fa, 0xfff5f6fa, 0xfff5f6fa, 0xfff5f6fa, 0xfff5f6fa, + 0xffdbe9e1, 0xff60ba49, 0xff61bb47, 0xff61bb47, 0xff61bb47, 0xff61bb47, 0xff61bb47, 0xff61bb47, + 0xff61bb47, 0xff61bb47, 0xff61bb47, 0xff61bb47, 0xff61bb47, 0xff61bb47, 0xff61bb47, 0xff61bb47, + 0xff61bb47, 0xff61bb47, 0xff61bb47, 0xff61bb47, 0xff61bb47, 0xff61bb47, 0xff61bb47, 0xff61bb47, + 0xff61bb47, 0xff61bb47, 0xff61bb47, 0xff60ba4c, 0x2854ae62, 0x00000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x00000000, 0x76f0f1f4, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, + 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, + 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, + 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, + 0xfff5f6f9, 0xffc8b863, 0xffcbb440, 0xffccb540, 0xffccb540, 0xffccb540, 0xffccb540, 0xffccb540, + 0xffccb540, 0xffccb540, 0xffccb540, 0xffccb540, 0xffccb540, 0xffccb540, 0xffccb540, 0xffccb540, + 0xffccb540, 0xffccb540, 0xffccb540, 0xffccb540, 0xffccb540, 0xffccb540, 0xffccb540, 0xffccb540, + 0xffccb540, 0xffccb540, 0xffccb540, 0xffccb541, 0x70cdaf48, 0x00000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x01000000, 0xa7f1f2f5, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, + 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, + 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, + 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, 0xfff5f6f9, + 0xfff5f6f9, 0xfffad0a0, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, + 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, + 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, + 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xa2f9b325, 0x01000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x02000000, 0xcef2f3f6, 0xfff4f5f9, 0xfff4f5f9, 0xfff4f5f9, 0xfff4f5f8, + 0xfff4f5f9, 0xfff4f5f9, 0xfff4f5f9, 0xfff4f5f9, 0xfff4f5f9, 0xfff4f5f9, 0xfff4f5f9, 0xfff4f5f9, + 0xfff4f5f8, 0xfff4f5f9, 0xfff4f5f9, 0xfff4f5f9, 0xfff4f5f9, 0xfff4f5f9, 0xfff4f5f9, 0xfff4f5f9, + 0xfff4f5f9, 0xfff4f5f8, 0xfff4f5f9, 0xfff4f5f9, 0xfff4f5f9, 0xfff4f5f9, 0xfff4f5f9, 0xfff4f5f9, + 0xfff4f5f9, 0xfff8e3d0, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, + 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, + 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, + 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xcafbb425, 0x02000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x04000000, 0xdef2f3f6, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, + 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, + 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, + 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, + 0xfff4f5f8, 0xfff6f2f0, 0xfffcb625, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, + 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, + 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, + 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xd7fab425, 0x03000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x05000000, 0xeff2f3f6, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, + 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, + 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, + 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, + 0xfff4f5f8, 0xfff5f5f8, 0xfffbc376, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, + 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, + 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, + 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xe6fbb526, 0x05000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x07000000, 0xf8f3f4f7, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, + 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, + 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, + 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, 0xfff4f5f8, + 0xfff4f5f8, 0xfff4f5f8, 0xfff9d4ad, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, + 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, + 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, + 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xf3fcb525, 0x06000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x07000000, 0xfdf3f5f7, 0xfff3f4f7, 0xfff3f5f8, 0xfff3f5f8, 0xfff3f4f7, + 0xfff4f5f8, 0xfff3f5f8, 0xfff3f4f7, 0xfff3f5f8, 0xfff3f5f8, 0xfff3f4f7, 0xfff3f5f8, 0xfff3f5f8, + 0xfff3f4f7, 0xfff4f5f8, 0xfff3f5f8, 0xfff3f4f7, 0xfff3f5f8, 0xfff3f5f8, 0xfff3f4f7, 0xfff3f5f8, + 0xfff3f5f8, 0xfff3f4f7, 0xfff4f5f8, 0xfff3f5f8, 0xfff3f4f7, 0xfff3f5f8, 0xfff3f5f8, 0xfff3f4f7, + 0xfff3f5f8, 0xfff3f5f8, 0xfff7e2cf, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, + 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb726, 0xfffdb827, 0xfffdb827, + 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, + 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xf9fcb526, 0x06000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x08000000, 0xfff3f5f8, 0xfff3f5f8, 0xfff3f5f8, 0xfff3f5f8, 0xfff3f5f7, + 0xfff3f5f7, 0xfff3f5f7, 0xfff3f5f8, 0xfff3f5f8, 0xfff3f5f8, 0xfff3f5f8, 0xfff3f5f8, 0xfff3f5f8, + 0xfff3f5f7, 0xfff3f5f7, 0xffe7e8eb, 0xff88898a, 0xff939496, 0xffeeeff2, 0xfff3f5f8, 0xfff3f5f8, + 0xfff3f5f8, 0xfff3f5f7, 0xfff3f5f7, 0xfff3f5f7, 0xfff3f5f8, 0xfff3f5f8, 0xfff3f5f8, 0xfff3f5f8, + 0xfff3f5f8, 0xfff3f5f8, 0xfff5efea, 0xfffcb726, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, + 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xffc58919, 0xff82540b, 0xffd7981d, 0xfffdb827, + 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, + 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfcfdb626, 0x07000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x08000000, 0xfff3f5f8, 0xfff3f5f7, 0xfff3f5f7, 0xfff3f5f7, 0xfff3f5f7, + 0xfff3f5f7, 0xfff3f5f7, 0xfff3f5f7, 0xfff3f5f7, 0xfff3f5f7, 0xfff3f5f7, 0xfff3f5f7, 0xfff3f5f7, + 0xfff3f5f7, 0xfff3f5f7, 0xffc9cbcd, 0xff616162, 0xff616162, 0xffd9dbdd, 0xfff3f5f7, 0xfff3f5f7, + 0xfff3f5f7, 0xfff3f5f7, 0xfff3f5f7, 0xfff3f5f7, 0xfff3f5f7, 0xfff3f5f7, 0xfff3f5f7, 0xfff3f5f7, + 0xfff3f5f7, 0xfff3f5f7, 0xfff4f6f8, 0xfffbbb53, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, + 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb726, 0xff784a0b, 0xff653b08, 0xff9d6912, 0xfffdb827, + 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, + 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb827, 0xfffdb626, 0x07000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x08000000, 0xfff3f4f7, 0xfff3f4f7, 0xfff3f4f7, 0xfff3f4f7, 0xfff3f4f7, + 0xfff3f4f7, 0xfff3f4f7, 0xfff3f4f7, 0xfff3f4f7, 0xfff3f4f7, 0xfff3f4f7, 0xfff3f4f7, 0xfff3f4f7, + 0xfff3f4f7, 0xfff3f4f7, 0xffc9cbcd, 0xff616162, 0xff616162, 0xffd7d9db, 0xfff3f4f7, 0xfff3f4f7, + 0xfff3f4f7, 0xfff3f4f7, 0xfff3f4f7, 0xfff3f4f7, 0xfff3f4f7, 0xfff3f4f7, 0xfff3f4f7, 0xfff3f4f7, + 0xfff3f4f7, 0xfff3f4f7, 0xfff3f4f7, 0xfff9c88d, 0xfffdb626, 0xfffdb626, 0xfffdb626, 0xfffdb626, + 0xfffdb626, 0xfffdb626, 0xfffdb626, 0xfffdb425, 0xfe76460f, 0xfe65370d, 0xfe9a6414, 0xfffdb525, + 0xfffdb626, 0xfffdb626, 0xfffdb626, 0xfffdb626, 0xfffdb626, 0xfffdb626, 0xfffdb626, 0xfffdb626, + 0xfffdb626, 0xfffdb626, 0xfffdb626, 0xfffdb626, 0xfffdb424, 0x08000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x08000000, 0xfff3f4f6, 0xfff2f4f6, 0xfff2f4f6, 0xfff2f4f6, 0xfff2f4f6, + 0xfff3f4f7, 0xfff2f4f6, 0xfff2f4f6, 0xfff2f4f6, 0xfff2f4f6, 0xfff2f4f6, 0xfff2f4f6, 0xfff2f4f6, + 0xfff2f4f6, 0xfff3f4f7, 0xffc9cacc, 0xff606162, 0xff606162, 0xffd7d9db, 0xfff2f4f6, 0xfff2f4f6, + 0xfff2f4f6, 0xfff2f4f6, 0xfff3f4f7, 0xfff2f4f6, 0xfff2f4f6, 0xfff2f4f6, 0xfff2f4f6, 0xfff2f4f6, + 0xfff2f4f6, 0xfff2f4f6, 0xfff2f4f6, 0xfff2bcb3, 0xfff86c3f, 0xfffd6e41, 0xfffe6e42, 0xfffe6e42, + 0xfffe6e42, 0xfffe6e42, 0xfffe6e42, 0xfffe6e42, 0xfe723618, 0xfe653113, 0xfe974425, 0xfffe6e42, + 0xfffe6e42, 0xfffe6e42, 0xfffe6e42, 0xfffe6e42, 0xfffe6e42, 0xfffe6e42, 0xfffe6e42, 0xfffe6e42, + 0xfffe6e42, 0xfffe6e42, 0xfffe6e42, 0xfffe6e42, 0xfffe6e43, 0x08000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x08000000, 0xfff2f4f6, 0xfff2f4f6, 0xfff2f4f6, 0xfff2f4f6, 0xfff2f4f6, + 0xfff2f4f6, 0xfff2f4f6, 0xfff2f4f6, 0xfff2f4f6, 0xfff2f4f6, 0xfff2f4f6, 0xfff2f4f6, 0xfff2f4f6, + 0xfff2f4f6, 0xfff2f4f6, 0xffc9cacc, 0xff606162, 0xff606162, 0xffd7d8da, 0xfff2f4f6, 0xfff2f4f6, + 0xfff2f4f6, 0xfff2f4f6, 0xfff2f4f6, 0xfff2f4f6, 0xfff2f4f6, 0xfff2f4f6, 0xfff2f4f6, 0xfff2f4f6, + 0xfff2f4f6, 0xfff2f4f6, 0xfff2f4f6, 0xfff2cdcb, 0xfff56240, 0xfffd6542, 0xfffe6643, 0xfffe6643, + 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6744, 0xff72311c, 0xff652c17, 0xff973f27, 0xfffe6643, + 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, + 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6745, 0x08000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x08000000, 0xfff2f3f6, 0xfff2f3f6, 0xfff2f3f6, 0xfff2f3f6, 0xfff2f3f6, + 0xfff2f4f6, 0xfff2f3f6, 0xfff2f3f6, 0xfff2f4f6, 0xfff2f3f6, 0xfff2f3f6, 0xfff2f3f6, 0xfff2f3f6, + 0xfff2f3f6, 0xfff2f4f6, 0xffd1d2d4, 0xff606162, 0xff606162, 0xffe1e3e5, 0xfff2f3f6, 0xfff2f3f6, + 0xfff2f3f6, 0xfff2f3f6, 0xfff2f4f6, 0xfff2f3f6, 0xfff2f3f6, 0xfff2f4f6, 0xfff2f3f6, 0xfff2f3f6, + 0xfff2f3f6, 0xfff2f3f6, 0xfff2f3f6, 0xfff3e0e0, 0xfff36140, 0xfffc6542, 0xfffe6643, 0xfffe6643, + 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xff8d3c27, 0xff65291a, 0xffac4931, 0xfffe6643, + 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, + 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6745, 0x08000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x08000000, 0xfff2f3f5, 0xfff2f3f5, 0xfff2f3f5, 0xfff2f3f5, 0xfff2f3f5, + 0xfff2f3f5, 0xfff2f3f5, 0xfff2f3f5, 0xfff2f3f5, 0xfff2f3f5, 0xfff2f3f5, 0xfff2f3f5, 0xfff2f3f5, + 0xfff2f3f5, 0xfff2f3f5, 0xfff2f3f5, 0xffd2d3d5, 0xffd8d9db, 0xfff2f3f5, 0xfff2f3f5, 0xfff2f3f5, + 0xfff2f3f5, 0xfff2f3f5, 0xfff2f3f5, 0xfff2f3f5, 0xfff2f3f5, 0xfff2f3f5, 0xfff2f3f5, 0xfff2f3f5, + 0xfff2f3f5, 0xfff2f3f5, 0xfff2f3f5, 0xfff4eff0, 0xfff0603f, 0xfffb6442, 0xfffe6643, 0xfffe6643, + 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfff16646, 0xffd35d42, 0xfff86847, 0xfffe6643, + 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, + 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6745, 0x08000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x08000000, 0xfff2f3f5, 0xfff1f3f5, 0xfff1f3f5, 0xfff2f3f5, 0xfff1f3f5, + 0xfff2f3f5, 0xfff1f3f5, 0xfff2f3f5, 0xfff2f3f5, 0xfff2f3f5, 0xfff1f3f5, 0xfff1f3f5, 0xfff2f3f5, + 0xfff1f3f5, 0xfff2f3f5, 0xfff1f3f5, 0xfff2f3f5, 0xfff2f3f5, 0xfff2f3f5, 0xfff1f3f5, 0xfff1f3f5, + 0xfff2f3f5, 0xfff1f3f5, 0xfff2f3f5, 0xfff1f3f5, 0xfff2f3f5, 0xfff2f3f5, 0xfff2f3f5, 0xfff1f3f5, + 0xfff1f3f5, 0xfff2f3f5, 0xfff1f3f5, 0xfff3f5f6, 0xffef6b51, 0xfffa6441, 0xfffe6643, 0xfffe6643, + 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, + 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, + 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6745, 0x08000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x08000000, 0xfff2f3f5, 0xfff1f3f5, 0xfff1f3f5, 0xfff2f3f5, 0xfff1f3f5, + 0xfff1f3f5, 0xfff2f3f5, 0xfff1f3f5, 0xfff1f3f5, 0xfff2f3f5, 0xfff1f3f5, 0xfff1f3f5, 0xfff2f3f5, + 0xfff1f3f5, 0xfff1f3f5, 0xfff2f3f5, 0xfff1f3f5, 0xfff1f3f5, 0xfff2f3f5, 0xfff1f3f5, 0xfff1f3f5, + 0xfff2f3f5, 0xfff1f3f5, 0xfff1f3f5, 0xfff2f3f5, 0xfff1f3f5, 0xfff1f3f5, 0xfff2f3f5, 0xfff1f3f5, + 0xfff1f3f5, 0xfff2f3f5, 0xfff1f3f5, 0xfff2f3f5, 0xfff08f82, 0xfff96441, 0xfffe6643, 0xfffe6643, + 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, + 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, + 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6745, 0x08000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x08000000, 0xfff1f2f5, 0xfff1f2f4, 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f4, + 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f4, 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f4, 0xfff1f2f5, 0xfff1f2f5, + 0xfff1f2f4, 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f4, 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f4, 0xfff1f2f5, + 0xfff1f2f5, 0xfff1f2f4, 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f4, 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f4, + 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f4, 0xfff2f3f5, 0xffef9f95, 0xfff86341, 0xfffd6643, 0xfffe6643, + 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, + 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, + 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6745, 0x08000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x08000000, 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f5, + 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f5, + 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f5, + 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f5, + 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f5, 0xfff1f2f5, 0xfff0b1ab, 0xfff66340, 0xfffd6542, 0xfffe6643, + 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, + 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, + 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6745, 0x08000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x08000000, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, + 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, + 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, + 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, + 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, 0xffefbab5, 0xfff56240, 0xfffd6542, 0xfffe6643, + 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, + 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, + 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6643, 0xfffe6745, 0x08000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x08000000, 0xfff1f2f4, 0xfff1f1f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f1f4, + 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f1f4, 0xfff1f2f4, 0xfff1f2f4, + 0xfff1f1f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f1f4, 0xfff1f2f4, + 0xfff1f2f4, 0xfff1f1f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f2f4, 0xfff1f1f4, + 0xfff1f2f4, 0xfff1f2f4, 0xfff1f1f4, 0xfff1f2f4, 0xfeedc8c6, 0xfdeb5a41, 0xfdf35e43, 0xfdf55f44, + 0xfdf55f44, 0xfdf55f44, 0xfdf55f44, 0xfdf55f44, 0xfdf55f44, 0xfdf55f44, 0xfdf55f44, 0xfdf55f44, + 0xfdf55f44, 0xfdf55f44, 0xfdf55f44, 0xfdf55f44, 0xfdf55f44, 0xfdf55f44, 0xfdf55f44, 0xfdf55f44, + 0xfdf55f44, 0xfdf55f44, 0xfdf55f44, 0xfdf55f44, 0xfef56046, 0x08000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x08000000, 0xfff1f1f4, 0xfff1f1f3, 0xfff1f1f4, 0xfff1f1f3, 0xfff1f1f4, + 0xfff1f1f4, 0xfff1f1f4, 0xfff1f1f3, 0xfff1f1f4, 0xfff1f1f4, 0xfff1f1f3, 0xfff1f1f4, 0xfff1f1f3, + 0xfff1f1f4, 0xfff1f1f4, 0xfff1f1f4, 0xfff1f1f3, 0xfff1f1f4, 0xfff1f1f4, 0xfff1f1f3, 0xfff1f1f4, + 0xfff1f1f3, 0xfff1f1f4, 0xfff1f1f4, 0xfff1f1f4, 0xfff1f1f3, 0xfff1f1f4, 0xfff1f1f4, 0xfff1f1f3, + 0xfff1f1f4, 0xfff1f1f3, 0xfff1f1f4, 0xfff1f1f4, 0xffe0c5c8, 0xffc6313f, 0xffce3341, 0xffd03442, + 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, + 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, + 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03644, 0x08000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x08000000, 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, + 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, + 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, + 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, + 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xffe5cfd1, 0xffc5313e, 0xffcd3341, 0xffd03442, + 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, + 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, + 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03644, 0x08000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x08000000, 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xfff0f1f3, 0xfff1f1f3, + 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xfff0f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xfff0f1f3, + 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xfff0f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, + 0xfff0f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xfff0f1f3, 0xfff1f1f3, 0xfff1f1f3, + 0xfff1f1f3, 0xfff0f1f3, 0xfff1f1f3, 0xfff1f1f3, 0xffe5d0d2, 0xffc4313e, 0xffcd3341, 0xffd03442, + 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, + 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, + 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03644, 0x08000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x08000000, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, + 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, + 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, + 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, + 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff1f2f3, 0xffd09da0, 0xffc4313e, 0xffcd3341, 0xffd03442, + 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, + 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, + 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xfed03644, 0x08000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x08000000, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, + 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, + 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, + 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff1f2f3, 0xffc9a8aa, 0xffbb898c, 0xffb98587, 0xffb88386, + 0xffb88386, 0xffb88285, 0xffb87e81, 0xffb6696e, 0xffb62d39, 0xffc5313e, 0xffcd3341, 0xffd03442, + 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, + 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, + 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xfcd03644, 0x07000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x08000000, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, + 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, + 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, + 0xfff0f1f2, 0xfff0f1f2, 0xfff0f1f2, 0xfff2f1f2, 0xffa42934, 0xffa72935, 0xffa82a35, 0xffa82a35, + 0xffa82a35, 0xffa82a35, 0xffaa2a36, 0xffb12c38, 0xffbc2f3b, 0xffc8313f, 0xffce3341, 0xffd03442, + 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, + 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, + 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xfbd03644, 0x07000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x08000000, 0xfff0f0f1, 0xffeff0f1, 0xffeff0f1, 0xfff0f0f1, 0xffeff0f1, + 0xfff0f0f1, 0xffeff0f1, 0xfff0f0f1, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, 0xfff0f0f1, + 0xffeff0f1, 0xfff0f0f1, 0xffeff0f1, 0xfff0f0f1, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, + 0xfff0f0f1, 0xffeff0f1, 0xfff0f0f1, 0xfff2f2f3, 0xffae2b37, 0xffb52d39, 0xffb82e3a, 0xffb82e3a, + 0xffb82e3a, 0xffb82e3a, 0xffba2e3b, 0xffbe2f3c, 0xffc5313e, 0xffcb3240, 0xffcf3341, 0xffd03442, + 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, + 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, + 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xfad03644, 0x07000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x08000000, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, + 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, 0xffe4e5e6, + 0xffedeeef, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, + 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, 0xfff2f3f4, 0xffb92f3b, 0xffc3303d, 0xffc6313f, 0xffc7313f, + 0xffc7313f, 0xffc7313f, 0xffc8313f, 0xffca323f, 0xffcc3340, 0xffce3341, 0xffd03442, 0xffd03442, + 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, + 0xffd03442, 0xffcf3442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, + 0xffd03442, 0xffd03442, 0xffd03442, 0xffd03442, 0xf8d03643, 0x07000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x08000000, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, + 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, 0xffdadbdc, 0xff626363, + 0xff919292, 0xffebeced, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, + 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, 0xfff2f2f4, 0xffbe3241, 0xffc83245, 0xffcc3347, 0xffcc3347, + 0xffcc3347, 0xffcc3347, 0xffcd3348, 0xffcd3448, 0xffcd3448, 0xffce3448, 0xffce3448, 0xffce3448, + 0xffce3448, 0xffce3448, 0xffce3448, 0xffce3448, 0xffce3448, 0xffce3448, 0xffce3448, 0xffce3446, + 0xfe892439, 0xfe461832, 0xffb83145, 0xffce3549, 0xffce3549, 0xffce3549, 0xffce3549, 0xffce3549, + 0xffce3549, 0xffce3549, 0xffce3549, 0xffce3549, 0xf7cd384c, 0x07000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x08000000, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, 0xffefeff1, + 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, 0xffefeff1, 0xffeff0f1, 0xffeff0f1, 0xffe3e3e5, 0xff6a6b6b, + 0xff5f5f60, 0xff949495, 0xffeaebec, 0xffeff0f1, 0xffefeff1, 0xffeff0f1, 0xffeff0f1, 0xffeff0f1, + 0xffeff0f1, 0xffefeff1, 0xffeff0f1, 0xfff2f2f3, 0xff933e9a, 0xff953d98, 0xff953d98, 0xff953d98, + 0xff953d98, 0xff953d98, 0xff953d98, 0xff953d98, 0xff953d98, 0xff953d98, 0xff953d98, 0xff953d98, + 0xff953d98, 0xff953d98, 0xff953d98, 0xff953d98, 0xff953d98, 0xff953d98, 0xff943d99, 0xfe672d75, + 0xfe36193d, 0xfe391c41, 0xff823f8f, 0xff953f99, 0xff953f99, 0xff953f99, 0xff953f99, 0xff953f99, + 0xff953f99, 0xff953f99, 0xff953f99, 0xff953f99, 0xfd92449d, 0x07000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x08000000, 0xffefeff1, 0xffeeeff0, 0xffefeff1, 0xffefeff1, 0xffefeff1, + 0xffefeff1, 0xffefeff1, 0xffefeff1, 0xffefeff1, 0xffefeff1, 0xffeeeff0, 0xffefeff1, 0xffd6d6d7, + 0xff686969, 0xff5f5f60, 0xff878788, 0xffe2e2e4, 0xffefeff1, 0xffefeff1, 0xffeeeff0, 0xffefeff1, + 0xffefeff1, 0xffefeff1, 0xffefeff1, 0xfff2f2f3, 0xff953d98, 0xff963d97, 0xff963d97, 0xff963d97, + 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, + 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff933c96, 0xff632b70, 0xfe2e1b49, + 0xfe321e4c, 0xff783a85, 0xff963e98, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, + 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xfe944099, 0x07000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x08000000, 0xffeeeff0, 0xffeeeff0, 0xffeeeff0, 0xffeeeff0, 0xffeeeff0, + 0xffeeeff0, 0xffeeeff0, 0xffeeeff0, 0xffeeeff0, 0xffeeeff0, 0xffeeeff0, 0xffeeeff0, 0xffeeeff0, + 0xffd6d7d8, 0xff6d6e6e, 0xff5f5f60, 0xff6c6c6d, 0xffc4c4c5, 0xffeeeef0, 0xffeeeff0, 0xffeeeff0, + 0xffeeeff0, 0xffeeeff0, 0xffeeeff0, 0xfff0eff1, 0xff953d98, 0xff963d97, 0xff963d97, 0xff963d97, + 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, + 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff86388e, 0xff4d2564, 0xff281c4f, 0xff332055, + 0xff7a3b89, 0xff963e98, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, + 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xfe944099, 0x07000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x08000000, 0xfdeeeff0, 0xffeeeff0, 0xffeeeff0, 0xffeeeff0, 0xffeeeff0, + 0xffeeeff0, 0xffeeeff0, 0xffeeeff0, 0xffeeeff0, 0xffeeeff0, 0xffeeeff0, 0xffeeeff0, 0xffeeeff0, + 0xffeeeff0, 0xffe0e1e2, 0xff868787, 0xff5f5f60, 0xff5f5f60, 0xff888889, 0xffcdcdce, 0xffedeeef, + 0xffeeeff0, 0xffeeeff0, 0xffeeeff0, 0xffeae8ed, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, + 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, + 0xff963d97, 0xff963d97, 0xff8a3991, 0xff632c74, 0xff2e1d54, 0xff261c51, 0xff442863, 0xff843e90, + 0xff963e97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, + 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xfd953f99, 0x07000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x07000000, 0xf9edeeef, 0xffeeeeef, 0xffeeeef0, 0xffeeeef0, 0xffeeeeef, + 0xffeeeef0, 0xffeeeeef, 0xffeeeeef, 0xffeeeeef, 0xffeeeeef, 0xffeeeeef, 0xffeeeef0, 0xffeeeef0, + 0xffeeeeef, 0xffeeeef0, 0xffebeced, 0xffb7b7b8, 0xff666667, 0xff5f5f5f, 0xff5f5f5f, 0xff7a7a7a, + 0xffb4b5b6, 0xffd6d6d8, 0xffededef, 0xffe2dde5, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, + 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff953d98, 0xff8e3b93, + 0xff7a3485, 0xff58296d, 0xff2e1d55, 0xff261c52, 0xff302057, 0xff67357c, 0xff904097, 0xff963d97, + 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, + 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xfb93429a, 0x06000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x07000000, 0xf0ececed, 0xffeeeeef, 0xffeeeeef, 0xffeeeeef, 0xffeeeeef, + 0xffeeeeef, 0xffeeeeef, 0xffeeeeef, 0xffeeeeef, 0xffeeeeef, 0xffeeeeef, 0xffeeeeef, 0xffeeeeef, + 0xffeeeeef, 0xffeeeeef, 0xffeeeeef, 0xffeeeeef, 0xffe2e2e3, 0xffababab, 0xff676767, 0xff5f5f5f, + 0xff5e5e5f, 0xff5a5a5e, 0xff5b5a64, 0xff7f7888, 0xff71377d, 0xff7a3584, 0xff7f3688, 0xff82368a, + 0xff85378b, 0xff83378a, 0xff803688, 0xff7a3485, 0xff70307d, 0xff612b73, 0xff492464, 0xff2f1e56, + 0xff261c52, 0xff261c52, 0xff332159, 0xff623378, 0xff893f93, 0xff963e98, 0xff963d97, 0xff963d97, + 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, + 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xe7934199, 0x06000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x06000000, 0xdfebebec, 0xffeeeeef, 0xffeeeeef, 0xffeeeeef, 0xffeeeeef, + 0xffeeeeef, 0xffeeeeef, 0xffedeeef, 0xffeeeeef, 0xffeeeeef, 0xffeeeeef, 0xffeeeeef, 0xffeeeeef, + 0xffeeeeef, 0xffeeeeef, 0xffeeeeef, 0xffedeeef, 0xffeeeeef, 0xffeeeeef, 0xffe7e7e7, 0xffc0c0c1, + 0xff939394, 0xff5b5b5f, 0xff514f5b, 0xff444057, 0xff362f54, 0xff2c2352, 0xff281e51, 0xff271c52, + 0xff281c53, 0xff271c53, 0xff271c52, 0xff261c52, 0xff261c52, 0xff261c52, 0xff281d53, 0xff35225a, + 0xff552e6f, 0xff753985, 0xff8e4096, 0xff963e98, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, + 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, + 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xdb924098, 0x05000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x04000000, 0xd0eaeaeb, 0xffeeeeee, 0xffededee, 0xffeeeeee, 0xffededee, + 0xffeeeeee, 0xffededee, 0xffeeeeee, 0xffededee, 0xffededee, 0xffeeeeee, 0xffededee, 0xffeeeeee, + 0xffededee, 0xffeeeeee, 0xffededee, 0xffeeeeee, 0xffededee, 0xffededee, 0xffeeeeee, 0xffededee, + 0xffeeeeee, 0xffe8e8e9, 0xffd1d1d3, 0xffa196ad, 0xff6f3d80, 0xff623577, 0xff583071, 0xff532f70, + 0xff4f2b6b, 0xff532e6f, 0xff572f71, 0xff5e3175, 0xff6a357e, 0xff773986, 0xff853d8f, 0xff914098, + 0xff953e98, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, + 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, + 0xff963d97, 0xff963d97, 0xff963d97, 0xff963d97, 0xcf914199, 0x04000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x03000000, 0xace7e7e8, 0xffeeeeee, 0xffeeeeee, 0xffeeeeee, 0xffeeeeee, + 0xffeeeeee, 0xffeeeeee, 0xffeeeeee, 0xffeeeeee, 0xffeeeeee, 0xffeeeeee, 0xffeeeeee, 0xffeeeeee, + 0xffeeeeee, 0xffeeeeee, 0xffeeeeee, 0xffeeeeee, 0xffeeeeee, 0xffeeeeee, 0xffeeeeee, 0xffeeeeee, + 0xffeeeeee, 0xffeeeeee, 0xffeeeeee, 0xffbfa8c6, 0xff953f98, 0xff953f98, 0xff954098, 0xff954099, + 0xff954099, 0xff954099, 0xff954098, 0xff954098, 0xff953f98, 0xff953f98, 0xff953f98, 0xff953f98, + 0xff953f98, 0xff953f98, 0xff953f98, 0xff953f98, 0xff953f98, 0xff953f98, 0xff953f98, 0xff953f98, + 0xff953f98, 0xff953f98, 0xff953f98, 0xff953f98, 0xff953f98, 0xff953f98, 0xff953f98, 0xff953f98, + 0xff953f98, 0xff953f98, 0xff953f98, 0xff953f98, 0xa78e4398, 0x02000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x01000000, 0x7ee2e2e3, 0xffeeeeee, 0xffededee, 0xffeeeeee, 0xffededed, + 0xffeeeeee, 0xffeeeeee, 0xffededed, 0xffeeeeee, 0xffeeeeee, 0xffeeeeee, 0xffededee, 0xffeeeeee, + 0xffededed, 0xffeeeeee, 0xffeeeeee, 0xffededed, 0xffeeeeee, 0xffeeeeee, 0xffeeeeee, 0xffededee, + 0xffeeeeee, 0xffededed, 0xffeeeeee, 0xff8aacd9, 0xff4f8bcd, 0xff518acc, 0xff5289cc, 0xff5289cc, + 0xff5289cc, 0xff5289cc, 0xff5289cc, 0xff5289cc, 0xff5289cc, 0xff5289cc, 0xff5289cc, 0xff5289cc, + 0xff5289cc, 0xff5289cc, 0xff5289cc, 0xff5289cc, 0xff5289cc, 0xff5289cc, 0xff5289cc, 0xff5289cc, + 0xff5289cc, 0xff5289cc, 0xff5289cc, 0xff5289cc, 0xff5289cc, 0xff5289cc, 0xff5289cc, 0xff5289cc, + 0xff5289cc, 0xff5289cc, 0xff5289cc, 0xff5289cc, 0x7a5b7abd, 0x01000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x00000000, 0x38cbcbcb, 0xffedeeee, 0xffedeeee, 0xffeeeeee, 0xffedeeee, + 0xffeeeeee, 0xffedeeee, 0xffedeeee, 0xffedeeee, 0xffedeeee, 0xffedeeee, 0xffedeeee, 0xffeeeeee, + 0xffedeeee, 0xffeeeeee, 0xffedeeee, 0xffedeeee, 0xffedeeee, 0xffedeeee, 0xffedeeee, 0xffedeeee, + 0xffeeeeee, 0xffedeeee, 0xfff0f0f0, 0xff3da2dc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff129bdb, 0x343579b4, 0x00000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x00000000, 0x0b000000, 0xd3e9e9e9, 0xffededee, 0xffededee, 0xffededee, + 0xffededee, 0xffededee, 0xffededee, 0xffededee, 0xffededee, 0xffededee, 0xffededee, 0xffededee, + 0xffededee, 0xffededee, 0xffededee, 0xffededee, 0xffededee, 0xffededee, 0xffededee, 0xffededee, + 0xffededee, 0xffededee, 0xffe1e7ed, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xcf2295d4, 0x0a000000, 0x00000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x00000000, 0x03000000, 0x62d5d5d5, 0xffededed, 0xffededed, 0xffededed, + 0xffededed, 0xffededed, 0xffededed, 0xffededed, 0xffededed, 0xffededed, 0xffededed, 0xffededed, + 0xffededed, 0xffededed, 0xffededed, 0xffededed, 0xffededed, 0xffededed, 0xffededed, 0xffededed, + 0xffededed, 0xffededed, 0xffc7d8ea, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xff009ddc, 0xff099cdb, 0x5c2884c0, 0x02000000, 0x00000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x0e000000, 0xc0e5e5e6, 0xffededed, 0xffededed, + 0xffededed, 0xffededed, 0xffededed, 0xffededed, 0xffededed, 0xffededed, 0xffededed, 0xffededed, + 0xffededed, 0xffededed, 0xffededed, 0xffededed, 0xffededed, 0xffededed, 0xffededed, 0xffededed, + 0xffededed, 0xffededed, 0xffa5c6e5, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xff019ddc, 0xbd2093d2, 0x0d000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x02000000, 0x2ca4a4a4, 0xe5eaeaea, 0xffececec, + 0xffededed, 0xffededed, 0xffececec, 0xffeceded, 0xffececed, 0xffececec, 0xffececed, 0xffeceded, + 0xffececec, 0xffededed, 0xffededed, 0xffececec, 0xffeceded, 0xffececed, 0xffececec, 0xffececed, + 0xffeceded, 0xffeeeeee, 0xff66acdf, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xe41798d8, 0x29216392, 0x02000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x06000000, 0x3cb1b1b1, 0xe6eaeaea, + 0xffececec, 0xffececec, 0xffececec, 0xffececec, 0xffececec, 0xffececec, 0xffececec, 0xffececec, + 0xffececec, 0xffececec, 0xffececec, 0xffececec, 0xffececec, 0xffececec, 0xffececec, 0xffececec, + 0xffececec, 0xffe3e8ed, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xe51598d8, 0x391d6ea2, 0x05000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x07000000, 0x339f9f9f, + 0xc4e4e4e4, 0xffededed, 0xffececec, 0xffececec, 0xffececec, 0xffececec, 0xffececec, 0xffececec, + 0xffececec, 0xffececec, 0xffededed, 0xffececec, 0xffececec, 0xffececec, 0xffececec, 0xffececec, + 0xffececec, 0xffbbd1e7, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff039ddc, 0xc31994d3, + 0x301e6290, 0x06000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x05000000, + 0x19020202, 0x73d0d0d0, 0xdce8e8e8, 0xffececec, 0xffececec, 0xffececec, 0xffececec, 0xffececec, + 0xffececec, 0xffececec, 0xffececec, 0xffececec, 0xffececec, 0xffececec, 0xffececec, 0xffececec, + 0xffededed, 0xff84b7e1, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, + 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff009ddc, 0xff069cdc, 0xdc1697d6, 0x712084be, 0x18000101, + 0x04000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x02000000, 0x0b000000, 0x1d000000, 0x55bdbdbd, 0x97dbdbdb, 0xc4e3e3e3, 0xdee7e7e7, 0xfaececec, + 0xffececec, 0xffededed, 0xffededed, 0xffececec, 0xffececec, 0xffededed, 0xffececec, 0xffededed, + 0xffe4e7ec, 0xff069cdb, 0xff009ddc, 0xff019ddc, 0xff089cdb, 0xff0b9cdb, 0xff0b9cdb, 0xff0b9cdb, + 0xff0b9cdb, 0xff0b9cdb, 0xff0b9cdb, 0xff0b9cdb, 0xff0b9cdb, 0xff0b9cdb, 0xff0b9cdb, 0xff0b9cdb, + 0xff0b9cdb, 0xff0b9cdb, 0xff0b9cdb, 0xff0c9cdb, 0xff0d9cdb, 0xff0e9cdb, 0xff0e9bdb, 0xff119bdb, + 0xfc159ada, 0xe31598d7, 0xc61794d3, 0x971b8dc9, 0x572378af, 0x1c000000, 0x0b000000, 0x01000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x02000000, 0x07000000, 0x10000000, 0x18000000, 0x1e000000, 0x22000000, + 0x261c1c1c, 0x29424242, 0x2a4f4f4f, 0x2b555555, 0x2b565656, 0x2b575757, 0x2b575757, 0x2b575757, + 0x2b484c51, 0x2b06304a, 0x2b05324c, 0x2b07324c, 0x2b0a314c, 0x2b0b304b, 0x2b0b304b, 0x2b0b304b, + 0x2b0b304b, 0x2b0b304b, 0x2b0b304b, 0x2b0b304b, 0x2b0b304b, 0x2b0b304b, 0x2b0b304b, 0x2b0b304b, + 0x2b0b304b, 0x2b0b304b, 0x2b0b304b, 0x2b0c304b, 0x2b0c304b, 0x2b0c2f4a, 0x2a0c2e48, 0x27082134, + 0x22000000, 0x1e000000, 0x17000000, 0x0f000000, 0x07000000, 0x02000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x01000000, 0x01000000, 0x02000000, + 0x03000000, 0x03000000, 0x04000000, 0x04000000, 0x04000000, 0x04000000, 0x04000000, 0x04000000, + 0x04000000, 0x04000000, 0x04000000, 0x04000000, 0x04000000, 0x04000000, 0x04000000, 0x04000000, + 0x04000000, 0x04000000, 0x04000000, 0x04000000, 0x04000000, 0x04000000, 0x04000000, 0x04000000, + 0x04000000, 0x04000000, 0x04000000, 0x04000000, 0x04000000, 0x04000000, 0x03000000, 0x03000000, + 0x02000000, 0x01000000, 0x01000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x00000000, + +}; diff --git a/contrib/mii-icon-64.png b/contrib/mii-icon-64.png new file mode 100644 index 0000000000000000000000000000000000000000..7ab865d2c1ab25c44d6c10adc59c2aac016baaa4 GIT binary patch literal 2755 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 - -// if STB_IMAGE_WRITE_STATIC causes problems, try defining STBIWDEF to 'inline' or 'static inline' -#ifndef STBIWDEF -#ifdef STB_IMAGE_WRITE_STATIC -#define STBIWDEF static -#else -#ifdef __cplusplus -#define STBIWDEF extern "C" -#else -#define STBIWDEF extern -#endif -#endif -#endif - -#ifndef STB_IMAGE_WRITE_STATIC // C++ forbids static forward declarations -STBIWDEF int stbi_write_tga_with_rle; -STBIWDEF int stbi_write_png_compression_level; -STBIWDEF int stbi_write_force_png_filter; -#endif - -#ifndef STBI_WRITE_NO_STDIO -STBIWDEF int stbi_write_png(char const *filename, int w, int h, int comp, const void *data, int stride_in_bytes); -STBIWDEF int stbi_write_bmp(char const *filename, int w, int h, int comp, const void *data); -STBIWDEF int stbi_write_tga(char const *filename, int w, int h, int comp, const void *data); -STBIWDEF int stbi_write_hdr(char const *filename, int w, int h, int comp, const float *data); -STBIWDEF int stbi_write_jpg(char const *filename, int x, int y, int comp, const void *data, int quality); - -#ifdef STBIW_WINDOWS_UTF8 -STBIWDEF int stbiw_convert_wchar_to_utf8(char *buffer, size_t bufferlen, const wchar_t* input); -#endif -#endif - -typedef void stbi_write_func(void *context, void *data, int size); - -STBIWDEF int stbi_write_png_to_func(stbi_write_func *func, void *context, int w, int h, int comp, const void *data, int stride_in_bytes); -STBIWDEF int stbi_write_bmp_to_func(stbi_write_func *func, void *context, int w, int h, int comp, const void *data); -STBIWDEF int stbi_write_tga_to_func(stbi_write_func *func, void *context, int w, int h, int comp, const void *data); -STBIWDEF int stbi_write_hdr_to_func(stbi_write_func *func, void *context, int w, int h, int comp, const float *data); -STBIWDEF int stbi_write_jpg_to_func(stbi_write_func *func, void *context, int x, int y, int comp, const void *data, int quality); - -STBIWDEF void stbi_flip_vertically_on_write(int flip_boolean); - -#endif//INCLUDE_STB_IMAGE_WRITE_H - -#ifdef STB_IMAGE_WRITE_IMPLEMENTATION - -#ifdef _WIN32 - #ifndef _CRT_SECURE_NO_WARNINGS - #define _CRT_SECURE_NO_WARNINGS - #endif - #ifndef _CRT_NONSTDC_NO_DEPRECATE - #define _CRT_NONSTDC_NO_DEPRECATE - #endif -#endif - -#ifndef STBI_WRITE_NO_STDIO -#include -#endif // STBI_WRITE_NO_STDIO - -#include -#include -#include -#include - -#if defined(STBIW_MALLOC) && defined(STBIW_FREE) && (defined(STBIW_REALLOC) || defined(STBIW_REALLOC_SIZED)) -// ok -#elif !defined(STBIW_MALLOC) && !defined(STBIW_FREE) && !defined(STBIW_REALLOC) && !defined(STBIW_REALLOC_SIZED) -// ok -#else -#error "Must define all or none of STBIW_MALLOC, STBIW_FREE, and STBIW_REALLOC (or STBIW_REALLOC_SIZED)." -#endif - -#ifndef STBIW_MALLOC -#define STBIW_MALLOC(sz) malloc(sz) -#define STBIW_REALLOC(p,newsz) realloc(p,newsz) -#define STBIW_FREE(p) free(p) -#endif - -#ifndef STBIW_REALLOC_SIZED -#define STBIW_REALLOC_SIZED(p,oldsz,newsz) STBIW_REALLOC(p,newsz) -#endif - - -#ifndef STBIW_MEMMOVE -#define STBIW_MEMMOVE(a,b,sz) memmove(a,b,sz) -#endif - - -#ifndef STBIW_ASSERT -#include -#define STBIW_ASSERT(x) assert(x) -#endif - -#define STBIW_UCHAR(x) (unsigned char) ((x) & 0xff) - -#ifdef STB_IMAGE_WRITE_STATIC -static int stbi_write_png_compression_level = 8; -static int stbi_write_tga_with_rle = 1; -static int stbi_write_force_png_filter = -1; -#else -int stbi_write_png_compression_level = 8; -int stbi_write_tga_with_rle = 1; -int stbi_write_force_png_filter = -1; -#endif - -static int stbi__flip_vertically_on_write = 0; - -STBIWDEF void stbi_flip_vertically_on_write(int flag) -{ - stbi__flip_vertically_on_write = flag; -} - -typedef struct -{ - stbi_write_func *func; - void *context; - unsigned char buffer[64]; - int buf_used; -} stbi__write_context; - -// initialize a callback-based context -static void stbi__start_write_callbacks(stbi__write_context *s, stbi_write_func *c, void *context) -{ - s->func = c; - s->context = context; -} - -#ifndef STBI_WRITE_NO_STDIO - -static void stbi__stdio_write(void *context, void *data, int size) -{ - fwrite(data,1,size,(FILE*) context); -} - -#if defined(_WIN32) && defined(STBIW_WINDOWS_UTF8) -#ifdef __cplusplus -#define STBIW_EXTERN extern "C" -#else -#define STBIW_EXTERN extern -#endif -STBIW_EXTERN __declspec(dllimport) int __stdcall MultiByteToWideChar(unsigned int cp, unsigned long flags, const char *str, int cbmb, wchar_t *widestr, int cchwide); -STBIW_EXTERN __declspec(dllimport) int __stdcall WideCharToMultiByte(unsigned int cp, unsigned long flags, const wchar_t *widestr, int cchwide, char *str, int cbmb, const char *defchar, int *used_default); - -STBIWDEF int stbiw_convert_wchar_to_utf8(char *buffer, size_t bufferlen, const wchar_t* input) -{ - return WideCharToMultiByte(65001 /* UTF8 */, 0, input, -1, buffer, (int) bufferlen, NULL, NULL); -} -#endif - -static FILE *stbiw__fopen(char const *filename, char const *mode) -{ - FILE *f; -#if defined(_WIN32) && defined(STBIW_WINDOWS_UTF8) - wchar_t wMode[64]; - wchar_t wFilename[1024]; - if (0 == MultiByteToWideChar(65001 /* UTF8 */, 0, filename, -1, wFilename, sizeof(wFilename)/sizeof(*wFilename))) - return 0; - - if (0 == MultiByteToWideChar(65001 /* UTF8 */, 0, mode, -1, wMode, sizeof(wMode)/sizeof(*wMode))) - return 0; - -#if defined(_MSC_VER) && _MSC_VER >= 1400 - if (0 != _wfopen_s(&f, wFilename, wMode)) - f = 0; -#else - f = _wfopen(wFilename, wMode); -#endif - -#elif defined(_MSC_VER) && _MSC_VER >= 1400 - if (0 != fopen_s(&f, filename, mode)) - f=0; -#else - f = fopen(filename, mode); -#endif - return f; -} - -static int stbi__start_write_file(stbi__write_context *s, const char *filename) -{ - FILE *f = stbiw__fopen(filename, "wb"); - stbi__start_write_callbacks(s, stbi__stdio_write, (void *) f); - return f != NULL; -} - -static void stbi__end_write_file(stbi__write_context *s) -{ - fclose((FILE *)s->context); -} - -#endif // !STBI_WRITE_NO_STDIO - -typedef unsigned int stbiw_uint32; -typedef int stb_image_write_test[sizeof(stbiw_uint32)==4 ? 1 : -1]; - -static void stbiw__writefv(stbi__write_context *s, const char *fmt, va_list v) -{ - while (*fmt) { - switch (*fmt++) { - case ' ': break; - case '1': { unsigned char x = STBIW_UCHAR(va_arg(v, int)); - s->func(s->context,&x,1); - break; } - case '2': { int x = va_arg(v,int); - unsigned char b[2]; - b[0] = STBIW_UCHAR(x); - b[1] = STBIW_UCHAR(x>>8); - s->func(s->context,b,2); - break; } - case '4': { stbiw_uint32 x = va_arg(v,int); - unsigned char b[4]; - b[0]=STBIW_UCHAR(x); - b[1]=STBIW_UCHAR(x>>8); - b[2]=STBIW_UCHAR(x>>16); - b[3]=STBIW_UCHAR(x>>24); - s->func(s->context,b,4); - break; } - default: - STBIW_ASSERT(0); - return; - } - } -} - -static void stbiw__writef(stbi__write_context *s, const char *fmt, ...) -{ - va_list v; - va_start(v, fmt); - stbiw__writefv(s, fmt, v); - va_end(v); -} - -static void stbiw__write_flush(stbi__write_context *s) -{ - if (s->buf_used) { - s->func(s->context, &s->buffer, s->buf_used); - s->buf_used = 0; - } -} - -static void stbiw__putc(stbi__write_context *s, unsigned char c) -{ - s->func(s->context, &c, 1); -} - -static void stbiw__write1(stbi__write_context *s, unsigned char a) -{ - if ((size_t)s->buf_used + 1 > sizeof(s->buffer)) - stbiw__write_flush(s); - s->buffer[s->buf_used++] = a; -} - -static void stbiw__write3(stbi__write_context *s, unsigned char a, unsigned char b, unsigned char c) -{ - int n; - if ((size_t)s->buf_used + 3 > sizeof(s->buffer)) - stbiw__write_flush(s); - n = s->buf_used; - s->buf_used = n+3; - s->buffer[n+0] = a; - s->buffer[n+1] = b; - s->buffer[n+2] = c; -} - -static void stbiw__write_pixel(stbi__write_context *s, int rgb_dir, int comp, int write_alpha, int expand_mono, unsigned char *d) -{ - unsigned char bg[3] = { 255, 0, 255}, px[3]; - int k; - - if (write_alpha < 0) - stbiw__write1(s, d[comp - 1]); - - switch (comp) { - case 2: // 2 pixels = mono + alpha, alpha is written separately, so same as 1-channel case - case 1: - if (expand_mono) - stbiw__write3(s, d[0], d[0], d[0]); // monochrome bmp - else - stbiw__write1(s, d[0]); // monochrome TGA - break; - case 4: - if (!write_alpha) { - // composite against pink background - for (k = 0; k < 3; ++k) - px[k] = bg[k] + ((d[k] - bg[k]) * d[3]) / 255; - stbiw__write3(s, px[1 - rgb_dir], px[1], px[1 + rgb_dir]); - break; - } - /* FALLTHROUGH */ - case 3: - stbiw__write3(s, d[1 - rgb_dir], d[1], d[1 + rgb_dir]); - break; - } - if (write_alpha > 0) - stbiw__write1(s, d[comp - 1]); -} - -static void stbiw__write_pixels(stbi__write_context *s, int rgb_dir, int vdir, int x, int y, int comp, void *data, int write_alpha, int scanline_pad, int expand_mono) -{ - stbiw_uint32 zero = 0; - int i,j, j_end; - - if (y <= 0) - return; - - if (stbi__flip_vertically_on_write) - vdir *= -1; - - if (vdir < 0) { - j_end = -1; j = y-1; - } else { - j_end = y; j = 0; - } - - for (; j != j_end; j += vdir) { - for (i=0; i < x; ++i) { - unsigned char *d = (unsigned char *) data + (j*x+i)*comp; - stbiw__write_pixel(s, rgb_dir, comp, write_alpha, expand_mono, d); - } - stbiw__write_flush(s); - s->func(s->context, &zero, scanline_pad); - } -} - -static int stbiw__outfile(stbi__write_context *s, int rgb_dir, int vdir, int x, int y, int comp, int expand_mono, void *data, int alpha, int pad, const char *fmt, ...) -{ - if (y < 0 || x < 0) { - return 0; - } else { - va_list v; - va_start(v, fmt); - stbiw__writefv(s, fmt, v); - va_end(v); - stbiw__write_pixels(s,rgb_dir,vdir,x,y,comp,data,alpha,pad, expand_mono); - return 1; - } -} - -static int stbi_write_bmp_core(stbi__write_context *s, int x, int y, int comp, const void *data) -{ - if (comp != 4) { - // write RGB bitmap - int pad = (-x*3) & 3; - return stbiw__outfile(s,-1,-1,x,y,comp,1,(void *) data,0,pad, - "11 4 22 4" "4 44 22 444444", - 'B', 'M', 14+40+(x*3+pad)*y, 0,0, 14+40, // file header - 40, x,y, 1,24, 0,0,0,0,0,0); // bitmap header - } else { - // RGBA bitmaps need a v4 header - // use BI_BITFIELDS mode with 32bpp and alpha mask - // (straight BI_RGB with alpha mask doesn't work in most readers) - return stbiw__outfile(s,-1,-1,x,y,comp,1,(void *)data,1,0, - "11 4 22 4" "4 44 22 444444 4444 4 444 444 444 444", - 'B', 'M', 14+108+x*y*4, 0, 0, 14+108, // file header - 108, x,y, 1,32, 3,0,0,0,0,0, 0xff0000,0xff00,0xff,0xff000000u, 0, 0,0,0, 0,0,0, 0,0,0, 0,0,0); // bitmap V4 header - } -} - -STBIWDEF int stbi_write_bmp_to_func(stbi_write_func *func, void *context, int x, int y, int comp, const void *data) -{ - stbi__write_context s = { 0 }; - stbi__start_write_callbacks(&s, func, context); - return stbi_write_bmp_core(&s, x, y, comp, data); -} - -#ifndef STBI_WRITE_NO_STDIO -STBIWDEF int stbi_write_bmp(char const *filename, int x, int y, int comp, const void *data) -{ - stbi__write_context s = { 0 }; - if (stbi__start_write_file(&s,filename)) { - int r = stbi_write_bmp_core(&s, x, y, comp, data); - stbi__end_write_file(&s); - return r; - } else - return 0; -} -#endif //!STBI_WRITE_NO_STDIO - -static int stbi_write_tga_core(stbi__write_context *s, int x, int y, int comp, void *data) -{ - int has_alpha = (comp == 2 || comp == 4); - int colorbytes = has_alpha ? comp-1 : comp; - int format = colorbytes < 2 ? 3 : 2; // 3 color channels (RGB/RGBA) = 2, 1 color channel (Y/YA) = 3 - - if (y < 0 || x < 0) - return 0; - - if (!stbi_write_tga_with_rle) { - return stbiw__outfile(s, -1, -1, x, y, comp, 0, (void *) data, has_alpha, 0, - "111 221 2222 11", 0, 0, format, 0, 0, 0, 0, 0, x, y, (colorbytes + has_alpha) * 8, has_alpha * 8); - } else { - int i,j,k; - int jend, jdir; - - stbiw__writef(s, "111 221 2222 11", 0,0,format+8, 0,0,0, 0,0,x,y, (colorbytes + has_alpha) * 8, has_alpha * 8); - - if (stbi__flip_vertically_on_write) { - j = 0; - jend = y; - jdir = 1; - } else { - j = y-1; - jend = -1; - jdir = -1; - } - for (; j != jend; j += jdir) { - unsigned char *row = (unsigned char *) data + j * x * comp; - int len; - - for (i = 0; i < x; i += len) { - unsigned char *begin = row + i * comp; - int diff = 1; - len = 1; - - if (i < x - 1) { - ++len; - diff = memcmp(begin, row + (i + 1) * comp, comp); - if (diff) { - const unsigned char *prev = begin; - for (k = i + 2; k < x && len < 128; ++k) { - if (memcmp(prev, row + k * comp, comp)) { - prev += comp; - ++len; - } else { - --len; - break; - } - } - } else { - for (k = i + 2; k < x && len < 128; ++k) { - if (!memcmp(begin, row + k * comp, comp)) { - ++len; - } else { - break; - } - } - } - } - - if (diff) { - unsigned char header = STBIW_UCHAR(len - 1); - stbiw__write1(s, header); - for (k = 0; k < len; ++k) { - stbiw__write_pixel(s, -1, comp, has_alpha, 0, begin + k * comp); - } - } else { - unsigned char header = STBIW_UCHAR(len - 129); - stbiw__write1(s, header); - stbiw__write_pixel(s, -1, comp, has_alpha, 0, begin); - } - } - } - stbiw__write_flush(s); - } - return 1; -} - -STBIWDEF int stbi_write_tga_to_func(stbi_write_func *func, void *context, int x, int y, int comp, const void *data) -{ - stbi__write_context s = { 0 }; - stbi__start_write_callbacks(&s, func, context); - return stbi_write_tga_core(&s, x, y, comp, (void *) data); -} - -#ifndef STBI_WRITE_NO_STDIO -STBIWDEF int stbi_write_tga(char const *filename, int x, int y, int comp, const void *data) -{ - stbi__write_context s = { 0 }; - if (stbi__start_write_file(&s,filename)) { - int r = stbi_write_tga_core(&s, x, y, comp, (void *) data); - stbi__end_write_file(&s); - return r; - } else - return 0; -} -#endif - -// ************************************************************************************************* -// Radiance RGBE HDR writer -// by Baldur Karlsson - -#define stbiw__max(a, b) ((a) > (b) ? (a) : (b)) - -#ifndef STBI_WRITE_NO_STDIO - -static void stbiw__linear_to_rgbe(unsigned char *rgbe, float *linear) -{ - int exponent; - float maxcomp = stbiw__max(linear[0], stbiw__max(linear[1], linear[2])); - - if (maxcomp < 1e-32f) { - rgbe[0] = rgbe[1] = rgbe[2] = rgbe[3] = 0; - } else { - float normalize = (float) frexp(maxcomp, &exponent) * 256.0f/maxcomp; - - rgbe[0] = (unsigned char)(linear[0] * normalize); - rgbe[1] = (unsigned char)(linear[1] * normalize); - rgbe[2] = (unsigned char)(linear[2] * normalize); - rgbe[3] = (unsigned char)(exponent + 128); - } -} - -static void stbiw__write_run_data(stbi__write_context *s, int length, unsigned char databyte) -{ - unsigned char lengthbyte = STBIW_UCHAR(length+128); - STBIW_ASSERT(length+128 <= 255); - s->func(s->context, &lengthbyte, 1); - s->func(s->context, &databyte, 1); -} - -static void stbiw__write_dump_data(stbi__write_context *s, int length, unsigned char *data) -{ - unsigned char lengthbyte = STBIW_UCHAR(length); - STBIW_ASSERT(length <= 128); // inconsistent with spec but consistent with official code - s->func(s->context, &lengthbyte, 1); - s->func(s->context, data, length); -} - -static void stbiw__write_hdr_scanline(stbi__write_context *s, int width, int ncomp, unsigned char *scratch, float *scanline) -{ - unsigned char scanlineheader[4] = { 2, 2, 0, 0 }; - unsigned char rgbe[4]; - float linear[3]; - int x; - - scanlineheader[2] = (width&0xff00)>>8; - scanlineheader[3] = (width&0x00ff); - - /* skip RLE for images too small or large */ - if (width < 8 || width >= 32768) { - for (x=0; x < width; x++) { - switch (ncomp) { - case 4: /* fallthrough */ - case 3: linear[2] = scanline[x*ncomp + 2]; - linear[1] = scanline[x*ncomp + 1]; - linear[0] = scanline[x*ncomp + 0]; - break; - default: - linear[0] = linear[1] = linear[2] = scanline[x*ncomp + 0]; - break; - } - stbiw__linear_to_rgbe(rgbe, linear); - s->func(s->context, rgbe, 4); - } - } else { - int c,r; - /* encode into scratch buffer */ - for (x=0; x < width; x++) { - switch(ncomp) { - case 4: /* fallthrough */ - case 3: linear[2] = scanline[x*ncomp + 2]; - linear[1] = scanline[x*ncomp + 1]; - linear[0] = scanline[x*ncomp + 0]; - break; - default: - linear[0] = linear[1] = linear[2] = scanline[x*ncomp + 0]; - break; - } - stbiw__linear_to_rgbe(rgbe, linear); - scratch[x + width*0] = rgbe[0]; - scratch[x + width*1] = rgbe[1]; - scratch[x + width*2] = rgbe[2]; - scratch[x + width*3] = rgbe[3]; - } - - s->func(s->context, scanlineheader, 4); - - /* RLE each component separately */ - for (c=0; c < 4; c++) { - unsigned char *comp = &scratch[width*c]; - - x = 0; - while (x < width) { - // find first run - r = x; - while (r+2 < width) { - if (comp[r] == comp[r+1] && comp[r] == comp[r+2]) - break; - ++r; - } - if (r+2 >= width) - r = width; - // dump up to first run - while (x < r) { - int len = r-x; - if (len > 128) len = 128; - stbiw__write_dump_data(s, len, &comp[x]); - x += len; - } - // if there's a run, output it - if (r+2 < width) { // same test as what we break out of in search loop, so only true if we break'd - // find next byte after run - while (r < width && comp[r] == comp[x]) - ++r; - // output run up to r - while (x < r) { - int len = r-x; - if (len > 127) len = 127; - stbiw__write_run_data(s, len, comp[x]); - x += len; - } - } - } - } - } -} - -static int stbi_write_hdr_core(stbi__write_context *s, int x, int y, int comp, float *data) -{ - if (y <= 0 || x <= 0 || data == NULL) - return 0; - else { - // Each component is stored separately. Allocate scratch space for full output scanline. - unsigned char *scratch = (unsigned char *) STBIW_MALLOC(x*4); - int i, len; - char buffer[128]; - char header[] = "#?RADIANCE\n# Written by stb_image_write.h\nFORMAT=32-bit_rle_rgbe\n"; - s->func(s->context, header, sizeof(header)-1); - -#ifdef __STDC_LIB_EXT1__ - len = sprintf_s(buffer, sizeof(buffer), "EXPOSURE= 1.0000000000000\n\n-Y %d +X %d\n", y, x); -#else - len = sprintf(buffer, "EXPOSURE= 1.0000000000000\n\n-Y %d +X %d\n", y, x); -#endif - s->func(s->context, buffer, len); - - for(i=0; i < y; i++) - stbiw__write_hdr_scanline(s, x, comp, scratch, data + comp*x*(stbi__flip_vertically_on_write ? y-1-i : i)); - STBIW_FREE(scratch); - return 1; - } -} - -STBIWDEF int stbi_write_hdr_to_func(stbi_write_func *func, void *context, int x, int y, int comp, const float *data) -{ - stbi__write_context s = { 0 }; - stbi__start_write_callbacks(&s, func, context); - return stbi_write_hdr_core(&s, x, y, comp, (float *) data); -} - -STBIWDEF int stbi_write_hdr(char const *filename, int x, int y, int comp, const float *data) -{ - stbi__write_context s = { 0 }; - if (stbi__start_write_file(&s,filename)) { - int r = stbi_write_hdr_core(&s, x, y, comp, (float *) data); - stbi__end_write_file(&s); - return r; - } else - return 0; -} -#endif // STBI_WRITE_NO_STDIO - - -////////////////////////////////////////////////////////////////////////////// -// -// PNG writer -// - -#ifndef STBIW_ZLIB_COMPRESS -// stretchy buffer; stbiw__sbpush() == vector<>::push_back() -- stbiw__sbcount() == vector<>::size() -#define stbiw__sbraw(a) ((int *) (void *) (a) - 2) -#define stbiw__sbm(a) stbiw__sbraw(a)[0] -#define stbiw__sbn(a) stbiw__sbraw(a)[1] - -#define stbiw__sbneedgrow(a,n) ((a)==0 || stbiw__sbn(a)+n >= stbiw__sbm(a)) -#define stbiw__sbmaybegrow(a,n) (stbiw__sbneedgrow(a,(n)) ? stbiw__sbgrow(a,n) : 0) -#define stbiw__sbgrow(a,n) stbiw__sbgrowf((void **) &(a), (n), sizeof(*(a))) - -#define stbiw__sbpush(a, v) (stbiw__sbmaybegrow(a,1), (a)[stbiw__sbn(a)++] = (v)) -#define stbiw__sbcount(a) ((a) ? stbiw__sbn(a) : 0) -#define stbiw__sbfree(a) ((a) ? STBIW_FREE(stbiw__sbraw(a)),0 : 0) - -static void *stbiw__sbgrowf(void **arr, int increment, int itemsize) -{ - int m = *arr ? 2*stbiw__sbm(*arr)+increment : increment+1; - void *p = STBIW_REALLOC_SIZED(*arr ? stbiw__sbraw(*arr) : 0, *arr ? (stbiw__sbm(*arr)*itemsize + sizeof(int)*2) : 0, itemsize * m + sizeof(int)*2); - STBIW_ASSERT(p); - if (p) { - if (!*arr) ((int *) p)[1] = 0; - *arr = (void *) ((int *) p + 2); - stbiw__sbm(*arr) = m; - } - return *arr; -} - -static unsigned char *stbiw__zlib_flushf(unsigned char *data, unsigned int *bitbuffer, int *bitcount) -{ - while (*bitcount >= 8) { - stbiw__sbpush(data, STBIW_UCHAR(*bitbuffer)); - *bitbuffer >>= 8; - *bitcount -= 8; - } - return data; -} - -static int stbiw__zlib_bitrev(int code, int codebits) -{ - int res=0; - while (codebits--) { - res = (res << 1) | (code & 1); - code >>= 1; - } - return res; -} - -static unsigned int stbiw__zlib_countm(unsigned char *a, unsigned char *b, int limit) -{ - int i; - for (i=0; i < limit && i < 258; ++i) - if (a[i] != b[i]) break; - return i; -} - -static unsigned int stbiw__zhash(unsigned char *data) -{ - stbiw_uint32 hash = data[0] + (data[1] << 8) + (data[2] << 16); - hash ^= hash << 3; - hash += hash >> 5; - hash ^= hash << 4; - hash += hash >> 17; - hash ^= hash << 25; - hash += hash >> 6; - return hash; -} - -#define stbiw__zlib_flush() (out = stbiw__zlib_flushf(out, &bitbuf, &bitcount)) -#define stbiw__zlib_add(code,codebits) \ - (bitbuf |= (code) << bitcount, bitcount += (codebits), stbiw__zlib_flush()) -#define stbiw__zlib_huffa(b,c) stbiw__zlib_add(stbiw__zlib_bitrev(b,c),c) -// default huffman tables -#define stbiw__zlib_huff1(n) stbiw__zlib_huffa(0x30 + (n), 8) -#define stbiw__zlib_huff2(n) stbiw__zlib_huffa(0x190 + (n)-144, 9) -#define stbiw__zlib_huff3(n) stbiw__zlib_huffa(0 + (n)-256,7) -#define stbiw__zlib_huff4(n) stbiw__zlib_huffa(0xc0 + (n)-280,8) -#define stbiw__zlib_huff(n) ((n) <= 143 ? stbiw__zlib_huff1(n) : (n) <= 255 ? stbiw__zlib_huff2(n) : (n) <= 279 ? stbiw__zlib_huff3(n) : stbiw__zlib_huff4(n)) -#define stbiw__zlib_huffb(n) ((n) <= 143 ? stbiw__zlib_huff1(n) : stbiw__zlib_huff2(n)) - -#define stbiw__ZHASH 16384 - -#endif // STBIW_ZLIB_COMPRESS - -STBIWDEF unsigned char * stbi_zlib_compress(unsigned char *data, int data_len, int *out_len, int quality) -{ -#ifdef STBIW_ZLIB_COMPRESS - // user provided a zlib compress implementation, use that - return STBIW_ZLIB_COMPRESS(data, data_len, out_len, quality); -#else // use builtin - static unsigned short lengthc[] = { 3,4,5,6,7,8,9,10,11,13,15,17,19,23,27,31,35,43,51,59,67,83,99,115,131,163,195,227,258, 259 }; - static unsigned char lengtheb[]= { 0,0,0,0,0,0,0, 0, 1, 1, 1, 1, 2, 2, 2, 2, 3, 3, 3, 3, 4, 4, 4, 4, 5, 5, 5, 5, 0 }; - static unsigned short distc[] = { 1,2,3,4,5,7,9,13,17,25,33,49,65,97,129,193,257,385,513,769,1025,1537,2049,3073,4097,6145,8193,12289,16385,24577, 32768 }; - static unsigned char disteb[] = { 0,0,0,0,1,1,2,2,3,3,4,4,5,5,6,6,7,7,8,8,9,9,10,10,11,11,12,12,13,13 }; - unsigned int bitbuf=0; - int i,j, bitcount=0; - unsigned char *out = NULL; - unsigned char ***hash_table = (unsigned char***) STBIW_MALLOC(stbiw__ZHASH * sizeof(unsigned char**)); - if (hash_table == NULL) - return NULL; - if (quality < 5) quality = 5; - - stbiw__sbpush(out, 0x78); // DEFLATE 32K window - stbiw__sbpush(out, 0x5e); // FLEVEL = 1 - stbiw__zlib_add(1,1); // BFINAL = 1 - stbiw__zlib_add(1,2); // BTYPE = 1 -- fixed huffman - - for (i=0; i < stbiw__ZHASH; ++i) - hash_table[i] = NULL; - - i=0; - while (i < data_len-3) { - // hash next 3 bytes of data to be compressed - int h = stbiw__zhash(data+i)&(stbiw__ZHASH-1), best=3; - unsigned char *bestloc = 0; - unsigned char **hlist = hash_table[h]; - int n = stbiw__sbcount(hlist); - for (j=0; j < n; ++j) { - if (hlist[j]-data > i-32768) { // if entry lies within window - int d = stbiw__zlib_countm(hlist[j], data+i, data_len-i); - if (d >= best) { best=d; bestloc=hlist[j]; } - } - } - // when hash table entry is too long, delete half the entries - if (hash_table[h] && stbiw__sbn(hash_table[h]) == 2*quality) { - STBIW_MEMMOVE(hash_table[h], hash_table[h]+quality, sizeof(hash_table[h][0])*quality); - stbiw__sbn(hash_table[h]) = quality; - } - stbiw__sbpush(hash_table[h],data+i); - - if (bestloc) { - // "lazy matching" - check match at *next* byte, and if it's better, do cur byte as literal - h = stbiw__zhash(data+i+1)&(stbiw__ZHASH-1); - hlist = hash_table[h]; - n = stbiw__sbcount(hlist); - for (j=0; j < n; ++j) { - if (hlist[j]-data > i-32767) { - int e = stbiw__zlib_countm(hlist[j], data+i+1, data_len-i-1); - if (e > best) { // if next match is better, bail on current match - bestloc = NULL; - break; - } - } - } - } - - if (bestloc) { - int d = (int) (data+i - bestloc); // distance back - STBIW_ASSERT(d <= 32767 && best <= 258); - for (j=0; best > lengthc[j+1]-1; ++j); - stbiw__zlib_huff(j+257); - if (lengtheb[j]) stbiw__zlib_add(best - lengthc[j], lengtheb[j]); - for (j=0; d > distc[j+1]-1; ++j); - stbiw__zlib_add(stbiw__zlib_bitrev(j,5),5); - if (disteb[j]) stbiw__zlib_add(d - distc[j], disteb[j]); - i += best; - } else { - stbiw__zlib_huffb(data[i]); - ++i; - } - } - // write out final bytes - for (;i < data_len; ++i) - stbiw__zlib_huffb(data[i]); - stbiw__zlib_huff(256); // end of block - // pad with 0 bits to byte boundary - while (bitcount) - stbiw__zlib_add(0,1); - - for (i=0; i < stbiw__ZHASH; ++i) - (void) stbiw__sbfree(hash_table[i]); - STBIW_FREE(hash_table); - - // store uncompressed instead if compression was worse - if (stbiw__sbn(out) > data_len + 2 + ((data_len+32766)/32767)*5) { - stbiw__sbn(out) = 2; // truncate to DEFLATE 32K window and FLEVEL = 1 - for (j = 0; j < data_len;) { - int blocklen = data_len - j; - if (blocklen > 32767) blocklen = 32767; - stbiw__sbpush(out, data_len - j == blocklen); // BFINAL = ?, BTYPE = 0 -- no compression - stbiw__sbpush(out, STBIW_UCHAR(blocklen)); // LEN - stbiw__sbpush(out, STBIW_UCHAR(blocklen >> 8)); - stbiw__sbpush(out, STBIW_UCHAR(~blocklen)); // NLEN - stbiw__sbpush(out, STBIW_UCHAR(~blocklen >> 8)); - memcpy(out+stbiw__sbn(out), data+j, blocklen); - stbiw__sbn(out) += blocklen; - j += blocklen; - } - } - - { - // compute adler32 on input - unsigned int s1=1, s2=0; - int blocklen = (int) (data_len % 5552); - j=0; - while (j < data_len) { - for (i=0; i < blocklen; ++i) { s1 += data[j+i]; s2 += s1; } - s1 %= 65521; s2 %= 65521; - j += blocklen; - blocklen = 5552; - } - stbiw__sbpush(out, STBIW_UCHAR(s2 >> 8)); - stbiw__sbpush(out, STBIW_UCHAR(s2)); - stbiw__sbpush(out, STBIW_UCHAR(s1 >> 8)); - stbiw__sbpush(out, STBIW_UCHAR(s1)); - } - *out_len = stbiw__sbn(out); - // make returned pointer freeable - STBIW_MEMMOVE(stbiw__sbraw(out), out, *out_len); - return (unsigned char *) stbiw__sbraw(out); -#endif // STBIW_ZLIB_COMPRESS -} - -static unsigned int stbiw__crc32(unsigned char *buffer, int len) -{ -#ifdef STBIW_CRC32 - return STBIW_CRC32(buffer, len); -#else - static unsigned int crc_table[256] = - { - 0x00000000, 0x77073096, 0xEE0E612C, 0x990951BA, 0x076DC419, 0x706AF48F, 0xE963A535, 0x9E6495A3, - 0x0eDB8832, 0x79DCB8A4, 0xE0D5E91E, 0x97D2D988, 0x09B64C2B, 0x7EB17CBD, 0xE7B82D07, 0x90BF1D91, - 0x1DB71064, 0x6AB020F2, 0xF3B97148, 0x84BE41DE, 0x1ADAD47D, 0x6DDDE4EB, 0xF4D4B551, 0x83D385C7, - 0x136C9856, 0x646BA8C0, 0xFD62F97A, 0x8A65C9EC, 0x14015C4F, 0x63066CD9, 0xFA0F3D63, 0x8D080DF5, - 0x3B6E20C8, 0x4C69105E, 0xD56041E4, 0xA2677172, 0x3C03E4D1, 0x4B04D447, 0xD20D85FD, 0xA50AB56B, - 0x35B5A8FA, 0x42B2986C, 0xDBBBC9D6, 0xACBCF940, 0x32D86CE3, 0x45DF5C75, 0xDCD60DCF, 0xABD13D59, - 0x26D930AC, 0x51DE003A, 0xC8D75180, 0xBFD06116, 0x21B4F4B5, 0x56B3C423, 0xCFBA9599, 0xB8BDA50F, - 0x2802B89E, 0x5F058808, 0xC60CD9B2, 0xB10BE924, 0x2F6F7C87, 0x58684C11, 0xC1611DAB, 0xB6662D3D, - 0x76DC4190, 0x01DB7106, 0x98D220BC, 0xEFD5102A, 0x71B18589, 0x06B6B51F, 0x9FBFE4A5, 0xE8B8D433, - 0x7807C9A2, 0x0F00F934, 0x9609A88E, 0xE10E9818, 0x7F6A0DBB, 0x086D3D2D, 0x91646C97, 0xE6635C01, - 0x6B6B51F4, 0x1C6C6162, 0x856530D8, 0xF262004E, 0x6C0695ED, 0x1B01A57B, 0x8208F4C1, 0xF50FC457, - 0x65B0D9C6, 0x12B7E950, 0x8BBEB8EA, 0xFCB9887C, 0x62DD1DDF, 0x15DA2D49, 0x8CD37CF3, 0xFBD44C65, - 0x4DB26158, 0x3AB551CE, 0xA3BC0074, 0xD4BB30E2, 0x4ADFA541, 0x3DD895D7, 0xA4D1C46D, 0xD3D6F4FB, - 0x4369E96A, 0x346ED9FC, 0xAD678846, 0xDA60B8D0, 0x44042D73, 0x33031DE5, 0xAA0A4C5F, 0xDD0D7CC9, - 0x5005713C, 0x270241AA, 0xBE0B1010, 0xC90C2086, 0x5768B525, 0x206F85B3, 0xB966D409, 0xCE61E49F, - 0x5EDEF90E, 0x29D9C998, 0xB0D09822, 0xC7D7A8B4, 0x59B33D17, 0x2EB40D81, 0xB7BD5C3B, 0xC0BA6CAD, - 0xEDB88320, 0x9ABFB3B6, 0x03B6E20C, 0x74B1D29A, 0xEAD54739, 0x9DD277AF, 0x04DB2615, 0x73DC1683, - 0xE3630B12, 0x94643B84, 0x0D6D6A3E, 0x7A6A5AA8, 0xE40ECF0B, 0x9309FF9D, 0x0A00AE27, 0x7D079EB1, - 0xF00F9344, 0x8708A3D2, 0x1E01F268, 0x6906C2FE, 0xF762575D, 0x806567CB, 0x196C3671, 0x6E6B06E7, - 0xFED41B76, 0x89D32BE0, 0x10DA7A5A, 0x67DD4ACC, 0xF9B9DF6F, 0x8EBEEFF9, 0x17B7BE43, 0x60B08ED5, - 0xD6D6A3E8, 0xA1D1937E, 0x38D8C2C4, 0x4FDFF252, 0xD1BB67F1, 0xA6BC5767, 0x3FB506DD, 0x48B2364B, - 0xD80D2BDA, 0xAF0A1B4C, 0x36034AF6, 0x41047A60, 0xDF60EFC3, 0xA867DF55, 0x316E8EEF, 0x4669BE79, - 0xCB61B38C, 0xBC66831A, 0x256FD2A0, 0x5268E236, 0xCC0C7795, 0xBB0B4703, 0x220216B9, 0x5505262F, - 0xC5BA3BBE, 0xB2BD0B28, 0x2BB45A92, 0x5CB36A04, 0xC2D7FFA7, 0xB5D0CF31, 0x2CD99E8B, 0x5BDEAE1D, - 0x9B64C2B0, 0xEC63F226, 0x756AA39C, 0x026D930A, 0x9C0906A9, 0xEB0E363F, 0x72076785, 0x05005713, - 0x95BF4A82, 0xE2B87A14, 0x7BB12BAE, 0x0CB61B38, 0x92D28E9B, 0xE5D5BE0D, 0x7CDCEFB7, 0x0BDBDF21, - 0x86D3D2D4, 0xF1D4E242, 0x68DDB3F8, 0x1FDA836E, 0x81BE16CD, 0xF6B9265B, 0x6FB077E1, 0x18B74777, - 0x88085AE6, 0xFF0F6A70, 0x66063BCA, 0x11010B5C, 0x8F659EFF, 0xF862AE69, 0x616BFFD3, 0x166CCF45, - 0xA00AE278, 0xD70DD2EE, 0x4E048354, 0x3903B3C2, 0xA7672661, 0xD06016F7, 0x4969474D, 0x3E6E77DB, - 0xAED16A4A, 0xD9D65ADC, 0x40DF0B66, 0x37D83BF0, 0xA9BCAE53, 0xDEBB9EC5, 0x47B2CF7F, 0x30B5FFE9, - 0xBDBDF21C, 0xCABAC28A, 0x53B39330, 0x24B4A3A6, 0xBAD03605, 0xCDD70693, 0x54DE5729, 0x23D967BF, - 0xB3667A2E, 0xC4614AB8, 0x5D681B02, 0x2A6F2B94, 0xB40BBE37, 0xC30C8EA1, 0x5A05DF1B, 0x2D02EF8D - }; - - unsigned int crc = ~0u; - int i; - for (i=0; i < len; ++i) - crc = (crc >> 8) ^ crc_table[buffer[i] ^ (crc & 0xff)]; - return ~crc; -#endif -} - -#define stbiw__wpng4(o,a,b,c,d) ((o)[0]=STBIW_UCHAR(a),(o)[1]=STBIW_UCHAR(b),(o)[2]=STBIW_UCHAR(c),(o)[3]=STBIW_UCHAR(d),(o)+=4) -#define stbiw__wp32(data,v) stbiw__wpng4(data, (v)>>24,(v)>>16,(v)>>8,(v)); -#define stbiw__wptag(data,s) stbiw__wpng4(data, s[0],s[1],s[2],s[3]) - -static void stbiw__wpcrc(unsigned char **data, int len) -{ - unsigned int crc = stbiw__crc32(*data - len - 4, len+4); - stbiw__wp32(*data, crc); -} - -static unsigned char stbiw__paeth(int a, int b, int c) -{ - int p = a + b - c, pa = abs(p-a), pb = abs(p-b), pc = abs(p-c); - if (pa <= pb && pa <= pc) return STBIW_UCHAR(a); - if (pb <= pc) return STBIW_UCHAR(b); - return STBIW_UCHAR(c); -} - -// @OPTIMIZE: provide an option that always forces left-predict or paeth predict -static void stbiw__encode_png_line(unsigned char *pixels, int stride_bytes, int width, int height, int y, int n, int filter_type, signed char *line_buffer) -{ - static int mapping[] = { 0,1,2,3,4 }; - static int firstmap[] = { 0,1,0,5,6 }; - int *mymap = (y != 0) ? mapping : firstmap; - int i; - int type = mymap[filter_type]; - unsigned char *z = pixels + stride_bytes * (stbi__flip_vertically_on_write ? height-1-y : y); - int signed_stride = stbi__flip_vertically_on_write ? -stride_bytes : stride_bytes; - - if (type==0) { - memcpy(line_buffer, z, width*n); - return; - } - - // first loop isn't optimized since it's just one pixel - for (i = 0; i < n; ++i) { - switch (type) { - case 1: line_buffer[i] = z[i]; break; - case 2: line_buffer[i] = z[i] - z[i-signed_stride]; break; - case 3: line_buffer[i] = z[i] - (z[i-signed_stride]>>1); break; - case 4: line_buffer[i] = (signed char) (z[i] - stbiw__paeth(0,z[i-signed_stride],0)); break; - case 5: line_buffer[i] = z[i]; break; - case 6: line_buffer[i] = z[i]; break; - } - } - switch (type) { - case 1: for (i=n; i < width*n; ++i) line_buffer[i] = z[i] - z[i-n]; break; - case 2: for (i=n; i < width*n; ++i) line_buffer[i] = z[i] - z[i-signed_stride]; break; - case 3: for (i=n; i < width*n; ++i) line_buffer[i] = z[i] - ((z[i-n] + z[i-signed_stride])>>1); break; - case 4: for (i=n; i < width*n; ++i) line_buffer[i] = z[i] - stbiw__paeth(z[i-n], z[i-signed_stride], z[i-signed_stride-n]); break; - case 5: for (i=n; i < width*n; ++i) line_buffer[i] = z[i] - (z[i-n]>>1); break; - case 6: for (i=n; i < width*n; ++i) line_buffer[i] = z[i] - stbiw__paeth(z[i-n], 0,0); break; - } -} - -STBIWDEF unsigned char *stbi_write_png_to_mem(const unsigned char *pixels, int stride_bytes, int x, int y, int n, int *out_len) -{ - int force_filter = stbi_write_force_png_filter; - int ctype[5] = { -1, 0, 4, 2, 6 }; - unsigned char sig[8] = { 137,80,78,71,13,10,26,10 }; - unsigned char *out,*o, *filt, *zlib; - signed char *line_buffer; - int j,zlen; - - if (stride_bytes == 0) - stride_bytes = x * n; - - if (force_filter >= 5) { - force_filter = -1; - } - - filt = (unsigned char *) STBIW_MALLOC((x*n+1) * y); if (!filt) return 0; - line_buffer = (signed char *) STBIW_MALLOC(x * n); if (!line_buffer) { STBIW_FREE(filt); return 0; } - for (j=0; j < y; ++j) { - int filter_type; - if (force_filter > -1) { - filter_type = force_filter; - stbiw__encode_png_line((unsigned char*)(pixels), stride_bytes, x, y, j, n, force_filter, line_buffer); - } else { // Estimate the best filter by running through all of them: - int best_filter = 0, best_filter_val = 0x7fffffff, est, i; - for (filter_type = 0; filter_type < 5; filter_type++) { - stbiw__encode_png_line((unsigned char*)(pixels), stride_bytes, x, y, j, n, filter_type, line_buffer); - - // Estimate the entropy of the line using this filter; the less, the better. - est = 0; - for (i = 0; i < x*n; ++i) { - est += abs((signed char) line_buffer[i]); - } - if (est < best_filter_val) { - best_filter_val = est; - best_filter = filter_type; - } - } - if (filter_type != best_filter) { // If the last iteration already got us the best filter, don't redo it - stbiw__encode_png_line((unsigned char*)(pixels), stride_bytes, x, y, j, n, best_filter, line_buffer); - filter_type = best_filter; - } - } - // when we get here, filter_type contains the filter type, and line_buffer contains the data - filt[j*(x*n+1)] = (unsigned char) filter_type; - STBIW_MEMMOVE(filt+j*(x*n+1)+1, line_buffer, x*n); - } - STBIW_FREE(line_buffer); - zlib = stbi_zlib_compress(filt, y*( x*n+1), &zlen, stbi_write_png_compression_level); - STBIW_FREE(filt); - if (!zlib) return 0; - - // each tag requires 12 bytes of overhead - out = (unsigned char *) STBIW_MALLOC(8 + 12+13 + 12+zlen + 12); - if (!out) return 0; - *out_len = 8 + 12+13 + 12+zlen + 12; - - o=out; - STBIW_MEMMOVE(o,sig,8); o+= 8; - stbiw__wp32(o, 13); // header length - stbiw__wptag(o, "IHDR"); - stbiw__wp32(o, x); - stbiw__wp32(o, y); - *o++ = 8; - *o++ = STBIW_UCHAR(ctype[n]); - *o++ = 0; - *o++ = 0; - *o++ = 0; - stbiw__wpcrc(&o,13); - - stbiw__wp32(o, zlen); - stbiw__wptag(o, "IDAT"); - STBIW_MEMMOVE(o, zlib, zlen); - o += zlen; - STBIW_FREE(zlib); - stbiw__wpcrc(&o, zlen); - - stbiw__wp32(o,0); - stbiw__wptag(o, "IEND"); - stbiw__wpcrc(&o,0); - - STBIW_ASSERT(o == out + *out_len); - - return out; -} - -#ifndef STBI_WRITE_NO_STDIO -STBIWDEF int stbi_write_png(char const *filename, int x, int y, int comp, const void *data, int stride_bytes) -{ - FILE *f; - int len; - unsigned char *png = stbi_write_png_to_mem((const unsigned char *) data, stride_bytes, x, y, comp, &len); - if (png == NULL) return 0; - - f = stbiw__fopen(filename, "wb"); - if (!f) { STBIW_FREE(png); return 0; } - fwrite(png, 1, len, f); - fclose(f); - STBIW_FREE(png); - return 1; -} -#endif - -STBIWDEF int stbi_write_png_to_func(stbi_write_func *func, void *context, int x, int y, int comp, const void *data, int stride_bytes) -{ - int len; - unsigned char *png = stbi_write_png_to_mem((const unsigned char *) data, stride_bytes, x, y, comp, &len); - if (png == NULL) return 0; - func(context, png, len); - STBIW_FREE(png); - return 1; -} - - -/* *************************************************************************** - * - * JPEG writer - * - * This is based on Jon Olick's jo_jpeg.cpp: - * public domain Simple, Minimalistic JPEG writer - http://www.jonolick.com/code.html - */ - -static const unsigned char stbiw__jpg_ZigZag[] = { 0,1,5,6,14,15,27,28,2,4,7,13,16,26,29,42,3,8,12,17,25,30,41,43,9,11,18, - 24,31,40,44,53,10,19,23,32,39,45,52,54,20,22,33,38,46,51,55,60,21,34,37,47,50,56,59,61,35,36,48,49,57,58,62,63 }; - -static void stbiw__jpg_writeBits(stbi__write_context *s, int *bitBufP, int *bitCntP, const unsigned short *bs) { - int bitBuf = *bitBufP, bitCnt = *bitCntP; - bitCnt += bs[1]; - bitBuf |= bs[0] << (24 - bitCnt); - while(bitCnt >= 8) { - unsigned char c = (bitBuf >> 16) & 255; - stbiw__putc(s, c); - if(c == 255) { - stbiw__putc(s, 0); - } - bitBuf <<= 8; - bitCnt -= 8; - } - *bitBufP = bitBuf; - *bitCntP = bitCnt; -} - -static void stbiw__jpg_DCT(float *d0p, float *d1p, float *d2p, float *d3p, float *d4p, float *d5p, float *d6p, float *d7p) { - float d0 = *d0p, d1 = *d1p, d2 = *d2p, d3 = *d3p, d4 = *d4p, d5 = *d5p, d6 = *d6p, d7 = *d7p; - float z1, z2, z3, z4, z5, z11, z13; - - float tmp0 = d0 + d7; - float tmp7 = d0 - d7; - float tmp1 = d1 + d6; - float tmp6 = d1 - d6; - float tmp2 = d2 + d5; - float tmp5 = d2 - d5; - float tmp3 = d3 + d4; - float tmp4 = d3 - d4; - - // Even part - float tmp10 = tmp0 + tmp3; // phase 2 - float tmp13 = tmp0 - tmp3; - float tmp11 = tmp1 + tmp2; - float tmp12 = tmp1 - tmp2; - - d0 = tmp10 + tmp11; // phase 3 - d4 = tmp10 - tmp11; - - z1 = (tmp12 + tmp13) * 0.707106781f; // c4 - d2 = tmp13 + z1; // phase 5 - d6 = tmp13 - z1; - - // Odd part - tmp10 = tmp4 + tmp5; // phase 2 - tmp11 = tmp5 + tmp6; - tmp12 = tmp6 + tmp7; - - // The rotator is modified from fig 4-8 to avoid extra negations. - z5 = (tmp10 - tmp12) * 0.382683433f; // c6 - z2 = tmp10 * 0.541196100f + z5; // c2-c6 - z4 = tmp12 * 1.306562965f + z5; // c2+c6 - z3 = tmp11 * 0.707106781f; // c4 - - z11 = tmp7 + z3; // phase 5 - z13 = tmp7 - z3; - - *d5p = z13 + z2; // phase 6 - *d3p = z13 - z2; - *d1p = z11 + z4; - *d7p = z11 - z4; - - *d0p = d0; *d2p = d2; *d4p = d4; *d6p = d6; -} - -static void stbiw__jpg_calcBits(int val, unsigned short bits[2]) { - int tmp1 = val < 0 ? -val : val; - val = val < 0 ? val-1 : val; - bits[1] = 1; - while(tmp1 >>= 1) { - ++bits[1]; - } - bits[0] = val & ((1<0)&&(DU[end0pos]==0); --end0pos) { - } - // end0pos = first element in reverse order !=0 - if(end0pos == 0) { - stbiw__jpg_writeBits(s, bitBuf, bitCnt, EOB); - return DU[0]; - } - for(i = 1; i <= end0pos; ++i) { - int startpos = i; - int nrzeroes; - unsigned short bits[2]; - for (; DU[i]==0 && i<=end0pos; ++i) { - } - nrzeroes = i-startpos; - if ( nrzeroes >= 16 ) { - int lng = nrzeroes>>4; - int nrmarker; - for (nrmarker=1; nrmarker <= lng; ++nrmarker) - stbiw__jpg_writeBits(s, bitBuf, bitCnt, M16zeroes); - nrzeroes &= 15; - } - stbiw__jpg_calcBits(DU[i], bits); - stbiw__jpg_writeBits(s, bitBuf, bitCnt, HTAC[(nrzeroes<<4)+bits[1]]); - stbiw__jpg_writeBits(s, bitBuf, bitCnt, bits); - } - if(end0pos != 63) { - stbiw__jpg_writeBits(s, bitBuf, bitCnt, EOB); - } - return DU[0]; -} - -static int stbi_write_jpg_core(stbi__write_context *s, int width, int height, int comp, const void* data, int quality) { - // Constants that don't pollute global namespace - static const unsigned char std_dc_luminance_nrcodes[] = {0,0,1,5,1,1,1,1,1,1,0,0,0,0,0,0,0}; - static const unsigned char std_dc_luminance_values[] = {0,1,2,3,4,5,6,7,8,9,10,11}; - static const unsigned char std_ac_luminance_nrcodes[] = {0,0,2,1,3,3,2,4,3,5,5,4,4,0,0,1,0x7d}; - static const unsigned char std_ac_luminance_values[] = { - 0x01,0x02,0x03,0x00,0x04,0x11,0x05,0x12,0x21,0x31,0x41,0x06,0x13,0x51,0x61,0x07,0x22,0x71,0x14,0x32,0x81,0x91,0xa1,0x08, - 0x23,0x42,0xb1,0xc1,0x15,0x52,0xd1,0xf0,0x24,0x33,0x62,0x72,0x82,0x09,0x0a,0x16,0x17,0x18,0x19,0x1a,0x25,0x26,0x27,0x28, - 0x29,0x2a,0x34,0x35,0x36,0x37,0x38,0x39,0x3a,0x43,0x44,0x45,0x46,0x47,0x48,0x49,0x4a,0x53,0x54,0x55,0x56,0x57,0x58,0x59, - 0x5a,0x63,0x64,0x65,0x66,0x67,0x68,0x69,0x6a,0x73,0x74,0x75,0x76,0x77,0x78,0x79,0x7a,0x83,0x84,0x85,0x86,0x87,0x88,0x89, - 0x8a,0x92,0x93,0x94,0x95,0x96,0x97,0x98,0x99,0x9a,0xa2,0xa3,0xa4,0xa5,0xa6,0xa7,0xa8,0xa9,0xaa,0xb2,0xb3,0xb4,0xb5,0xb6, - 0xb7,0xb8,0xb9,0xba,0xc2,0xc3,0xc4,0xc5,0xc6,0xc7,0xc8,0xc9,0xca,0xd2,0xd3,0xd4,0xd5,0xd6,0xd7,0xd8,0xd9,0xda,0xe1,0xe2, - 0xe3,0xe4,0xe5,0xe6,0xe7,0xe8,0xe9,0xea,0xf1,0xf2,0xf3,0xf4,0xf5,0xf6,0xf7,0xf8,0xf9,0xfa - }; - static const unsigned char std_dc_chrominance_nrcodes[] = {0,0,3,1,1,1,1,1,1,1,1,1,0,0,0,0,0}; - static const unsigned char std_dc_chrominance_values[] = {0,1,2,3,4,5,6,7,8,9,10,11}; - static const unsigned char std_ac_chrominance_nrcodes[] = {0,0,2,1,2,4,4,3,4,7,5,4,4,0,1,2,0x77}; - static const unsigned char std_ac_chrominance_values[] = { - 0x00,0x01,0x02,0x03,0x11,0x04,0x05,0x21,0x31,0x06,0x12,0x41,0x51,0x07,0x61,0x71,0x13,0x22,0x32,0x81,0x08,0x14,0x42,0x91, - 0xa1,0xb1,0xc1,0x09,0x23,0x33,0x52,0xf0,0x15,0x62,0x72,0xd1,0x0a,0x16,0x24,0x34,0xe1,0x25,0xf1,0x17,0x18,0x19,0x1a,0x26, - 0x27,0x28,0x29,0x2a,0x35,0x36,0x37,0x38,0x39,0x3a,0x43,0x44,0x45,0x46,0x47,0x48,0x49,0x4a,0x53,0x54,0x55,0x56,0x57,0x58, - 0x59,0x5a,0x63,0x64,0x65,0x66,0x67,0x68,0x69,0x6a,0x73,0x74,0x75,0x76,0x77,0x78,0x79,0x7a,0x82,0x83,0x84,0x85,0x86,0x87, - 0x88,0x89,0x8a,0x92,0x93,0x94,0x95,0x96,0x97,0x98,0x99,0x9a,0xa2,0xa3,0xa4,0xa5,0xa6,0xa7,0xa8,0xa9,0xaa,0xb2,0xb3,0xb4, - 0xb5,0xb6,0xb7,0xb8,0xb9,0xba,0xc2,0xc3,0xc4,0xc5,0xc6,0xc7,0xc8,0xc9,0xca,0xd2,0xd3,0xd4,0xd5,0xd6,0xd7,0xd8,0xd9,0xda, - 0xe2,0xe3,0xe4,0xe5,0xe6,0xe7,0xe8,0xe9,0xea,0xf2,0xf3,0xf4,0xf5,0xf6,0xf7,0xf8,0xf9,0xfa - }; - // Huffman tables - static const unsigned short YDC_HT[256][2] = { {0,2},{2,3},{3,3},{4,3},{5,3},{6,3},{14,4},{30,5},{62,6},{126,7},{254,8},{510,9}}; - static const unsigned short UVDC_HT[256][2] = { {0,2},{1,2},{2,2},{6,3},{14,4},{30,5},{62,6},{126,7},{254,8},{510,9},{1022,10},{2046,11}}; - static const unsigned short YAC_HT[256][2] = { - {10,4},{0,2},{1,2},{4,3},{11,4},{26,5},{120,7},{248,8},{1014,10},{65410,16},{65411,16},{0,0},{0,0},{0,0},{0,0},{0,0},{0,0}, - {12,4},{27,5},{121,7},{502,9},{2038,11},{65412,16},{65413,16},{65414,16},{65415,16},{65416,16},{0,0},{0,0},{0,0},{0,0},{0,0},{0,0}, - {28,5},{249,8},{1015,10},{4084,12},{65417,16},{65418,16},{65419,16},{65420,16},{65421,16},{65422,16},{0,0},{0,0},{0,0},{0,0},{0,0},{0,0}, - {58,6},{503,9},{4085,12},{65423,16},{65424,16},{65425,16},{65426,16},{65427,16},{65428,16},{65429,16},{0,0},{0,0},{0,0},{0,0},{0,0},{0,0}, - {59,6},{1016,10},{65430,16},{65431,16},{65432,16},{65433,16},{65434,16},{65435,16},{65436,16},{65437,16},{0,0},{0,0},{0,0},{0,0},{0,0},{0,0}, - {122,7},{2039,11},{65438,16},{65439,16},{65440,16},{65441,16},{65442,16},{65443,16},{65444,16},{65445,16},{0,0},{0,0},{0,0},{0,0},{0,0},{0,0}, - {123,7},{4086,12},{65446,16},{65447,16},{65448,16},{65449,16},{65450,16},{65451,16},{65452,16},{65453,16},{0,0},{0,0},{0,0},{0,0},{0,0},{0,0}, - {250,8},{4087,12},{65454,16},{65455,16},{65456,16},{65457,16},{65458,16},{65459,16},{65460,16},{65461,16},{0,0},{0,0},{0,0},{0,0},{0,0},{0,0}, - {504,9},{32704,15},{65462,16},{65463,16},{65464,16},{65465,16},{65466,16},{65467,16},{65468,16},{65469,16},{0,0},{0,0},{0,0},{0,0},{0,0},{0,0}, - {505,9},{65470,16},{65471,16},{65472,16},{65473,16},{65474,16},{65475,16},{65476,16},{65477,16},{65478,16},{0,0},{0,0},{0,0},{0,0},{0,0},{0,0}, - {506,9},{65479,16},{65480,16},{65481,16},{65482,16},{65483,16},{65484,16},{65485,16},{65486,16},{65487,16},{0,0},{0,0},{0,0},{0,0},{0,0},{0,0}, - {1017,10},{65488,16},{65489,16},{65490,16},{65491,16},{65492,16},{65493,16},{65494,16},{65495,16},{65496,16},{0,0},{0,0},{0,0},{0,0},{0,0},{0,0}, - {1018,10},{65497,16},{65498,16},{65499,16},{65500,16},{65501,16},{65502,16},{65503,16},{65504,16},{65505,16},{0,0},{0,0},{0,0},{0,0},{0,0},{0,0}, - {2040,11},{65506,16},{65507,16},{65508,16},{65509,16},{65510,16},{65511,16},{65512,16},{65513,16},{65514,16},{0,0},{0,0},{0,0},{0,0},{0,0},{0,0}, - {65515,16},{65516,16},{65517,16},{65518,16},{65519,16},{65520,16},{65521,16},{65522,16},{65523,16},{65524,16},{0,0},{0,0},{0,0},{0,0},{0,0}, - {2041,11},{65525,16},{65526,16},{65527,16},{65528,16},{65529,16},{65530,16},{65531,16},{65532,16},{65533,16},{65534,16},{0,0},{0,0},{0,0},{0,0},{0,0} - }; - static const unsigned short UVAC_HT[256][2] = { - {0,2},{1,2},{4,3},{10,4},{24,5},{25,5},{56,6},{120,7},{500,9},{1014,10},{4084,12},{0,0},{0,0},{0,0},{0,0},{0,0},{0,0}, - {11,4},{57,6},{246,8},{501,9},{2038,11},{4085,12},{65416,16},{65417,16},{65418,16},{65419,16},{0,0},{0,0},{0,0},{0,0},{0,0},{0,0}, - {26,5},{247,8},{1015,10},{4086,12},{32706,15},{65420,16},{65421,16},{65422,16},{65423,16},{65424,16},{0,0},{0,0},{0,0},{0,0},{0,0},{0,0}, - {27,5},{248,8},{1016,10},{4087,12},{65425,16},{65426,16},{65427,16},{65428,16},{65429,16},{65430,16},{0,0},{0,0},{0,0},{0,0},{0,0},{0,0}, - {58,6},{502,9},{65431,16},{65432,16},{65433,16},{65434,16},{65435,16},{65436,16},{65437,16},{65438,16},{0,0},{0,0},{0,0},{0,0},{0,0},{0,0}, - {59,6},{1017,10},{65439,16},{65440,16},{65441,16},{65442,16},{65443,16},{65444,16},{65445,16},{65446,16},{0,0},{0,0},{0,0},{0,0},{0,0},{0,0}, - {121,7},{2039,11},{65447,16},{65448,16},{65449,16},{65450,16},{65451,16},{65452,16},{65453,16},{65454,16},{0,0},{0,0},{0,0},{0,0},{0,0},{0,0}, - {122,7},{2040,11},{65455,16},{65456,16},{65457,16},{65458,16},{65459,16},{65460,16},{65461,16},{65462,16},{0,0},{0,0},{0,0},{0,0},{0,0},{0,0}, - {249,8},{65463,16},{65464,16},{65465,16},{65466,16},{65467,16},{65468,16},{65469,16},{65470,16},{65471,16},{0,0},{0,0},{0,0},{0,0},{0,0},{0,0}, - {503,9},{65472,16},{65473,16},{65474,16},{65475,16},{65476,16},{65477,16},{65478,16},{65479,16},{65480,16},{0,0},{0,0},{0,0},{0,0},{0,0},{0,0}, - {504,9},{65481,16},{65482,16},{65483,16},{65484,16},{65485,16},{65486,16},{65487,16},{65488,16},{65489,16},{0,0},{0,0},{0,0},{0,0},{0,0},{0,0}, - {505,9},{65490,16},{65491,16},{65492,16},{65493,16},{65494,16},{65495,16},{65496,16},{65497,16},{65498,16},{0,0},{0,0},{0,0},{0,0},{0,0},{0,0}, - {506,9},{65499,16},{65500,16},{65501,16},{65502,16},{65503,16},{65504,16},{65505,16},{65506,16},{65507,16},{0,0},{0,0},{0,0},{0,0},{0,0},{0,0}, - {2041,11},{65508,16},{65509,16},{65510,16},{65511,16},{65512,16},{65513,16},{65514,16},{65515,16},{65516,16},{0,0},{0,0},{0,0},{0,0},{0,0},{0,0}, - {16352,14},{65517,16},{65518,16},{65519,16},{65520,16},{65521,16},{65522,16},{65523,16},{65524,16},{65525,16},{0,0},{0,0},{0,0},{0,0},{0,0}, - {1018,10},{32707,15},{65526,16},{65527,16},{65528,16},{65529,16},{65530,16},{65531,16},{65532,16},{65533,16},{65534,16},{0,0},{0,0},{0,0},{0,0},{0,0} - }; - static const int YQT[] = {16,11,10,16,24,40,51,61,12,12,14,19,26,58,60,55,14,13,16,24,40,57,69,56,14,17,22,29,51,87,80,62,18,22, - 37,56,68,109,103,77,24,35,55,64,81,104,113,92,49,64,78,87,103,121,120,101,72,92,95,98,112,100,103,99}; - static const int UVQT[] = {17,18,24,47,99,99,99,99,18,21,26,66,99,99,99,99,24,26,56,99,99,99,99,99,47,66,99,99,99,99,99,99, - 99,99,99,99,99,99,99,99,99,99,99,99,99,99,99,99,99,99,99,99,99,99,99,99,99,99,99,99,99,99,99,99}; - static const float aasf[] = { 1.0f * 2.828427125f, 1.387039845f * 2.828427125f, 1.306562965f * 2.828427125f, 1.175875602f * 2.828427125f, - 1.0f * 2.828427125f, 0.785694958f * 2.828427125f, 0.541196100f * 2.828427125f, 0.275899379f * 2.828427125f }; - - int row, col, i, k, subsample; - float fdtbl_Y[64], fdtbl_UV[64]; - unsigned char YTable[64], UVTable[64]; - - if(!data || !width || !height || comp > 4 || comp < 1) { - return 0; - } - - quality = quality ? quality : 90; - subsample = quality <= 90 ? 1 : 0; - quality = quality < 1 ? 1 : quality > 100 ? 100 : quality; - quality = quality < 50 ? 5000 / quality : 200 - quality * 2; - - for(i = 0; i < 64; ++i) { - int uvti, yti = (YQT[i]*quality+50)/100; - YTable[stbiw__jpg_ZigZag[i]] = (unsigned char) (yti < 1 ? 1 : yti > 255 ? 255 : yti); - uvti = (UVQT[i]*quality+50)/100; - UVTable[stbiw__jpg_ZigZag[i]] = (unsigned char) (uvti < 1 ? 1 : uvti > 255 ? 255 : uvti); - } - - for(row = 0, k = 0; row < 8; ++row) { - for(col = 0; col < 8; ++col, ++k) { - fdtbl_Y[k] = 1 / (YTable [stbiw__jpg_ZigZag[k]] * aasf[row] * aasf[col]); - fdtbl_UV[k] = 1 / (UVTable[stbiw__jpg_ZigZag[k]] * aasf[row] * aasf[col]); - } - } - - // Write Headers - { - static const unsigned char head0[] = { 0xFF,0xD8,0xFF,0xE0,0,0x10,'J','F','I','F',0,1,1,0,0,1,0,1,0,0,0xFF,0xDB,0,0x84,0 }; - static const unsigned char head2[] = { 0xFF,0xDA,0,0xC,3,1,0,2,0x11,3,0x11,0,0x3F,0 }; - const unsigned char head1[] = { 0xFF,0xC0,0,0x11,8,(unsigned char)(height>>8),STBIW_UCHAR(height),(unsigned char)(width>>8),STBIW_UCHAR(width), - 3,1,(unsigned char)(subsample?0x22:0x11),0,2,0x11,1,3,0x11,1,0xFF,0xC4,0x01,0xA2,0 }; - s->func(s->context, (void*)head0, sizeof(head0)); - s->func(s->context, (void*)YTable, sizeof(YTable)); - stbiw__putc(s, 1); - s->func(s->context, UVTable, sizeof(UVTable)); - s->func(s->context, (void*)head1, sizeof(head1)); - s->func(s->context, (void*)(std_dc_luminance_nrcodes+1), sizeof(std_dc_luminance_nrcodes)-1); - s->func(s->context, (void*)std_dc_luminance_values, sizeof(std_dc_luminance_values)); - stbiw__putc(s, 0x10); // HTYACinfo - s->func(s->context, (void*)(std_ac_luminance_nrcodes+1), sizeof(std_ac_luminance_nrcodes)-1); - s->func(s->context, (void*)std_ac_luminance_values, sizeof(std_ac_luminance_values)); - stbiw__putc(s, 1); // HTUDCinfo - s->func(s->context, (void*)(std_dc_chrominance_nrcodes+1), sizeof(std_dc_chrominance_nrcodes)-1); - s->func(s->context, (void*)std_dc_chrominance_values, sizeof(std_dc_chrominance_values)); - stbiw__putc(s, 0x11); // HTUACinfo - s->func(s->context, (void*)(std_ac_chrominance_nrcodes+1), sizeof(std_ac_chrominance_nrcodes)-1); - s->func(s->context, (void*)std_ac_chrominance_values, sizeof(std_ac_chrominance_values)); - s->func(s->context, (void*)head2, sizeof(head2)); - } - - // Encode 8x8 macroblocks - { - static const unsigned short fillBits[] = {0x7F, 7}; - int DCY=0, DCU=0, DCV=0; - int bitBuf=0, bitCnt=0; - // comp == 2 is grey+alpha (alpha is ignored) - int ofsG = comp > 2 ? 1 : 0, ofsB = comp > 2 ? 2 : 0; - const unsigned char *dataR = (const unsigned char *)data; - const unsigned char *dataG = dataR + ofsG; - const unsigned char *dataB = dataR + ofsB; - int x, y, pos; - if(subsample) { - for(y = 0; y < height; y += 16) { - for(x = 0; x < width; x += 16) { - float Y[256], U[256], V[256]; - for(row = y, pos = 0; row < y+16; ++row) { - // row >= height => use last input row - int clamped_row = (row < height) ? row : height - 1; - int base_p = (stbi__flip_vertically_on_write ? (height-1-clamped_row) : clamped_row)*width*comp; - for(col = x; col < x+16; ++col, ++pos) { - // if col >= width => use pixel from last input column - int p = base_p + ((col < width) ? col : (width-1))*comp; - float r = dataR[p], g = dataG[p], b = dataB[p]; - Y[pos]= +0.29900f*r + 0.58700f*g + 0.11400f*b - 128; - U[pos]= -0.16874f*r - 0.33126f*g + 0.50000f*b; - V[pos]= +0.50000f*r - 0.41869f*g - 0.08131f*b; - } - } - DCY = stbiw__jpg_processDU(s, &bitBuf, &bitCnt, Y+0, 16, fdtbl_Y, DCY, YDC_HT, YAC_HT); - DCY = stbiw__jpg_processDU(s, &bitBuf, &bitCnt, Y+8, 16, fdtbl_Y, DCY, YDC_HT, YAC_HT); - DCY = stbiw__jpg_processDU(s, &bitBuf, &bitCnt, Y+128, 16, fdtbl_Y, DCY, YDC_HT, YAC_HT); - DCY = stbiw__jpg_processDU(s, &bitBuf, &bitCnt, Y+136, 16, fdtbl_Y, DCY, YDC_HT, YAC_HT); - - // subsample U,V - { - float subU[64], subV[64]; - int yy, xx; - for(yy = 0, pos = 0; yy < 8; ++yy) { - for(xx = 0; xx < 8; ++xx, ++pos) { - int j = yy*32+xx*2; - subU[pos] = (U[j+0] + U[j+1] + U[j+16] + U[j+17]) * 0.25f; - subV[pos] = (V[j+0] + V[j+1] + V[j+16] + V[j+17]) * 0.25f; - } - } - DCU = stbiw__jpg_processDU(s, &bitBuf, &bitCnt, subU, 8, fdtbl_UV, DCU, UVDC_HT, UVAC_HT); - DCV = stbiw__jpg_processDU(s, &bitBuf, &bitCnt, subV, 8, fdtbl_UV, DCV, UVDC_HT, UVAC_HT); - } - } - } - } else { - for(y = 0; y < height; y += 8) { - for(x = 0; x < width; x += 8) { - float Y[64], U[64], V[64]; - for(row = y, pos = 0; row < y+8; ++row) { - // row >= height => use last input row - int clamped_row = (row < height) ? row : height - 1; - int base_p = (stbi__flip_vertically_on_write ? (height-1-clamped_row) : clamped_row)*width*comp; - for(col = x; col < x+8; ++col, ++pos) { - // if col >= width => use pixel from last input column - int p = base_p + ((col < width) ? col : (width-1))*comp; - float r = dataR[p], g = dataG[p], b = dataB[p]; - Y[pos]= +0.29900f*r + 0.58700f*g + 0.11400f*b - 128; - U[pos]= -0.16874f*r - 0.33126f*g + 0.50000f*b; - V[pos]= +0.50000f*r - 0.41869f*g - 0.08131f*b; - } - } - - DCY = stbiw__jpg_processDU(s, &bitBuf, &bitCnt, Y, 8, fdtbl_Y, DCY, YDC_HT, YAC_HT); - DCU = stbiw__jpg_processDU(s, &bitBuf, &bitCnt, U, 8, fdtbl_UV, DCU, UVDC_HT, UVAC_HT); - DCV = stbiw__jpg_processDU(s, &bitBuf, &bitCnt, V, 8, fdtbl_UV, DCV, UVDC_HT, UVAC_HT); - } - } - } - - // Do the bit alignment of the EOI marker - stbiw__jpg_writeBits(s, &bitBuf, &bitCnt, fillBits); - } - - // EOI - stbiw__putc(s, 0xFF); - stbiw__putc(s, 0xD9); - - return 1; -} - -STBIWDEF int stbi_write_jpg_to_func(stbi_write_func *func, void *context, int x, int y, int comp, const void *data, int quality) -{ - stbi__write_context s = { 0 }; - stbi__start_write_callbacks(&s, func, context); - return stbi_write_jpg_core(&s, x, y, comp, (void *) data, quality); -} - - -#ifndef STBI_WRITE_NO_STDIO -STBIWDEF int stbi_write_jpg(char const *filename, int x, int y, int comp, const void *data, int quality) -{ - stbi__write_context s = { 0 }; - if (stbi__start_write_file(&s,filename)) { - int r = stbi_write_jpg_core(&s, x, y, comp, data, quality); - stbi__end_write_file(&s); - return r; - } else - return 0; -} -#endif - -#endif // STB_IMAGE_WRITE_IMPLEMENTATION - -/* Revision history - 1.16 (2021-07-11) - make Deflate code emit uncompressed blocks when it would otherwise expand - support writing BMPs with alpha channel - 1.15 (2020-07-13) unknown - 1.14 (2020-02-02) updated JPEG writer to downsample chroma channels - 1.13 - 1.12 - 1.11 (2019-08-11) - - 1.10 (2019-02-07) - support utf8 filenames in Windows; fix warnings and platform ifdefs - 1.09 (2018-02-11) - fix typo in zlib quality API, improve STB_I_W_STATIC in C++ - 1.08 (2018-01-29) - add stbi__flip_vertically_on_write, external zlib, zlib quality, choose PNG filter - 1.07 (2017-07-24) - doc fix - 1.06 (2017-07-23) - writing JPEG (using Jon Olick's code) - 1.05 ??? - 1.04 (2017-03-03) - monochrome BMP expansion - 1.03 ??? - 1.02 (2016-04-02) - avoid allocating large structures on the stack - 1.01 (2016-01-16) - STBIW_REALLOC_SIZED: support allocators with no realloc support - avoid race-condition in crc initialization - minor compile issues - 1.00 (2015-09-14) - installable file IO function - 0.99 (2015-09-13) - warning fixes; TGA rle support - 0.98 (2015-04-08) - added STBIW_MALLOC, STBIW_ASSERT etc - 0.97 (2015-01-18) - fixed HDR asserts, rewrote HDR rle logic - 0.96 (2015-01-17) - add HDR output - fix monochrome BMP - 0.95 (2014-08-17) - add monochrome TGA output - 0.94 (2014-05-31) - rename private functions to avoid conflicts with stb_image.h - 0.93 (2014-05-27) - warning fixes - 0.92 (2010-08-01) - casts to unsigned char to fix warnings - 0.91 (2010-07-17) - first public release - 0.90 first internal release -*/ - -/* ------------------------------------------------------------------------------- -This software is available under 2 licenses -- choose whichever you prefer. ------------------------------------------------------------------------------- -ALTERNATIVE A - MIT License -Copyright (c) 2017 Sean Barrett -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. ------------------------------------------------------------------------------- -ALTERNATIVE B - Public Domain (www.unlicense.org) -This is free and unencumbered software released into the public domain. -Anyone is free to copy, modify, publish, use, compile, sell, or distribute this -software, either in source code form or as a compiled binary, for any purpose, -commercial or non-commercial, and by any means. -In jurisdictions that recognize copyright laws, the author or authors of this -software dedicate any and all copyright interest in the software to the public -domain. We make this dedication for the benefit of the public at large and to -the detriment of our heirs and successors. We intend this dedication to be an -overt act of relinquishment in perpetuity of all present and future rights to -this software under copyright law. -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 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. ------------------------------------------------------------------------------- -*/ diff --git a/disks/Games1MB.po b/disks/GamesWithFirmware.po similarity index 100% rename from disks/Games1MB.po rename to disks/GamesWithFirmware.po diff --git a/docs/screen_config.png b/docs/screen_config.png new file mode 100644 index 0000000000000000000000000000000000000000..a488392a5b12fd85e19c44244adf05ad3095cb2c GIT binary patch literal 65970 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%0Ab<+sZMLpM(BiL^YYEeOXh)1}Q?-0B{xz!z*i;`Y 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

8SzcXWEg4<_N3D;UuBO5a10;3AogI&D)L_A< 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­s>-VqKsefIMwW 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 YY6CL#|bx5n*f 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 XWT&7d4n8LbOzA@Zon7}( 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⋘YXYz&p9&hBl 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 literal 0 HcmV?d00001 diff --git a/fonts/DroidSans.ttf b/fonts/DroidSans.ttf deleted file mode 100644 index 767c63ad000e3eea20f3cb7a43ba9f4154ed7a5d..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 190044 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%0D=-PBu&{ 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

t{GVzT*WUhU>N*S-D!@jA{mzETcBr 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 D2>n@| diff --git a/fonts/PrintChar21.ttf b/fonts/PrintChar21.ttf deleted file mode 100644 index 7204e971878966081d007e0fbf644e1d99e28086..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 501992 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 z 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�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%0Dw~UCs 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

    ;@enH`FZ*5_qQXW5KtiC=;-n8Z*z?4mwf@4!I}W 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 z<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW z<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW z<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW z<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW z<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<$vXW<^TKV z2EYC{{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N z{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N z{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N z{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N z{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N{x|+N z{x|+N{x|+N{x|+N{x|+N{x|+N?l;~y-Z$Pi-Z$Pi-Z$Pi-Z$Pi-Z$Pi-Z$Pi-Z$Pi z-Z$Pi-Z$Pi-Z#!S&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C z&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C z&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C z&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C&Nt3C&Nt3?&Uemt&Uemt&Uemt z&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt z&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt z&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt z&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt z&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt&Uemt z&Uemt&Ufeeo%h{&e&>GYe&>GYe&>GYe&>GYe&>GYe&>GYe&>GYedm1VeCK@Ue0N^o zdEa^8dEa^8dEa^8dEa^8dEa^8dEa^8dEa^8dEYtTIo~f!TsQVa6h;o+z;*t_k;Vv{osB$ z&kz0w|AYU*|KNY{KlmT~5B>-Lga5(*;D7Kx_#gZa{s;eq|H1#@fABx}AN&vg2mgcr z!T;cY@IUw;{15&I|AYU*|KNY{KlmT~5B>-Lga5(*;D7Kx_#gZa{s;eq|H1#@fABx} zAN&vg2mgcr!T;cY@IUw;{15&I|AYU*|KNY{KlmT~5B>-Lga5(*;D7Qz`Jenx{wM#F z|H=R4fAT;1pZrh$C;yZG$^Ybk@;~{X{7?QT|C9g8|Kxx2Klz{hPyQ$WlmE&8b z{Ga@PKNxRaeg04WPySE-PySE-PyW9jeDyf$^MCSx^8fu{;r@60e?O>au0H=K|KAUI z`MUk*|K$JV|K$JV|K$JV|K$JV|K$JV|K$JV|K$JV|K$JV|K$JV|K$JV|K$JV|K$JV z|K$JV|K$JV|K$JV|K$JV|K$JV|K$JV|K$JV|K$JV|K$JV|K$JV|K$JV|K$JV|K$JV z|K$JV|K$JV|K$JV|K$JV|K$JV|K$JV|K$JV|K$JV|K$JV|K$JV|K$JV|K$JV|K$JV z|K$JV|K$JV|K$JV|K$JV|K$JV|K$JsL7e^b%>T*%$^Xg!$^Xg!$^Xg!$^Xg!$^Xg! z$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg! z$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg! z$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg! z$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^Xg! z$^Xg!$^Xg!$^Xg!$^Xg!$^Xg!$^XUw_g0v{$AR~TmbTl!_x_dU+yBM?<@|r~e>wkO z{9pWE&i@zxm-GL{|Mw=J`|hJ!4=j!kNf9LA&{(tA{@BV-1>hJ!4=j!kNf9LA& z{(tA{@BV)ezq|ftzyEpXyZ&eY{^OnR`k(##k9WT7fA;&IcfRX?_WPfAzUzPX`=58d z>wot9pLf3NfA;&I_pq=2e*g2%)!*-b-nshw{m(o1c>Dd#JNJ0|{mjdN$J>4T9{%_1 z_St>=&iCu~*?s%Y_v`lAef!S$>-O1w`_A|4_St>^^g_&# z=j!b~edp@!K7Hrv?LK|yr=I=Vefn~>_sjhrUaofC_j9TL@4cV#d;6=Oc?MSAKh4`K z@89N~K|g~ygFJ(A1|H}A3q32J|IXk!gK`Fc&fw<^{+_|VGx(gr$@4jRJ}1xT3o;>yBsV7f8dFshiPo8%2w3DZu zJniIZCr>-o+o{z~t#)d)Q>&d??bPb0RzJ1+snt)derolz@BQq1Kh^u$_kQ-hpFI8K z=_gM=dHTuIPo6k=;^c{wCr+L?dE(@WlP6A|ICQ(~PG>$k)!*1z8!{0@G1 z@H_b3!SCRA2j9VN#_!;_`gibK{X6)r{vG^Q{|;Mcje^S&G@eDX8aCLp52V!m6K;T<9Frc+0FP}IeB(7epgPO-HhLrlV>;M zcjY`kyBWVL=lR*q_^#|`{0`3Zvzzg|a-N@d^0f2(?DqWroAdm%lV>;V_e&?wZrJb2 z$+H{wyK?gEhJ9Cd!+r-R&u-Z7%E_}E_PcWO?1ufWoIJbr-j&^Y-@$q2ck6vu&NIJT z@4IsLbhqAj9MMT{(NYTkl=jt@j9MMT{(NY zTkpGa_H?)2cjfHqZoTiyZ|i*rCr_TA%shGW{AA|IGk*K@{jPDIpK_gML%lpvu9h^M-(DYq7dG?{{yK?gEL(_NVoA1xf_vhyObMyVV`F?J`pPTRJ=KHz%er~>>oA2l5`?>jk zZoZ$J@8{zMq@#=jQvl`TpK~e{a6OH{aiz@9)j`_vZV1^ZmW~{@#3lZ@#}b z-`|_>@6Gr3=KFi|{k{4A-F*LUzJE90znkyh&G+x-`*-vGyZQdzeE)8~e>dO1oA2Mv z_wVNWck}(b`93$_=jQv|e4m@|bMt*}zR%6~x%oaf-{o;G&`Qqk_n=fv@xcTDdi<>WQ zzPS0~=8KyzZoatr;^vE+FK)iL`SRw=n=fy^y!rCx%bPE6zP$PJ=F6KeZ@#?w^5)B% zFK@oQ`NqvRZoYBzjhk=WeBXLQ`PR+1ZoYN%t($M%eCy_0H{ZJX*3GwW XzIF4hn{VBG>*iZG-&+6s-~av(%il(i diff --git a/fonts/ProggyClean.ttf b/fonts/ProggyClean.ttf deleted file mode 100644 index 0270cdfe3caa811e1b252cdcc5b179989c1b8ad6..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 41208 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.fd == -1 || (c->flags & MISH_CLIENT_DELETE)) mish_client_delete(m, c); } + + unsigned int max_lines = m->backlog.max_lines; + if (m->flags & MISH_CLEAR_BACKLOG) { + max_lines = 1; // zero is unlimited, we don't want that + m->flags &= ~MISH_CLEAR_BACKLOG; + printf("Clearing backlog has %d lines\n", m->backlog.size); + } + /* + * It is not enough just to remove the top lines from the backlog, + * We also need to check all the current clients in case they have + * a line we are going to remove in their display... We just have to + * check the top line in this case, and 'scroll' their display to the + * next line, if applicable + */ + if (max_lines && m->backlog.size > max_lines) { + mish_line_p l; + while ((l = TAILQ_FIRST(&m->backlog.log)) != NULL) { + TAILQ_REMOVE(&m->backlog.log, l, self); + m->backlog.size--; + m->backlog.alloc -= sizeof(*l) + l->size; + // now check the clients for this line + TAILQ_FOREACH_SAFE(c, &m->clients, self, safe) { + if (c->bottom == l) + c->bottom = TAILQ_NEXT(l, self); + if (c->sending == l) + c->sending = TAILQ_NEXT(l, self); + } + free(l); + if (m->backlog.size <= max_lines) + break; + } + } } + if ((m->flags & MISH_CONSOLE_TTY) && tcsetattr(0, TCSAFLUSH, &m->orig_termios)) perror("thread tcsetattr"); diff --git a/libmish/src/mish_cmd.c b/libmish/src/mish_cmd.c index ad0ddc2..e5d07b6 100644 --- a/libmish/src/mish_cmd.c +++ b/libmish/src/mish_cmd.c @@ -166,9 +166,10 @@ mish_argv_make( state = s_copy; break; case s_copyquote: - if (*dup == '\\') + if (*dup == '\\') { state = s_skip; - else if (*dup == quote) { + dup++; + } else if (*dup == quote) { state = s_newarg; dup++; if (*dup) *dup++ = 0; diff --git a/libmish/src/mish_input.c b/libmish/src/mish_input.c index 964c1d5..a25021e 100644 --- a/libmish/src/mish_input.c +++ b/libmish/src/mish_input.c @@ -37,7 +37,7 @@ _mish_input_init( int flags = fcntl(fd, F_GETFL, NULL); if (flags == 1) { - perror("mish: input F_GETFL"); + // perror("mish: input F_GETFL"); flags = 0; } if (fcntl(fd, F_SETFL, flags | O_NONBLOCK) == -1) diff --git a/libmish/src/mish_priv.h b/libmish/src/mish_priv.h index 1509ae3..c12d236 100644 --- a/libmish/src/mish_priv.h +++ b/libmish/src/mish_priv.h @@ -134,6 +134,8 @@ enum { MISH_QUIT = (1 << 31), // the process console we started was a tty, so has terminal settings MISH_CONSOLE_TTY = (1 << 30), + // request to clear the backlog + MISH_CLEAR_BACKLOG = (1 << 29), }; typedef struct mish_t { @@ -151,6 +153,7 @@ typedef struct mish_t { pthread_t main; // todo: allow pause/stop/resume? struct { + unsigned int max_lines; // max lines in backlog (0 = unlimited) mish_line_queue_t log; unsigned int size; // number of lines in backlog size_t alloc; // number of bytes in the backlog diff --git a/libmish/src/mish_session.c b/libmish/src/mish_session.c index 47228ae..b321990 100644 --- a/libmish/src/mish_session.c +++ b/libmish/src/mish_session.c @@ -11,7 +11,7 @@ #include #include #include // for isatty etc - +#include // for isdigit #ifdef __MACH__ #include #include @@ -106,12 +106,12 @@ mish_prepare( if (tty) m->flags |= MISH_CONSOLE_TTY; if (tcgetattr(0, &m->orig_termios)) - perror("tcgetattr"); + ;//perror("tcgetattr"); struct termios raw = m->orig_termios; raw.c_iflag &= ~(ICRNL | IXON); raw.c_lflag &= ~(ECHO | ICANON | IEXTEN); // ISIG if (tcsetattr(0, TCSAFLUSH, &raw)) - perror("tcsetattr"); + ;//perror("tcsetattr"); } #endif if (!(caps & MISH_CAP_NO_TELNET)) { @@ -271,12 +271,36 @@ _mish_cmd_mish( printf(" max sizes: vector: %d input: %d\n", c->output.size, c->input.line ? c->input.line->size : 0); } - + if (argv[1] && !strcmp(argv[1], "clear")) { + printf("Clearing backlog\n"); + m->flags |= MISH_CLEAR_BACKLOG; + } + if (argv[1] && !strcmp(argv[1], "backlog")) { + if (argv[2]) { + if (!strcmp(argv[2], "clear")) { + m->flags |= MISH_CLEAR_BACKLOG; + } else if (!strcmp(argv[2], "max") && argv[3] && isdigit(argv[3][0])) { + m->backlog.max_lines = atoi(argv[3]); + printf("Backlog max lines set to %d\n", m->backlog.max_lines); + } else if (isdigit(argv[2][0])) { + m->backlog.max_lines = atoi(argv[2]); + printf("Backlog max lines set to %d\n", m->backlog.max_lines); + } else + fprintf(stderr, "Unknown backlog command '%s'\n", argv[2]); + } else { + printf("Backlog: %6d/%6d lines (%5dKB)\n", + m->backlog.size, m->backlog.max_lines, + (int)m->backlog.alloc / 1024); + } + } } MISH_CMD_NAMES(mish, "mish"); MISH_CMD_HELP(mish, - "Displays mish status.", + "[cmd...] Displays mish status.", + "backlog [clear] [max ] - show backlog status\n" + " also set the maximum lines in the backlog\n" + " (0 = unlimited)\n" "Show status and a few bits of internals."); MISH_CMD_REGISTER(mish, _mish_cmd_mish); diff --git a/libmish/src/mish_telnet.c b/libmish/src/mish_telnet.c index febd93c..d9d6fe3 100644 --- a/libmish/src/mish_telnet.c +++ b/libmish/src/mish_telnet.c @@ -176,11 +176,12 @@ mish_telnet_prepare( port = port & 0x3fff; } - int tries = 5; + int tries = 10; do { b.sin_port = htons(port); if (bind(m->telnet.listen, (struct sockaddr *)&b, sizeof(b)) == -1) { fprintf(stderr, "%s can't bind %d\n", __func__, ntohs(b.sin_port)); + perror("bind"); port = port + (random() & 0x3ff); continue; } diff --git a/libmish/tests/mish_argv_make_test.c b/libmish/tests/mish_argv_make_test.c index 3fabd23..903f1f0 100644 --- a/libmish/tests/mish_argv_make_test.c +++ b/libmish/tests/mish_argv_make_test.c @@ -47,9 +47,10 @@ mish_argv_make( state = s_copy; break; case s_copyquote: - if (*dup == '\\') + if (*dup == '\\') { state = s_skip; - else if (*dup == quote) { + dup++; + } else if (*dup == quote) { state = s_newarg; dup++; if (*dup) *dup++ = 0; diff --git a/libmish/tests/mish_cmd_test.c b/libmish/tests/mish_cmd_test.c index d48de73..dd2c9ef 100644 --- a/libmish/tests/mish_cmd_test.c +++ b/libmish/tests/mish_cmd_test.c @@ -5,7 +5,7 @@ * * SPDX-License-Identifier: Apache-2.0 */ - +#define MISH_DEBUG 1 #include "mish_cmd.c" int main() diff --git a/libmui/Makefile b/libmui/Makefile new file mode 100644 index 0000000..0a6bb9c --- /dev/null +++ b/libmui/Makefile @@ -0,0 +1,101 @@ + +CC = gcc +CPPFLAGS += -Imui +# for bsd_queue.h +CPPFLAGS += -I../libmish/src +CPPFLAGS += -DUI_HAS_XCB=1 -DUI_HAS_XKB=1 +OPTIMIZE ?= -O0 -g +CFLAGS += --std=gnu99 -Wall -Wextra +CFLAGS += $(OPTIMIZE) +CFLAGS += -Wno-unused-parameter -Wno-unused-function +# PIC is necessary for the shared library/plugin to work +CFLAGS += -fPIC + +CPPFLAGS += ${shell pkg-config --cflags pixman-1} +LDLIBS += ${shell pkg-config --libs pixman-1} +LDLIBS += $(shell pkg-config --libs \ + xcb xcb-shm xcb-randr \ + xkbcommon-x11) +LDLIBS += -lm -ldl + +O := $(BUILD_DIR)build-$(shell $(CC) -dumpmachine) +BIN := $(O)/bin +OBJ := $(O)/obj/libmui +LIB := $(O)/lib + +MUI_SRC := $(wildcard mui/*.c) +SRC := $(MUI_SRC) +MUI_OBJ := ${patsubst %, $(OBJ)/%, ${notdir ${SRC:.c=.o}}} + +SRC_VPATH := mui tests +SRC_VPATH += ../ui_gl +vpath %.c $(SRC_VPATH) + +CPPFLAGS += -I../contrib + +VERSION := ${shell git log -1 --date=short --pretty="%h %cd"} +CPPFLAGS += -DUI_VERSION="\"$(VERSION)\"" + +TARGET_LIB := $(LIB)/libmui.a + +all : $(BIN)/mui_playground $(LIB)/ui_tests.so + +.PHONY : static + +static : $(TARGET_LIB) + +ifeq ($(V),1) +Q := +else +Q := @ +endif + +$(TARGET_LIB) : $(MUI_OBJ) | $(LIB) + @echo " AR $@" + $(Q)$(AR) rcs $@ $^ + +$(OBJ)/ui_tests.o : CPPFLAGS += -Itests -I../ui_gl +$(LIB)/ui_tests.so : $(OBJ)/mii_mui_slots.o +$(LIB)/ui_tests.so : $(OBJ)/mii_mui_loadbin.o +$(LIB)/ui_tests.so : $(OBJ)/mii_mui_1mb.o +$(LIB)/ui_tests.so : $(OBJ)/mii_mui_2dsk.o + +# use a .temp file, otherwise the playground tries to reload before the file +# is fully written, and it fails. +# the ${filter} are there to make the sure object files are linked before the .a +$(LIB)/ui_tests.so : $(OBJ)/ui_tests.o $(LIB)/libmui.a | $(O) + @echo " LDSO $@" + $(Q)$(CC) $(CPPFLAGS) $(CFLAGS) -shared -fPIC -o $@.temp \ + ${filter %.o, $^} ${filter %.a, $^} $(LDLIBS) && \ + mv $@.temp $@ + +$(BIN)/mui_playground : $(OBJ)/mui_playground.o $(LIB)/libmui.a + +$(OBJ)/%.o : %.c | $(OBJ) + @echo " CC " ${filter -O%, $(CPPFLAGS) $(CFLAGS)} " $<" + $(Q)$(CC) -MMD $(CPPFLAGS) $(CFLAGS) -c -o $@ $< + +$(BIN)/% : | $(BIN) + @echo " LD $@" + $(Q)$(CC) $(LDFLAGS) -o $@ $^ $(LDLIBS) + +$(O) $(OBJ) $(BIN) $(LIB): + @mkdir -p $@ + +clean : + rm -rf $(O) + +# This is for development purpose. This will recompile the project +# everytime a file is modified. +watch : + while true; do \ + clear; $(MAKE) -j all; \ + inotifywait -qre close_write mui tests ../ui_gl; \ + done + +compile_commands.json: lsp +lsp: + { $$(which gmake) CC=gcc V=1 --always-make --dry-run all ; } | \ + sh ../utils/clangd_gen.sh >compile_commands.json + +-include $(OBJ)/*.d diff --git a/libmui/README.md b/libmui/README.md new file mode 100644 index 0000000..941461c --- /dev/null +++ b/libmui/README.md @@ -0,0 +1,132 @@ + +# What is this? +This is a contender for the World Record for Feature Creep Side Project. It is pretty high in the contender list as it's a bolt on to *another* contender for the World Record for Feature Creep Side Project (the MII Apple //e emulator). + +It is a library that duplicate a lot of a Macintosh Classic "Toolbox" APIs. It is not a complete implementation, but it is enough to make a few simple applications, also, all the bits I needed for the MII emulator. + +# Why? +Well I wanted a UI library for MII -- something without tons of dependencies, and I didn't want the typical 'game like' style with "Arrow Keys + Return + Escape" sort of menus. + +I started with Nuklear immediate mode UI, but it both look super ugly, AND is very limited as soon as you want to do something 'custom', and I didn't see myself hacking into that codebase. One thing I particularly dislike is the 'layout' engine that decide to put stuff where it wants, and it's *never* in the 'right' place, like a hard case of 'computer say so' -- typicaly result into +something like Programmer's Art. That's why Linux On The Deskop is famous for it's Pixel Perfection polished UIs. *Cough*. + +The other things I don't like with the trendy IM UIs is that they promise you that you don't have to keep a separate state around blah blah, however they forget to mention that there IS a state kept for you, based on hash values, and if you are unlucky enough to have a hash clash, you are screwed. I've seen that happen in real life, and it's not pretty to debug. + +Also, I realized I had quite a few bits that I could use to make my own library anyway, so I started to do that. + +Also, I like making my own toys. There, happy now? + +# What can it do? +Well, it has a few of the classic 'managers' that the Macintosh Toolbox had. Or the later GS/OS. However it behaves more like a modern system, it's not 'synchronous' like the old toolbox. Stuff gets redrawn even if you are clicking/dragging etc without having to 'do it yourself'. + +It is designed to draw into a 'screen' that is an ARGB buffer. You can then copy that to wherever you fancy. + * In the case of MII, it's an OpenGL texture that gets overlayed; + * In the 'example' folder, the playground demo copies it to an X11 window via a XCB 'shared' pixmap, so works great even via remote X11. +The library is 'smart', like the old OSes, it keeps track of 'invalid' regions, and only redraws what is needed, so theres very very little overdraw. + +One small drawback is that the output *has* to be ARGB -- so if you want to render say on a 16 bit framebuffer, you'll have to do the conversion yourself, +luckily, you only have to draw/convert the 'dirty' regions, so it's not too bad. + +It could be possible to 'vectorize' the rendering to vertice buffers and stuff, but really, it's not needed, it's fast enough as it is and it would fall back the 'lets redraw everything' behaviour of the IMmediate UI libraries. + +# How does it differ from the original? +Well, in terms of looks, it's kinda like I started with MacOS8/9, but removed all the grayscale bits. +Bizarelly, I think System 7 'flat' looks has aged better than the 'grayscale' look of 8/9, so I went with that. Most of the 'visible' difference are really related to the dialog's popup menus, which are a lot more 'OS8' than 'OS7'. + +In terms of of the API, one massive change is that it is fully asynchronous, and +you *can't* just spinloop and draw things in a window or GrafPort whenever you feel like it, like on the original. Instead, you change the state of the UI, and it will redraw itself when it needs to. This is a lot more like modern UI libraries in that respect. + +Many things will look familiar, if like me you were a developer back then, +I tried to use most of the names of things as is; and I even have all the +elements drawn by WDEF, CDEF, LDEF functions, which is fun. Also pretty easy +to customize. by adding your own, like in the old days. + +Also, for the API, it's all 'callback' based. No more polling. You don't have +to 'ask' the UI if something happened, it will tell you when it does. This is +also a lot more like modern UI libraries. + +It is also a lot simpler than the original in concept; everything is either a *mui_window* (Well, windows, menubars and menus) OR a *mui_control* (Menu titles, menu items, everything in windows, even separator lines). + +## Window Manager +It can create windows, and it can draw into them. Has up to 15 'layers', and can do clipping and stuff. Got the typical 'BringToFront' behaviour, and you can drag windows around. + +I deliberately limited the number of coordinate systems to 2 -- like the old one; so you get the 'screen coordinates' and the 'window content coordinates'. I was half tempted to create a fully hierarchical system, but realized it was really not neeeded, and would just make things more complicated. + +It's a 'smart' window manager, it keeps track of an 'invalid' list of rectangles, and clips to that when redrawing, so it doesn't redraw the whole window every time, yeah, like the original. None of that 'lets redraw absolutely everything every frame' stuff. + - It's missing bits like 'zooming' (TODO), and 'resizing' (TODO). + - It doesn't do transparent windows. It is by design, it draws windows 'top down' to optimize clipping -- adding transparency wouldn't be hard, but I'd have to draw the windows 'bottom up' to handle blending, and we'd revert back to drawing a lot of stuff for very little return. + - Also, you can always alpha blend the whole *ui* 'screen' to wherever you want, so it's not like you can't have transparency. + +## Menu Manager +Menubar, menus, checkmarks, keyboard shortcuts, all that stuff. Made to looks like System 7/8, or GS/OS. This was the most complicated bit to do, and it's still not perfect -- turns out the original had quite a few features that are not obvious at first glance. + - It has hierarchical menus, they don't behave *perfectly* like the original yet, I'll have to revisit that for a better match. + - It's missing displaying and scrolling super large popups (TODO). + + It will call your 'action' callback when you click on a menu item, or when you press the key of a keyboard shortcut. Easy, peasy. + + There is half baked support for sticky menus, but it's disabled for now as it's not quite right yet. + +## Control Manager +Buttons, checkboxes, radio buttons, scrollbars (vertical), wrapping textboxes, all that stuff. + - It's missing bits like Edit Field (TODO), and a Slider. +## List Manager +More or less hard coded to display filenames so far, but plain lists are actually easier than this so. Handle arrow keys/page up/down, scroll wheel, etc. + - It's missing a way to 'compress' the font and/or use ellipsis abreviations (TODO) when the item text is too long. +## Alerts +It has the typical 'Cancel'+'OK' alert. + - Could do with more types of alerts (TODO). +## Standard File +It has the classic 'Open' a file dialog. Haven't needed the other one. yet. + - Could do with a 'Save' dialog (TODO). + - Maybe a 'period correct' way to handle previously visited folders... +## Resource Manager +Nope! Not there; I'd need some sort of ResEdit and stuff -- and now that is *ONE* Feature Creep Too Far thank you very much. +I have a vague idea of making some sort of MessagePack format for resources, but that's for another day. + +# So, what *are* the dependencies? +Well, *external* dependencies are just *libpixman* -- that's it. It's a library that does pixel pushing, it's used everywhere, it's massively optimized, and it has 'regions' which are super useful for clipping -- they aren't *as good* as the regions in QuickDraw, but they are good enough. + +Other bits I used to make it are rolled into the source code anyway, so it's not like you need to install them. +* libpixman: as mentioned, you'll need libpixman-dev installed. +* [libcg](https://github.com/xboot/libcg): a small antialiased renderer that looks vaguely like *cairo* but comes in just 2 files, with a nice licence. It's likely far from being the most optimized, but it's largely good enough. +* [stb_truetype.h](https://github.com/nothings/stb): a small library to load truetype fonts, it's used to load the system fonts. +* stb_ttc.h: my own extension to stb_truetype.h, makes up a font/glyph dictionary with hash tables, font textures etc. + +That's it, all the other bits I already had 'in stock' -- the 2D geometry bits I made 25+ years ago, they were rolled in [libc3](https://github.com/buserror/simreprap) as well. + +# How do I build it? +It's a simple Makefile, so you just need to do a 'make' in the root directory. It will build the library, and the tests/demos/samples. +To build the tests/demos/samples, you'll need: +* xcb xcb-shm xcb-randr xkbcommon-x11 -- this is just to run the 'playground' demo, which is a simple window with a few buttons and stuff. +## Nvidia Driver gotcha +* *Note* that if you use the nvidia binary driver (I do), you will need to add a flag to your config, otherwise the playground won't work. + - Add: 'Option \"AllowSHMPixmaps\" \"1\"' to the "Device" In your /etc/X11/xorg.conf file. + +# How do I use it? +Well the best way is to hack around *mui_playground.c* and *ui_tests.c*. It's a simple window with a few buttons and stuff, and it's a good place to start. + +The cool thing about ui_playground is that it loads ui_test.so as a *plugin* and auto-reload it if it detects a change. So you can hack around ui_test.c, and it will reload it and run it again! You can code a new dialog insanely fast with that, pretty must as fast as you would with a resource editor. + +A good trick is to use 'make watch' on the *libmui* directory in a terminal tab, and it will rebuild the library and the playground automatically when you change something, that with the 'auto save' of your editor, and you will have a constantly building/running playground as you hack around. + +Have fun! + +# FAQ! + * Ok so, is there a dark mode? + - nope + * Fine, but at least it is themable, right? + - nope + * Tranparent Windows and a Cube Effect ?!@?! + - nope + * What, menu aren't sticky? They don't stay up when you release the mouse button? Common throw us a bone here man, it's 2023! + - nope + * But, is it at least using the super trendy tools? cmake? meson? ninja? + - nope + * What about autotools, seriously? + - nope + * Are there any bindings for any of the fancy new languages? Rust? Go? Python? + - nope + * Is it using any of the super trendy framerworks? GTK? QT? + - nope + * What about SDL? It's cross Platform, it is Ugly Everywhere Equally! + - nope + * And Wayland then? Wayland is The Future after all! + - nope diff --git a/libmui/fonts/Cairo.ttf b/libmui/fonts/Cairo.ttf new file mode 100644 index 0000000000000000000000000000000000000000..7e7213c9ff9ca8a1d4652409553c75e674410c86 GIT binary patch literal 66044 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 literal 0 HcmV?d00001 diff --git a/libmui/fonts/Charcoal_mui_italic.sfd b/libmui/fonts/Charcoal_mui_italic.sfd new file mode 100644 index 0000000..8c6d33a --- /dev/null +++ b/libmui/fonts/Charcoal_mui_italic.sfd @@ -0,0 +1,31334 @@ +SplineFontDB: 3.2 +FontName: Charcoal +FullName: Charcoal +FamilyName: Charcoal +Weight: Book +Copyright: (c)1995-1997 The Font Bureau, Inc. All Rights Reserved.\n* 2023 Michel Pollet -- Added some glyphs for libmui +Version: 3.2 +ItalicAngle: 0 +UnderlinePosition: 0 +UnderlineWidth: 0 +Ascent: 1536 +Descent: 512 +InvalidEm: 0 +sfntRevision: 0x00010000 +LayerCount: 2 +Layer: 0 1 "Back" 1 +Layer: 1 1 "Fore" 0 +XUID: [1021 220 -731427345 9134032] +StyleMap: 0x0040 +FSType: 0 +OS2Version: 0 +OS2_WeightWidthSlopeOnly: 0 +OS2_UseTypoMetrics: 0 +CreationTime: 830956166 +ModificationTime: 1699771305 +PfmFamily: 81 +TTFWeight: 400 +TTFWidth: 5 +LineGap: 171 +VLineGap: 0 +Panose: 0 0 0 0 0 0 0 0 0 0 +OS2TypoAscent: 1536 +OS2TypoAOffset: 0 +OS2TypoDescent: -512 +OS2TypoDOffset: 0 +OS2TypoLinegap: 171 +OS2WinAscent: 2048 +OS2WinAOffset: 0 +OS2WinDescent: 512 +OS2WinDOffset: 0 +HheadAscent: 2048 +HheadAOffset: 0 +HheadDescent: -512 +HheadDOffset: 0 +OS2SubXSize: 1434 +OS2SubYSize: 1331 +OS2SubXOff: 0 +OS2SubYOff: 293 +OS2SupXSize: 1434 +OS2SupYSize: 1331 +OS2SupXOff: 0 +OS2SupYOff: 928 +OS2StrikeYSize: 102 +OS2StrikeYPos: 530 +OS2Vendor: 'Alts' +OS2UnicodeRanges: 00000000.00000000.00000000.00000000 +MarkAttachClasses: 1 +DEI: 91125 +TtTable: prep +NPUSHB + 98 + 63 + 37 + 63 + 38 + 79 + 37 + 3 + 39 + 37 + 38 + 31 + 39 + 37 + 38 + 31 + 38 + 37 + 38 + 31 + 36 + 37 + 38 + 31 + 15 + 37 + 47 + 37 + 95 + 37 + 143 + 37 + 191 + 37 + 208 + 37 + 6 + 208 + 72 + 1 + 116 + 117 + 160 + 15 + 16 + 86 + 1 + 115 + 34 + 224 + 15 + 26 + 42 + 160 + 15 + 97 + 42 + 160 + 15 + 64 + 51 + 80 + 51 + 96 + 51 + 112 + 51 + 128 + 51 + 144 + 51 + 6 + 45 + 47 + 160 + 15 + 95 + 47 + 1 + 80 + 58 + 1 + 208 + 58 + 1 + 58 + 42 + 224 + 15 + 42 + 30 + 200 + 15 + 43 + 35 + 160 + 15 + 34 + 35 +PUSHW_1 + 416 +NPUSHB + 20 + 15 + 127 + 46 + 1 + 144 + 46 + 1 + 16 + 62 + 32 + 62 + 48 + 62 + 3 + 33 + 31 + 224 + 15 + 35 + 31 +PUSHW_1 + 352 +NPUSHB + 51 + 15 + 30 + 31 + 224 + 15 + 41 + 31 + 224 + 15 + 63 + 211 + 1 + 204 + 205 + 224 + 15 + 16 + 90 + 80 + 90 + 96 + 90 + 112 + 87 + 144 + 89 + 144 + 90 + 6 + 110 + 111 + 160 + 15 + 113 + 114 + 18 + 31 + 76 + 75 + 224 + 15 + 15 + 74 + 31 + 74 + 2 + 48 + 95 + 1 + 68 + 69 +PUSHW_1 + 288 +NPUSHB + 12 + 15 + 63 + 69 + 79 + 69 + 2 + 73 + 89 + 224 + 15 + 65 + 22 +PUSHW_1 + 416 +PUSHB_3 + 15 + 71 + 22 +PUSHW_1 + 288 +PUSHB_7 + 15 + 84 + 22 + 160 + 15 + 27 + 25 +PUSHW_1 + 352 +PUSHB_3 + 15 + 25 + 23 +PUSHW_1 + 352 +PUSHB_3 + 15 + 24 + 22 +PUSHW_1 + 288 +NPUSHB + 17 + 15 + 48 + 49 + 224 + 15 + 80 + 59 + 208 + 59 + 2 + 112 + 59 + 128 + 59 + 2 + 63 + 22 +PUSHW_1 + 416 +PUSHB_3 + 15 + 60 + 45 +PUSHW_1 + 288 +PUSHB_3 + 15 + 23 + 22 +PUSHW_1 + 416 +PUSHB_3 + 15 + 22 + 20 +PUSHW_1 + 352 +PUSHB_7 + 15 + 57 + 20 + 224 + 15 + 28 + 21 +PUSHW_1 + 288 +NPUSHB + 74 + 15 + 21 + 20 + 224 + 15 + 20 + 22 + 224 + 15 + 16 + 22 + 1 + 48 + 103 + 1 + 78 + 77 + 15 + 31 + 63 + 77 + 1 + 15 + 112 + 31 + 112 + 2 + 15 + 11 + 55 + 31 + 13 + 12 + 46 + 31 + 16 + 12 + 1 + 95 + 11 + 159 + 11 + 223 + 11 + 3 + 11 + 10 + 38 + 31 + 82 + 81 + 38 + 31 + 63 + 81 + 1 + 78 + 77 + 46 + 31 + 5 + 4 + 46 + 31 + 3 + 2 + 46 + 31 + 1 + 0 + 38 + 31 + 50 + 1 +PUSHW_1 + 511 +SCANCTRL +SCANTYPE +MPPEM +GTEQ +IF +PUSHB_1 + 128 +SCVTCI +EIF +SVTCA[y-axis] +CALL +CALL +CALL +CALL +DELTAC1 +CALL +CALL +DELTAC3 +DELTAC1 +CALL +CALL +DELTAC1 +DELTAC1 +CALL +DELTAC1 +SVTCA[x-axis] +DELTAC1 +CALL +CALL +CALL +CALL +CALL +CALL +CALL +CALL +DELTAC2 +DELTAC3 +CALL +CALL +CALL +CALL +CALL +CALL +CALL +CALL +DELTAC1 +CALL +DELTAC1 +DELTAC1 +CALL +CALL +CALL +DELTAC1 +CALL +DELTAC1 +SVTCA[y-axis] +CALL +CALL +CALL +CALL +DELTAC1 +DELTAC3 +DELTAC1 +CALL +CALL +CALL +CALL +DELTAC2 +DELTAC3 +DELTAC1 +CALL +DELTAC1 +CALL +CALL +CALL +DELTAC1 +CALL +DELTAC1 +DELTAC1 +CALL +CALL +CALL +CALL +DELTAC1 +EndTTInstrs +TtTable: fpgm +PUSHB_3 + 16 + 31 + 15 +FDEF +SWAP +DUP +ROLL +ROLL +RCVT +LT +IF +POP +POP +ELSE +RCVT +WCVTP +EIF +ENDF +FDEF +MPPEM +GT +IF +RCVT +WCVTP +ELSE +POP +POP +EIF +ENDF +FDEF +SWAP +DUP +ROLL +ROLL +RCVT +LT +IF +POP +POP +ELSE +RCVT +PUSHB_1 + 64 +SUB +WCVTP +EIF +ENDF +EndTTInstrs +ShortTable: cvt 229 + 0 + -25 + 0 + -25 + 0 + -25 + -1 + -1 + -1 + -1 + 1509 + 1530 + 1151 + 1180 + 1509 + 1532 + 1509 + 1530 + -342 + -362 + 340 + 342 + 319 + 332 + 309 + 338 + 235 + 322 + 355 + -1 + 232 + 213 + 174 + 191 + 225 + 230 + 1323 + 1354 + 1317 + 1335 + 57 + 186 + 252 + 255 + -1 + 244 + 185 + 217 + 245 + 285 + -1 + 82 + 213 + 60 + 31 + 308 + 307 + 294 + 245 + 292 + 235 + 295 + 338 + 336 + 45 + 319 + 319 + 23 + 238 + 226 + 799 + 326 + 326 + 184 + 1327 + 144 + 158 + 580 + 562 + 463 + 66 + 590 + 569 + 738 + 340 + 340 + 336 + 707 + 258 + 170 + 518 + 354 + 202 + 209 + 164 + 619 + 272 + 223 + 242 + 227 + 209 + 349 + 152 + -178 + 363 + 186 + 279 + 263 + 279 + 291 + 188 + 169 + 696 + 209 + 175 + 197 + 195 + 143 + 160 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + -1 + 268 + 208 + 168 + 168 + 172 + 170 + 108 + 106 + 180 + 570 + 131 + 1690 + 120 + 140 + 137 + 156 + 163 + 189 + -171 + 288 + 912 + 280 + 598 + 226 + 244 + 220 + 250 + 130 + 48 +EndShort +ShortTable: maxp 16 + 1 + 0 + 456 + 96 + 7 + 70 + 4 + 2 + 16 + 22 + 32 + 0 + 326 + 450 + 3 + 1 +EndShort +LangName: 1033 "+AKkA-1995-1997 The Font Bureau, Inc. All Rights Reserved." "" "Regular" "Charcoal" "" "3.2" "" "Charcoal+AKoA is a trademark of Apple Computer, Inc." +Encoding: Custom +UnicodeInterp: none +NameList: AGL For New Fonts +DisplaySize: -96 +AntiAlias: 1 +FitToEm: 1 +WinInfo: 247 19 9 +BeginPrivate: 0 +EndPrivate +BeginChars: 487 469 + +StartChar: .notdef +Encoding: 409 -1 0 +Width: 1024 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 11 + 6 + 2 + 7 + 1 + 0 + 4 + 0 + 9 + 7 + 1 + 8 +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rp0,min,rnd,black] +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rp0,min,rnd,black] +MDAP[rnd] +MDRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +877 0 m 1,0,-1 + 147 0 l 1,1,-1 + 147 1552 l 1,2,-1 + 877 1552 l 1,3,-1 + 877 0 l 1,0,-1 +727 150 m 1,4,-1 + 727 1403 l 1,5,-1 + 297 1403 l 1,6,-1 + 297 150 l 1,7,-1 + 727 150 l 1,4,-1 +EndSplineSet +EndChar + +StartChar: .null +Encoding: 5 8 1 +AltUni2: 002400.ffffffff.0 00001d.ffffffff.0 002400.ffffffff.0 00001d.ffffffff.0 +Width: 0 +GlyphClass: 1 +Flags: W +LayerCount: 2 +EndChar + +StartChar: nonmarkingreturn +Encoding: 10 13 2 +Width: 510 +GlyphClass: 1 +Flags: W +LayerCount: 2 +EndChar + +StartChar: space +Encoding: 6 9 3 +AltUni2: 000020.ffffffff.0 000020.ffffffff.0 +Width: 510 +GlyphClass: 1 +Flags: W +LayerCount: 2 +EndChar + +StartChar: exclam +Encoding: 24 33 4 +Width: 652 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 28 + 3 + 10 + 2 + 116 + 6 + 66 + 5 + 0 + 63 + 9 + 1 + 1 + 2 + 0 + 3 + 0 + 67 + 7 + 65 + 6 + 67 + 63 + 3 + 144 + 3 + 2 + 3 + 8 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MIRP[rnd,black] +MIRP[rp0,min,rnd,black] +MIRP[rnd,black] +SRP1 +SRP2 +IP +IP +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIRP[rp0,min,rnd,white] +MIAP[rnd] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +844 1509 m 1,0,-1 + 578 514 l 1,1,-1 + 311 514 l 1,2,-1 + 505 1509 l 1,3,-1 + 844 1509 l 1,0,-1 +475 0 m 1,4,-1 + 178 0 l 1,5,-1 + 252 319 l 1,6,-1 + 548 319 l 1,7,-1 + 475 0 l 1,4,-1 +EndSplineSet +EndChar + +StartChar: quotedbl +Encoding: 25 34 5 +Width: 945 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 34 + 2 + 6 + 3 + 7 + 16 + 1 + 53 + 0 + 2 + 53 + 3 + 5 + 53 + 4 + 6 + 53 + 7 + 0 + 84 + 16 + 3 + 144 + 3 + 2 + 3 + 4 + 84 + 15 + 7 + 31 + 7 + 2 + 7 + 8 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +SRP0 +MIRP[rnd,black] +SRP0 +MIRP[rnd,black] +SRP0 +MIRP[rnd,black] +SRP0 +MIRP[rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MDRP[rp0,min,rnd,black] +MDRP[rnd,grey] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1219 1509 m 1,0,-1 + 1046 1024 l 1,1,-1 + 838 1024 l 1,2,-1 + 901 1509 l 1,3,-1 + 1219 1509 l 1,0,-1 +743 1509 m 1,4,-1 + 570 1024 l 1,5,-1 + 362 1024 l 1,6,-1 + 425 1509 l 1,7,-1 + 743 1509 l 1,4,-1 +EndSplineSet +EndChar + +StartChar: numbersign +Encoding: 26 35 6 +Width: 1421 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 78 + 5 + 4 + 1 + 0 + 9 + 31 + 28 + 24 + 25 + 27 + 2 + 3 + 6 + 7 + 10 + 26 + 8 + 12 + 19 + 18 + 15 + 14 + 12 + 13 + 16 + 17 + 20 + 21 + 23 + 29 + 30 + 10 + 11 + 10 + 22 + 12 + 12 + 33 + 32 + 10 + 9 + 6 + 3 + 13 + 68 + 30 + 31 + 3 + 3 + 79 + 16 + 1 + 16 + 20 + 29 + 28 + 2 + 3 + 17 + 68 + 23 + 24 + 27 + 3 + 20 + 12 + 22 + 25 + 26 + 3 + 21 + 10 + 11 + 8 + 7 + 3 + 12 + 0 +SVTCA[y-axis] +MIAP[rnd] +SLOOP +ALIGNRP +MIAP[rnd] +SLOOP +ALIGNRP +MIAP[rnd] +SLOOP +ALIGNRP +MIRP[rp0,min,rnd,black] +SLOOP +ALIGNRP +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +SLOOP +ALIGNRP +MIRP[rp0,min,rnd,black] +SLOOP +ALIGNRP +SVTCA[x-axis] +SRP1 +SRP2 +IP +MDAP[rnd] +MDRP[rp0,min,rnd,black] +SLOOP +IP +SRP0 +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +MDRP[rp0,min,rnd,black] +SLOOP +IP +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1687 1151 m 1,0,-1 + 1574 911 l 1,1,-1 + 1224 911 l 1,2,-1 + 1083 598 l 1,3,-1 + 1445 598 l 1,4,-1 + 1336 358 l 1,5,-1 + 976 358 l 1,6,-1 + 811 0 l 1,7,-1 + 651 0 l 1,8,-1 + 814 358 l 1,9,-1 + 597 358 l 1,10,-1 + 434 0 l 1,11,-1 + 274 0 l 1,12,-1 + 435 358 l 1,13,-1 + 83 358 l 1,14,-1 + 187 598 l 1,15,-1 + 544 598 l 1,16,-1 + 685 911 l 1,17,-1 + 325 911 l 1,18,-1 + 429 1151 l 1,19,-1 + 793 1151 l 1,20,-1 + 954 1509 l 1,21,-1 + 1119 1509 l 1,22,-1 + 957 1151 l 1,23,-1 + 1171 1151 l 1,24,-1 + 1331 1509 l 1,25,-1 + 1495 1509 l 1,26,-1 + 1332 1151 l 1,27,-1 + 1687 1151 l 1,0,-1 +1061 911 m 1,28,-1 + 848 911 l 1,29,-1 + 705 598 l 1,30,-1 + 921 598 l 1,31,-1 + 1061 911 l 1,28,-1 +EndSplineSet +EndChar + +StartChar: dollar +Encoding: 27 36 7 +Width: 1107 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 160 + 54 + 14 + 56 + 47 + 85 + 35 + 101 + 18 + 101 + 22 + 119 + 47 + 217 + 13 + 231 + 48 + 247 + 48 + 9 + 5 + 48 + 1 + 43 + 34 + 36 + 47 + 88 + 47 + 3 + 26 + 33 + 34 + 47 + 48 + 3 + 6 + 25 + 117 + 4 + 1 + 4 + 43 + 43 + 71 + 43 + 2 + 23 + 37 + 43 + 14 + 13 + 6 + 6 + 24 + 117 + 5 + 1 + 5 + 25 + 24 + 4 + 32 + 5 + 1 + 5 + 5 + 24 + 6 + 34 + 6 + 43 + 22 + 34 + 22 + 43 + 4 + 34 + 43 + 0 + 40 + 57 + 47 + 1 + 47 + 14 + 44 + 17 + 17 + 40 + 44 + 0 + 4 + 6 + 23 + 10 + 70 + 48 + 13 + 30 + 9 + 3 + 4 + 6 + 5 + 30 + 70 + 33 + 37 + 30 + 29 + 26 + 24 + 23 + 15 + 9 + 44 + 73 + 44 + 2 + 6 + 13 + 14 + 43 + 37 + 23 + 24 + 26 + 33 + 34 + 47 + 48 + 3 + 4 + 14 + 25 + 5 + 5 + 0 + 20 + 79 + 30 + 95 + 30 + 111 + 30 + 3 + 30 + 0 + 23 + 32 + 44 + 80 + 44 + 2 + 44 + 40 + 23 + 9 + 20 + 51 +SRP0 +MDRP[rp0,rnd,white] +MDRP[rnd,grey] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +DELTAP1 +SRP1 +SRP2 +IP +MDAP[rnd] +MDRP[rp0,min,rnd,black] +SLOOP +IP +DELTAP2 +SVTCA[y-axis] +MIAP[rnd] +MDRP[min,rnd,grey] +MDRP[grey] +MDRP[grey] +MIRP[rp0,min,rnd,black] +MDRP[grey] +SROUND +MDRP[rnd,grey] +RTG +MIAP[rnd] +MDRP[min,rnd,grey] +MDRP[grey] +MDRP[grey] +MIRP[rp0,min,rnd,black] +MDRP[grey] +SROUND +MDRP[rnd,grey] +RTG +SRP1 +SRP2 +SLOOP +IP +SRP1 +SRP2 +IP +IP +DELTAP1 +SRP1 +SRP2 +IP +IP +DELTAP1 +SVTCA[x-axis] +SPVTL[orthog] +SRP0 +DELTAP1 +MDRP[rp0,min,rnd,black] +SRP0 +MDRP[rp0,min,rnd,black] +SPVTCA[x-axis] +SRP1 +DELTAP1 +SRP2 +SLOOP +IP +DELTAP1 +SRP1 +DELTAP1 +SRP2 +SLOOP +IP +DELTAP1 +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP2 +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1139 451 m 0,0,1 + 1093 252 1093 252 939.5 126 c 128,-1,2 + 786 0 786 0 564 -16 c 1,3,-1 + 514 -170 l 1,4,-1 + 379 -170 l 1,5,-1 + 429 -20 l 1,6,7 + 311 -18 311 -18 236 -4 c 128,-1,8 + 161 10 161 10 94 33 c 1,9,-1 + 153 279 l 1,10,11 + 195 256 195 256 272 228.5 c 128,-1,12 + 349 201 349 201 500 199 c 1,13,-1 + 658 668 l 1,14,15 + 649 672 649 672 638 677 c 128,-1,16 + 627 682 627 682 614 688 c 0,17,18 + 484 750 484 750 408 849 c 128,-1,19 + 332 948 332 948 371 1114 c 0,20,21 + 416 1307 416 1307 575.5 1413.5 c 128,-1,22 + 735 1520 735 1520 944 1528 c 1,23,-1 + 1001 1700 l 1,24,-1 + 1138 1700 l 1,25,-1 + 1078 1524 l 1,26,27 + 1144 1518 1144 1518 1201 1506.5 c 128,-1,28 + 1258 1495 1258 1495 1301 1481 c 1,29,-1 + 1248 1253 l 1,30,31 + 1186 1282 1186 1282 1131.5 1292.5 c 128,-1,32 + 1077 1303 1077 1303 1006 1307 c 1,33,-1 + 874 911 l 1,34,35 + 1073 811 1073 811 1121.5 698.5 c 128,-1,36 + 1170 586 1170 586 1139 451 c 0,0,1 +870 1309 m 1,37,38 + 787 1294 787 1294 722.5 1262.5 c 128,-1,39 + 658 1231 658 1231 643 1165 c 0,40,41 + 628 1100 628 1100 646.5 1060 c 128,-1,42 + 665 1020 665 1020 758 971 c 1,43,-1 + 870 1309 l 1,37,38 +845 391 m 0,44,45 + 859 451 859 451 846.5 502 c 128,-1,46 + 834 553 834 553 770 600 c 1,47,-1 + 638 203 l 1,48,49 + 708 217 708 217 766 259 c 128,-1,50 + 824 301 824 301 845 391 c 0,44,45 +EndSplineSet +EndChar + +StartChar: percent +Encoding: 28 37 8 +Width: 1741 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 129 + 4 + 1 + 11 + 5 + 11 + 8 + 4 + 12 + 7 + 15 + 4 + 19 + 11 + 23 + 11 + 26 + 4 + 30 + 20 + 1 + 27 + 5 + 27 + 8 + 20 + 12 + 21 + 15 + 20 + 19 + 27 + 23 + 27 + 26 + 20 + 30 + 37 + 15 + 42 + 17 + 53 + 15 + 54 + 16 + 56 + 17 + 23 + 14 + 14 + 16 + 4 + 43 + 89 + 3 + 87 + 36 + 89 + 10 + 15 + 50 + 89 + 28 + 87 + 57 + 89 + 21 + 5 + 63 + 17 + 1 + 17 + 1 + 14 + 18 + 14 + 52 + 14 + 112 + 14 + 128 + 14 + 5 + 48 + 14 + 1 + 48 + 14 + 1 + 14 + 49 + 15 + 1 + 15 + 63 + 16 + 1 + 63 + 16 + 1 + 16 + 6 + 18 + 88 + 46 + 90 + 54 + 88 + 63 + 24 + 159 + 24 + 2 + 24 + 6 + 0 + 88 + 32 + 90 + 40 + 88 + 31 + 6 + 48 + 6 + 176 + 6 + 3 + 6 + 60 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MIRP[min,rnd,black] +MIRP[rp0,min,rnd,grey] +MIRP[rp0,min,rnd,black] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[min,rnd,black] +MIRP[rp0,min,rnd,grey] +MIRP[rp0,min,rnd,black] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +DELTAP1 +MDRP[min,grey] +DELTAP1 +MDRP[rp0,min,rnd,black] +DELTAP1 +DELTAP1 +DELTAP1 +MDRP[rp0,min,grey] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIRP[min,rnd,black] +MIRP[rp0,min,rnd,black] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIRP[min,rnd,black] +MIRP[rp0,min,rnd,black] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIAP[rnd] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1067 1153 m 2,0,1 + 1033 1006 1033 1006 915 915.5 c 128,-1,2 + 797 825 797 825 637 825 c 0,3,4 + 479 825 479 825 397 919.5 c 128,-1,5 + 315 1014 315 1014 347 1151 c 2,6,-1 + 359 1206 l 2,7,8 + 388 1333 388 1333 505.5 1432.5 c 128,-1,9 + 623 1532 623 1532 800 1532 c 0,10,11 + 967 1532 967 1532 1039.5 1439 c 128,-1,12 + 1112 1346 1112 1346 1079 1206 c 2,13,-1 + 1067 1153 l 2,0,1 +1816 1509 m 1,14,-1 + 505 0 l 1,15,-1 + 267 0 l 1,16,-1 + 1578 1509 l 1,17,-1 + 1816 1509 l 1,14,-1 +1722 305 m 2,18,19 + 1684 139 1684 139 1567 58 c 128,-1,20 + 1450 -23 1450 -23 1295 -23 c 0,21,22 + 1127 -23 1127 -23 1049 69.5 c 128,-1,23 + 971 162 971 162 1004 303 c 2,24,-1 + 1016 358 l 2,25,26 + 1044 481 1044 481 1163 582.5 c 128,-1,27 + 1282 684 1282 684 1458 684 c 0,28,29 + 1625 684 1625 684 1696 592 c 128,-1,30 + 1767 500 1767 500 1734 358 c 2,31,-1 + 1722 305 l 2,18,19 +831 1163 m 2,32,-1 + 835 1180 l 2,33,34 + 854 1262 854 1262 843.5 1312 c 128,-1,35 + 833 1362 833 1362 761 1362 c 0,36,37 + 690 1362 690 1362 651.5 1314 c 128,-1,38 + 613 1266 613 1266 593 1180 c 2,39,-1 + 589 1163 l 2,40,41 + 573 1094 573 1094 586.5 1037.5 c 128,-1,42 + 600 981 600 981 675 981 c 0,43,44 + 742 981 742 981 777.5 1033 c 128,-1,45 + 813 1085 813 1085 831 1163 c 2,32,-1 +1488 315 m 2,46,-1 + 1492 332 l 2,47,48 + 1511 414 1511 414 1500.5 464 c 128,-1,49 + 1490 514 1490 514 1419 514 c 0,50,51 + 1348 514 1348 514 1309 466 c 128,-1,52 + 1270 418 1270 418 1250 332 c 2,53,-1 + 1246 315 l 2,54,55 + 1230 246 1230 246 1244 189.5 c 128,-1,56 + 1258 133 1258 133 1333 133 c 0,57,58 + 1399 133 1399 133 1434.5 185.5 c 128,-1,59 + 1470 238 1470 238 1488 315 c 2,46,-1 +EndSplineSet +EndChar + +StartChar: ampersand +Encoding: 29 38 9 +Width: 1346 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 143 + 6 + 7 + 9 + 10 + 11 + 16 + 10 + 19 + 5 + 23 + 5 + 26 + 8 + 30 + 25 + 11 + 26 + 16 + 27 + 19 + 21 + 23 + 21 + 26 + 22 + 35 + 29 + 52 + 37 + 23 + 153 + 11 + 234 + 14 + 17 + 36 + 49 + 68 + 49 + 84 + 49 + 102 + 49 + 4 + 49 + 27 + 30 + 37 + 4 + 0 + 46 + 37 + 52 + 41 + 55 + 58 + 55 + 74 + 55 + 90 + 55 + 185 + 6 + 6 + 15 + 55 + 52 + 6 + 4 + 0 + 18 + 34 + 33 + 33 + 18 + 46 + 3 + 9 + 43 + 31 + 21 + 11 + 61 + 31 + 9 + 1 + 0 + 0 + 47 + 30 + 1 + 49 + 27 + 30 + 37 + 4 + 0 + 46 + 29 + 6 + 41 + 52 + 37 + 55 + 3 + 15 + 55 + 52 + 6 + 4 + 18 + 40 + 88 + 32 + 24 + 1 + 128 + 24 + 208 + 24 + 2 + 24 + 46 + 88 + 18 + 24 + 18 + 34 + 12 + 33 + 34 + 31 + 3 + 43 + 3 + 2 + 3 + 0 + 65 + 58 + 24 + 12 + 64 +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[rp0,min,rnd,black] +SRP0 +MDRP[rp0,rnd,white] +MDRP[min,grey] +DELTAP1 +MDRP[rp0,rnd,grey] +MDRP[rp0,min,grey] +SRP1 +SRP2 +IP +IP +MDAP[rnd] +MIRP[rp0,min,rnd,black] +MDAP[rnd] +DELTAP1 +DELTAP2 +MIRP[rp0,min,rnd,black] +SRP1 +SLOOP +IP +DELTAP1 +SRP1 +SRP2 +SLOOP +IP +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIRP[min,rnd,black] +SRP2 +SLOOP +IP +MDAP[rnd] +MDRP[grey] +SRP1 +SRP2 +SLOOP +IP +DELTAP1 +SRP1 +SRP2 +SLOOP +IP +DELTAP1 +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1334 0 m 1,0,1 + 1258 0 1258 0 1185.5 0 c 128,-1,2 + 1113 0 1113 0 1040 0 c 1,3,4 + 1006 49 1006 49 986 82 c 128,-1,5 + 966 115 966 115 933 168 c 1,6,7 + 816 59 816 59 699 17 c 128,-1,8 + 582 -25 582 -25 475 -25 c 0,9,10 + 266 -25 266 -25 183 79.5 c 128,-1,11 + 100 184 100 184 140 358 c 0,12,13 + 176 512 176 512 278 620.5 c 128,-1,14 + 380 729 380 729 574 848 c 1,15,16 + 494 991 494 991 486.5 1060.5 c 128,-1,17 + 479 1130 479 1130 493 1190 c 0,18,19 + 527 1339 527 1339 646 1434.5 c 128,-1,20 + 765 1530 765 1530 921 1530 c 0,21,22 + 1106 1530 1106 1530 1176 1435.5 c 128,-1,23 + 1246 1341 1246 1341 1215 1206 c 0,24,25 + 1187 1085 1187 1085 1111.5 1001.5 c 128,-1,26 + 1036 918 1036 918 863 805 c 1,27,28 + 896 733 896 733 929.5 675.5 c 128,-1,29 + 963 618 963 618 1014 518 c 1,30,31 + 1110 608 1110 608 1167.5 683 c 128,-1,32 + 1225 758 1225 758 1299 885 c 1,33,-1 + 1479 762 l 1,34,35 + 1416 645 1416 645 1316 521 c 128,-1,36 + 1216 397 1216 397 1136 328 c 1,37,38 + 1220 190 1220 190 1260 122.5 c 128,-1,39 + 1300 55 1300 55 1334 0 c 1,0,1 +961 1176 m 0,40,41 + 975 1235 975 1235 958.5 1274 c 128,-1,42 + 942 1313 942 1313 875 1313 c 0,43,44 + 813 1313 813 1313 779 1269 c 128,-1,45 + 745 1225 745 1225 735 1182 c 0,46,47 + 730 1159 730 1159 738.5 1119 c 128,-1,48 + 747 1079 747 1079 786 993 c 1,49,50 + 866 1051 866 1051 908.5 1092 c 128,-1,51 + 951 1133 951 1133 961 1176 c 0,40,41 +821 360 m 1,52,53 + 767 455 767 455 737 510 c 128,-1,54 + 707 565 707 565 661 657 c 1,55,56 + 552 573 552 573 497.5 509.5 c 128,-1,57 + 443 446 443 446 428 381 c 0,58,59 + 412 311 412 311 442 271 c 128,-1,60 + 472 231 472 231 551 231 c 0,61,62 + 611 231 611 231 677 259 c 128,-1,63 + 743 287 743 287 821 360 c 1,52,53 +EndSplineSet +EndChar + +StartChar: quotesingle +Encoding: 30 39 10 +Width: 472 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 17 + 2 + 3 + 16 + 1 + 53 + 0 + 2 + 53 + 0 + 84 + 15 + 3 + 31 + 3 + 2 + 3 + 4 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MIRP[min,rnd,black] +MIRP[rnd,black] +SRP0 +MIRP[rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +734 1509 m 1,0,-1 + 587 1020 l 1,1,-1 + 356 1020 l 1,2,-1 + 436 1509 l 1,3,-1 + 734 1509 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: parenleft +Encoding: 31 40 11 +Width: 786 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 26 + 87 + 5 + 1 + 13 + 75 + 0 + 103 + 7 + 75 + 6 + 7 + 0 + 23 + 10 + 39 + 10 + 119 + 10 + 3 + 10 + 22 + 239 + 3 + 1 + 3 + 14 +SRP0 +MDRP[rp0,min,rnd,white] +DELTAP1 +MIRP[min,rnd,black] +DELTAP1 +MDRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SVTCA[y-axis] +MDAP[rnd] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +646 -178 m 1,0,1 + 404 -170 404 -170 274 61.5 c 128,-1,2 + 144 293 144 293 235 688 c 0,3,4 + 312 1022 312 1022 556.5 1285 c 128,-1,5 + 801 1548 801 1548 1047 1556 c 1,6,-1 + 1012 1407 l 1,7,8 + 884 1358 884 1358 735 1141 c 128,-1,9 + 586 924 586 924 532 688 c 0,10,11 + 473 434 473 434 524.5 227 c 128,-1,12 + 576 20 576 20 681 -29 c 1,13,-1 + 646 -178 l 1,0,1 +EndSplineSet +EndChar + +StartChar: parenright +Encoding: 32 41 12 +Width: 786 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 21 + 54 + 6 + 1 + 4 + 75 + 3 + 103 + 10 + 75 + 11 + 11 + 3 + 7 + 22 + 47 + 0 + 63 + 0 + 2 + 0 + 15 +SRP0 +MDRP[rp0,min,rnd,white] +DELTAP1 +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SVTCA[y-axis] +MDAP[rnd] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +869 688 m 0,0,1 + 780 303 780 303 546 69.5 c 128,-1,2 + 312 -164 312 -164 58 -178 c 1,3,-1 + 92 -29 l 1,4,5 + 255 27 255 27 380 214 c 128,-1,6 + 505 401 505 401 571 688 c 0,7,8 + 625 924 625 924 584.5 1141 c 128,-1,9 + 544 1358 544 1358 424 1407 c 1,10,-1 + 458 1556 l 1,11,12 + 704 1556 704 1556 830 1309.5 c 128,-1,13 + 956 1063 956 1063 869 688 c 0,0,1 +EndSplineSet +EndChar + +StartChar: asterisk +Encoding: 33 42 13 +Width: 1021 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 50 + 63 + 0 + 54 + 4 + 48 + 8 + 48 + 9 + 58 + 13 + 63 + 17 + 122 + 11 + 117 + 15 + 8 + 4 + 1 + 7 + 89 + 13 + 16 + 10 + 63 + 19 + 1 + 52 + 0 + 1 + 0 + 59 + 9 + 1 + 9 + 4 + 47 + 13 + 59 + 13 + 239 + 13 + 3 + 63 + 13 + 1 + 13 + 13 + 19 + 18 +SRP1 +SRP2 +IP +RTHG +MDAP[rnd] +DELTAP1 +DELTAP1 +ALIGNRP +MDRP[min,rnd,grey] +DELTAP1 +MDRP[rp0,min,rnd,grey] +DELTAP1 +RTG +DELTAP1 +SVTCA[y-axis] +MDAP[rnd] +MDRP[rnd,grey] +MDRP[grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1139 850 m 1,0,-1 + 892 850 l 1,1,-1 + 1049 625 l 1,2,-1 + 907 502 l 1,3,-1 + 695 805 l 1,4,-1 + 343 502 l 1,5,-1 + 257 625 l 1,6,-1 + 518 850 l 1,7,-1 + 271 850 l 1,8,-1 + 310 1020 l 1,9,-1 + 544 1020 l 1,10,-1 + 397 1231 l 1,11,-1 + 540 1354 l 1,12,-1 + 752 1051 l 1,13,-1 + 1103 1354 l 1,14,-1 + 1189 1231 l 1,15,-1 + 944 1020 l 1,16,-1 + 1179 1020 l 1,17,-1 + 1139 850 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: plus +Encoding: 34 43 14 +Width: 1100 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 25 + 3 + 104 + 1 + 4 + 69 + 8 + 104 + 10 + 7 + 70 + 0 + 104 + 10 + 1 + 69 + 5 + 104 + 7 + 47 + 4 + 1 + 4 + 4 + 13 + 12 +SRP1 +SRP2 +IP +MDAP[rnd] +DELTAP1 +MDRP[rnd,grey] +MIRP[min,rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,grey] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIRP[min,rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,grey] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1132 582 m 1,0,-1 + 790 582 l 1,1,-1 + 707 225 l 1,2,-1 + 497 225 l 1,3,-1 + 579 582 l 1,4,-1 + 237 582 l 1,5,-1 + 287 799 l 1,6,-1 + 630 799 l 1,7,-1 + 712 1155 l 1,8,-1 + 922 1155 l 1,9,-1 + 840 799 l 1,10,-1 + 1182 799 l 1,11,-1 + 1132 582 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: comma +Encoding: 35 44 15 +Width: 538 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 24 + 5 + 1 + 22 + 1 + 2 + 4 + 3 + 86 + 9 + 85 + 7 + 0 + 7 + 102 + 0 + 71 + 0 + 4 + 16 + 4 + 2 + 4 + 8 + 11 +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MIRP[min,rnd,black] +MIRP[rp0,min,rnd,black] +MDRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +430 0 m 2,0,1 + 389 -178 389 -178 274.5 -257 c 128,-1,2 + 160 -336 160 -336 36 -336 c 1,3,-1 + 65 -209 l 1,4,5 + 143 -209 143 -209 198 -174 c 128,-1,6 + 253 -139 253 -139 285 0 c 1,7,-1 + 114 0 l 1,8,-1 + 192 340 l 1,9,-1 + 508 340 l 1,10,-1 + 430 0 l 2,0,1 +EndSplineSet +EndChar + +StartChar: hyphen +Encoding: 36 45 16 +AltUni2: 002010.ffffffff.0 0000ad.ffffffff.0 002010.ffffffff.0 0000ad.ffffffff.0 +Width: 1102 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 14 + 1 + 68 + 2 + 70 + 48 + 0 + 80 + 0 + 144 + 0 + 3 + 0 + 1 + 4 +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rp0,min,rnd,black] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1135 561 m 1,0,-1 + 222 561 l 1,1,-1 + 277 799 l 1,2,-1 + 1190 799 l 1,3,-1 + 1135 561 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: period +Encoding: 37 46 17 +Width: 537 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_8 + 2 + 72 + 1 + 0 + 0 + 71 + 1 + 4 +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +424 0 m 1,0,-1 + 121 0 l 1,1,-1 + 196 326 l 1,2,-1 + 499 326 l 1,3,-1 + 424 0 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: slash +Encoding: 38 47 18 +Width: 883 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 36 + 41 + 3 + 70 + 1 + 86 + 1 + 121 + 3 + 153 + 3 + 5 + 0 + 100 + 3 + 1 + 100 + 2 + 2 + 3 + 51 + 3 + 1 + 3 + 61 + 2 + 1 + 2 + 4 + 1 + 63 + 2 + 1 + 2 + 0 + 3 + 10 +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MDAP[rnd] +DELTAP1 +MDRP[rnd,grey] +SVTCA[x-axis] +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MDRP[rp0,min,rnd,grey] +DELTAP1 +SPVTL[orthog] +SRP0 +MIRP[rp0,min,rnd,black] +SRP0 +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1287 1509 m 1,0,-1 + 129 -170 l 1,1,-1 + -80 -170 l 1,2,-1 + 1077 1509 l 1,3,-1 + 1287 1509 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: zero +Encoding: 39 48 19 +Width: 1102 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 48 + 134 + 26 + 150 + 26 + 2 + 18 + 35 + 10 + 15 + 25 + 35 + 3 + 5 + 0 + 25 + 47 + 14 + 48 + 14 + 127 + 14 + 3 + 32 + 14 + 59 + 14 + 79 + 14 + 112 + 14 + 160 + 14 + 5 + 14 + 70 + 22 + 25 + 54 + 6 + 1 + 223 + 6 + 1 + 159 + 6 + 1 + 6 + 28 +SRP0 +MDRP[rp0,min,rnd,white] +DELTAP3 +DELTAP2 +DELTAP1 +MIRP[min,rnd,black] +SROUND +MDRP[rp0,min,rnd,grey] +RTG +DELTAP1 +DELTAP2 +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1210 688 m 2,0,1 + 1127 328 1127 328 952.5 151.5 c 128,-1,2 + 778 -25 778 -25 546 -25 c 256,3,4 + 314 -25 314 -25 221 151.5 c 128,-1,5 + 128 328 128 328 211 688 c 2,6,-1 + 239 811 l 2,7,8 + 318 1155 318 1155 488.5 1343.5 c 128,-1,9 + 659 1532 659 1532 905 1532 c 256,10,11 + 1151 1532 1151 1532 1234 1343.5 c 128,-1,12 + 1317 1155 1317 1155 1238 811 c 2,13,-1 + 1210 688 l 2,0,1 +906 729 m 2,14,-1 + 915 770 l 2,15,16 + 985 1075 985 1075 974 1189 c 128,-1,17 + 963 1303 963 1303 852 1303 c 256,18,19 + 741 1303 741 1303 677.5 1189 c 128,-1,20 + 614 1075 614 1075 544 770 c 2,21,-1 + 535 729 l 2,22,23 + 465 424 465 424 477.5 314.5 c 128,-1,24 + 490 205 490 205 599 205 c 0,25,26 + 710 205 710 205 773 314.5 c 128,-1,27 + 836 424 836 424 906 729 c 2,14,-1 +EndSplineSet +EndChar + +StartChar: one +Encoding: 40 49 20 +Width: 1091 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 15 + 3 + 46 + 4 + 7 + 14 + 1 + 4 + 0 + 24 + 7 + 3 + 1 + 1 + 10 + 9 +SRP1 +SRP2 +IP +MDAP[rnd] +MDRP[min,rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +MDRP[rp0,min,rnd,grey] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +772 0 m 1,0,-1 + 485 0 l 1,1,-1 + 750 1149 l 1,2,-1 + 554 1149 l 1,3,-1 + 596 1331 l 1,4,5 + 732 1343 732 1343 817.5 1408 c 128,-1,6 + 903 1473 903 1473 932 1509 c 1,7,-1 + 1120 1509 l 1,8,-1 + 772 0 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: two +Encoding: 41 50 21 +Width: 1101 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 50 + 52 + 20 + 213 + 20 + 2 + 22 + 20 + 1 + 12 + 8 + 35 + 15 + 15 + 57 + 2 + 217 + 2 + 2 + 2 + 21 + 34 + 1 + 4 + 5 + 21 + 107 + 21 + 118 + 21 + 3 + 21 + 0 + 18 + 25 + 32 + 5 + 81 + 5 + 128 + 5 + 226 + 5 + 4 + 5 + 2 + 11 + 49 + 12 + 2 + 23 +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rp0,rnd,grey] +MIRP[rp0,min,rnd,black] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[grey] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +DELTAP1 +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MDRP[rp0,min,rnd,grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP2 +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1031 0 m 1,0,-1 + 75 0 l 1,1,-1 + 133 254 l 1,2,3 + 572 590 572 590 745 756 c 128,-1,4 + 918 922 918 922 949 1055 c 0,5,6 + 972 1155 972 1155 945 1227.5 c 128,-1,7 + 918 1300 918 1300 824 1300 c 0,8,9 + 720 1300 720 1300 677 1258.5 c 128,-1,10 + 634 1217 634 1217 603 1159 c 1,11,-1 + 333 1159 l 1,12,13 + 419 1352 419 1352 569 1441 c 128,-1,14 + 719 1530 719 1530 921 1530 c 0,15,16 + 1139 1530 1139 1530 1227.5 1406 c 128,-1,17 + 1316 1282 1316 1282 1273 1098 c 0,18,19 + 1228 901 1228 901 1020 697.5 c 128,-1,20 + 812 494 812 494 387 225 c 1,21,-1 + 1083 225 l 1,22,-1 + 1031 0 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: three +Encoding: 42 51 22 +Width: 1102 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 60 + 38 + 17 + 52 + 18 + 18 + 3 + 32 + 29 + 25 + 35 + 32 + 15 + 7 + 10 + 35 + 6 + 3 + 5 + 38 + 18 + 18 + 0 + 6 + 35 + 22 + 22 + 0 + 25 + 80 + 13 + 1 + 16 + 13 + 32 + 13 + 112 + 13 + 240 + 13 + 4 + 13 + 6 + 28 + 49 + 29 + 71 + 7 + 15 + 6 + 31 + 6 + 47 + 6 + 63 + 6 + 111 + 6 + 5 + 6 + 41 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MDRP[grey] +SROUND +MDRP[rp0,rnd,grey] +RTG +MIRP[rp0,min,rnd,black] +SRP0 +MDRP[rp0,min,rnd,black] +DELTAP1 +DELTAP2 +MIRP[min,rnd,black] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +MDAP[rnd] +IP +SVTCA[y-axis] +MIAP[rnd] +MDRP[grey] +MIRP[rp0,min,rnd,black] +MDRP[grey] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MDRP[rp0,min,rnd,grey] +SRP1 +SRP2 +IP +MDAP[rnd] +MIRP[rp0,min,rnd,black] +IP +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1129 438 m 0,0,1 + 1081 229 1081 229 916 102 c 128,-1,2 + 751 -25 751 -25 532 -25 c 0,3,4 + 339 -25 339 -25 234.5 53 c 128,-1,5 + 130 131 130 131 97 252 c 1,6,-1 + 319 379 l 1,7,8 + 329 311 329 311 392 260 c 128,-1,9 + 455 209 455 209 558 209 c 0,10,11 + 660 209 660 209 735.5 275.5 c 128,-1,12 + 811 342 811 342 839 465 c 0,13,14 + 861 561 861 561 821.5 623.5 c 128,-1,15 + 782 686 782 686 672 686 c 2,16,-1 + 530 686 l 1,17,-1 + 580 899 l 1,18,-1 + 728 899 l 2,19,20 + 825 899 825 899 885 948 c 128,-1,21 + 945 997 945 997 968 1096 c 0,22,23 + 990 1192 990 1192 965 1247.5 c 128,-1,24 + 940 1303 940 1303 852 1303 c 0,25,26 + 772 1303 772 1303 722 1273 c 128,-1,27 + 672 1243 672 1243 638 1190 c 1,28,-1 + 385 1190 l 1,29,30 + 444 1329 444 1329 577.5 1430.5 c 128,-1,31 + 711 1532 711 1532 905 1532 c 0,32,33 + 1135 1532 1135 1532 1221 1417.5 c 128,-1,34 + 1307 1303 1307 1303 1269 1139 c 0,35,36 + 1235 991 1235 991 1148.5 912.5 c 128,-1,37 + 1062 834 1062 834 977 811 c 1,38,39 + 1067 772 1067 772 1115.5 680 c 128,-1,40 + 1164 588 1164 588 1129 438 c 0,0,1 +EndSplineSet +EndChar + +StartChar: four +Encoding: 43 52 23 +Width: 1102 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_6 + 150 + 13 + 1 + 12 + 16 + 13 +PUSHW_1 + -16 +NPUSHB + 48 + 13 + 33 + 6 + 12 + 33 + 7 + 6 + 7 + 6 + 9 + 13 + 34 + 1 + 4 + 4 + 3 + 12 + 7 + 14 + 3 + 4 + 0 + 9 + 2 + 24 + 7 + 12 + 3 + 13 + 43 + 6 + 59 + 6 + 2 + 6 + 31 + 5 + 79 + 5 + 143 + 5 + 144 + 5 + 240 + 5 + 5 + 5 + 14 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MDRP[rnd,grey] +DELTAP1 +MDRP[grey] +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[rp0,min,rnd,grey] +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +MDRP[grey] +SRP2 +IP +MDAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[grey] +SPVTL[orthog] +SRP0 +MIRP[rp0,min,rnd,black] +SRP0 +SFVTCA[x-axis] +MIRP[rp0,min,rnd,black] +SVTCA[x-axis] +SHPIX +SVTCA[y-axis] +SHPIX +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP3 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1154 438 m 1,0,-1 + 1054 438 l 1,1,-1 + 953 0 l 1,2,-1 + 668 0 l 1,3,-1 + 769 438 l 1,4,-1 + 154 438 l 1,5,-1 + 206 729 l 1,6,-1 + 1112 1509 l 1,7,-1 + 1302 1509 l 1,8,-1 + 1105 657 l 1,9,-1 + 1205 657 l 1,10,-1 + 1154 438 l 1,0,-1 +823 657 m 1,11,-1 + 929 1116 l 1,12,-1 + 396 657 l 1,13,-1 + 823 657 l 1,11,-1 +EndSplineSet +EndChar + +StartChar: five +Encoding: 44 53 24 +Width: 1103 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 83 + 87 + 1 + 87 + 15 + 2 + 63 + 24 + 79 + 24 + 2 + 24 + 43 + 21 + 19 + 63 + 25 + 79 + 25 + 2 + 25 + 43 + 20 + 20 + 21 + 20 + 19 + 70 + 16 + 35 + 25 + 28 + 28 + 3 + 24 + 34 + 21 + 14 + 7 + 10 + 35 + 6 + 3 + 5 + 23 + 68 + 0 + 25 + 47 + 13 + 1 + 16 + 13 + 32 + 13 + 48 + 13 + 224 + 13 + 4 + 13 + 6 + 24 + 19 + 25 + 21 + 20 + 71 + 7 + 15 + 6 + 31 + 6 + 47 + 6 + 63 + 6 + 79 + 6 + 95 + 6 + 6 + 6 + 31 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MDRP[grey] +SROUND +MDRP[rp0,rnd,grey] +RTG +MDRP[rnd,grey] +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +DELTAP2 +MIRP[rp0,min,rnd,black] +SROUND +MDRP[rnd,grey] +RTG +SVTCA[y-axis] +MIAP[rnd] +MDRP[grey] +MIRP[rp0,min,rnd,black] +MDRP[grey] +MIAP[rnd] +MIRP[min,rnd,black] +SRP2 +IP +MDAP[rnd] +MDRP[grey] +MIRP[rp0,min,rnd,black] +SROUND +MDRP[rnd,grey] +RTG +MDRP[grey] +SPVTL[orthog] +SFVTCA[x-axis] +SRP0 +MIRP[min,rnd,black] +DELTAP1 +MDRP[grey] +SRP0 +MIRP[rp0,min,rnd,black] +DELTAP1 +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1183 551 m 0,0,1 + 1113 246 1113 246 929.5 109.5 c 128,-1,2 + 746 -27 746 -27 506 -27 c 0,3,4 + 327 -27 327 -27 221 51 c 128,-1,5 + 115 129 115 129 79 297 c 1,6,-1 + 318 426 l 1,7,8 + 335 311 335 311 394 259 c 128,-1,9 + 453 207 453 207 545 207 c 0,10,11 + 663 207 663 207 741.5 286 c 128,-1,12 + 820 365 820 365 855 520 c 0,13,14 + 889 668 889 668 847 743.5 c 128,-1,15 + 805 819 805 819 694 819 c 0,16,17 + 638 819 638 819 573.5 795.5 c 128,-1,18 + 509 772 509 772 441 721 c 1,19,-1 + 283 834 l 1,20,-1 + 466 1509 l 1,21,-1 + 1264 1509 l 1,22,-1 + 1212 1284 l 1,23,-1 + 645 1284 l 1,24,-1 + 568 1004 l 1,25,26 + 629 1044 629 1044 714 1057.5 c 128,-1,27 + 799 1071 799 1071 834 1071 c 0,28,29 + 1040 1071 1040 1071 1137.5 923.5 c 128,-1,30 + 1235 776 1235 776 1183 551 c 0,0,1 +EndSplineSet +EndChar + +StartChar: six +Encoding: 45 54 25 +Width: 1103 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +SplineSet +1210 539 m 0,0,1 + 1150 281 1150 281 978.5 128 c 128,-1,2 + 807 -25 807 -25 565 -25 c 0,3,4 + 306 -25 306 -25 213 122 c 0,5,6 + 117 270 117 270 172 573 c 0,7,8 + 175 592 175 592 182 619 c 0,9,10 + 184 627 184 627 189 647 c 0,11,12 + 200 692 200 692 203 705 c 0,13,14 + 303 1067 303 1067 510.5 1296.5 c 128,-1,15 + 718 1526 718 1526 1000 1526 c 0,16,17 + 1074 1526 1074 1526 1128 1515.5 c 128,-1,18 + 1182 1505 1182 1505 1227 1495 c 1,19,-1 + 1167 1235 l 1,20,21 + 1107 1251 1107 1251 1051.5 1261.5 c 128,-1,22 + 996 1272 996 1272 941 1272 c 0,23,24 + 810 1272 810 1272 717.5 1178.5 c 128,-1,25 + 625 1085 625 1085 576 926 c 1,26,27 + 615 977 615 977 707 1018 c 128,-1,28 + 799 1059 799 1059 871 1059 c 0,29,30 + 1088 1059 1088 1059 1177.5 923.5 c 128,-1,31 + 1267 788 1267 788 1210 539 c 0,0,1 +888 514 m 0,32,33 + 925 674 925 674 885 749.5 c 128,-1,34 + 845 825 845 825 749 825 c 0,35,36 + 663 825 663 825 582.5 750.5 c 128,-1,37 + 502 676 502 676 465 514 c 0,38,39 + 427 348 427 348 466.5 276.5 c 128,-1,40 + 506 205 506 205 605 205 c 0,41,42 + 702 205 702 205 775.5 275.5 c 128,-1,43 + 849 346 849 346 888 514 c 0,32,33 +EndSplineSet +EndChar + +StartChar: seven +Encoding: 46 55 26 +Width: 1102 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +SplineSet +1321 1253 m 1,0,1 + 1064 1103 1064 1103 855 802 c 0,2,3 + 671 537 671 537 575 119 c 0,4,5 + 561 60 561 60 550 0 c 1,6,-1 + 243 0 l 1,7,8 + 323 346 323 346 545 712.5 c 128,-1,9 + 767 1079 767 1079 1090 1284 c 1,10,-1 + 355 1284 l 1,11,-1 + 407 1509 l 1,12,-1 + 1380 1509 l 1,13,-1 + 1321 1253 l 1,0,1 +EndSplineSet +EndChar + +StartChar: eight +Encoding: 47 56 27 +Width: 1101 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 45 + 104 + 2 + 121 + 2 + 2 + 22 + 10 + 41 + 52 + 35 + 29 + 35 + 35 + 3 + 16 + 15 + 47 + 35 + 3 + 5 + 26 + 57 + 19 + 64 + 38 + 57 + 22 + 10 + 16 + 0 + 79 + 0 + 143 + 0 + 3 + 0 + 6 + 32 + 57 + 13 + 64 + 44 + 57 + 6 + 50 +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[min,rnd,black] +MIRP[rp0,rnd,grey] +MIRP[rp0,min,rnd,black] +SRP0 +MDRP[rp0,min,rnd,black] +DELTAP1 +IP +IP +MIRP[min,rnd,black] +MIRP[rp0,rnd,grey] +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +SRP2 +IP +MIRP[rp0,min,rnd,black] +MDAP[rnd] +MIRP[rp0,min,rnd,black] +IP +IP +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1124 446 m 0,0,1 + 1073 227 1073 227 914.5 101 c 128,-1,2 + 756 -25 756 -25 545 -25 c 0,3,4 + 318 -25 318 -25 224.5 90 c 128,-1,5 + 131 205 131 205 181 421 c 0,6,7 + 207 534 207 534 292.5 640 c 128,-1,8 + 378 746 378 746 501 791 c 1,9,-1 + 506 813 l 1,10,11 + 417 848 417 848 386 932 c 128,-1,12 + 355 1016 355 1016 383 1139 c 0,13,14 + 420 1300 420 1300 560 1415 c 128,-1,15 + 700 1530 700 1530 904 1530 c 0,16,17 + 1111 1530 1111 1530 1193 1416.5 c 128,-1,18 + 1275 1303 1275 1303 1234 1126 c 0,19,20 + 1208 1012 1208 1012 1139.5 929 c 128,-1,21 + 1071 846 1071 846 978 813 c 1,22,-1 + 973 791 l 1,23,24 + 1070 756 1070 756 1113.5 672 c 128,-1,25 + 1157 588 1157 588 1124 446 c 0,0,1 +959 1100 m 0,26,27 + 983 1202 983 1202 955 1251 c 128,-1,28 + 927 1300 927 1300 851 1300 c 256,29,30 + 775 1300 775 1300 725 1250 c 128,-1,31 + 675 1200 675 1200 652 1100 c 0,32,33 + 627 993 627 993 656.5 946 c 128,-1,34 + 686 899 686 899 758 899 c 0,35,36 + 821 899 821 899 877 944 c 128,-1,37 + 933 989 933 989 959 1100 c 0,26,27 +844 438 m 0,38,39 + 874 567 874 567 838 626.5 c 128,-1,40 + 802 686 802 686 709 686 c 0,41,42 + 619 686 619 686 555.5 629.5 c 128,-1,43 + 492 573 492 573 461 438 c 0,44,45 + 433 315 433 315 470 260 c 128,-1,46 + 507 205 507 205 598 205 c 0,47,48 + 682 205 682 205 749 260 c 128,-1,49 + 816 315 816 315 844 438 c 0,38,39 +EndSplineSet +EndChar + +StartChar: nine +Encoding: 48 57 28 +Width: 1103 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +SplineSet +1256 801 m 0,0,1 + 1139 413 1139 413 949 204 c 0,2,3 + 747 -20 747 -20 459 -20 c 0,4,5 + 385 -20 385 -20 331 -9.5 c 128,-1,6 + 277 1 277 1 232 11 c 1,7,-1 + 292 271 l 1,8,9 + 352 255 352 255 407.5 244.5 c 128,-1,10 + 463 234 463 234 517 234 c 0,11,12 + 652 234 652 234 746 327 c 128,-1,13 + 840 420 840 420 883 580 c 1,14,15 + 844 529 844 529 752 488 c 128,-1,16 + 660 447 660 447 587 447 c 0,17,18 + 370 447 370 447 282 582.5 c 128,-1,19 + 194 718 194 718 251 967 c 0,20,21 + 311 1225 311 1225 481 1378 c 128,-1,22 + 651 1531 651 1531 893 1531 c 0,23,24 + 1152 1531 1152 1531 1245 1384 c 0,25,26 + 1334 1248 1334 1248 1286 933 c 0,27,28 + 1283 910 1283 910 1275 876 c 0,29,30 + 1273 868 1273 868 1271 859 c 0,31,32 + 1260 814 1260 814 1256 801 c 0,0,1 +993 989 m 0,33,34 + 1031 1155 1031 1155 991.5 1226.5 c 128,-1,35 + 952 1298 952 1298 852 1298 c 0,36,37 + 755 1298 755 1298 683.5 1227.5 c 128,-1,38 + 612 1157 612 1157 573 989 c 0,39,40 + 536 829 536 829 574.5 753.5 c 128,-1,41 + 613 678 613 678 709 678 c 0,42,43 + 795 678 795 678 875.5 752.5 c 128,-1,44 + 956 827 956 827 993 989 c 0,33,34 +EndSplineSet +EndChar + +StartChar: colon +Encoding: 49 58 29 +Width: 470 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 19 + 1 + 71 + 2 + 12 + 6 + 71 + 5 + 0 + 4 + 0 + 71 + 5 + 144 + 1 + 160 + 1 + 2 + 1 + 8 +SRP0 +MDRP[rp0,min,rnd,white] +DELTAP1 +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +582 823 m 1,0,-1 + 279 823 l 1,1,-1 + 354 1149 l 1,2,-1 + 658 1149 l 1,3,-1 + 582 823 l 1,0,-1 +392 0 m 1,4,-1 + 89 0 l 1,5,-1 + 164 326 l 1,6,-1 + 468 326 l 1,7,-1 + 392 0 l 1,4,-1 +EndSplineSet +EndChar + +StartChar: semicolon +Encoding: 50 59 30 +AltUni2: 00037e.ffffffff.0 00037e.ffffffff.0 +Width: 470 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 38 + 6 + 6 + 22 + 6 + 2 + 1 + 72 + 2 + 12 + 8 + 7 + 86 + 15 + 13 + 1 + 13 + 72 + 12 + 0 + 11 + 102 + 4 + 0 + 71 + 0 + 8 + 16 + 8 + 2 + 8 + 13 + 144 + 1 + 160 + 1 + 2 + 1 + 15 +SRP0 +MDRP[rp0,min,rnd,white] +DELTAP1 +MDRP[rnd,grey] +MDRP[rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +MDRP[rp0,rnd,grey] +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MIRP[min,rnd,black] +DELTAP1 +MIRP[rp0,min,rnd,black] +MDRP[rp0,min,rnd,black] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +582 823 m 1,0,-1 + 279 823 l 1,1,-1 + 354 1149 l 1,2,-1 + 658 1149 l 1,3,-1 + 582 823 l 1,0,-1 +399 27 m 2,4,5 + 358 -152 358 -152 246.5 -244 c 128,-1,6 + 135 -336 135 -336 12 -336 c 1,7,-1 + 41 -209 l 1,8,9 + 119 -209 119 -209 169 -174 c 128,-1,10 + 219 -139 219 -139 251 0 c 1,11,-1 + 89 0 l 1,12,-1 + 164 326 l 1,13,-1 + 468 326 l 1,14,-1 + 399 27 l 2,4,5 +EndSplineSet +EndChar + +StartChar: less +Encoding: 51 60 31 +Width: 948 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 44 + 48 + 0 + 48 + 3 + 2 + 59 + 4 + 53 + 6 + 2 + 4 + 3 + 2 + 6 + 0 + 1 + 14 + 0 + 30 + 0 + 2 + 0 + 1 + 1 + 3 + 17 + 3 + 2 + 3 + 5 + 48 + 1 + 1 + 1 + 92 + 2 + 5 + 0 + 1 + 4 + 0 + 8 + 1 + 7 +SRP0 +MDRP[rp0,min,rnd,white] +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rnd,grey] +SRP1 +SRP2 +IP +SVTCA[y-axis] +MDAP[rnd] +MIRP[min,rnd,black] +DELTAP1 +IP +MDRP[rp0,min,rnd,grey] +DELTAP1 +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +SRP1 +SRP2 +IP +SRP1 +SRP2 +IP +IUP[x] +IUP[y] +DELTAP1 +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +880 94 m 1,0,-1 + 183 496 l 1,1,-1 + 229 698 l 1,2,-1 + 1113 1100 l 1,3,-1 + 1055 852 l 1,4,-1 + 502 594 l 1,5,-1 + 938 342 l 1,6,-1 + 880 94 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: equal +Encoding: 52 61 32 +Width: 1099 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 12 + 5 + 68 + 6 + 1 + 68 + 2 + 0 + 4 + 9 + 1 + 5 + 8 +SRP0 +MDRP[rp0,rnd,white] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,rnd,white] +MDRP[rnd,grey] +SVTCA[y-axis] +MDAP[rnd] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1153 715 m 1,0,-1 + 279 715 l 1,1,-1 + 334 952 l 1,2,-1 + 1208 952 l 1,3,-1 + 1153 715 l 1,0,-1 +1042 236 m 1,4,-1 + 168 236 l 1,5,-1 + 223 473 l 1,6,-1 + 1097 473 l 1,7,-1 + 1042 236 l 1,4,-1 +EndSplineSet +EndChar + +StartChar: greater +Encoding: 53 62 33 +Width: 948 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +SplineSet +112 94 m 1,0,-1 + 169 342 l 1,1,-1 + 721 594 l 1,2,-1 + 287 852 l 1,3,-1 + 344 1100 l 1,4,-1 + 1042 698 l 1,5,-1 + 995 496 l 1,6,-1 + 112 94 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: question +Encoding: 54 63 34 +Width: 853 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 53 + 5 + 2 + 5 + 31 + 20 + 31 + 134 + 2 + 149 + 2 + 170 + 16 + 234 + 16 + 7 + 25 + 69 + 22 + 34 + 26 + 69 + 29 + 15 + 10 + 117 + 34 + 72 + 33 + 0 + 63 + 37 + 1 + 0 + 22 + 112 + 19 + 224 + 19 + 2 + 19 + 33 + 9 + 32 + 65 + 10 + 33 + 63 + 25 + 79 + 25 + 2 + 25 + 36 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIRP[rp0,min,rnd,white] +MIAP[rnd] +SROUND +MDRP[rnd,grey] +RTG +MIRP[rp0,min,rnd,black] +SROUND +MDRP[rnd,grey] +RTG +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1126 1208 m 0,0,1 + 1104 1114 1104 1114 1053.5 1035 c 128,-1,2 + 1003 956 1003 956 881 889 c 0,3,4 + 785 836 785 836 725 765 c 128,-1,5 + 665 694 665 694 646 612 c 0,6,7 + 636 567 636 567 630 541.5 c 128,-1,8 + 624 516 624 516 612 465 c 1,9,-1 + 358 465 l 1,10,11 + 358 465 358 465 374 534.5 c 128,-1,12 + 390 604 390 604 401 653 c 0,13,14 + 422 743 422 743 486 825 c 128,-1,15 + 550 907 550 907 628 956 c 0,16,17 + 731 1022 731 1022 767.5 1061 c 128,-1,18 + 804 1100 804 1100 821 1174 c 0,19,20 + 835 1235 835 1235 802.5 1272 c 128,-1,21 + 770 1309 770 1309 663 1309 c 0,22,23 + 568 1309 568 1309 466 1266 c 128,-1,24 + 364 1223 364 1223 296 1176 c 1,25,-1 + 348 1399 l 1,26,27 + 435 1456 435 1456 539.5 1493 c 128,-1,28 + 644 1530 644 1530 791 1530 c 0,29,30 + 1014 1530 1014 1530 1084.5 1431.5 c 128,-1,31 + 1155 1333 1155 1333 1126 1208 c 0,0,1 +525 0 m 1,32,-1 + 234 0 l 1,33,-1 + 308 322 l 1,34,-1 + 599 322 l 1,35,-1 + 525 0 l 1,32,-1 +EndSplineSet +EndChar + +StartChar: at +Encoding: 55 64 35 +Width: 1734 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 115 + 42 + 10 + 44 + 48 + 44 + 52 + 54 + 52 + 54 + 54 + 86 + 55 + 119 + 33 + 135 + 13 + 8 + 6 + 28 + 68 + 94 + 3 + 9 + 62 + 94 + 18 + 21 + 15 + 9 + 15 + 47 + 53 + 34 + 46 + 53 + 15 + 43 + 40 + 46 + 64 + 44 + 1 + 44 + 47 + 18 + 22 + 63 + 25 + 1 + 25 + 73 + 60 + 18 + 1 + 18 + 59 + 6 + 91 + 6 + 107 + 6 + 123 + 6 + 139 + 6 + 155 + 6 + 6 + 6 + 64 + 56 + 128 + 56 + 2 + 56 + 63 + 65 + 1 + 65 + 73 + 9 + 12 + 48 + 12 + 192 + 12 + 3 + 12 + 44 + 12 + 44 + 0 + 50 + 0 + 93 + 96 + 31 + 127 + 31 + 2 + 31 + 23 + 37 + 71 + 37 + 87 + 37 + 3 + 37 + 93 + 40 + 50 + 1 + 50 + 71 +SRP0 +MDRP[rp0,min,rnd,white] +DELTAP1 +MIRP[min,rnd,black] +DELTAP1 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +IP +MDAP[rnd] +MDAP[rnd] +DELTAP1 +MIRP[min,rnd,black] +DELTAP1 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MDRP[rnd,grey] +DELTAP1 +MDRP[rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +DELTAP1 +MDRP[rnd,grey] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +IP +MDAP[rnd] +MDRP[rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[rnd,grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1812 688 m 0,0,1 + 1767 494 1767 494 1636 332 c 128,-1,2 + 1505 170 1505 170 1351 170 c 0,3,4 + 1237 170 1237 170 1190.5 213 c 128,-1,5 + 1144 256 1144 256 1151 362 c 1,6,7 + 1100 260 1100 260 1035 214 c 128,-1,8 + 970 168 970 168 859 168 c 0,9,10 + 737 168 737 168 675.5 265 c 128,-1,11 + 614 362 614 362 666 588 c 0,12,13 + 710 778 710 778 827.5 902 c 128,-1,14 + 945 1026 945 1026 1110 1026 c 0,15,16 + 1173 1026 1173 1026 1211.5 998.5 c 128,-1,17 + 1250 971 1250 971 1261 928 c 1,18,19 + 1267 956 1267 956 1274.5 988 c 128,-1,20 + 1282 1020 1282 1020 1282 1020 c 1,21,-1 + 1460 1020 l 1,22,23 + 1460 1020 1460 1020 1414 825.5 c 128,-1,24 + 1368 631 1368 631 1336 494 c 0,25,26 + 1317 412 1317 412 1329.5 382 c 128,-1,27 + 1342 352 1342 352 1383 352 c 0,28,29 + 1480 352 1480 352 1544.5 465 c 128,-1,30 + 1609 578 1609 578 1629 668 c 0,31,32 + 1704 995 1704 995 1587.5 1177.5 c 128,-1,33 + 1471 1360 1471 1360 1211 1360 c 0,34,35 + 975 1360 975 1360 750.5 1144 c 128,-1,36 + 526 928 526 928 449 594 c 0,37,38 + 366 233 366 233 485 35.5 c 128,-1,39 + 604 -162 604 -162 863 -162 c 0,40,41 + 943 -162 943 -162 1046.5 -142.5 c 128,-1,42 + 1150 -123 1150 -123 1223 -86 c 1,43,-1 + 1182 -262 l 1,44,45 + 1116 -295 1116 -295 1025 -319.5 c 128,-1,46 + 934 -344 934 -344 821 -344 c 0,47,48 + 463 -344 463 -344 309.5 -89 c 128,-1,49 + 156 166 156 166 255 594 c 0,50,51 + 350 1004 350 1004 634 1272 c 128,-1,52 + 918 1540 918 1540 1254 1540 c 0,53,54 + 1591 1540 1591 1540 1749 1319 c 128,-1,55 + 1907 1098 1907 1098 1812 688 c 0,0,1 +1181 584 m 0,56,57 + 1190 623 1190 623 1190 623 c 129,-1,58 + 1190 623 1190 623 1200 666 c 0,59,60 + 1221 756 1221 756 1191.5 809 c 128,-1,61 + 1162 862 1162 862 1072 862 c 0,62,63 + 996 862 996 862 938.5 794.5 c 128,-1,64 + 881 727 881 727 848 582 c 0,65,66 + 829 500 829 500 842.5 422 c 128,-1,67 + 856 344 856 344 942 344 c 0,68,69 + 1024 344 1024 344 1090.5 412.5 c 128,-1,70 + 1157 481 1157 481 1181 584 c 0,56,57 +EndSplineSet +EndChar + +StartChar: A +Encoding: 56 65 36 +Width: 1262 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 111 + 24 + 6 + 40 + 6 + 55 + 5 + 57 + 6 + 135 + 7 + 150 + 7 + 6 + 8 + 9 + 123 + 9 + 137 + 9 + 169 + 9 + 186 + 9 + 5 + 9 + 57 + 4 + 9 + 7 + 57 + 8 + 1 + 8 + 2 + 8 + 1 + 1 + 1 + 57 + 42 + 0 + 57 + 0 + 2 + 0 + 0 + 7 + 61 + 9 + 183 + 9 + 2 + 9 + 58 + 6 + 10 + 3 + 6 + 4 + 38 + 4 + 59 + 4 + 203 + 4 + 219 + 4 + 237 + 4 + 6 + 4 + 58 + 5 + 5 + 6 + 6 + 7 + 9 + 4 + 1 + 3 + 2 + 10 + 8 + 9 + 0 + 5 + 235 + 0 + 1 + 0 + 12 + 5 + 11 + 2 + 3 + 30 + 8 + 10 + 9 + 10 + 4 + 6 + 7 + 6 + 10 + 1 + 4 + 0 + 5 + 0 +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +MIAP[rnd] +MDRP[rnd,grey] +SRP1 +SRP2 +IP +IP +MDAP[rnd] +MDRP[rp0,rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SVTCA[x-axis] +SRP0 +MDRP[rp0,rnd,white] +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +SRP1 +SRP2 +SLOOP +IP +SPVTL[orthog] +SRP0 +MIRP[rp0,min,rnd,black] +DELTAP1 +MDRP[grey] +MDRP[grey] +SRP0 +MIRP[rp0,min,rnd,black] +DELTAP1 +SPVTL[orthog] +SRP0 +DELTAP1 +MIRP[rp0,min,rnd,black] +DELTAP1 +MDRP[grey] +MDRP[grey] +DELTAP1 +SRP0 +SFVTL[parallel] +MIRP[rp0,min,rnd,black] +DELTAP1 +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1257 0 m 1,0,-1 + 934 0 l 1,1,-1 + 914 362 l 1,2,-1 + 451 362 l 1,3,-1 + 261 0 l 1,4,-1 + 4 0 l 1,5,-1 + 824 1509 l 1,6,-1 + 1119 1509 l 1,7,-1 + 1257 0 l 1,0,-1 +897 580 m 1,8,-1 + 853 1135 l 1,9,-1 + 563 580 l 1,10,-1 + 897 580 l 1,8,-1 +EndSplineSet +EndChar + +StartChar: B +Encoding: 57 66 37 +Width: 1202 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 63 + 5 + 8 + 22 + 8 + 69 + 14 + 73 + 31 + 87 + 14 + 117 + 10 + 150 + 10 + 184 + 2 + 182 + 10 + 215 + 13 + 10 + 6 + 12 + 22 + 12 + 2 + 12 + 28 + 33 + 20 + 20 + 5 + 4 + 19 + 31 + 5 + 10 + 29 + 31 + 4 + 0 + 215 + 9 + 1 + 9 + 65 + 15 + 0 + 23 + 61 + 24 + 111 + 24 + 240 + 24 + 3 + 24 + 4 + 20 + 28 + 20 + 4 + 33 +SRP0 +MDRP[rp0,rnd,white] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[min,rnd,black] +MDRP[rp0,rnd,grey] +MIRP[rp0,min,rnd,black] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +MDAP[rnd] +MIRP[rp0,min,rnd,black] +IP +DELTAP1 +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1233 479 m 0,0,1 + 1180 250 1180 250 1018 125 c 128,-1,2 + 856 0 856 0 622 0 c 2,3,-1 + 134 0 l 1,4,-1 + 482 1509 l 1,5,-1 + 980 1509 l 2,6,7 + 1193 1509 1193 1509 1264 1428.5 c 128,-1,8 + 1335 1348 1335 1348 1302 1204 c 0,9,10 + 1273 1077 1273 1077 1200.5 1003.5 c 128,-1,11 + 1128 930 1128 930 1008 907 c 1,12,13 + 1145 887 1145 887 1210.5 776.5 c 128,-1,14 + 1276 666 1276 666 1233 479 c 0,0,1 +985 1122 m 0,15,16 + 1006 1212 1006 1212 970.5 1252 c 128,-1,17 + 935 1292 935 1292 846 1292 c 2,18,-1 + 737 1292 l 1,19,-1 + 660 958 l 1,20,-1 + 771 958 l 2,21,22 + 868 958 868 958 918 1002.5 c 128,-1,23 + 968 1047 968 1047 985 1122 c 0,15,16 +924 496 m 0,24,25 + 952 618 952 618 906.5 688 c 128,-1,26 + 861 758 861 758 721 758 c 2,27,-1 + 614 758 l 1,28,-1 + 489 217 l 1,29,-1 + 613 217 l 2,30,31 + 747 217 747 217 820 288.5 c 128,-1,32 + 893 360 893 360 924 496 c 0,24,25 +EndSplineSet +EndChar + +StartChar: C +Encoding: 58 67 38 +Width: 1103 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 55 + 21 + 9 + 21 + 18 + 53 + 15 + 182 + 9 + 198 + 9 + 5 + 48 + 14 + 1 + 14 + 40 + 17 + 30 + 13 + 40 + 10 + 11 + 27 + 40 + 24 + 30 + 0 + 40 + 3 + 1 + 14 + 0 + 0 + 32 + 0 + 64 + 0 + 160 + 0 + 192 + 0 + 5 + 0 + 6 + 21 + 21 + 21 + 2 + 21 + 21 + 223 + 6 + 1 + 6 + 28 +SRP0 +MDRP[rp0,min,rnd,white] +DELTAP2 +MIRP[min,rnd,black] +DELTAP1 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MDRP[rnd,grey] +SVTCA[y-axis] +MIAP[rnd] +MIRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MIRP[rnd,grey] +MIAP[rnd] +MIRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MIRP[rnd,grey] +DELTAP1 +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1038 41 m 1,0,1 + 1000 18 1000 18 909.5 -3.5 c 128,-1,2 + 819 -25 819 -25 707 -25 c 0,3,4 + 423 -25 423 -25 287.5 163.5 c 128,-1,5 + 152 352 152 352 237 717 c 2,6,-1 + 251 778 l 2,7,8 + 329 1118 329 1118 558.5 1324 c 128,-1,9 + 788 1530 788 1530 1062 1530 c 0,10,11 + 1180 1530 1180 1530 1256 1515.5 c 128,-1,12 + 1332 1501 1332 1501 1370 1477 c 1,13,-1 + 1314 1237 l 1,14,15 + 1279 1260 1279 1260 1215 1278 c 128,-1,16 + 1151 1296 1151 1296 1059 1296 c 0,17,18 + 888 1296 888 1296 759 1153 c 128,-1,19 + 630 1010 630 1010 571 758 c 2,20,-1 + 567 737 l 2,21,22 + 505 467 505 467 564.5 338 c 128,-1,23 + 624 209 624 209 780 209 c 0,24,25 + 885 209 885 209 964.5 231.5 c 128,-1,26 + 1044 254 1044 254 1092 274 c 1,27,-1 + 1038 41 l 1,0,1 +EndSplineSet +EndChar + +StartChar: D +Encoding: 59 68 39 +Width: 1260 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 25 + 136 + 12 + 153 + 12 + 2 + 15 + 30 + 5 + 11 + 16 + 30 + 4 + 1 + 0 + 21 + 26 + 10 + 40 + 10 + 2 + 10 + 16 + 20 + 4 + 20 +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1351 723 m 2,0,1 + 1266 356 1266 356 1089.5 178 c 128,-1,2 + 913 0 913 0 648 0 c 2,3,-1 + 134 0 l 1,4,-1 + 482 1509 l 1,5,-1 + 943 1509 l 2,6,7 + 1259 1509 1259 1509 1350.5 1314.5 c 128,-1,8 + 1442 1120 1442 1120 1365 784 c 2,9,-1 + 1351 723 l 2,0,1 +1035 733 m 2,10,-1 + 1044 774 l 2,11,12 + 1101 1022 1101 1022 1051.5 1153 c 128,-1,13 + 1002 1284 1002 1284 865 1284 c 2,14,-1 + 746 1284 l 1,15,-1 + 502 225 l 1,16,-1 + 647 225 l 2,17,18 + 782 225 782 225 874 330.5 c 128,-1,19 + 966 436 966 436 1035 733 c 2,10,-1 +EndSplineSet +EndChar + +StartChar: E +Encoding: 60 69 40 +Width: 1081 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 27 + 9 + 30 + 6 + 73 + 6 + 2 + 1 + 5 + 30 + 2 + 10 + 10 + 30 + 1 + 0 + 8 + 4 + 192 + 0 + 1 + 0 + 1 + 6 + 10 + 20 + 1 + 12 +SRP0 +MDRP[rp0,rnd,white] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MDRP[rnd,grey] +MDRP[rnd,grey] +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +SROUND +MDAP[rnd] +RTG +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1026 0 m 1,0,-1 + 134 0 l 1,1,-1 + 482 1509 l 1,2,-1 + 1336 1509 l 1,3,-1 + 1284 1284 l 1,4,-1 + 746 1284 l 1,5,-1 + 653 879 l 1,6,-1 + 1070 879 l 1,7,-1 + 1018 651 l 1,8,-1 + 600 651 l 1,9,-1 + 502 225 l 1,10,-1 + 1078 225 l 1,11,-1 + 1026 0 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: F +Encoding: 61 70 41 +Width: 1028 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 24 + 5 + 30 + 2 + 73 + 2 + 7 + 8 + 1 + 30 + 8 + 10 + 7 + 0 + 48 + 11 + 1 + 4 + 0 + 7 + 2 + 6 + 20 + 7 + 10 +SRP0 +MDRP[rp0,rnd,white] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +SROUND +MDAP[rnd] +RTG +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1284 1284 m 1,0,-1 + 746 1284 l 1,1,-1 + 653 879 l 1,2,-1 + 1070 879 l 1,3,-1 + 1018 651 l 1,4,-1 + 600 651 l 1,5,-1 + 450 0 l 1,6,-1 + 134 0 l 1,7,-1 + 482 1509 l 1,8,-1 + 1336 1509 l 1,9,-1 + 1284 1284 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: G +Encoding: 62 71 42 +Width: 1261 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 66 + 21 + 9 + 21 + 18 + 39 + 5 + 38 + 9 + 37 + 18 + 148 + 19 + 6 + 29 + 33 + 30 + 30 + 24 + 17 + 48 + 14 + 1 + 14 + 17 + 31 + 13 + 10 + 11 + 24 + 31 + 3 + 1 + 29 + 29 + 31 + 6 + 14 + 70 + 31 + 49 + 15 + 27 + 1 + 0 + 27 + 32 + 27 + 64 + 27 + 96 + 27 + 144 + 27 + 176 + 27 + 6 + 27 + 6 + 21 + 22 + 21 + 2 + 21 + 21 + 6 + 32 +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[min,rnd,black] +DELTAP1 +MDRP[rp0,min,rnd,grey] +RTG +DELTAP1 +DELTAP2 +MIRP[rp0,min,rnd,black] +SROUND +MDRP[rnd,grey] +RTG +SRP1 +SRP2 +IP +MDAP[rnd] +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +DELTAP1 +SRP1 +SRP2 +IP +MDAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1236 227 m 1,0,1 + 1146 117 1146 117 1004.5 46 c 128,-1,2 + 863 -25 863 -25 663 -25 c 0,3,4 + 387 -25 387 -25 272.5 164.5 c 128,-1,5 + 158 354 158 354 242 717 c 2,6,-1 + 261 799 l 2,7,8 + 336 1122 336 1122 557 1326 c 128,-1,9 + 778 1530 778 1530 1093 1530 c 0,10,11 + 1211 1530 1211 1530 1286.5 1506.5 c 128,-1,12 + 1362 1483 1362 1483 1395 1460 c 1,13,-1 + 1339 1217 l 1,14,15 + 1296 1241 1296 1241 1229 1268.5 c 128,-1,16 + 1162 1296 1162 1296 1067 1296 c 0,17,18 + 898 1296 898 1296 766 1171.5 c 128,-1,19 + 634 1047 634 1047 568 758 c 2,20,-1 + 563 737 l 2,21,22 + 494 436 494 436 547 320.5 c 128,-1,23 + 600 205 600 205 741 205 c 0,24,25 + 818 205 818 205 888 233.5 c 128,-1,26 + 958 262 958 262 987 299 c 1,27,-1 + 1058 608 l 1,28,-1 + 844 608 l 1,29,-1 + 890 809 l 1,30,-1 + 1370 809 l 1,31,-1 + 1236 227 l 1,0,1 +EndSplineSet +EndChar + +StartChar: H +Encoding: 63 72 43 +Width: 1258 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 31 + 3 + 30 + 8 + 73 + 8 + 5 + 6 + 10 + 6 + 10 + 1 + 5 + 0 + 0 + 20 + 9 + 127 + 2 + 1 + 63 + 2 + 111 + 2 + 2 + 2 + 5 + 8 + 3 + 20 + 5 + 12 +SRP0 +MDRP[rp0,rnd,white] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +DELTAP2 +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIAP[rnd] +MDRP[rnd,grey] +SRP1 +SRP2 +IP +SROUND +MDAP[rnd] +RTG +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1124 0 m 1,0,-1 + 808 0 l 1,1,-1 + 958 649 l 1,2,-1 + 600 649 l 1,3,-1 + 450 0 l 1,4,-1 + 134 0 l 1,5,-1 + 482 1509 l 1,6,-1 + 798 1509 l 1,7,-1 + 653 881 l 1,8,-1 + 1011 881 l 1,9,-1 + 1156 1509 l 1,10,-1 + 1473 1509 l 1,11,-1 + 1124 0 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: I +Encoding: 64 73 44 +Width: 689 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 16 + 2 + 10 + 1 + 0 + 63 + 5 + 1 + 0 + 20 + 63 + 1 + 255 + 1 + 2 + 1 + 4 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MIRP[rp0,min,rnd,grey] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +503 0 m 1,0,-1 + 187 0 l 1,1,-1 + 535 1509 l 1,2,-1 + 851 1509 l 1,3,-1 + 503 0 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: J +Encoding: 65 74 45 +Width: 882 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 30 + 5 + 2 + 21 + 2 + 2 + 7 + 40 + 10 + 31 + 6 + 40 + 3 + 1 + 14 + 10 + 15 + 20 + 31 + 14 + 112 + 14 + 2 + 14 + 63 + 6 + 112 + 6 + 2 + 6 + 16 +SRP0 +MDRP[rnd,white] +DELTAP1 +MDRP[rp0,rnd,white] +DELTAP1 +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +MIRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MIRP[rnd,grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +819 350 m 2,0,1 + 784 197 784 197 665 91.5 c 128,-1,2 + 546 -14 546 -14 353 -14 c 0,3,4 + 240 -14 240 -14 176 2 c 128,-1,5 + 112 18 112 18 74 45 c 1,6,-1 + 127 276 l 1,7,8 + 160 260 160 260 216.5 245.5 c 128,-1,9 + 273 231 273 231 334 231 c 0,10,11 + 392 231 392 231 441.5 265 c 128,-1,12 + 491 299 491 299 509 379 c 2,13,-1 + 770 1509 l 1,14,-1 + 1086 1509 l 1,15,-1 + 819 350 l 2,0,1 +EndSplineSet +EndChar + +StartChar: K +Encoding: 66 75 46 +Width: 1243 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 82 + 25 + 2 + 45 + 2 + 123 + 2 + 3 + 2 + 57 + 3 + 10 + 11 + 9 + 1 + 27 + 1 + 43 + 1 + 123 + 1 + 4 + 1 + 57 + 140 + 0 + 1 + 0 + 0 + 11 + 9 + 9 + 41 + 9 + 2 + 9 + 58 + 10 + 52 + 8 + 70 + 8 + 86 + 8 + 3 + 8 + 58 + 3 + 3 + 10 + 8 + 3 + 11 + 2 + 4 + 5 + 6 + 0 + 1 + 5 + 0 + 10 + 9 + 6 + 10 + 2 + 11 + 1 + 9 + 4 + 0 + 3 + 10 + 62 + 0 + 1 + 0 + 13 + 8 + 3 + 20 + 5 + 12 +SRP0 +MDRP[rp0,rnd,white] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rnd,grey] +DELTAP1 +MDRP[rnd,grey] +SRP1 +SRP2 +SLOOP +IP +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MDRP[rnd,grey] +MIAP[rnd] +MDRP[rnd,grey] +MDRP[rnd,grey] +SRP1 +SRP2 +SLOOP +IP +SPVTL[orthog] +SRP0 +MIRP[rp0,min,rnd,black] +DELTAP1 +SRP0 +SFVTCA[x-axis] +MIRP[rp0,min,rnd,black] +DELTAP1 +SPVTL[orthog] +SRP0 +DELTAP1 +MIRP[rp0,min,rnd,black] +DELTAP1 +SRP0 +SFVTL[parallel] +MIRP[rp0,min,rnd,black] +DELTAP1 +IUP[x] +IUP[y] +SVTCA[x-axis] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1279 0 m 1,0,-1 + 908 0 l 1,1,-1 + 625 555 l 1,2,-1 + 564 492 l 1,3,-1 + 450 0 l 1,4,-1 + 134 0 l 1,5,-1 + 482 1509 l 1,6,-1 + 798 1509 l 1,7,-1 + 652 874 l 1,8,-1 + 1236 1509 l 1,9,-1 + 1551 1509 l 1,10,-1 + 857 803 l 1,11,-1 + 1279 0 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: L +Encoding: 67 76 47 +Width: 1042 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 18 + 2 + 10 + 4 + 30 + 1 + 0 + 95 + 5 + 1 + 5 + 7 + 48 + 7 + 1 + 4 + 20 + 1 + 6 +SRP0 +MDRP[rp0,rnd,white] +MIRP[min,rnd,black] +DELTAP1 +SRP0 +MDRP[rnd,grey] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +971 0 m 1,0,-1 + 134 0 l 1,1,-1 + 482 1509 l 1,2,-1 + 798 1509 l 1,3,-1 + 502 225 l 1,4,-1 + 1023 225 l 1,5,-1 + 971 0 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: M +Encoding: 68 77 48 +Width: 1743 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 86 + 58 + 2 + 57 + 5 + 218 + 5 + 3 + 11 + 4 + 54 + 3 + 2 + 25 + 10 + 237 + 10 + 2 + 10 + 31 + 9 + 10 + 3 + 52 + 11 + 120 + 11 + 235 + 11 + 3 + 11 + 31 + 2 + 3 + 2 + 10 + 57 + 4 + 8 + 9 + 52 + 9 + 180 + 9 + 196 + 9 + 4 + 9 + 57 + 5 + 5 + 4 + 5 + 2 + 10 + 3 + 9 + 7 + 1 + 3 + 4 + 7 + 0 + 11 + 9 + 10 + 9 + 11 + 10 + 4 + 3 + 5 + 0 + 7 + 0 + 24 + 6 + 2 + 1 + 2 + 10 + 5 + 1 + 5 + 26 + 7 + 13 +SVTCA[x-axis] +SRP0 +MDRP[rp0,rnd,white] +MIRP[min,rnd,black] +DELTAP1 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +SLOOP +IP +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIAP[rnd] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +SRP1 +SRP2 +SLOOP +IP +SPVTL[orthog] +SRP0 +SFVTCA[x-axis] +MIRP[rp0,min,rnd,black] +DELTAP1 +SRP0 +MIRP[rp0,min,rnd,black] +SPVTL[orthog] +SRP0 +MIRP[rp0,min,rnd,black] +DELTAP1 +SRP0 +SFVTL[parallel] +MIRP[rp0,min,rnd,black] +DELTAP1 +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +SVTCA[y-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1620 0 m 1,0,-1 + 1331 0 l 1,1,-1 + 1531 866 l 1,2,-1 + 900 0 l 1,3,-1 + 737 0 l 1,4,-1 + 534 866 l 1,5,-1 + 334 0 l 1,6,-1 + 116 0 l 1,7,-1 + 464 1509 l 1,8,-1 + 698 1509 l 1,9,-1 + 972 410 l 1,10,-1 + 1761 1509 l 1,11,-1 + 1969 1509 l 1,12,-1 + 1620 0 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: N +Encoding: 69 78 49 +Width: 1294 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 66 + 22 + 7 + 69 + 6 + 88 + 2 + 201 + 1 + 198 + 7 + 5 + 88 + 1 + 87 + 6 + 2 + 7 + 24 + 1 + 1 + 2 + 2 + 24 + 6 + 6 + 7 + 7 + 2 + 4 + 6 + 8 + 6 + 10 + 1 + 4 + 0 + 1 + 7 + 74 + 0 + 0 + 16 + 0 + 143 + 0 + 160 + 0 + 192 + 0 + 5 + 0 + 4 + 6 + 2 + 74 + 207 + 4 + 223 + 4 + 239 + 4 + 255 + 4 + 4 + 4 + 10 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MIRP[rp0,min,rnd,black] +MDRP[grey] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +MDRP[grey] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIAP[rnd] +MDRP[rnd,grey] +SRP1 +SRP2 +IP +IP +SPVTL[orthog] +SFVTCA[y-axis] +SRP0 +MIRP[rp0,min,rnd,black] +SPVTL[orthog] +SRP0 +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +SVTCA[y-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1179 0 m 1,0,-1 + 1043 0 l 1,1,-1 + 559 948 l 1,2,-1 + 341 0 l 1,3,-1 + 116 0 l 1,4,-1 + 464 1509 l 1,5,-1 + 633 1509 l 1,6,-1 + 1096 616 l 1,7,-1 + 1302 1509 l 1,8,-1 + 1527 1509 l 1,9,-1 + 1179 0 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: O +Encoding: 70 79 50 +Width: 1266 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 98 + 38 + 5 + 39 + 8 + 39 + 9 + 40 + 12 + 71 + 19 + 72 + 26 + 86 + 19 + 89 + 26 + 104 + 24 + 121 + 26 + 134 + 9 + 167 + 8 + 168 + 12 + 201 + 27 + 216 + 27 + 231 + 8 + 233 + 12 + 250 + 16 + 250 + 27 + 19 + 54 + 18 + 1 + 18 + 31 + 10 + 11 + 57 + 25 + 1 + 25 + 31 + 3 + 1 + 231 + 0 + 1 + 0 + 21 + 80 + 14 + 112 + 14 + 2 + 32 + 14 + 1 + 0 + 14 + 32 + 14 + 80 + 14 + 112 + 14 + 145 + 14 + 208 + 14 + 6 + 14 + 70 + 247 + 22 + 1 + 22 + 21 + 191 + 6 + 223 + 6 + 2 + 151 + 6 + 223 + 6 + 2 + 6 + 28 +SRP0 +MDRP[rp0,min,rnd,white] +DELTAP1 +DELTAP2 +MIRP[min,rnd,black] +DELTAP1 +SROUND +MDRP[rp0,min,rnd,grey] +RTG +DELTAP1 +DELTAP2 +DELTAP3 +MIRP[rp0,min,rnd,black] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +DELTAP1 +MIAP[rnd] +MIRP[rp0,min,rnd,black] +DELTAP1 +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1376 707 m 2,0,1 + 1294 350 1294 350 1092.5 162.5 c 128,-1,2 + 891 -25 891 -25 632 -25 c 0,3,4 + 375 -25 375 -25 259.5 162.5 c 128,-1,5 + 144 350 144 350 225 702 c 2,6,-1 + 245 788 l 2,7,8 + 321 1116 321 1116 517 1323 c 128,-1,9 + 713 1530 713 1530 999 1530 c 0,10,11 + 1279 1530 1279 1530 1374 1320 c 128,-1,12 + 1469 1110 1469 1110 1395 788 c 2,13,-1 + 1376 707 l 2,0,1 +1063 727 m 2,14,-1 + 1072 768 l 2,15,16 + 1136 1047 1136 1047 1123 1182 c 128,-1,17 + 1110 1317 1110 1317 942 1317 c 0,18,19 + 787 1317 787 1317 708 1197 c 128,-1,20 + 629 1077 629 1077 558 768 c 2,21,-1 + 550 731 l 2,22,23 + 478 420 478 420 496.5 304 c 128,-1,24 + 515 188 515 188 681 188 c 0,25,26 + 817 188 817 188 906.5 313 c 128,-1,27 + 996 438 996 438 1063 727 c 2,14,-1 +EndSplineSet +EndChar + +StartChar: P +Encoding: 71 80 51 +Width: 1102 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 40 + 217 + 20 + 232 + 20 + 2 + 18 + 31 + 4 + 4 + 6 + 7 + 17 + 31 + 7 + 10 + 6 + 0 + 151 + 0 + 167 + 0 + 246 + 0 + 3 + 0 + 21 + 6 + 12 + 22 + 12 + 32 + 12 + 3 + 12 + 6 + 18 + 4 + 20 + 6 + 22 +SRP0 +MDRP[rp0,rnd,white] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[min,rnd,black] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +MDAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1294 999 m 2,0,1 + 1250 807 1250 807 1118.5 709.5 c 128,-1,2 + 987 612 987 612 782 612 c 2,3,-1 + 591 612 l 1,4,-1 + 450 0 l 1,5,-1 + 134 0 l 1,6,-1 + 482 1509 l 1,7,-1 + 1009 1509 l 2,8,9 + 1235 1509 1235 1509 1296.5 1391.5 c 128,-1,10 + 1358 1274 1358 1274 1322 1118 c 2,11,-1 + 1294 999 l 2,0,1 +966 977 m 2,12,-1 + 999 1122 l 2,13,14 + 1019 1208 1019 1208 984 1249 c 128,-1,15 + 949 1290 949 1290 846 1290 c 2,16,-1 + 748 1290 l 1,17,-1 + 639 819 l 1,18,-1 + 754 819 l 2,19,20 + 828 819 828 819 888 860 c 128,-1,21 + 948 901 948 901 966 977 c 2,12,-1 +EndSplineSet +EndChar + +StartChar: Q +Encoding: 72 81 52 +Width: 1266 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 93 + 11 + 12 + 27 + 12 + 39 + 19 + 73 + 37 + 89 + 37 + 122 + 37 + 135 + 20 + 218 + 37 + 250 + 38 + 9 + 9 + 32 + 10 + 18 + 54 + 29 + 249 + 29 + 2 + 29 + 31 + 21 + 11 + 57 + 36 + 247 + 36 + 2 + 36 + 41 + 3 + 14 + 0 + 9 + 5 + 3 + 1 + 3 + 57 + 14 + 14 + 3 + 9 + 3 + 0 + 17 + 0 + 21 + 32 + 25 + 64 + 25 + 2 + 0 + 25 + 32 + 25 + 64 + 25 + 80 + 25 + 96 + 25 + 112 + 25 + 136 + 25 + 145 + 25 + 176 + 25 + 208 + 25 + 10 + 25 + 33 + 21 + 135 + 17 + 151 + 17 + 2 + 17 + 39 +SRP0 +MDRP[rp0,min,rnd,white] +DELTAP1 +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +DELTAP1 +DELTAP2 +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +SLOOP +IP +MDAP[rnd] +MIRP[rp0,min,rnd,black] +DELTAP2 +MDAP[rnd] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +DELTAP1 +MIAP[rnd] +MIRP[rp0,min,rnd,black] +DELTAP1 +MIAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1372 707 m 2,0,1 + 1301 401 1301 401 1134 212 c 128,-1,2 + 967 23 967 23 769 2 c 1,3,4 + 754 -86 754 -86 780.5 -125 c 128,-1,5 + 807 -164 807 -164 904 -164 c 0,6,7 + 956 -164 956 -164 1012.5 -164 c 128,-1,8 + 1069 -164 1069 -164 1069 -164 c 1,9,-1 + 1029 -340 l 1,10,-1 + 751 -340 l 2,11,12 + 585 -340 585 -340 522.5 -236.5 c 128,-1,13 + 460 -133 460 -133 486 2 c 1,14,15 + 269 78 269 78 215 261 c 128,-1,16 + 161 444 161 444 221 707 c 2,17,-1 + 240 788 l 2,18,19 + 318 1126 318 1126 520 1328 c 128,-1,20 + 722 1530 722 1530 986 1530 c 0,21,22 + 1267 1530 1267 1530 1368.5 1331.5 c 128,-1,23 + 1470 1133 1470 1133 1391 788 c 2,24,-1 + 1372 707 l 2,0,1 +1058 727 m 2,25,-1 + 1068 768 l 2,26,27 + 1142 1090 1142 1090 1114 1203.5 c 128,-1,28 + 1086 1317 1086 1317 937 1317 c 0,29,30 + 789 1317 789 1317 709 1206.5 c 128,-1,31 + 629 1096 629 1096 554 768 c 2,32,-1 + 544 727 l 2,33,34 + 474 424 474 424 493.5 306 c 128,-1,35 + 513 188 513 188 676 188 c 0,36,37 + 808 188 808 188 899 310 c 128,-1,38 + 990 432 990 432 1058 727 c 2,25,-1 +EndSplineSet +EndChar + +StartChar: R +Encoding: 73 82 53 +Width: 1159 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 60 + 9 + 3 + 28 + 3 + 2 + 3 + 11 + 20 + 11 + 39 + 11 + 3 + 25 + 16 + 1 + 16 + 5 + 31 + 25 + 25 + 7 + 8 + 24 + 31 + 8 + 10 + 0 + 7 + 0 + 59 + 16 + 1 + 16 + 20 + 13 + 0 + 23 + 1 + 182 + 13 + 198 + 13 + 2 + 13 + 23 + 95 + 20 + 191 + 20 + 2 + 20 + 7 + 25 + 5 + 20 + 208 + 7 + 1 + 7 + 29 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[min,rnd,black] +DELTAP1 +MDRP[rp0,rnd,grey] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +MDAP[rnd] +MIRP[rp0,min,rnd,black] +IP +DELTAP1 +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1094 0 m 1,0,-1 + 782 0 l 1,1,2 + 863 365 863 365 838 521.5 c 128,-1,3 + 813 678 813 678 681 678 c 2,4,-1 + 606 678 l 1,5,-1 + 450 0 l 1,6,-1 + 134 0 l 1,7,-1 + 482 1509 l 1,8,-1 + 1035 1509 l 2,9,10 + 1237 1509 1237 1509 1299 1415 c 128,-1,11 + 1361 1321 1361 1321 1330 1184 c 2,12,-1 + 1311 1102 l 2,13,14 + 1281 973 1281 973 1189.5 895 c 128,-1,15 + 1098 817 1098 817 948 803 c 1,16,17 + 1101 750 1101 750 1142 567.5 c 128,-1,18 + 1183 385 1183 385 1094 0 c 1,0,-1 +992 1071 m 2,19,-1 + 1002 1112 l 2,20,21 + 1025 1210 1025 1210 986 1252 c 128,-1,22 + 947 1294 947 1294 852 1294 c 2,23,-1 + 749 1294 l 1,24,-1 + 657 895 l 1,25,-1 + 742 895 l 2,26,27 + 855 895 855 895 913.5 939 c 128,-1,28 + 972 983 972 983 992 1071 c 2,19,-1 +EndSplineSet +EndChar + +StartChar: S +Encoding: 74 83 54 +Width: 1072 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 104 + 43 + 18 + 36 + 37 + 82 + 15 + 82 + 36 + 103 + 1 + 104 + 20 + 117 + 36 + 133 + 34 + 169 + 2 + 186 + 16 + 177 + 36 + 226 + 36 + 12 + 6 + 35 + 22 + 35 + 37 + 35 + 3 + 35 + 0 + 32 + 11 + 16 + 1 + 169 + 16 + 1 + 16 + 13 + 19 + 19 + 32 + 13 + 0 + 4 + 3 + 22 + 26 + 40 + 29 + 30 + 25 + 40 + 22 + 11 + 7 + 40 + 10 + 30 + 6 + 40 + 3 + 1 + 35 + 0 + 32 + 74 + 16 + 1 + 16 + 13 + 19 + 112 + 25 + 1 + 25 + 71 + 0 + 28 + 31 + 13 + 32 + 13 + 2 + 32 + 13 + 48 + 13 + 64 + 13 + 144 + 13 + 4 + 13 + 32 + 27 + 6 + 71 + 19 + 38 +SRP0 +MDRP[rp0,min,rnd,white] +SROUND +MDRP[rnd,grey] +RTG +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,black] +DELTAP1 +DELTAP2 +MIRP[rp0,min,rnd,black] +SROUND +MDRP[rnd,grey] +RTG +DELTAP1 +SRP1 +SRP2 +IP +DELTAP1 +SRP1 +SRP2 +IP +SVTCA[y-axis] +MIAP[rnd] +MIRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MIRP[rnd,grey] +MIAP[rnd] +MIRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MIRP[rnd,grey] +SRP1 +SRP2 +SLOOP +IP +SRP1 +SRP2 +IP +DELTAP1 +DELTAP2 +SRP1 +SRP2 +IP +DELTAP1 +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1119 449 m 0,0,1 + 1064 209 1064 209 888.5 94.5 c 128,-1,2 + 713 -20 713 -20 448 -20 c 0,3,4 + 296 -20 296 -20 196.5 -1 c 128,-1,5 + 97 18 97 18 35 47 c 1,6,-1 + 95 307 l 1,7,8 + 140 272 140 272 247 242.5 c 128,-1,9 + 354 213 354 213 463 213 c 0,10,11 + 617 213 617 213 685.5 257 c 128,-1,12 + 754 301 754 301 778 403 c 0,13,14 + 793 469 793 469 763 520 c 128,-1,15 + 733 571 733 571 556 655 c 0,16,17 + 415 723 415 723 354 828.5 c 128,-1,18 + 293 934 293 934 335 1116 c 0,19,20 + 375 1290 375 1290 541 1410 c 128,-1,21 + 707 1530 707 1530 951 1530 c 0,22,23 + 1069 1530 1069 1530 1158.5 1509.5 c 128,-1,24 + 1248 1489 1248 1489 1283 1473 c 1,25,-1 + 1229 1243 l 1,26,27 + 1154 1264 1154 1264 1079 1280 c 128,-1,28 + 1004 1296 1004 1296 914 1296 c 0,29,30 + 788 1296 788 1296 722.5 1254 c 128,-1,31 + 657 1212 657 1212 642 1149 c 0,32,33 + 624 1069 624 1069 656.5 1028 c 128,-1,34 + 689 987 689 987 874 907 c 0,35,36 + 1047 833 1047 833 1101 719.5 c 128,-1,37 + 1155 606 1155 606 1119 449 c 0,0,1 +EndSplineSet +EndChar + +StartChar: T +Encoding: 75 84 55 +Width: 951 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 18 + 1 + 5 + 30 + 6 + 10 + 3 + 0 + 0 + 2 + 20 + 5 + 112 + 3 + 1 + 3 + 3 + 9 + 8 +SRP1 +SRP2 +IP +MDAP[rnd] +DELTAP1 +MDRP[rnd,white] +MIRP[rp0,min,rnd,black] +MDRP[rp0,min,rnd,grey] +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1239 1284 m 1,0,-1 + 934 1284 l 1,1,-1 + 638 0 l 1,2,-1 + 322 0 l 1,3,-1 + 618 1284 l 1,4,-1 + 306 1284 l 1,5,-1 + 357 1509 l 1,6,-1 + 1291 1509 l 1,7,-1 + 1239 1284 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: U +Encoding: 76 85 56 +Width: 1243 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 48 + 50 + 1 + 50 + 2 + 60 + 4 + 60 + 5 + 60 + 10 + 49 + 14 + 6 + 16 + 7 + 10 + 12 + 30 + 3 + 1 + 103 + 0 + 1 + 0 + 42 + 0 + 15 + 63 + 15 + 79 + 15 + 3 + 15 + 89 + 9 + 105 + 9 + 120 + 9 + 136 + 9 + 152 + 9 + 5 + 9 + 20 + 6 + 18 +SRP0 +MDRP[rp0,rnd,white] +MIRP[min,rnd,black] +DELTAP1 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MDRP[rnd,grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1237 436 m 2,0,1 + 1188 223 1188 223 1038.5 102.5 c 128,-1,2 + 889 -18 889 -18 633 -18 c 0,3,4 + 359 -18 359 -18 268.5 111 c 128,-1,5 + 178 240 178 240 237 494 c 2,6,-1 + 471 1509 l 1,7,-1 + 787 1509 l 1,8,-1 + 544 453 l 2,9,10 + 516 330 516 330 553 268.5 c 128,-1,11 + 590 207 590 207 717 207 c 0,12,13 + 835 207 835 207 905 262 c 128,-1,14 + 975 317 975 317 1008 459 c 2,15,-1 + 1251 1509 l 1,16,-1 + 1485 1509 l 1,17,-1 + 1237 436 l 2,0,1 +EndSplineSet +EndChar + +StartChar: V +Encoding: 77 86 57 +Width: 1160 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 91 + 57 + 5 + 117 + 5 + 144 + 5 + 3 + 40 + 3 + 57 + 1 + 54 + 2 + 51 + 5 + 55 + 6 + 71 + 1 + 105 + 2 + 7 + 0 + 58 + 6 + 1 + 58 + 5 + 5 + 6 + 153 + 3 + 1 + 3 + 22 + 4 + 2 + 22 + 5 + 5 + 4 + 5 + 2 + 3 + 1 + 2 + 0 + 6 + 4 + 0 + 3 + 10 + 63 + 8 + 1 + 1 + 6 + 0 + 2 + 4 + 3 + 156 + 5 + 1 + 5 + 5 + 0 + 3 + 60 + 0 + 1 + 63 + 0 + 1 + 0 + 32 + 3 + 1 + 0 + 3 + 47 + 3 + 51 + 3 + 180 + 3 + 208 + 3 + 5 + 3 + 7 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +DELTAP2 +MDRP[rp0,min,rnd,black] +DELTAP1 +DELTAP1 +SRP1 +SRP2 +IP +RTHG +MDAP[rnd] +RTG +DELTAP1 +SRP2 +IP +IP +SRP2 +IP +IP +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +MIAP[rnd] +MDRP[rnd,grey] +SRP1 +SRP2 +IP +SPVTL[orthog] +SFVTCA[x-axis] +SRP0 +MIRP[rp0,min,rnd,black] +MDAP[rnd] +MIRP[rp0,min,rnd,black] +DELTAP1 +SPVTL[orthog] +SRP0 +MIRP[rp0,min,rnd,black] +MDAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +SVTCA[y-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1530 1509 m 1,0,-1 + 705 0 l 1,1,-1 + 440 0 l 1,2,-1 + 327 1509 l 1,3,-1 + 669 1509 l 1,4,-1 + 730 467 l 1,5,-1 + 1270 1509 l 1,6,-1 + 1530 1509 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: W +Encoding: 78 87 58 +Width: 1796 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 142 + 63 + 14 + 1 + 22 + 10 + 39 + 1 + 56 + 0 + 51 + 1 + 54 + 4 + 57 + 5 + 54 + 7 + 7 + 15 + 3 + 22 + 8 + 22 + 11 + 52 + 3 + 57 + 8 + 57 + 11 + 6 + 3 + 58 + 3 + 2 + 9 + 8 + 9 + 1 + 73 + 0 + 89 + 0 + 2 + 0 + 58 + 60 + 12 + 1 + 12 + 11 + 12 + 9 + 2 + 132 + 3 + 150 + 3 + 228 + 3 + 3 + 3 + 22 + 10 + 10 + 11 + 5 + 6 + 22 + 7 + 8 + 7 + 3 + 10 + 9 + 1 + 12 + 0 + 2 + 10 + 9 + 11 + 11 + 3 + 0 + 1 + 0 + 16 + 9 + 32 + 9 + 48 + 9 + 3 + 9 + 4 + 10 + 5 + 7 + 6 + 16 + 8 + 1 + 8 + 9 + 8 + 10 + 8 + 6 + 18 + 6 + 95 + 6 + 127 + 6 + 143 + 6 + 148 + 6 + 6 + 6 + 13 + 8 + 3 + 11 + 3 + 5 + 6 + 7 + 9 + 10 + 12 + 0 + 5 + 6 + 10 + 4 + 2 + 1 + 3 + 5 + 0 +SVTCA[y-axis] +MIAP[rnd] +SLOOP +ALIGNRP +MIAP[rnd] +SLOOP +ALIGNRP +SRP1 +SRP2 +SLOOP +IP +SVTCA[x-axis] +SRP0 +MDRP[rnd,grey] +DELTAP1 +MDRP[rp0,min,rnd,black] +IP +IP +RTHG +MDAP[rnd] +DELTAP1 +SRP2 +IP +IP +SRP2 +IP +RTG +MDAP[rnd] +DELTAP1 +MDRP[rp0,min,rnd,black] +DELTAP1 +IP +RTHG +MDAP[rnd] +SRP2 +IP +IP +SRP2 +IP +IP +RTG +SRP1 +SRP2 +IP +SPVTL[orthog] +MDAP[rnd] +MIRP[rp0,min,rnd,black] +MDRP[grey] +SPVTL[orthog] +SRP0 +MIRP[rp0,min,rnd,black] +DELTAP1 +MDRP[grey] +MDRP[rnd,grey] +SPVTL[orthog] +MDAP[rnd] +DELTAP1 +MIRP[rp0,min,rnd,black] +DELTAP1 +MDRP[grey] +SPVTL[orthog] +SRP0 +SFVTL[parallel] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[y-axis] +DELTAP1 +SVTCA[x-axis] +DELTAP1 +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +2160 1509 m 1,0,-1 + 1423 0 l 1,1,-1 + 1195 0 l 1,2,-1 + 1127 983 l 1,3,-1 + 598 0 l 1,4,-1 + 370 0 l 1,5,-1 + 330 1509 l 1,6,-1 + 650 1509 l 1,7,-1 + 673 553 l 1,8,-1 + 1172 1509 l 1,9,-1 + 1396 1509 l 1,10,-1 + 1455 553 l 1,11,-1 + 1905 1509 l 1,12,-1 + 2160 1509 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: X +Encoding: 79 88 59 +Width: 1163 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 111 + 89 + 2 + 83 + 8 + 146 + 8 + 3 + 40 + 5 + 59 + 3 + 89 + 3 + 88 + 11 + 4 + 89 + 5 + 1 + 6 + 8 + 22 + 8 + 54 + 8 + 3 + 11 + 8 + 0 + 7 + 7 + 0 + 2 + 5 + 1 + 6 + 6 + 9 + 6 + 1 + 1 + 22 + 0 + 7 + 0 + 7 + 22 + 6 + 6 + 1 + 11 + 2 + 10 + 3 + 3 + 10 + 8 + 5 + 9 + 4 + 4 + 9 + 9 + 58 + 10 + 3 + 10 + 3 + 58 + 4 + 4 + 9 + 5 + 8 + 2 + 11 + 4 + 4 + 6 + 7 + 9 + 10 + 3 + 6 + 10 + 3 + 1 + 0 + 3 + 4 + 0 + 3 + 1 + 2 + 5 + 8 + 11 + 7 + 9 + 8 + 0 + 12 + 10 + 0 + 63 + 13 + 1 + 13 + 6 + 4 + 12 +SRP0 +MDRP[rp0,rnd,white] +MDRP[rnd,grey] +SRP0 +DELTAP1 +MDRP[rp0,rnd,white] +MDRP[rp0,rnd,grey] +SRP1 +SRP2 +SLOOP +IP +SVTCA[y-axis] +MIAP[rnd] +SLOOP +ALIGNRP +MIAP[rnd] +SLOOP +ALIGNRP +SRP1 +SRP2 +SLOOP +IP +SPVTL[orthog] +SFVTCA[x-axis] +SRP0 +MIRP[min,rnd,black] +SPVTL[orthog] +SRP0 +MIRP[min,rnd,black] +SPVTL[orthog] +SFVTCA[x-axis] +SRP1 +SRP2 +IP +IP +SPVTL[orthog] +SRP1 +SRP2 +IP +IP +SPVTL[orthog] +SRP0 +MIRP[rp0,min,rnd,black] +SPVTL[orthog] +SRP0 +MIRP[rp0,min,rnd,black] +SPVTL[orthog] +SFVTL[parallel] +SRP1 +SRP2 +IP +IP +SPVTL[orthog] +SRP1 +SRP2 +IP +IP +DELTAP1 +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP2 +DELTAP1 +SVTCA[y-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1174 0 m 1,0,-1 + 816 0 l 1,1,-1 + 668 518 l 1,2,-1 + 257 0 l 1,3,-1 + -7 0 l 1,4,-1 + 599 754 l 1,5,-1 + 387 1509 l 1,6,-1 + 745 1509 l 1,7,-1 + 877 1044 l 1,8,-1 + 1246 1509 l 1,9,-1 + 1510 1509 l 1,10,-1 + 945 809 l 1,11,-1 + 1174 0 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: Y +Encoding: 80 89 60 +Width: 1234 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 74 + 54 + 7 + 1 + 53 + 1 + 54 + 4 + 54 + 7 + 102 + 7 + 4 + 7 + 58 + 6 + 7 + 1 + 8 + 58 + 0 + 0 + 1 + 72 + 7 + 88 + 7 + 2 + 7 + 22 + 4 + 6 + 22 + 5 + 5 + 4 + 8 + 6 + 0 + 5 + 7 + 1 + 4 + 55 + 0 + 1 + 0 + 1 + 20 + 5 + 15 + 4 + 31 + 4 + 144 + 4 + 160 + 4 + 4 + 4 + 4 + 10 + 9 + 0 + 8 + 6 + 7 + 4 + 1 + 3 + 3 + 5 + 10 + 3 + 0 +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +SRP2 +SLOOP +IP +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +SVTCA[x-axis] +SRP1 +SRP2 +IP +MDAP[rnd] +DELTAP1 +MDRP[min,rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rp0,min,rnd,grey] +DELTAP1 +SRP1 +SRP2 +IP +SRP1 +SRP2 +IP +IP +SPVTL[orthog] +SRP0 +MIRP[rp0,min,rnd,black] +SRP0 +MIRP[rp0,min,rnd,black] +DELTAP1 +SPVTL[orthog] +SRP0 +MIRP[rp0,min,rnd,black] +SRP0 +SFVTL[parallel] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[y-axis] +DELTAP1 +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1596 1509 m 1,0,-1 + 929 639 l 1,1,-1 + 781 0 l 1,2,-1 + 465 0 l 1,3,-1 + 605 606 l 1,4,-1 + 339 1509 l 1,5,-1 + 708 1509 l 1,6,-1 + 877 903 l 1,7,-1 + 1345 1509 l 1,8,-1 + 1596 1509 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: Z +Encoding: 81 90 61 +Width: 1029 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHW_2 + 8 + -20 +NPUSHB + 43 + 3 + 20 + 8 + 22 + 2 + 2 + 3 + 3 + 22 + 7 + 7 + 8 + 7 + 3 + 6 + 3 + 30 + 6 + 10 + 2 + 8 + 1 + 8 + 30 + 1 + 0 + 3 + 7 + 0 + 0 + 11 + 10 + 4 + 8 + 47 + 2 + 63 + 2 + 95 + 2 + 3 + 2 + 10 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MDRP[grey] +MDRP[rnd,grey] +SRP1 +SRP2 +IP +MDAP[rnd] +MDRP[rp0,rnd,grey] +MDRP[grey] +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +MIAP[rnd] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +SPVTL[orthog] +SFVTCA[x-axis] +SRP0 +MIRP[rp0,min,rnd,black] +SPVTL[orthog] +SRP0 +MIRP[rp0,min,rnd,black] +SVTCA[x-axis] +SHPIX +SHPIX +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1001 0 m 1,0,-1 + 34 0 l 1,1,-1 + 81 205 l 1,2,-1 + 947 1284 l 1,3,-1 + 357 1284 l 1,4,-1 + 409 1509 l 1,5,-1 + 1349 1509 l 1,6,-1 + 1303 1309 l 1,7,-1 + 434 225 l 1,8,-1 + 1053 225 l 1,9,-1 + 1001 0 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: bracketleft +Encoding: 82 91 62 +Width: 747 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 19 + 5 + 73 + 2 + 16 + 7 + 73 + 48 + 1 + 1 + 1 + 4 + 0 + 6 + 24 + 239 + 1 + 1 + 1 + 8 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +SVTCA[y-axis] +MDAP[rnd] +DELTAP1 +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +603 -223 m 1,0,-1 + 66 -223 l 1,1,-1 + 466 1509 l 1,2,-1 + 1003 1509 l 1,3,-1 + 961 1325 l 1,4,-1 + 701 1325 l 1,5,-1 + 390 -25 l 1,6,-1 + 649 -25 l 1,7,-1 + 603 -223 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: backslash +Encoding: 83 92 63 +Width: 883 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 33 + 40 + 1 + 55 + 1 + 89 + 1 + 152 + 1 + 201 + 1 + 5 + 0 + 100 + 1 + 3 + 100 + 2 + 2 + 1 + 0 + 52 + 2 + 1 + 2 + 4 + 0 + 63 + 1 + 1 + 1 + 2 + 3 + 10 +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MDAP[rnd] +DELTAP1 +MDRP[rnd,grey] +SVTCA[x-axis] +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MDRP[rp0,min,rnd,grey] +SPVTL[orthog] +SRP0 +MIRP[rp0,min,rnd,black] +SRP0 +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +899 -170 m 1,0,-1 + 689 -170 l 1,1,-1 + 307 1509 l 1,2,-1 + 517 1509 l 1,3,-1 + 899 -170 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: bracketright +Encoding: 84 93 64 +Width: 747 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 16 + 5 + 73 + 6 + 10 + 2 + 73 + 48 + 1 + 1 + 1 + 5 + 1 + 4 + 24 + 0 + 8 +SRP0 +MDRP[rp0,rnd,white] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +SVTCA[y-axis] +MDAP[rnd] +DELTAP1 +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +584 -223 m 1,0,-1 + 48 -223 l 1,1,-1 + 93 -25 l 1,2,-1 + 352 -25 l 1,3,-1 + 664 1325 l 1,4,-1 + 405 1325 l 1,5,-1 + 448 1509 l 1,6,-1 + 984 1509 l 1,7,-1 + 584 -223 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: asciicircum +Encoding: 85 94 65 +Width: 1047 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 42 + 10 + 1 + 6 + 3 + 26 + 1 + 22 + 3 + 53 + 5 + 58 + 6 + 6 + 1 + 3 + 15 + 2 + 31 + 2 + 2 + 2 + 5 + 10 + 50 + 0 + 1 + 144 + 0 + 1 + 0 + 8 + 31 + 4 + 47 + 4 + 61 + 4 + 95 + 4 + 4 + 4 + 7 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +PUSHB_2 + 4 + 0 +MD[grid] +EVEN +IF +PUSHB_2 + 0 + 64 +SHPIX +EIF +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MDRP[rp0,min,rnd,grey] +DELTAP1 +MDAP[rnd] +MDRP[rnd,grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1199 657 m 1,0,-1 + 906 657 l 1,1,-1 + 773 1122 l 1,2,-1 + 445 657 l 1,3,-1 + 152 657 l 1,4,-1 + 759 1509 l 1,5,-1 + 986 1509 l 1,6,-1 + 1199 657 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: underscore +Encoding: 86 95 66 +Width: 948 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 10 + 1 + 201 + 2 + 0 + 5 + 63 + 1 + 1 + 1 + 4 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +SRP0 +MDRP[rp0,rnd,white] +SVTCA[y-axis] +MDAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +865 -360 m 1,0,-1 + -83 -360 l 1,1,-1 + -35 -150 l 1,2,-1 + 914 -150 l 1,3,-1 + 865 -360 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: grave +Encoding: 87 96 67 +Width: 942 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 18 + 2 + 91 + 1 + 36 + 3 + 1 + 0 + 2 + 0 + 63 + 2 + 79 + 2 + 80 + 2 + 3 + 2 + 4 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MDRP[rp0,min,rnd,grey] +SRP1 +SRP2 +IP +IP +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1017 1323 m 1,0,-1 + 815 1323 l 1,1,-1 + 592 1677 l 1,2,-1 + 866 1677 l 1,3,-1 + 1017 1323 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: a +Encoding: 88 97 68 +Width: 1145 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 89 + 11 + 7 + 11 + 10 + 4 + 28 + 25 + 6 + 25 + 7 + 25 + 10 + 20 + 28 + 51 + 43 + 186 + 7 + 9 + 34 + 32 + 15 + 12 + 12 + 5 + 26 + 0 + 2 + 41 + 33 + 59 + 2 + 1 + 2 + 5 + 3 + 22 + 51 + 255 + 19 + 1 + 19 + 33 + 23 + 51 + 26 + 13 + 16 + 45 + 63 + 45 + 192 + 45 + 240 + 45 + 4 + 22 + 22 + 0 + 8 + 0 + 22 + 1 + 2 + 15 + 0 + 30 + 1 + 64 + 30 + 80 + 30 + 112 + 30 + 128 + 30 + 160 + 30 + 5 + 30 + 38 + 23 + 47 + 8 + 63 + 8 + 2 + 8 + 44 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +DELTAP1 +DELTAP2 +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[grey] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +MDAP[rnd] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIRP[rnd,grey] +MIRP[rp0,min,rnd,black] +DELTAP1 +MIRP[rp0,rnd,grey] +MIAP[rnd] +MDRP[min,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +MIAP[rnd] +SRP1 +SRP2 +IP +MDAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1009 0 m 1,0,-1 + 737 0 l 1,1,-1 + 766 158 l 1,2,3 + 683 63 683 63 609.5 23.5 c 128,-1,4 + 536 -16 536 -16 441 -16 c 0,5,6 + 289 -16 289 -16 211.5 92 c 128,-1,7 + 134 200 134 200 197 469 c 2,8,-1 + 223 583 l 2,9,10 + 280 830 280 830 421 968.5 c 128,-1,11 + 562 1107 562 1107 772 1107 c 0,12,13 + 822 1107 822 1107 877 1091 c 128,-1,14 + 932 1075 932 1075 963 1025 c 1,15,-1 + 1262 1093 l 0,16,17 + 1261 1093 1261 1093 1009 0 c 1,0,-1 +799 317 m 1,18,-1 + 922 848 l 1,19,20 + 911 886 911 886 879 911.5 c 128,-1,21 + 847 937 847 937 786 937 c 0,22,23 + 700 937 700 937 631 859.5 c 128,-1,24 + 562 782 562 782 515 575 c 2,25,-1 + 497 501 l 2,26,27 + 459 336 459 336 484 268.5 c 128,-1,28 + 509 201 509 201 593 201 c 0,29,30 + 650 201 650 201 707.5 235 c 128,-1,31 + 765 269 765 269 799 317 c 1,18,-1 +EndSplineSet +EndChar + +StartChar: b +Encoding: 89 98 69 +Width: 1145 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 63 + 7 + 15 + 23 + 15 + 89 + 30 + 152 + 30 + 4 + 9 + 16 + 26 + 25 + 6 + 3 + 3 + 14 + 64 + 11 + 96 + 14 + 22 + 47 + 14 + 13 + 29 + 32 + 3 + 3 + 8 + 2 + 96 + 33 + 1 + 0 + 23 + 47 + 18 + 48 + 18 + 64 + 18 + 160 + 18 + 4 + 64 + 18 + 113 + 18 + 144 + 18 + 223 + 18 + 224 + 18 + 5 + 18 + 11 + 26 + 6 + 22 + 8 + 32 +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[rp0,min,rnd,grey] +DELTAP1 +DELTAP2 +MIRP[rp0,min,rnd,black] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIRP[min,rnd,black] +SRP0 +SMD +MDRP[min,grey] +SMD +SRP1 +SRP2 +SLOOP +IP +MIAP[rnd] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1166 500 m 2,0,1 + 1112 267 1112 267 970.5 125.5 c 128,-1,2 + 829 -16 829 -16 623 -16 c 0,3,4 + 572 -16 572 -16 513.5 3.5 c 128,-1,5 + 455 23 455 23 424 66 c 1,6,-1 + 405 0 l 1,7,-1 + 130 0 l 1,8,-1 + 478 1509 l 1,9,-1 + 762 1509 l 1,10,-1 + 629 931 l 1,11,12 + 720 1046 720 1046 798.5 1081 c 128,-1,13 + 877 1116 877 1116 955 1116 c 0,14,15 + 1113 1116 1113 1116 1182 992 c 128,-1,16 + 1251 868 1251 868 1194 624 c 2,17,-1 + 1166 500 l 2,0,1 +878 524 m 2,18,-1 + 896 602 l 2,19,20 + 936 775 936 775 903 832.5 c 128,-1,21 + 870 890 870 890 796 890 c 0,22,23 + 743 890 743 890 682.5 853 c 128,-1,24 + 622 816 622 816 590 765 c 1,25,-1 + 473 256 l 1,26,27 + 473 218 473 218 510 186 c 128,-1,28 + 547 154 547 154 616 154 c 0,29,30 + 691 154 691 154 758.5 227 c 128,-1,31 + 826 300 826 300 878 524 c 2,18,-1 +EndSplineSet +EndChar + +StartChar: c +Encoding: 90 99 70 +Width: 984 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 59 + 53 + 15 + 70 + 18 + 2 + 48 + 14 + 1 + 14 + 51 + 17 + 33 + 13 + 51 + 10 + 13 + 27 + 51 + 24 + 33 + 0 + 51 + 3 + 3 + 14 + 128 + 0 + 1 + 64 + 0 + 96 + 0 + 2 + 0 + 55 + 21 + 117 + 21 + 198 + 21 + 3 + 21 + 23 + 15 + 6 + 79 + 6 + 96 + 6 + 3 + 15 + 6 + 207 + 6 + 239 + 6 + 3 + 6 + 28 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +DELTAP2 +MIRP[min,rnd,black] +DELTAP1 +MDRP[rp0,min,rnd,grey] +DELTAP1 +DELTAP2 +MDRP[rnd,grey] +SVTCA[y-axis] +MIAP[rnd] +MIRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MIRP[rnd,grey] +MIAP[rnd] +MIRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MIRP[rnd,grey] +DELTAP1 +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +899 61 m 1,0,1 + 879 45 879 45 791 10 c 128,-1,2 + 703 -25 703 -25 616 -25 c 0,3,4 + 373 -25 373 -25 259.5 112.5 c 128,-1,5 + 146 250 146 250 210 525 c 2,6,-1 + 220 572 l 2,7,8 + 278 821 278 821 456 971.5 c 128,-1,9 + 634 1122 634 1122 874 1122 c 0,10,11 + 967 1122 967 1122 1035 1094 c 128,-1,12 + 1103 1066 1103 1066 1124 1034 c 1,13,-1 + 1079 838 l 1,14,15 + 1051 874 1051 874 995 898.5 c 128,-1,16 + 939 923 939 923 868 923 c 0,17,18 + 743 923 743 923 654 826.5 c 128,-1,19 + 565 730 565 730 527 561 c 2,20,-1 + 522 542 l 2,21,22 + 478 353 478 353 514 274 c 128,-1,23 + 550 195 550 195 669 195 c 0,24,25 + 749 195 749 195 831.5 221.5 c 128,-1,26 + 914 248 914 248 949 275 c 1,27,-1 + 899 61 l 1,0,1 +EndSplineSet +EndChar + +StartChar: d +Encoding: 91 100 71 +Width: 1145 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 60 + 9 + 6 + 25 + 6 + 118 + 10 + 3 + 16 + 16 + 18 + 19 + 15 + 3 + 5 + 12 + 22 + 32 + 12 + 13 + 29 + 47 + 64 + 2 + 96 + 5 + 3 + 1 + 2 + 96 + 33 + 1 + 0 + 22 + 15 + 63 + 2 + 1 + 2 + 16 + 18 + 64 + 18 + 112 + 18 + 3 + 18 + 26 + 23 + 15 + 8 + 31 + 8 + 2 + 15 + 8 + 239 + 8 + 2 + 8 + 32 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +DELTAP2 +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +DELTAP1 +MDRP[rnd,grey] +DELTAP1 +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +SMD +MDRP[min,rnd,grey] +SMD +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +SLOOP +IP +MIAP[rnd] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1009 0 m 1,0,-1 + 737 0 l 1,1,-1 + 766 158 l 1,2,3 + 683 63 683 63 609.5 23.5 c 128,-1,4 + 536 -16 536 -16 441 -16 c 0,5,6 + 289 -16 289 -16 211.5 92 c 128,-1,7 + 134 200 134 200 197 469 c 2,8,-1 + 223 583 l 2,9,10 + 280 830 280 830 421 968.5 c 128,-1,11 + 562 1107 562 1107 772 1107 c 0,12,13 + 822 1107 822 1107 877 1091 c 128,-1,14 + 932 1075 932 1075 963 1025 c 1,15,-1 + 1074 1509 l 1,16,-1 + 1358 1509 l 1,17,-1 + 1009 0 l 1,0,-1 +799 317 m 1,18,-1 + 922 848 l 1,19,20 + 911 886 911 886 879 911.5 c 128,-1,21 + 847 937 847 937 786 937 c 0,22,23 + 700 937 700 937 631 859.5 c 128,-1,24 + 562 782 562 782 515 575 c 2,25,-1 + 497 501 l 2,26,27 + 459 336 459 336 484 268.5 c 128,-1,28 + 509 201 509 201 593 201 c 0,29,30 + 650 201 650 201 707.5 235 c 128,-1,31 + 765 269 765 269 799 317 c 1,18,-1 +EndSplineSet +EndChar + +StartChar: e +Encoding: 92 101 72 +Width: 1135 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 78 + 90 + 12 + 84 + 20 + 2 + 1 + 46 + 127 + 30 + 1 + 30 + 73 + 30 + 11 + 18 + 80 + 7 + 96 + 7 + 2 + 7 + 71 + 4 + 33 + 8 + 70 + 11 + 3 + 169 + 27 + 1 + 27 + 33 + 152 + 18 + 1 + 18 + 13 + 8 + 0 + 23 + 80 + 24 + 96 + 24 + 112 + 24 + 3 + 0 + 24 + 16 + 24 + 48 + 24 + 64 + 24 + 80 + 24 + 144 + 24 + 200 + 24 + 7 + 24 + 14 + 30 + 1 + 23 + 31 + 14 + 1 + 15 + 14 + 199 + 14 + 2 + 14 + 31 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +DELTAP2 +MIRP[rp0,min,rnd,black] +MDRP[grey] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +DELTAP2 +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SVTCA[y-axis] +MIAP[rnd] +DELTAP1 +MIRP[rp0,min,rnd,black] +DELTAP1 +MIAP[rnd] +SROUND +MDRP[rnd,grey] +RTG +MIRP[rp0,min,rnd,black] +SROUND +MDRP[rnd,grey] +RTG +DELTAP1 +SRP1 +SRP2 +IP +SROUND +MDAP[rnd] +DELTAP1 +RTG +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1155 484 m 1,0,-1 + 487 484 l 1,1,2 + 447 309 447 309 505 240.5 c 128,-1,3 + 563 172 563 172 682 172 c 0,4,5 + 761 172 761 172 848.5 193.5 c 128,-1,6 + 936 215 936 215 1033 258 c 1,7,-1 + 990 74 l 1,8,9 + 904 25 904 25 807.5 0 c 128,-1,10 + 711 -25 711 -25 593 -25 c 0,11,12 + 362 -25 362 -25 253.5 109.5 c 128,-1,13 + 145 244 145 244 205 503 c 2,14,-1 + 223 582 l 2,15,16 + 275 809 275 809 439 965.5 c 128,-1,17 + 603 1122 603 1122 842 1122 c 0,18,19 + 1065 1122 1065 1122 1149.5 973.5 c 128,-1,20 + 1234 825 1234 825 1179 587 c 2,21,22 + 1179 587 1179 587 1168.5 541.5 c 128,-1,23 + 1158 496 1158 496 1155 484 c 1,0,-1 +939 648 m 1,24,25 + 966 803 966 803 926.5 870 c 128,-1,26 + 887 937 887 937 798 937 c 0,27,28 + 720 937 720 937 643.5 871 c 128,-1,29 + 567 805 567 805 526 648 c 1,30,-1 + 939 648 l 1,24,25 +EndSplineSet +EndChar + +StartChar: f +Encoding: 93 102 73 +Width: 584 +GlyphClass: 1 +Flags: W +HStem: 0 21G<146 433.622> 952 176<276 366 692 836> 1333 197<750.335 973.598> +DStem2: 146 0 429 0 0.227046 0.973884<64.2539 979.035 1222.51 1395.05> +TtInstrs: +NPUSHB + 41 + 9 + 20 + 25 + 20 + 185 + 20 + 3 + 11 + 14 + 32 + 8 + 17 + 12 + 13 + 2 + 1 + 4 + 33 + 0 + 21 + 17 + 0 + 25 + 10 + 8 + 12 + 22 + 17 + 15 + 31 + 13 + 95 + 13 + 144 + 13 + 3 + 144 + 13 + 1 + 13 + 24 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +DELTAP2 +MDRP[min,rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[rp0,min,rnd,grey] +SRP0 +MDRP[rp0,rnd,white] +SVTCA[y-axis] +MIAP[rnd] +MDRP[grey] +MIRP[rp0,min,rnd,black] +MDRP[grey] +MIAP[rnd] +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1056 1454 m 1,0,-1 + 945 1284 l 1,1,2 + 925 1307 925 1307 902 1320 c 128,-1,3 + 879 1333 879 1333 839 1333 c 0,4,5 + 780 1333 780 1333 750.5 1293 c 128,-1,6 + 721 1253 721 1253 702 1171 c 2,7,-1 + 692 1128 l 1,8,-1 + 876 1128 l 1,9,-1 + 836 952 l 1,10,-1 + 649 952 l 1,11,-1 + 429 0 l 1,12,-1 + 146 0 l 1,13,-1 + 366 954 l 1,14,-1 + 236 954 l 1,15,-1 + 276 1128 l 1,16,-1 + 411 1128 l 1,17,-1 + 419 1161 l 2,18,19 + 459 1333 459 1333 567.5 1431.5 c 128,-1,20 + 676 1530 676 1530 826 1530 c 0,21,22 + 896 1530 896 1530 957 1509.5 c 128,-1,23 + 1018 1489 1018 1489 1056 1454 c 1,0,-1 +EndSplineSet +EndChar + +StartChar: g +Encoding: 94 103 74 +Width: 1179 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 99 + 9 + 18 + 52 + 42 + 107 + 18 + 123 + 18 + 123 + 23 + 138 + 11 + 154 + 11 + 170 + 11 + 185 + 11 + 9 + 134 + 41 + 1 + 41 + 47 + 17 + 17 + 24 + 10 + 59 + 14 + 52 + 27 + 2 + 27 + 14 + 17 + 24 + 7 + 51 + 80 + 10 + 240 + 10 + 2 + 10 + 47 + 6 + 51 + 3 + 19 + 28 + 12 + 34 + 32 + 24 + 13 + 0 + 22 + 63 + 27 + 1 + 27 + 31 + 48 + 14 + 144 + 14 + 2 + 48 + 14 + 64 + 14 + 112 + 14 + 176 + 14 + 224 + 14 + 5 + 14 + 38 + 23 + 7 + 31 + 20 + 47 + 20 + 63 + 20 + 3 + 207 + 20 + 239 + 20 + 255 + 20 + 3 + 20 + 44 +SRP0 +MDRP[rp0,min,rnd,white] +DELTAP1 +DELTAP2 +MDRP[rnd,grey] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +DELTAP1 +DELTAP2 +MDRP[rnd,grey] +MDRP[rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIAP[rnd] +MIRP[rnd,grey] +MIRP[rp0,min,rnd,black] +DELTAP1 +MIRP[rnd,grey] +SRP1 +SRP2 +IP +IP +DELTAP1 +SRP1 +SRP2 +IP +MDAP[rnd] +MIRP[rp0,min,rnd,black] +DELTAP1 +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1066 138 m 2,0,1 + 1011 -99 1011 -99 841 -234 c 128,-1,2 + 671 -369 671 -369 428 -369 c 0,3,4 + 356 -369 356 -369 265 -352.5 c 128,-1,5 + 174 -336 174 -336 110 -295 c 1,6,-1 + 160 -78 l 1,7,8 + 226 -115 226 -115 305.5 -129 c 128,-1,9 + 385 -143 385 -143 446 -143 c 0,10,11 + 613 -143 613 -143 687 -79.5 c 128,-1,12 + 761 -16 761 -16 785 87 c 2,13,-1 + 810 197 l 1,14,15 + 773 137 773 137 684.5 70.5 c 128,-1,16 + 596 4 596 4 456 4 c 0,17,18 + 292 4 292 4 219 126.5 c 128,-1,19 + 146 249 146 249 212 534 c 2,20,-1 + 219 566 l 2,21,22 + 282 839 282 839 428 973 c 128,-1,23 + 574 1107 574 1107 775 1107 c 0,24,25 + 852 1107 852 1107 912 1078.5 c 128,-1,26 + 972 1050 972 1050 998 993 c 1,27,-1 + 1021 1093 l 1,28,-1 + 1287 1093 l 1,29,-1 + 1066 138 l 2,0,1 +881 569 m 2,30,-1 + 897 636 l 2,31,32 + 934 797 934 797 907 863 c 128,-1,33 + 880 929 880 929 787 929 c 0,34,35 + 698 929 698 929 629.5 852.5 c 128,-1,36 + 561 776 561 776 520 598 c 2,37,-1 + 513 569 l 2,38,39 + 466 366 466 366 496.5 294.5 c 128,-1,40 + 527 223 527 223 617 223 c 0,41,42 + 707 223 707 223 772.5 302 c 128,-1,43 + 838 381 838 381 881 569 c 2,30,-1 +EndSplineSet +EndChar + +StartChar: h +Encoding: 95 104 75 +Width: 1131 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 48 + 37 + 18 + 70 + 17 + 231 + 17 + 246 + 17 + 4 + 5 + 17 + 1 + 8 + 13 + 96 + 16 + 1 + 10 + 2 + 11 + 16 + 5 + 45 + 55 + 16 + 1 + 16 + 13 + 96 + 21 + 1 + 0 + 22 + 64 + 1 + 160 + 1 + 2 + 175 + 1 + 1 + 1 + 10 + 13 + 9 + 22 + 10 + 20 +SRP0 +MDRP[rp0,rnd,white] +MIRP[rp0,min,rnd,black] +MDRP[grey] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +DELTAP2 +MIRP[rp0,min,rnd,black] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +DELTAP1 +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIAP[rnd] +MDRP[rnd,grey] +SRP0 +SMD +MDRP[rp0,min,grey] +MDRP[min,grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP2 +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1079 328 m 1,0,1 + 1075 310 1075 310 1069 265 c 128,-1,2 + 1063 220 1063 220 1065 212 c 0,3,4 + 1065 197 1065 197 1075 186.5 c 128,-1,5 + 1085 176 1085 176 1099 176 c 0,6,7 + 1150 176 1150 176 1252 342 c 1,8,-1 + 1265 366 l 1,9,-1 + 1319 331 l 1,10,11 + 1198 124 1198 124 1120.5 51.5 c 128,-1,12 + 1043 -21 1043 -21 938 -21 c 0,13,14 + 902 -21 902 -21 874 -12 c 128,-1,15 + 846 -3 846 -3 828 9 c 128,-1,16 + 810 21 810 21 798.5 43.5 c 128,-1,17 + 787 66 787 66 780.5 83.5 c 128,-1,18 + 774 101 774 101 774 131 c 128,-1,19 + 774 161 774 161 774.5 176.5 c 128,-1,20 + 775 192 775 192 780.5 223.5 c 128,-1,21 + 786 255 786 255 787.5 262.5 c 128,-1,22 + 789 270 789 270 795 295 c 1,23,-1 + 902 757 l 2,24,25 + 913 807 913 807 893 839.5 c 128,-1,26 + 873 872 873 872 814 872 c 0,27,28 + 754 872 754 872 695.5 846.5 c 128,-1,29 + 637 821 637 821 593 777 c 1,30,-1 + 414 0 l 1,31,-1 + 130 0 l 1,32,-1 + 478 1509 l 1,33,-1 + 762 1509 l 1,34,-1 + 625 917 l 1,35,36 + 688 993 688 993 786.5 1054.5 c 128,-1,37 + 885 1116 885 1116 1010 1116 c 0,38,39 + 1135 1116 1135 1116 1178 1031 c 128,-1,40 + 1221 946 1221 946 1181 771 c 2,41,-1 + 1079 328 l 1,0,1 +EndSplineSet +EndChar + +StartChar: i +Encoding: 96 105 76 +Width: 532 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 22 + 214 + 1 + 214 + 11 + 2 + 9 + 62 + 3 + 14 + 12 + 13 + 2 + 0 + 63 + 6 + 12 + 22 + 207 + 13 + 1 + 13 + 16 +SRP0 +MDRP[rp0,min,rnd,white] +DELTAP1 +MIRP[min,rnd,black] +MDRP[rp0,rnd,grey] +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +MDRP[rp0,min,rnd,white] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +742 1407 m 0,0,1 + 725 1335 725 1335 670 1286 c 128,-1,2 + 615 1237 615 1237 553 1237 c 0,3,4 + 492 1237 492 1237 459.5 1286 c 128,-1,5 + 427 1335 427 1335 443 1407 c 0,6,7 + 459 1477 459 1477 514.5 1526 c 128,-1,8 + 570 1575 570 1575 631 1575 c 0,9,10 + 693 1575 693 1575 725.5 1526 c 128,-1,11 + 758 1477 758 1477 742 1407 c 0,0,1 +488 337 m 1,12,13 + 484 318 484 318 477.5 270.5 c 128,-1,14 + 471 223 471 223 472 214 c 0,15,16 + 472 198 472 198 482 187 c 128,-1,17 + 492 176 492 176 505 176 c 0,18,19 + 555 176 555 176 659 356 c 1,20,-1 + 673 382 l 1,21,-1 + 725 344 l 1,22,23 + 602 129 602 129 523.5 54 c 128,-1,24 + 445 -21 445 -21 339 -21 c 0,25,26 + 302 -21 302 -21 274 -11.5 c 128,-1,27 + 246 -2 246 -2 228 11 c 128,-1,28 + 210 24 210 24 198.5 48 c 128,-1,29 + 187 72 187 72 181 90.5 c 128,-1,30 + 175 109 175 109 175 141 c 128,-1,31 + 175 173 175 173 175.5 189.5 c 128,-1,32 + 176 206 176 206 182 239.5 c 128,-1,33 + 188 273 188 273 190 281 c 128,-1,34 + 192 289 192 289 198 315 c 1,35,-1 + 391 1151 l 1,36,-1 + 676 1151 l 1,37,-1 + 488 337 l 1,12,13 +EndSplineSet +EndChar + +StartChar: j +Encoding: 97 106 77 +Width: 549 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 39 + 6 + 14 + 22 + 14 + 216 + 5 + 216 + 7 + 4 + 9 + 62 + 3 + 26 + 12 + 19 + 22 + 34 + 18 + 15 + 19 + 0 + 63 + 6 + 12 + 22 + 15 + 25 + 1 + 80 + 25 + 96 + 25 + 2 + 111 + 25 + 1 + 25 + 19 + 28 +SRP0 +MDRP[min,rnd,white] +MDRP[rp0,min,rnd,white] +DELTAP1 +DELTAP2 +DELTAP3 +MIRP[min,rnd,black] +MDRP[rp0,rnd,grey] +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MIAP[rnd] +MDRP[rp0,min,rnd,white] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +762 1407 m 0,0,1 + 745 1335 745 1335 690 1286 c 128,-1,2 + 635 1237 635 1237 574 1237 c 0,3,4 + 512 1237 512 1237 479.5 1286 c 128,-1,5 + 447 1335 447 1335 463 1407 c 0,6,7 + 479 1477 479 1477 534 1526 c 128,-1,8 + 589 1575 589 1575 652 1575 c 0,9,10 + 713 1575 713 1575 745.5 1526 c 128,-1,11 + 778 1477 778 1477 762 1407 c 0,0,1 +448 76 m 2,12,13 + 400 -131 400 -131 284 -243.5 c 128,-1,14 + 168 -356 168 -356 21 -356 c 0,15,16 + -42 -356 -42 -356 -109 -348 c 128,-1,17 + -176 -340 -176 -340 -222 -324 c 1,18,-1 + -171 -106 l 1,19,20 + -145 -117 -145 -117 -105 -124 c 128,-1,21 + -65 -131 -65 -131 -18 -131 c 0,22,23 + 49 -131 49 -131 91 -93 c 128,-1,24 + 133 -55 133 -55 153 33 c 2,25,-1 + 411 1151 l 1,26,-1 + 696 1151 l 1,27,-1 + 448 76 l 2,12,13 +EndSplineSet +EndChar + +StartChar: k +Encoding: 98 107 78 +Width: 1030 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 43 + 13 + 1 + 11 + 8 + 28 + 1 + 27 + 8 + 4 + 5 + 16 + 7 + 2 + 10 + 3 + 4 + 9 + 8 + 12 + 0 + 1 + 4 + 2 + 54 + 10 + 1 + 1 + 0 + 10 + 8 + 9 + 2 + 9 + 16 + 0 + 1 + 0 + 12 + 7 + 2 + 22 + 4 + 11 +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MDRP[rnd,grey] +SRP1 +SRP2 +IP +IP +SRP2 +IP +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MDRP[rnd,grey] +MIAP[rnd] +MDRP[rnd,grey] +SRP2 +SLOOP +IP +MIAP[rnd] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +967 182 m 1,0,1 + 978 159 978 159 992 159 c 0,2,3 + 1040 159 1040 159 1131 317 c 2,4,-1 + 1158 364 l 1,5,-1 + 1205 328 l 1,6,7 + 1091 124 1091 124 1016.5 50.5 c 128,-1,8 + 942 -23 942 -23 846 -23 c 0,9,10 + 740 -23 740 -23 708 44 c 1,11,-1 + 522 469 l 1,12,-1 + 414 0 l 1,13,-1 + 130 0 l 1,14,-1 + 478 1509 l 1,15,-1 + 762 1509 l 1,16,-1 + 571 684 l 1,17,-1 + 1008 1151 l 1,18,-1 + 1261 1151 l 1,19,-1 + 763 639 l 1,20,-1 + 967 182 l 1,0,1 +EndSplineSet +EndChar + +StartChar: l +Encoding: 99 108 79 +Width: 555 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 11 + 2 + 16 + 1 + 2 + 0 + 22 + 176 + 1 + 1 + 1 + 4 +SRP0 +MDRP[rp0,min,rnd,white] +DELTAP1 +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +498 337 m 1,0,1 + 494 318 494 318 487.5 270.5 c 128,-1,2 + 481 223 481 223 483 214 c 0,3,4 + 483 198 483 198 492.5 187 c 128,-1,5 + 502 176 502 176 515 176 c 0,6,7 + 565 176 565 176 668 355 c 1,8,-1 + 683 382 l 1,9,-1 + 736 344 l 1,10,11 + 613 129 613 129 534 54 c 128,-1,12 + 455 -21 455 -21 349 -21 c 0,13,14 + 304 -21 304 -21 272 -7 c 128,-1,15 + 240 7 240 7 222.5 27.5 c 128,-1,16 + 205 48 205 48 196.5 81 c 128,-1,17 + 188 114 188 114 186 139.5 c 128,-1,18 + 184 165 184 165 188.5 203.5 c 128,-1,19 + 193 242 193 242 197 260 c 128,-1,20 + 201 278 201 278 208 308 c 0,21,22 + 209 313 209 313 209 315 c 1,23,-1 + 485 1509 l 1,24,-1 + 769 1509 l 1,25,-1 + 498 337 l 1,0,1 +EndSplineSet +EndChar + +StartChar: m +Encoding: 100 109 80 +Width: 1704 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +SplineSet +1601 1116 m 0,0,1 + 1698 1115 1698 1115 1751 1024 c 0,2,3 + 1809 925 1809 925 1775 776 c 2,4,-1 + 1671 328 l 1,5,6 + 1667 310 1667 310 1661.5 265 c 128,-1,7 + 1656 220 1656 220 1658 212 c 0,8,9 + 1658 197 1658 197 1668 186.5 c 128,-1,10 + 1678 176 1678 176 1691 176 c 0,11,12 + 1742 176 1742 176 1845 342 c 1,13,-1 + 1858 366 l 1,14,-1 + 1911 331 l 1,15,16 + 1790 124 1790 124 1712.5 51.5 c 128,-1,17 + 1635 -21 1635 -21 1531 -21 c 0,18,19 + 1495 -21 1495 -21 1467 -12 c 128,-1,20 + 1439 -3 1439 -3 1420.5 9 c 128,-1,21 + 1402 21 1402 21 1391 43.5 c 128,-1,22 + 1380 66 1380 66 1373.5 83.5 c 128,-1,23 + 1367 101 1367 101 1366.5 131 c 128,-1,24 + 1366 161 1366 161 1366.5 176.5 c 128,-1,25 + 1367 192 1367 192 1372.5 223.5 c 128,-1,26 + 1378 255 1378 255 1380 262.5 c 128,-1,27 + 1382 270 1382 270 1387 295 c 1,28,-1 + 1494 757 l 2,29,30 + 1504 801 1504 801 1475.5 836.5 c 128,-1,31 + 1447 872 1447 872 1396 872 c 0,32,33 + 1329 872 1329 872 1275 840 c 128,-1,34 + 1221 808 1221 808 1181 759 c 1,35,-1 + 1006 0 l 1,36,-1 + 722 0 l 1,37,-1 + 896 757 l 2,38,39 + 907 803 907 803 882 837.5 c 128,-1,40 + 857 872 857 872 799 872 c 0,41,42 + 733 872 733 872 683 846 c 128,-1,43 + 633 820 633 820 593 777 c 1,44,-1 + 414 0 l 1,45,-1 + 130 0 l 1,46,-1 + 382 1093 l 1,47,-1 + 655 1093 l 1,48,-1 + 619 911 l 1,49,50 + 685 1005 685 1005 779.5 1060.5 c 128,-1,51 + 874 1116 874 1116 991 1116 c 0,52,53 + 1096 1116 1096 1116 1144 1058.5 c 128,-1,54 + 1192 1001 1192 1001 1197 900 c 1,55,56 + 1281 1049 1281 1049 1379 1113 c 0,57,58 + 1474 1174 1474 1174 1609 1174 c 0,59,60 + 1612 1174 1612 1174 1601 1116 c 0,0,1 +EndSplineSet +EndChar + +StartChar: n +Encoding: 101 110 81 +Width: 1131 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 40 + 37 + 17 + 1 + 8 + 13 + 96 + 16 + 1 + 10 + 2 + 11 + 12 + 5 + 45 + 16 + 13 + 96 + 21 + 1 + 0 + 22 + 112 + 1 + 1 + 64 + 1 + 160 + 1 + 2 + 175 + 1 + 1 + 1 + 10 + 12 + 13 + 9 + 22 + 10 + 20 +SRP0 +MDRP[rp0,rnd,white] +MIRP[rp0,min,rnd,black] +MDRP[grey] +MDRP[grey] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +DELTAP2 +DELTAP3 +MIRP[rp0,min,rnd,black] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIAP[rnd] +MDRP[rnd,grey] +SRP0 +SMD +MDRP[rp0,min,grey] +MDRP[min,grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1079 328 m 1,0,1 + 1075 310 1075 310 1069 265 c 128,-1,2 + 1063 220 1063 220 1065 212 c 0,3,4 + 1065 197 1065 197 1075 186.5 c 128,-1,5 + 1085 176 1085 176 1099 176 c 0,6,7 + 1150 176 1150 176 1252 342 c 1,8,-1 + 1265 366 l 1,9,-1 + 1319 331 l 1,10,11 + 1198 124 1198 124 1120.5 51.5 c 128,-1,12 + 1043 -21 1043 -21 938 -21 c 0,13,14 + 902 -21 902 -21 874 -12 c 128,-1,15 + 846 -3 846 -3 828 9 c 128,-1,16 + 810 21 810 21 798.5 43.5 c 128,-1,17 + 787 66 787 66 780.5 83.5 c 128,-1,18 + 774 101 774 101 774 131 c 128,-1,19 + 774 161 774 161 774.5 176.5 c 128,-1,20 + 775 192 775 192 780.5 223.5 c 128,-1,21 + 786 255 786 255 787.5 262.5 c 128,-1,22 + 789 270 789 270 795 295 c 1,23,-1 + 902 757 l 2,24,25 + 912 799 912 799 886.5 835.5 c 128,-1,26 + 861 872 861 872 799 872 c 0,27,28 + 739 872 739 872 687 845.5 c 128,-1,29 + 635 819 635 819 593 777 c 1,30,-1 + 414 0 l 1,31,-1 + 130 0 l 1,32,-1 + 382 1093 l 1,33,-1 + 655 1093 l 1,34,-1 + 618 909 l 1,35,36 + 667 991 667 991 768 1053.5 c 128,-1,37 + 869 1116 869 1116 993 1116 c 0,38,39 + 1118 1116 1118 1116 1169.5 1031 c 128,-1,40 + 1221 946 1221 946 1181 771 c 2,41,-1 + 1079 328 l 1,0,1 +EndSplineSet +EndChar + +StartChar: o +Encoding: 102 111 82 +Width: 1196 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 78 + 88 + 24 + 150 + 4 + 150 + 9 + 198 + 2 + 247 + 8 + 5 + 104 + 18 + 1 + 18 + 33 + 10 + 13 + 103 + 25 + 1 + 25 + 33 + 3 + 3 + 0 + 23 + 96 + 14 + 112 + 14 + 144 + 14 + 3 + 0 + 14 + 48 + 14 + 80 + 14 + 129 + 14 + 208 + 14 + 224 + 14 + 240 + 14 + 7 + 14 + 134 + 22 + 1 + 22 + 23 + 159 + 6 + 1 + 15 + 6 + 31 + 6 + 63 + 6 + 3 + 15 + 6 + 207 + 6 + 223 + 6 + 239 + 6 + 255 + 6 + 5 + 6 + 28 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +DELTAP2 +DELTAP3 +MIRP[min,rnd,black] +DELTAP1 +MDRP[rp0,min,rnd,grey] +DELTAP1 +DELTAP2 +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +DELTAP1 +MIAP[rnd] +MIRP[rp0,min,rnd,black] +DELTAP1 +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1217 514 m 2,0,1 + 1158 260 1158 260 990.5 117.5 c 128,-1,2 + 823 -25 823 -25 590 -25 c 0,3,4 + 359 -25 359 -25 254.5 120.5 c 128,-1,5 + 150 266 150 266 207 514 c 2,6,-1 + 223 583 l 2,7,8 + 279 825 279 825 450 973.5 c 128,-1,9 + 621 1122 621 1122 853 1122 c 0,10,11 + 1088 1122 1088 1122 1188.5 972.5 c 128,-1,12 + 1289 823 1289 823 1234 583 c 2,13,-1 + 1217 514 l 2,0,1 +931 541 m 2,14,-1 + 935 557 l 2,15,16 + 985 774 985 774 941.5 852.5 c 128,-1,17 + 898 931 898 931 807 931 c 0,18,19 + 716 931 716 931 638 846.5 c 128,-1,20 + 560 762 560 762 512 557 c 2,21,-1 + 509 541 l 2,22,23 + 457 315 457 315 496.5 240.5 c 128,-1,24 + 536 166 536 166 632 166 c 0,25,26 + 719 166 719 166 798.5 238.5 c 128,-1,27 + 878 311 878 311 931 541 c 2,14,-1 +EndSplineSet +EndChar + +StartChar: p +Encoding: 103 112 83 +Width: 1145 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 50 + 6 + 16 + 22 + 16 + 121 + 1 + 3 + 9 + 12 + 26 + 25 + 6 + 3 + 3 + 14 + 64 + 11 + 96 + 14 + 22 + 47 + 14 + 13 + 29 + 32 + 3 + 3 + 8 + 18 + 96 + 33 + 1 + 0 + 23 + 16 + 18 + 64 + 18 + 112 + 18 + 3 + 18 + 8 + 10 + 11 + 26 + 6 + 22 + 8 + 32 +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIRP[min,rnd,black] +SRP0 +SMD +MDRP[min,grey] +SMD +SRP1 +SRP2 +SLOOP +IP +MIAP[rnd] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1167 506 m 2,0,1 + 1109 257 1109 257 965.5 121.5 c 128,-1,2 + 822 -14 822 -14 623 -14 c 0,3,4 + 572 -14 572 -14 515.5 2 c 128,-1,5 + 459 18 459 18 428 61 c 1,6,-1 + 335 -342 l 1,7,-1 + 51 -342 l 1,8,-1 + 382 1093 l 1,9,-1 + 655 1093 l 1,10,-1 + 623 923 l 1,11,12 + 676 1011 676 1011 770 1063.5 c 128,-1,13 + 864 1116 864 1116 962 1116 c 0,14,15 + 1102 1116 1102 1116 1179.5 1006 c 128,-1,16 + 1257 896 1257 896 1196 633 c 2,17,-1 + 1167 506 l 2,0,1 +877 520 m 2,18,-1 + 893 591 l 2,19,20 + 933 763 933 763 905 826.5 c 128,-1,21 + 877 890 877 890 803 890 c 0,22,23 + 745 890 745 890 685.5 854 c 128,-1,24 + 626 818 626 818 593 776 c 1,25,-1 + 474 260 l 1,26,27 + 472 224 472 224 510 190 c 128,-1,28 + 548 156 548 156 616 156 c 0,29,30 + 677 156 677 156 750.5 223 c 128,-1,31 + 824 290 824 290 877 520 c 2,18,-1 +EndSplineSet +EndChar + +StartChar: q +Encoding: 104 113 84 +Width: 1145 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 65 + 9 + 6 + 24 + 6 + 121 + 11 + 137 + 6 + 155 + 6 + 5 + 16 + 12 + 52 + 15 + 1 + 18 + 19 + 15 + 3 + 5 + 12 + 22 + 32 + 12 + 13 + 29 + 47 + 64 + 2 + 96 + 5 + 3 + 1 + 18 + 96 + 33 + 1 + 0 + 22 + 16 + 15 + 2 + 16 + 18 + 64 + 18 + 112 + 18 + 3 + 18 + 26 + 23 + 15 + 8 + 31 + 8 + 2 + 15 + 8 + 239 + 8 + 2 + 8 + 32 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +DELTAP2 +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +DELTAP1 +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +SMD +MDRP[min,rnd,grey] +SMD +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +SLOOP +IP +DELTAP1 +MIAP[rnd] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +931 -342 m 1,0,-1 + 647 -342 l 1,1,-1 + 761 152 l 1,2,3 + 724 100 724 100 638.5 42 c 128,-1,4 + 553 -16 553 -16 448 -16 c 0,5,6 + 280 -16 280 -16 208 95 c 128,-1,7 + 136 206 136 206 199 481 c 2,8,-1 + 232 624 l 2,9,10 + 279 829 279 829 423 968 c 128,-1,11 + 567 1107 567 1107 770 1107 c 0,12,13 + 850 1107 850 1107 902.5 1080.5 c 128,-1,14 + 955 1054 955 1054 977 993 c 1,15,-1 + 1058 1093 l 1,16,-1 + 1262 1093 l 1,17,-1 + 931 -342 l 1,0,-1 +799 317 m 1,18,-1 + 923 854 l 1,19,20 + 901 903 901 903 864 920 c 128,-1,21 + 827 937 827 937 787 937 c 0,22,23 + 698 937 698 937 629.5 856.5 c 128,-1,24 + 561 776 561 776 517 586 c 2,25,-1 + 502 518 l 2,26,27 + 462 345 462 345 483.5 273 c 128,-1,28 + 505 201 505 201 593 201 c 0,29,30 + 654 201 654 201 706 232 c 128,-1,31 + 758 263 758 263 799 317 c 1,18,-1 +EndSplineSet +EndChar + +StartChar: r +Encoding: 105 114 85 +Width: 891 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +SplineSet +1051 751 m 1,0,-1 + 799 751 l 1,1,2 + 810 797 810 797 800 829.5 c 128,-1,3 + 790 862 790 862 738 862 c 0,4,5 + 689 862 689 862 649.5 833 c 128,-1,6 + 610 804 610 804 592 772 c 1,7,-1 + 414 0 l 1,8,-1 + 130 0 l 1,9,-1 + 382 1093 l 1,10,-1 + 655 1093 l 1,11,-1 + 619 911 l 1,12,13 + 680 1009 680 1009 753.5 1062.5 c 128,-1,14 + 827 1116 827 1116 910 1116 c 0,15,16 + 1034 1116 1034 1116 1073 1022 c 0,17,18 + 1103 952 1103 952 1076 835 c 0,19,20 + 1067 796 1067 796 1051 751 c 1,0,-1 +EndSplineSet +EndChar + +StartChar: s +Encoding: 106 115 86 +Width: 820 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 146 + 4 + 1 + 11 + 17 + 11 + 20 + 2 + 37 + 20 + 1 + 27 + 17 + 27 + 20 + 18 + 37 + 52 + 15 + 52 + 37 + 70 + 36 + 119 + 37 + 139 + 17 + 152 + 2 + 155 + 17 + 166 + 35 + 184 + 20 + 181 + 35 + 235 + 17 + 19 + 70 + 35 + 166 + 35 + 182 + 35 + 214 + 35 + 232 + 35 + 248 + 35 + 6 + 35 + 0 + 32 + 103 + 16 + 233 + 16 + 246 + 16 + 3 + 16 + 13 + 19 + 19 + 32 + 13 + 0 + 4 + 3 + 22 + 143 + 26 + 159 + 26 + 175 + 26 + 3 + 26 + 29 + 46 + 143 + 25 + 159 + 25 + 175 + 25 + 3 + 25 + 22 + 13 + 160 + 7 + 1 + 7 + 51 + 10 + 46 + 6 + 71 + 3 + 3 + 63 + 39 + 1 + 63 + 39 + 1 + 35 + 0 + 32 + 16 + 13 + 19 + 143 + 25 + 1 + 25 + 0 + 49 + 127 + 13 + 144 + 13 + 2 + 32 + 13 + 80 + 13 + 144 + 13 + 207 + 13 + 4 + 13 + 32 + 48 + 6 + 47 + 19 + 63 + 19 + 2 + 19 + 38 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MDRP[rnd,grey] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +DELTAP1 +DELTAP2 +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +DELTAP1 +SRP1 +SRP2 +IP +SRP1 +SRP2 +IP +DELTAP1 +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +SROUND +MDRP[rnd,grey] +RTG +MIRP[rp0,min,rnd,black] +MIRP[rnd,grey] +DELTAP1 +MIAP[rnd] +MDRP[rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +DELTAP1 +SRP1 +SRP2 +SLOOP +IP +SRP1 +SRP2 +IP +DELTAP1 +SRP1 +SRP2 +IP +DELTAP1 +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +845 329 m 0,0,1 + 810 178 810 178 685 76.5 c 128,-1,2 + 560 -25 560 -25 326 -25 c 0,3,4 + 229 -25 229 -25 153.5 -6.5 c 128,-1,5 + 78 12 78 12 48 39 c 1,6,-1 + 93 236 l 1,7,8 + 128 211 128 211 197.5 185.5 c 128,-1,9 + 267 160 267 160 345 160 c 0,10,11 + 445 160 445 160 507 197 c 128,-1,12 + 569 234 569 234 583 291 c 0,13,14 + 596 347 596 347 568 381.5 c 128,-1,15 + 540 416 540 416 405 465 c 0,16,17 + 278 509 278 509 246.5 590 c 128,-1,18 + 215 671 215 671 245 799 c 0,19,20 + 275 930 275 930 385.5 1020.5 c 128,-1,21 + 496 1111 496 1111 689 1111 c 0,22,23 + 778 1111 778 1111 836.5 1095.5 c 128,-1,24 + 895 1080 895 1080 942 1051 c 1,25,-1 + 901 871 l 1,26,27 + 867 895 867 895 798.5 911 c 128,-1,28 + 730 927 730 927 664 927 c 0,29,30 + 587 927 587 927 533 898.5 c 128,-1,31 + 479 870 479 870 468 821 c 0,32,33 + 455 765 455 765 486.5 743 c 128,-1,34 + 518 721 518 721 636 680 c 0,35,36 + 786 628 786 628 830.5 544.5 c 128,-1,37 + 875 461 875 461 845 329 c 0,0,1 +EndSplineSet +EndChar + +StartChar: t +Encoding: 107 116 87 +Width: 742 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 51 + 10 + 14 + 26 + 14 + 185 + 13 + 3 + 16 + 1 + 32 + 18 + 19 + 22 + 12 + 20 + 16 + 8 + 5 + 33 + 9 + 12 + 3 + 20 + 20 + 2 + 15 + 0 + 47 + 9 + 1 + 9 + 15 + 22 + 2 + 22 + 19 + 17 + 31 + 15 + 63 + 15 + 95 + 15 + 3 + 239 + 15 + 255 + 15 + 2 + 15 + 24 +SRP0 +MDRP[rp0,min,rnd,white] +DELTAP1 +DELTAP2 +MDRP[min,rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,black] +DELTAP1 +MDRP[rnd,grey] +SRP1 +SRP2 +IP +MDAP[rnd] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MIAP[rnd] +MIAP[rnd] +MDRP[rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +909 952 m 1,0,-1 + 649 952 l 1,1,-1 + 513 362 l 2,2,3 + 492 272 492 272 508 230 c 128,-1,4 + 524 188 524 188 598 188 c 0,5,6 + 633 188 633 188 667 194.5 c 128,-1,7 + 701 201 701 201 731 211 c 1,8,-1 + 689 31 l 1,9,10 + 640 10 640 10 572.5 0 c 128,-1,11 + 505 -10 505 -10 442 -10 c 0,12,13 + 304 -10 304 -10 247 91.5 c 128,-1,14 + 190 193 190 193 234 383 c 2,15,-1 + 366 954 l 1,16,-1 + 249 954 l 1,17,-1 + 287 1118 l 1,18,-1 + 416 1126 l 1,19,-1 + 600 1509 l 1,20,-1 + 778 1509 l 1,21,-1 + 690 1128 l 1,22,-1 + 950 1128 l 1,23,-1 + 909 952 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: u +Encoding: 108 117 88 +Width: 1145 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 54 + 43 + 6 + 235 + 6 + 251 + 6 + 3 + 10 + 6 + 140 + 6 + 154 + 6 + 172 + 6 + 4 + 17 + 2 + 96 + 5 + 0 + 2 + 14 + 45 + 57 + 5 + 72 + 5 + 2 + 5 + 3 + 18 + 9 + 12 + 96 + 21 + 1 + 19 + 22 + 1 + 2 + 64 + 18 + 160 + 18 + 2 + 175 + 18 + 1 + 18 + 10 + 22 + 9 + 20 +SRP0 +MDRP[rp0,rnd,white] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +DELTAP1 +DELTAP2 +MDRP[rnd,grey] +MDRP[grey] +MIRP[rp0,min,rnd,black] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIAP[rnd] +DELTAP1 +MIRP[rp0,min,rnd,black] +MIAP[rnd] +SRP0 +SMD +MDRP[rp0,min,grey] +MDRP[min,rnd,grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP2 +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1006 0 m 1,0,-1 + 733 0 l 1,1,-1 + 770 180 l 1,2,3 + 704 104 704 104 603.5 40.5 c 128,-1,4 + 503 -23 503 -23 377 -23 c 0,5,6 + 259 -23 259 -23 209.5 54 c 128,-1,7 + 160 131 160 131 206 328 c 2,8,-1 + 382 1093 l 1,9,-1 + 666 1093 l 1,10,-1 + 498 366 l 2,11,12 + 479 282 479 282 504.5 252.5 c 128,-1,13 + 530 223 530 223 577 223 c 0,14,15 + 637 223 637 223 693 247 c 128,-1,16 + 749 271 749 271 796 321 c 1,17,-1 + 974 1093 l 1,18,-1 + 1259 1093 l 1,19,-1 + 1006 0 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: v +Encoding: 109 118 89 +Width: 998 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 112 + 25 + 5 + 56 + 5 + 59 + 6 + 84 + 1 + 89 + 6 + 106 + 2 + 118 + 1 + 121 + 6 + 150 + 1 + 156 + 2 + 169 + 2 + 184 + 2 + 12 + 16 + 8 + 1 + 21 + 5 + 85 + 5 + 154 + 5 + 3 + 63 + 0 + 140 + 0 + 2 + 0 + 60 + 22 + 6 + 1 + 6 + 59 + 1 + 1 + 1 + 60 + 5 + 5 + 6 + 116 + 3 + 132 + 3 + 2 + 3 + 59 + 123 + 4 + 1 + 4 + 25 + 2 + 60 + 2 + 2 + 2 + 59 + 121 + 5 + 1 + 5 + 5 + 4 + 5 + 2 + 3 + 1 + 2 + 0 + 6 + 4 + 0 + 3 + 12 + 1 + 6 + 0 + 2 + 4 + 3 + 106 + 5 + 1 + 5 + 5 + 0 + 3 + 0 + 0 + 1 + 0 + 8 + 18 + 3 + 1 + 3 + 7 +SVTCA[x-axis] +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +SRP1 +SRP2 +IP +RTHG +MDAP[rnd] +RTG +DELTAP1 +SRP2 +IP +IP +SRP2 +IP +IP +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +MIAP[rnd] +MDRP[rnd,grey] +SRP1 +SRP2 +IP +SPVTL[orthog] +SFVTCA[x-axis] +SRP0 +DELTAP1 +MIRP[rp0,min,rnd,black] +DELTAP1 +MDAP[rnd] +DELTAP1 +MIRP[rp0,min,rnd,black] +DELTAP1 +SPVTL[orthog] +SRP0 +MIRP[rp0,min,rnd,black] +DELTAP1 +MDAP[rnd] +DELTAP1 +MIRP[rp0,min,rnd,black] +DELTAP1 +IUP[x] +IUP[y] +SVTCA[y-axis] +DELTAP1 +SVTCA[x-axis] +DELTAP1 +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1266 1093 m 1,0,-1 + 590 0 l 1,1,-1 + 391 0 l 1,2,-1 + 246 1093 l 1,3,-1 + 269 896 l 0,4,5 + 268 909 268 909 258.5 918.5 c 128,-1,6 + 249 928 249 928 237 928 c 0,7,8 + 189 928 189 928 95 772 c 2,9,-1 + 73 734 l 1,10,-1 + 24 769 l 1,11,12 + 139 972 139 972 213.5 1043.5 c 128,-1,13 + 288 1115 288 1115 388 1115 c 0,14,15 + 522 1115 522 1115 536 983 c 0,16,-1 + 523 1093 l 1,17,-1 + 602 349 l 1,18,-1 + 1018 1024 l 0,19,20 + 1077 1120 1077 1120 1170 1119 c 0,21,22 + 1247 1119 1247 1119 1282 1052.5 c 128,-1,23 + 1317 986 1317 986 1332 811 c 1,24,-1 + 1285 782 l 1,25,-1 + 1270 873 l 2,26,27 + 1263 921 1263 921 1247.5 947.5 c 128,-1,28 + 1232 974 1232 974 1214 974 c 0,29,30 + 1194 974 1194 974 1174 939 c 0,31,-1 + 1266 1093 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: w +Encoding: 110 119 90 +Width: 1497 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 199 + 11 + 3 + 31 + 8 + 70 + 8 + 68 + 11 + 80 + 8 + 80 + 11 + 111 + 3 + 103 + 11 + 159 + 3 + 9 + 12 + 2 + 8 + 5 + 6 + 9 + 26 + 2 + 20 + 4 + 41 + 8 + 57 + 2 + 57 + 4 + 56 + 5 + 54 + 7 + 51 + 10 + 74 + 2 + 70 + 10 + 87 + 1 + 88 + 2 + 88 + 5 + 86 + 8 + 88 + 9 + 87 + 10 + 88 + 11 + 136 + 12 + 151 + 4 + 22 + 63 + 14 + 1 + 8 + 60 + 7 + 8 + 4 + 48 + 9 + 1 + 9 + 60 + 3 + 4 + 3 + 132 + 11 + 1 + 11 + 59 + 11 + 12 + 2 + 116 + 10 + 132 + 10 + 2 + 10 + 59 + 3 + 3 + 2 + 246 + 11 + 1 + 11 + 60 + 23 + 1 + 1 + 1 + 59 + 12 + 68 + 12 + 246 + 12 + 3 + 12 + 60 + 5 + 0 + 53 + 0 + 2 + 0 + 0 + 1 + 118 + 8 + 134 + 8 + 2 + 8 + 59 + 5 + 116 + 7 + 132 + 7 + 2 + 7 + 59 + 5 + 6 + 6 + 5 + 1 + 12 + 0 + 2 + 9 + 11 + 4 + 10 + 5 + 7 + 6 + 8 + 3 + 11 + 8 + 0 + 6 + 11 + 0 + 63 + 0 + 176 + 0 + 3 + 0 + 9 + 10 + 8 + 6 + 47 + 6 + 55 + 6 + 3 + 6 + 13 + 8 + 3 + 11 + 3 + 5 + 6 + 7 + 9 + 10 + 12 + 0 + 5 + 6 + 12 + 4 + 2 + 1 + 3 + 5 + 0 +SVTCA[y-axis] +MIAP[rnd] +SLOOP +ALIGNRP +MIAP[rnd] +SLOOP +ALIGNRP +SRP1 +SRP2 +SLOOP +IP +SVTCA[x-axis] +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MDRP[rp0,min,rnd,grey] +MDRP[rp0,min,rnd,black] +MDRP[rp0,min,rnd,grey] +DELTAP1 +SRP1 +SRP2 +IP +IP +IP +RTHG +MDAP[rnd] +SRP2 +IP +IP +SRP2 +IP +MDAP[rnd] +SRP2 +IP +SRP2 +IP +IP +RTG +SPVTL[orthog] +SRP0 +MDRP[grey] +MIRP[rp0,min,rnd,black] +DELTAP1 +SRP0 +MIRP[rp0,min,rnd,black] +DELTAP1 +SPVTL[orthog] +SRP0 +DELTAP1 +MIRP[rp0,min,rnd,black] +DELTAP1 +MDAP[no-rnd] +DELTAP1 +MIRP[rp0,min,rnd,black] +DELTAP1 +SPVTL[orthog] +SRP0 +MIRP[rp0,min,rnd,black] +DELTAP1 +MDAP[no-rnd] +SFVTL[parallel] +MIRP[rp0,min,rnd,black] +DELTAP1 +SPVTL[orthog] +SRP0 +SFVTCA[x-axis] +MIRP[rp0,min,rnd,black] +DELTAP1 +MDAP[no-rnd] +SFVTL[parallel] +MIRP[rp0,min,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +DELTAP1 +SVTCA[y-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1769 1093 m 1,0,-1 + 1116 0 l 1,1,-1 + 947 0 l 1,2,-1 + 882 556 l 1,3,-1 + 546 0 l 1,4,-1 + 378 0 l 1,5,-1 + 247 1093 l 1,6,-1 + 268 898 l 0,7,8 + 267 911 267 911 257.5 920.5 c 128,-1,9 + 248 930 248 930 236 930 c 0,10,11 + 190 930 190 930 96 777 c 2,12,-1 + 73 737 l 1,13,-1 + 23 771 l 1,14,15 + 137 972 137 972 212 1043.5 c 128,-1,16 + 287 1115 287 1115 385 1115 c 0,17,18 + 517 1115 517 1115 531 981 c 0,19,-1 + 520 1093 l 1,20,-1 + 597 416 l 1,21,-1 + 976 1093 l 1,22,-1 + 1101 1093 l 1,23,-1 + 1161 416 l 1,24,-1 + 1531 1068 l 0,25,26 + 1593 1177 1593 1177 1688 1177 c 0,27,28 + 1765 1177 1765 1177 1798 1097.5 c 128,-1,29 + 1831 1018 1831 1018 1838 810 c 1,30,-1 + 1790 781 l 1,31,-1 + 1775 871 l 2,32,33 + 1768 919 1768 919 1752.5 945.5 c 128,-1,34 + 1737 972 1737 972 1719 972 c 0,35,36 + 1699 972 1699 972 1679 937 c 0,37,-1 + 1769 1093 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: x +Encoding: 111 120 91 +Width: 1028 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 160 + 54 + 2 + 67 + 8 + 86 + 8 + 102 + 8 + 118 + 8 + 149 + 8 + 6 + 11 + 1 + 5 + 3 + 5 + 7 + 11 + 9 + 25 + 1 + 21 + 3 + 21 + 7 + 38 + 11 + 91 + 5 + 85 + 11 + 154 + 1 + 154 + 2 + 167 + 11 + 198 + 11 + 215 + 11 + 15 + 11 + 8 + 0 + 7 + 8 + 9 + 7 + 0 + 137 + 5 + 1 + 2 + 5 + 1 + 6 + 6 + 5 + 6 + 1 + 116 + 1 + 128 + 1 + 2 + 1 + 59 + 0 + 0 + 7 + 116 + 7 + 132 + 7 + 2 + 7 + 59 + 6 + 6 + 1 + 11 + 2 + 10 + 3 + 3 + 10 + 8 + 5 + 9 + 4 + 4 + 9 + 9 + 60 + 10 + 3 + 10 + 3 + 60 + 4 + 4 + 9 + 8 + 5 + 11 + 2 + 4 + 4 + 6 + 7 + 8 + 10 + 3 + 6 + 12 + 3 + 1 + 0 + 3 + 4 + 2 + 9 + 3 + 4 + 10 + 10 + 7 + 1 + 3 + 0 + 159 + 6 + 1 + 6 + 5 + 8 + 2 + 11 + 6 + 5 + 16 + 0 + 32 + 0 + 112 + 0 + 144 + 0 + 4 + 0 + 31 + 4 + 47 + 4 + 2 + 4 + 12 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MDRP[rp0,min,rnd,black] +DELTAP1 +SLOOP +IP +MDAP[rnd] +DELTAP1 +SRP2 +SLOOP +IP +MDAP[rnd] +SRP2 +IP +IP +SVTCA[y-axis] +MIAP[rnd] +SLOOP +ALIGNRP +MIAP[rnd] +SLOOP +ALIGNRP +SRP1 +SRP2 +SLOOP +IP +SPVTL[orthog] +SFVTCA[x-axis] +SRP0 +MIRP[rp0,min,rnd,black] +SPVTL[orthog] +SRP0 +MIRP[rp0,min,rnd,black] +SPVTL[orthog] +SFVTCA[x-axis] +SRP1 +SRP2 +IP +IP +SPVTL[orthog] +SRP1 +SRP2 +IP +IP +SPVTL[orthog] +SFVTCA[x-axis] +SRP0 +MIRP[rp0,min,rnd,black] +DELTAP1 +SPVTL[orthog] +SRP0 +MIRP[rp0,min,rnd,black] +DELTAP1 +SPVTL[parallel] +SFVTL[parallel] +SRP1 +SRP2 +IP +IP +DELTAP1 +SPVTL[orthog] +SFVTL[parallel] +SRP1 +SRP2 +IP +IP +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +SVTCA[y-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +978 191 m 1,0,1 + 986 167 986 167 1005 167 c 0,2,3 + 1053 167 1053 167 1148 325 c 2,4,-1 + 1169 361 l 1,5,-1 + 1219 326 l 1,6,7 + 1103 122 1103 122 1027.5 50 c 128,-1,8 + 952 -22 952 -22 852 -22 c 0,9,10 + 738 -22 738 -22 709 60 c 1,11,-1 + 588 382 l 1,12,-1 + 281 34 l 1,13,14 + 229 -25 229 -25 148 -25 c 0,15,16 + 67 -25 67 -25 31.5 43 c 128,-1,17 + -4 111 -4 111 -21 291 c 1,18,-1 + 30 322 l 1,19,-1 + 43 241 l 2,20,21 + 51 188 51 188 66.5 158.5 c 128,-1,22 + 82 129 82 129 101 129 c 0,23,24 + 104 129 104 129 107 130 c 128,-1,25 + 110 131 110 131 112.5 132.5 c 128,-1,26 + 115 134 115 134 118 137 c 128,-1,27 + 121 140 121 140 122.5 142 c 128,-1,28 + 124 144 124 144 127.5 148 c 128,-1,29 + 131 152 131 152 133 154 c 1,30,-1 + 522 567 l 1,31,-1 + 297 1093 l 1,32,-1 + 614 1093 l 1,33,-1 + 727 788 l 1,34,-1 + 1003 1093 l 1,35,-1 + 1259 1093 l 1,36,-1 + 797 604 l 1,37,-1 + 978 191 l 1,0,1 +EndSplineSet +EndChar + +StartChar: y +Encoding: 112 121 92 +Width: 998 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 115 + 84 + 6 + 1 + 21 + 6 + 56 + 6 + 90 + 3 + 109 + 3 + 153 + 3 + 152 + 4 + 6 + 62 + 2 + 1 + 2 + 60 + 8 + 1 + 18 + 1 + 38 + 1 + 60 + 1 + 4 + 1 + 4 + 3 + 1 + 3 + 4 + 3 + 4 + 6 + 20 + 6 + 38 + 6 + 169 + 6 + 4 + 6 + 5 + 6 + 61 + 7 + 1 + 7 + 60 + 11 + 0 + 52 + 0 + 2 + 0 + 0 + 1 + 13 + 3 + 1 + 60 + 3 + 173 + 3 + 2 + 3 + 59 + 6 + 115 + 4 + 132 + 4 + 2 + 4 + 59 + 120 + 5 + 1 + 5 + 6 + 5 + 3 + 2 + 7 + 0 + 2 + 5 + 4 + 6 + 1 + 6 + 0 + 4 + 90 + 0 + 1 + 0 + 9 + 4 + 8 + 3 + 2 + 0 + 7 + 5 + 4 + 12 + 1 + 2 + 18 +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIAP[rnd] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +SRP2 +IP +SVTCA[x-axis] +SRP0 +MDRP[rp0,rnd,white] +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +SRP1 +SRP2 +IP +IP +RTHG +MDAP[rnd] +RTG +SRP2 +IP +IP +SRP2 +IP +MDAP[rnd] +IP +SPVTL[orthog] +MDAP[rnd] +DELTAP1 +MIRP[rp0,min,rnd,black] +DELTAP1 +SRP0 +MIRP[rp0,min,rnd,black] +DELTAP1 +DELTAP2 +SPVTL[orthog] +SRP0 +DELTAP1 +MIRP[rp0,min,rnd,black] +DELTAP1 +SFVTL[parallel] +MDRP[grey] +DELTAP1 +SFVTL[parallel] +MDRP[grey] +DELTAP1 +SFVTCA[x-axis] +MDAP[rnd] +DELTAP1 +MIRP[rp0,min,rnd,black] +DELTAP1 +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +SVTCA[y-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1266 1093 m 1,0,-1 + 367 -342 l 1,1,-1 + 142 -342 l 1,2,-1 + 382 27 l 1,3,-1 + 244 1093 l 1,4,-1 + 268 897 l 0,5,6 + 267 910 267 910 257 920 c 128,-1,7 + 247 930 247 930 236 930 c 0,8,9 + 188 930 188 930 93 771 c 1,10,-1 + 72 733 l 1,11,-1 + 22 768 l 1,12,13 + 137 971 137 971 212 1043 c 128,-1,14 + 287 1115 287 1115 386 1115 c 0,15,16 + 520 1115 520 1115 535 983 c 0,17,-1 + 522 1093 l 1,18,-1 + 600 347 l 1,19,-1 + 1044 1093 l 1,20,-1 + 996 1014 l 0,21,22 + 1058 1118 1058 1118 1158 1118 c 0,23,24 + 1241 1118 1241 1118 1277.5 1050 c 128,-1,25 + 1314 982 1314 982 1330 800 c 1,26,-1 + 1278 768 l 1,27,-1 + 1266 847 l 2,28,29 + 1258 903 1258 903 1242 932.5 c 128,-1,30 + 1226 962 1226 962 1206 962 c 0,31,32 + 1187 962 1187 962 1166 930 c 0,33,-1 + 1266 1093 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: z +Encoding: 113 122 93 +Width: 874 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 11 + 56 + 7 + 136 + 3 + 134 + 8 + 3 + 96 + 8 + 1 + 8 +PUSHW_1 + -16 +NPUSHB + 43 + 3 + 16 + 8 + 59 + 2 + 2 + 3 + 3 + 59 + 7 + 7 + 8 + 7 + 3 + 6 + 3 + 33 + 6 + 12 + 2 + 8 + 1 + 8 + 33 + 1 + 0 + 63 + 11 + 1 + 3 + 7 + 16 + 0 + 1 + 0 + 11 + 4 + 8 + 128 + 2 + 1 + 2 + 10 +SRP0 +MDRP[rp0,min,rnd,white] +DELTAP1 +MDRP[grey] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MDRP[rp0,rnd,grey] +MDRP[grey] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +MIAP[rnd] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +SPVTL[orthog] +SFVTCA[x-axis] +SRP0 +MIRP[rp0,min,rnd,black] +SPVTL[orthog] +SRP0 +MIRP[rp0,min,rnd,black] +SVTCA[x-axis] +SHPIX +SHPIX +DELTAP1 +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +855 0 m 1,0,-1 + 35 0 l 1,1,-1 + 84 212 l 1,2,-1 + 743 888 l 1,3,-1 + 249 888 l 1,4,-1 + 296 1093 l 1,5,-1 + 1085 1093 l 1,6,-1 + 1036 881 l 1,7,-1 + 362 205 l 1,8,-1 + 902 205 l 1,9,-1 + 855 0 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: braceleft +Encoding: 114 123 94 +Width: 749 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 63 + 11 + 3 + 11 + 18 + 27 + 3 + 27 + 18 + 246 + 32 + 5 + 6 + 32 + 1 + 31 + 10 + 45 + 11 + 74 + 11 + 21 + 0 + 40 + 73 + 0 + 18 + 22 + 73 + 21 + 16 + 4 + 31 + 1 + 22 + 31 + 1 + 31 + 10 + 22 + 0 + 4 + 28 + 34 + 24 + 10 + 14 + 17 + 9 + 4 + 1 + 40 + 4 + 56 + 4 + 72 + 4 + 88 + 4 + 249 + 4 + 5 + 4 + 41 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +DELTAP2 +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[min,rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +SRP0 +MDRP[grey] +DELTAP1 +DELTAP2 +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +SROUND +MDAP[rnd] +RTG +MIRP[rp0,min,rnd,black] +IP +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP2 +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +604 -344 m 1,0,-1 + 414 -344 l 2,1,2 + 299 -344 299 -344 247 -265 c 128,-1,3 + 195 -186 195 -186 218 -86 c 0,4,5 + 246 35 246 35 269.5 139.5 c 128,-1,6 + 293 244 293 244 323 373 c 0,7,8 + 337 434 337 434 310.5 461.5 c 128,-1,9 + 284 489 284 489 188 489 c 1,10,-1 + 243 731 l 1,11,12 + 343 731 343 731 380 747.5 c 128,-1,13 + 417 764 417 764 431 825 c 0,14,15 + 461 956 461 956 480 1043 c 128,-1,16 + 499 1130 499 1130 527 1251 c 0,17,18 + 550 1352 550 1352 642 1430.5 c 128,-1,19 + 734 1509 734 1509 842 1509 c 2,20,-1 + 1032 1509 l 1,21,-1 + 988 1317 l 1,22,23 + 988 1317 988 1317 956 1317 c 128,-1,24 + 924 1317 924 1317 889 1317 c 0,25,26 + 842 1317 842 1317 824 1270 c 128,-1,27 + 806 1223 806 1223 780 1110 c 0,28,29 + 725 870 725 870 672 755.5 c 128,-1,30 + 619 641 619 641 498 612 c 1,31,32 + 606 575 606 575 613.5 498.5 c 128,-1,33 + 621 422 621 422 569 197 c 0,34,35 + 516 -33 516 -33 509.5 -92.5 c 128,-1,36 + 503 -152 503 -152 550 -152 c 0,37,38 + 584 -152 584 -152 616.5 -152 c 128,-1,39 + 649 -152 649 -152 649 -152 c 1,40,-1 + 604 -344 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: bar +Encoding: 115 124 95 +Width: 778 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 10 + 1 + 2 + 16 + 63 + 0 + 1 + 0 + 58 + 1 + 4 +SRP0 +MDRP[rp0,rnd,white] +MIRP[rp0,min,rnd,black] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MDAP[rnd] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +416 -410 m 1,0,-1 + 176 -410 l 1,1,-1 + 619 1511 l 1,2,-1 + 859 1511 l 1,3,-1 + 416 -410 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: braceright +Encoding: 116 125 96 +Width: 749 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 70 + 9 + 19 + 1 + 3 + 7 + 3 + 33 + 19 + 7 + 19 + 33 + 249 + 19 + 5 + 20 + 0 + 45 + 0 + 40 + 16 + 40 + 96 + 40 + 3 + 40 + 74 + 40 + 11 + 30 + 11 + 73 + 10 + 18 + 29 + 73 + 30 + 16 + 11 + 20 + 1 + 25 + 20 + 1 + 20 + 0 + 29 + 12 + 6 + 23 + 17 + 24 + 0 + 37 + 34 + 6 + 6 + 1 + 39 + 6 + 55 + 6 + 71 + 6 + 87 + 6 + 246 + 6 + 5 + 6 + 42 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +DELTAP2 +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[min,rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +SRP0 +MDRP[grey] +DELTAP1 +DELTAP2 +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +SROUND +MDAP[rnd] +RTG +DELTAP1 +MIRP[rp0,min,rnd,black] +IP +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +DELTAP2 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +788 489 m 1,0,1 + 691 489 691 489 652 461.5 c 128,-1,2 + 613 434 613 434 599 373 c 0,3,4 + 569 244 569 244 544 139.5 c 128,-1,5 + 519 35 519 35 491 -86 c 0,6,7 + 468 -186 468 -186 380 -265 c 128,-1,8 + 292 -344 292 -344 176 -344 c 2,9,-1 + -14 -344 l 1,10,-1 + 30 -152 l 1,11,12 + 30 -152 30 -152 62.5 -152 c 128,-1,13 + 95 -152 95 -152 129 -152 c 0,14,15 + 176 -152 176 -152 197 -92.5 c 128,-1,16 + 218 -33 218 -33 271 197 c 0,17,18 + 323 422 323 422 366 498.5 c 128,-1,19 + 409 575 409 575 534 612 c 1,20,21 + 426 641 426 641 426 755.5 c 128,-1,22 + 426 870 426 870 482 1110 c 0,23,24 + 508 1223 508 1223 512 1270 c 128,-1,25 + 516 1317 516 1317 468 1317 c 0,26,27 + 434 1317 434 1317 402 1317 c 128,-1,28 + 370 1317 370 1317 370 1317 c 1,29,-1 + 414 1509 l 1,30,-1 + 604 1509 l 2,31,32 + 713 1509 713 1509 768 1430.5 c 128,-1,33 + 823 1352 823 1352 800 1251 c 0,34,35 + 772 1130 772 1130 751 1043 c 128,-1,36 + 730 956 730 956 700 825 c 0,37,38 + 686 764 686 764 714 747.5 c 128,-1,39 + 742 731 742 731 843 731 c 1,40,-1 + 788 489 l 1,0,1 +EndSplineSet +EndChar + +StartChar: asciitilde +Encoding: 117 126 97 +Width: 1210 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 78 + 54 + 2 + 63 + 14 + 48 + 36 + 69 + 2 + 4 + 6 + 9 + 3 + 12 + 31 + 28 + 34 + 25 + 34 + 34 + 3 + 18 + 12 + 34 + 25 + 40 + 6 + 52 + 9 + 1 + 9 + 3 + 12 + 59 + 31 + 1 + 31 + 28 + 34 + 25 + 25 + 34 + 12 + 3 + 4 + 0 + 22 + 18 + 15 + 41 + 0 + 34 + 40 + 112 + 37 + 1 + 0 + 37 + 16 + 37 + 144 + 37 + 3 + 37 + 70 + 15 + 34 + 19 + 15 + 22 + 47 + 22 + 63 + 22 + 239 + 22 + 4 + 22 + 44 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MDRP[rnd,grey] +MIRP[min,rnd,black] +SROUND +MDRP[rp0,min,rnd,grey] +RTG +DELTAP1 +DELTAP2 +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rnd,grey] +SRP1 +SRP2 +SLOOP +IP +SRP1 +SRP2 +IP +IP +DELTAP1 +SRP1 +SRP2 +IP +DELTAP1 +IP +SVTCA[y-axis] +MDAP[rnd] +MDRP[rp0,rnd,grey] +MIRP[rp0,min,rnd,black] +MDAP[rnd] +MDRP[rp0,rnd,grey] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +IP +SRP1 +SRP2 +IP +IP +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1353 731 m 0,0,1 + 1326 616 1326 616 1230.5 531 c 128,-1,2 + 1135 446 1135 446 989 446 c 0,3,4 + 903 446 903 446 829.5 492.5 c 128,-1,5 + 756 539 756 539 700 596 c 0,6,7 + 693 604 693 604 688 607 c 128,-1,8 + 683 610 683 610 675 618 c 0,9,10 + 636 664 636 664 598 699.5 c 128,-1,11 + 560 735 560 735 512 735 c 0,12,13 + 449 735 449 735 414 699.5 c 128,-1,14 + 379 664 379 664 370 621 c 0,15,16 + 358 571 358 571 365 530 c 128,-1,17 + 372 489 372 489 392 449 c 1,18,-1 + 215 449 l 1,19,20 + 188 489 188 489 178.5 537.5 c 128,-1,21 + 169 586 169 586 189 672 c 0,22,23 + 214 782 214 782 316 869 c 128,-1,24 + 418 956 418 956 551 956 c 0,25,26 + 637 956 637 956 710.5 911 c 128,-1,27 + 784 866 784 866 839 809 c 0,28,29 + 850 797 850 797 852.5 795 c 128,-1,30 + 855 793 855 793 861 782 c 0,31,32 + 900 737 900 737 941 702.5 c 128,-1,33 + 982 668 982 668 1030 668 c 0,34,35 + 1093 668 1093 668 1127.5 703.5 c 128,-1,36 + 1162 739 1162 739 1172 782 c 0,37,38 + 1183 831 1183 831 1176.5 872 c 128,-1,39 + 1170 913 1170 913 1149 954 c 1,40,-1 + 1326 954 l 1,41,42 + 1353 913 1353 913 1363 865 c 128,-1,43 + 1373 817 1373 817 1353 731 c 0,0,1 +EndSplineSet +EndChar + +StartChar: Adieresis +Encoding: 154 196 98 +Width: 1358 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 36 65 N 1 0 0 1 0 0 2 +Refer: 142 168 N 1 0 0 1 197 247 2 +EndChar + +StartChar: Aring +Encoding: 155 197 99 +Width: 1262 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 161 + 58 + 6 + 55 + 18 + 102 + 4 + 218 + 13 + 233 + 13 + 248 + 6 + 6 + 9 + 32 + 202 + 32 + 218 + 32 + 235 + 32 + 4 + 32 + 57 + 4 + 32 + 18 + 31 + 2 + 40 + 1 + 1 + 1 + 57 + 56 + 0 + 1 + 0 + 0 + 18 + 100 + 32 + 1 + 59 + 32 + 1 + 32 + 58 + 6 + 33 + 3 + 100 + 4 + 1 + 39 + 4 + 63 + 4 + 203 + 4 + 219 + 4 + 237 + 4 + 5 + 4 + 58 + 5 + 5 + 6 + 15 + 75 + 47 + 19 + 127 + 19 + 2 + 15 + 19 + 63 + 19 + 64 + 19 + 80 + 19 + 127 + 19 + 143 + 19 + 159 + 19 + 207 + 19 + 223 + 19 + 240 + 19 + 10 + 19 + 25 + 75 + 0 + 9 + 1 + 9 + 6 + 18 + 32 + 33 + 31 + 3 + 2 + 4 + 1 + 9 + 15 + 11 + 0 + 5 + 0 + 35 + 55 + 5 + 1 + 5 + 34 + 2 + 3 + 30 + 31 + 48 + 33 + 1 + 33 + 33 + 5 + 6 + 18 + 63 + 6 + 1 + 6 + 22 + 210 + 7 + 12 + 55 + 12 + 71 + 12 + 87 + 12 + 4 + 12 + 28 + 4 + 1 + 0 + 3 + 5 + 0 +SVTCA[y-axis] +MIAP[rnd] +SLOOP +IP +MDAP[rnd] +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[min,rnd,black] +MDRP[rp0,rnd,grey] +DELTAP1 +MDRP[rnd,grey] +SRP1 +SRP2 +IP +MDAP[rnd] +DELTAP1 +MDRP[rp0,rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SVTCA[x-axis] +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +SRP0 +MDRP[rp0,rnd,white] +SRP1 +SRP2 +SLOOP +IP +MDAP[rnd] +DELTAP1 +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +DELTAP1 +DELTAP2 +MIRP[rp0,min,rnd,black] +SPVTL[orthog] +SRP0 +MIRP[rp0,min,rnd,black] +DELTAP1 +DELTAP2 +MDRP[grey] +MDRP[grey] +SRP0 +MIRP[rp0,min,black] +DELTAP1 +DELTAP2 +SPVTL[orthog] +SRP0 +DELTAP1 +MIRP[rp0,min,rnd,black] +DELTAP1 +MDRP[grey] +MDRP[grey] +SRP0 +SFVTL[parallel] +MIRP[rp0,min,black] +DELTAP1 +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1257 0 m 1,0,-1 + 934 0 l 1,1,-1 + 914 362 l 1,2,-1 + 451 362 l 1,3,-1 + 261 0 l 1,4,-1 + 4 0 l 1,5,-1 + 758 1386 l 1,6,7 + 729 1419 729 1419 719 1470.5 c 128,-1,8 + 709 1522 709 1522 723 1581 c 0,9,10 + 749 1692 749 1692 841 1768.5 c 128,-1,11 + 933 1845 933 1845 1042 1845 c 0,12,13 + 1152 1845 1152 1845 1207.5 1768.5 c 128,-1,14 + 1263 1692 1263 1692 1237 1581 c 0,15,16 + 1225 1528 1225 1528 1197 1481 c 128,-1,17 + 1169 1434 1169 1434 1131 1401 c 1,18,-1 + 1257 0 l 1,0,-1 +1102 1581 m 0,19,20 + 1114 1632 1114 1632 1087 1672 c 128,-1,21 + 1060 1712 1060 1712 1011 1712 c 0,22,23 + 963 1712 963 1712 916.5 1672 c 128,-1,24 + 870 1632 870 1632 858 1581 c 0,25,26 + 846 1528 846 1528 874 1489 c 128,-1,27 + 902 1450 902 1450 950 1450 c 0,28,29 + 999 1450 999 1450 1044.5 1489 c 128,-1,30 + 1090 1528 1090 1528 1102 1581 c 0,19,20 +897 580 m 1,31,-1 + 853 1135 l 1,32,-1 + 563 580 l 1,33,-1 + 897 580 l 1,31,-1 +EndSplineSet +EndChar + +StartChar: Ccedilla +Encoding: 157 199 100 +Width: 1190 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 38 67 N 1 0 0 1 0 0 2 +Refer: 222 184 N 1 0 0 1 248 10 2 +EndChar + +StartChar: Eacute +Encoding: 159 201 101 +Width: 1167 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 40 69 N 1 0 0 1 0 0 2 +Refer: 141 180 N 1 0 0 1 216 260 2 +EndChar + +StartChar: Ntilde +Encoding: 167 209 102 +Width: 1397 +GlyphClass: 1 +Flags: W +TtInstrs: +SVTCA[y-axis] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 49 78 N 1 0 0 1 0 0 2 +Refer: 217 732 N 1 0 0 1 248 232 2 +EndChar + +StartChar: Odieresis +Encoding: 172 214 103 +Width: 1365 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_8 + 63 + 34 + 63 + 46 + 2 + 63 + 53 + 1 +DELTAP1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 50 79 N 1 0 0 1 0 0 3 +Refer: 142 168 N 1 0 0 1 246 247 2 +EndChar + +StartChar: Udieresis +Encoding: 178 220 104 +Width: 1343 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_5 + 63 + 24 + 63 + 36 + 2 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 56 85 N 1 0 0 1 0 0 2 +Refer: 142 168 N 1 0 0 1 255 247 2 +EndChar + +StartChar: aacute +Encoding: 183 225 105 +Width: 1238 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 29 + 0 + 46 + 16 + 46 + 2 + 143 + 46 + 159 + 46 + 175 + 46 + 191 + 46 + 4 + 16 + 49 + 63 + 49 + 159 + 46 + 175 + 46 + 191 + 46 + 192 + 49 + 240 + 49 + 7 +DELTAP1 +DELTAP1 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 68 97 N 1 0 0 1 0 0 2 +Refer: 141 180 N 1 0 0 1 105 -75 2 +EndChar + +StartChar: agrave +Encoding: 182 224 106 +Width: 1238 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 19 + 0 + 46 + 16 + 46 + 2 + 47 + 46 + 63 + 46 + 2 + 16 + 49 + 63 + 49 + 192 + 49 + 240 + 49 + 4 +DELTAP1 +DELTAP1 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 68 97 N 1 0 0 1 0 0 2 +Refer: 67 96 N 1 0 0 1 58 -75 2 +EndChar + +StartChar: acircumflex +Encoding: 184 226 107 +Width: 1238 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 19 + 0 + 46 + 16 + 46 + 2 + 16 + 52 + 63 + 52 + 192 + 52 + 240 + 52 + 4 + 0 + 48 + 16 + 48 + 2 +DELTAP1 +DELTAP1 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 68 97 N 1 0 0 1 0 0 2 +Refer: 216 710 N 1 0 0 1 101 -75 2 +EndChar + +StartChar: adieresis +Encoding: 186 228 108 +Width: 1238 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 29 + 16 + 47 + 16 + 59 + 2 + 0 + 47 + 0 + 59 + 2 + 48 + 47 + 48 + 59 + 2 + 16 + 50 + 16 + 62 + 2 + 16 + 69 + 63 + 69 + 192 + 69 + 240 + 69 + 4 +DELTAP1 +DELTAP1 +SVTCA[y-axis] +DELTAP1 +DELTAP1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 68 97 N 1 0 0 1 0 0 2 +Refer: 142 168 N 1 0 0 1 93 -73 2 +EndChar + +StartChar: atilde +Encoding: 185 227 109 +Width: 1238 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 21 + 0 + 47 + 16 + 47 + 2 + 0 + 57 + 16 + 57 + 47 + 57 + 3 + 16 + 71 + 63 + 71 + 192 + 71 + 240 + 71 + 4 +DELTAP1 +DELTAP1 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 68 97 N 1 0 0 1 0 0 2 +Refer: 217 732 N 1 0 0 1 99 -90 2 +EndChar + +StartChar: aring +Encoding: 187 229 110 +Width: 1238 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 61 + 0 + 47 + 0 + 66 + 16 + 47 + 16 + 66 + 4 + 16 + 50 + 16 + 62 + 95 + 50 + 95 + 62 + 4 + 223 + 50 + 223 + 62 + 2 + 0 + 50 + 0 + 62 + 16 + 50 + 16 + 62 + 159 + 50 + 159 + 62 + 175 + 50 + 175 + 62 + 191 + 50 + 191 + 62 + 207 + 50 + 207 + 62 + 239 + 50 + 239 + 62 + 14 + 16 + 69 + 63 + 69 + 192 + 69 + 240 + 69 + 4 +DELTAP1 +DELTAP1 +DELTAP2 +DELTAP3 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 68 97 N 1 0 0 1 0 0 2 +Refer: 221 730 N 1 0 0 1 103 -82 2 +EndChar + +StartChar: ccedilla +Encoding: 189 231 111 +Width: 1124 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_7 + 0 + 46 + 16 + 46 + 215 + 4 + 3 +SVTCA[x-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 70 99 N 1 0 0 1 0 0 2 +Refer: 222 184 N 1 0 0 1 143 10 2 +EndChar + +StartChar: eacute +Encoding: 191 233 112 +Width: 1303 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_5 + 0 + 33 + 16 + 33 + 2 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 72 101 N 1 0 0 1 0 0 2 +Refer: 141 180 N 1 0 0 1 117 -75 2 +EndChar + +StartChar: egrave +Encoding: 190 232 113 +Width: 1303 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_8 + 0 + 32 + 16 + 32 + 2 + 48 + 33 + 1 +DELTAP1 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 72 101 N 1 0 0 1 0 0 2 +Refer: 67 96 N 1 0 0 1 72 -75 2 +EndChar + +StartChar: ecircumflex +Encoding: 192 234 114 +Width: 1303 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_5 + 0 + 33 + 16 + 35 + 2 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 72 101 N 1 0 0 1 0 0 2 +Refer: 216 710 N 1 0 0 1 115 -75 2 +EndChar + +StartChar: edieresis +Encoding: 193 235 115 +Width: 1303 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 9 + 0 + 35 + 0 + 47 + 16 + 35 + 16 + 47 + 4 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 72 101 N 1 0 0 1 0 0 2 +Refer: 142 168 N 1 0 0 1 120 -88 2 +EndChar + +StartChar: iacute +Encoding: 195 237 116 +Width: 610 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_5 + 0 + 6 + 16 + 6 + 2 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 215 305 N 1 0 0 1 0 0 2 +Refer: 141 180 N 1 0 0 1 -129 -75 2 +EndChar + +StartChar: igrave +Encoding: 194 236 117 +Width: 610 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_8 + 0 + 5 + 16 + 5 + 2 + 48 + 6 + 1 +DELTAP1 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 215 305 N 1 0 0 1 0 0 2 +Refer: 67 96 N 1 0 0 1 -260 -75 2 +EndChar + +StartChar: icircumflex +Encoding: 196 238 118 +Width: 610 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_5 + 0 + 5 + 16 + 5 + 2 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 215 305 N 1 0 0 1 0 0 2 +Refer: 216 710 N 1 0 0 1 -218 -75 2 +EndChar + +StartChar: idieresis +Encoding: 197 239 119 +Width: 610 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 9 + 0 + 7 + 0 + 19 + 16 + 7 + 16 + 19 + 4 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 215 305 N 1 0 0 1 0 0 2 +Refer: 142 168 N 1 0 0 1 -227 -88 2 +EndChar + +StartChar: ntilde +Encoding: 199 241 120 +Width: 1296 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_5 + 0 + 22 + 16 + 22 + 2 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 81 110 N 1 0 0 1 0 0 2 +Refer: 217 732 N 1 0 0 1 113 -88 2 +EndChar + +StartChar: oacute +Encoding: 201 243 121 +Width: 1360 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_5 + 0 + 30 + 16 + 30 + 2 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 82 111 N 1 0 0 1 0 0 2 +Refer: 141 180 N 1 0 0 1 183 -75 2 +EndChar + +StartChar: ograve +Encoding: 200 242 122 +Width: 1360 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_8 + 0 + 30 + 16 + 30 + 2 + 48 + 30 + 1 +DELTAP1 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 82 111 N 1 0 0 1 0 0 2 +Refer: 67 96 N 1 0 0 1 104 -75 2 +EndChar + +StartChar: ocircumflex +Encoding: 202 244 123 +Width: 1360 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_5 + 0 + 32 + 16 + 32 + 2 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 82 111 N 1 0 0 1 0 0 2 +Refer: 216 710 N 1 0 0 1 165 -75 2 +EndChar + +StartChar: odieresis +Encoding: 204 246 124 +Width: 1360 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 9 + 0 + 31 + 0 + 43 + 16 + 31 + 16 + 43 + 4 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 82 111 N 1 0 0 1 0 0 2 +Refer: 142 168 N 1 0 0 1 150 -89 2 +EndChar + +StartChar: otilde +Encoding: 203 245 125 +Width: 1360 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_8 + 0 + 31 + 16 + 31 + 2 + 48 + 41 + 1 +DELTAP1 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 82 111 N 1 0 0 1 0 0 2 +Refer: 217 732 N 1 0 0 1 142 -89 2 +EndChar + +StartChar: uacute +Encoding: 208 250 126 +Width: 1296 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_8 + 0 + 22 + 16 + 22 + 2 + 96 + 25 + 1 +DELTAP1 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 88 117 N 1 0 0 1 0 0 2 +Refer: 141 180 N 1 0 0 1 157 -75 2 +EndChar + +StartChar: ugrave +Encoding: 207 249 127 +Width: 1296 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 10 + 0 + 21 + 16 + 21 + 2 + 48 + 22 + 96 + 25 + 2 +DELTAP1 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 88 117 N 1 0 0 1 0 0 2 +Refer: 67 96 N 1 0 0 1 74 -75 2 +EndChar + +StartChar: ucircumflex +Encoding: 209 251 128 +Width: 1296 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_8 + 0 + 23 + 16 + 23 + 2 + 96 + 28 + 1 +DELTAP1 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 88 117 N 1 0 0 1 0 0 2 +Refer: 216 710 N 1 0 0 1 116 -75 2 +EndChar + +StartChar: udieresis +Encoding: 210 252 129 +Width: 1296 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 12 + 0 + 23 + 0 + 35 + 16 + 23 + 16 + 35 + 4 + 96 + 45 + 1 +DELTAP1 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 88 117 N 1 0 0 1 0 0 2 +Refer: 142 168 N 1 0 0 1 111 -88 2 +EndChar + +StartChar: dagger +Encoding: 356 8224 130 +Width: 974 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 33 + 1 + 5 + 73 + 10 + 6 + 12 + 3 + 18 + 8 + 16 + 48 + 13 + 1 + 2 + 64 + 1 + 48 + 0 + 1 + 0 + 101 + 9 + 22 + 3 + 64 + 4 + 63 + 5 + 1 + 5 + 101 + 8 + 12 +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[min,rnd,grey] +DELTAP1 +MDRP[rnd,grey] +MIRP[rnd,black] +MIRP[rp0,min,rnd,black] +MIRP[min,rnd,grey] +DELTAP1 +MDRP[rnd,grey] +MIRP[rnd,black] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1181 952 m 1,0,-1 + 848 952 l 1,1,-1 + 511 -342 l 1,2,-1 + 288 -342 l 1,3,-1 + 549 952 l 1,4,-1 + 233 952 l 1,5,-1 + 278 1149 l 1,6,-1 + 594 1149 l 1,7,-1 + 677 1509 l 1,8,-1 + 977 1509 l 1,9,-1 + 894 1149 l 1,10,-1 + 1227 1149 l 1,11,-1 + 1181 952 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: degree +Encoding: 134 176 131 +Width: 763 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 32 + 5 + 1 + 9 + 5 + 9 + 7 + 5 + 11 + 4 + 3 + 73 + 21 + 15 + 73 + 9 + 15 + 0 + 73 + 12 + 18 + 73 + 63 + 6 + 143 + 6 + 160 + 6 + 176 + 6 + 4 + 6 + 24 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1005 1163 m 0,0,1 + 970 1012 970 1012 843.5 903.5 c 128,-1,2 + 717 795 717 795 577 795 c 0,3,4 + 436 795 436 795 360 903.5 c 128,-1,5 + 284 1012 284 1012 319 1163 c 0,6,7 + 354 1315 354 1315 480 1423.5 c 128,-1,8 + 606 1532 606 1532 747 1532 c 0,9,10 + 887 1532 887 1532 963.5 1423.5 c 128,-1,11 + 1040 1315 1040 1315 1005 1163 c 0,0,1 +811 1163 m 0,12,13 + 827 1231 827 1231 794.5 1277 c 128,-1,14 + 762 1323 762 1323 699 1323 c 256,15,16 + 636 1323 636 1323 582.5 1277 c 128,-1,17 + 529 1231 529 1231 513 1163 c 0,18,19 + 498 1096 498 1096 530 1050 c 128,-1,20 + 562 1004 562 1004 625 1004 c 256,21,22 + 688 1004 688 1004 742 1050 c 128,-1,23 + 796 1096 796 1096 811 1163 c 0,12,13 +EndSplineSet +EndChar + +StartChar: cent +Encoding: 120 162 132 +Width: 1100 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 38 + 22 + 51 + 25 + 34 + 21 + 51 + 16 + 18 + 15 + 35 + 51 + 32 + 35 + 0 + 51 + 3 + 5 + 6 + 3 + 18 + 3 + 32 + 15 + 6 + 6 + 0 + 9 + 22 + 0 + 39 + 29 + 55 + 29 + 2 + 29 + 23 + 8 + 36 +SRP0 +MDRP[rp0,rnd,white] +MIRP[min,rnd,black] +DELTAP1 +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +SRP1 +SRP2 +IP +MDAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SVTCA[y-axis] +MIAP[rnd] +MDRP[min,rnd,grey] +MDRP[rnd,grey] +MIRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MIRP[rnd,grey] +MDAP[rnd] +MDRP[rnd,grey] +MDRP[min,rnd,grey] +MIRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MIRP[rnd,grey] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +972 43 m 1,0,1 + 960 31 960 31 908 18.5 c 128,-1,2 + 856 6 856 6 788 2 c 1,3,-1 + 755 -143 l 1,4,-1 + 597 -143 l 1,5,-1 + 630 0 l 1,6,7 + 399 16 399 16 301.5 173 c 128,-1,8 + 204 330 204 330 271 616 c 0,9,10 + 274 627 274 627 276.5 639 c 128,-1,11 + 279 651 279 651 282 666 c 0,12,13 + 341 922 341 922 516.5 1067 c 128,-1,14 + 692 1212 692 1212 913 1229 c 1,15,-1 + 950 1389 l 1,16,-1 + 1108 1389 l 1,17,-1 + 1071 1227 l 1,18,19 + 1130 1221 1130 1221 1171.5 1208.5 c 128,-1,20 + 1213 1196 1213 1196 1234 1184 c 1,21,-1 + 1184 969 l 1,22,23 + 1148 987 1148 987 1096 1003.5 c 128,-1,24 + 1044 1020 1044 1020 966 1020 c 0,25,26 + 829 1020 829 1020 728.5 924.5 c 128,-1,27 + 628 829 628 829 587 651 c 2,28,-1 + 583 631 l 2,29,30 + 540 444 540 444 577.5 336.5 c 128,-1,31 + 615 229 615 229 751 229 c 0,32,33 + 838 229 838 229 910 239.5 c 128,-1,34 + 982 250 982 250 1023 262 c 1,35,-1 + 972 43 l 1,0,1 +EndSplineSet +EndChar + +StartChar: sterling +Encoding: 121 163 133 +Width: 1105 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 38 + 23 + 2 + 73 + 20 + 5 + 5 + 1 + 9 + 192 + 13 + 1 + 13 + 16 + 35 + 12 + 9 + 15 + 24 + 34 + 1 + 4 + 48 + 13 + 1 + 13 + 22 + 0 + 1 + 20 + 24 + 20 + 3 + 5 + 143 + 1 + 1 + 1 + 26 +SRP0 +MDRP[rp0,min,rnd,white] +DELTAP1 +MDRP[rnd,grey] +MDRP[min,rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,black] +IP +MDRP[rnd,grey] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +DELTAP1 +SRP1 +SRP2 +IP +MDAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1049 0 m 1,0,-1 + 158 0 l 1,1,-1 + 303 631 l 1,2,-1 + 167 631 l 1,3,-1 + 213 831 l 1,4,-1 + 349 831 l 1,5,-1 + 393 1020 l 2,6,7 + 444 1241 444 1241 628 1385.5 c 128,-1,8 + 812 1530 812 1530 1025 1530 c 0,9,10 + 1143 1530 1143 1530 1203 1517.5 c 128,-1,11 + 1263 1505 1263 1505 1339 1481 c 1,12,-1 + 1287 1253 l 1,13,14 + 1211 1280 1211 1280 1151.5 1290 c 128,-1,15 + 1092 1300 1092 1300 1021 1300 c 0,16,17 + 890 1300 890 1300 816 1231.5 c 128,-1,18 + 742 1163 742 1163 703 993 c 2,19,-1 + 666 831 l 1,20,-1 + 936 831 l 1,21,-1 + 890 631 l 1,22,-1 + 620 631 l 1,23,-1 + 526 225 l 1,24,-1 + 1101 225 l 1,25,-1 + 1049 0 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: section +Encoding: 125 167 134 +Width: 840 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 99 + 5 + 7 + 9 + 32 + 20 + 7 + 26 + 32 + 37 + 7 + 41 + 32 + 70 + 3 + 70 + 46 + 86 + 3 + 85 + 21 + 87 + 48 + 100 + 4 + 105 + 24 + 117 + 3 + 121 + 28 + 15 + 53 + 28 + 59 + 3 + 28 + 3 + 28 + 9 + 34 + 38 + 70 + 41 + 46 + 127 + 37 + 1 + 37 + 34 + 15 + 13 + 71 + 16 + 46 + 12 + 9 + 5 + 22 + 19 + 25 + 166 + 3 + 1 + 3 + 59 + 6 + 56 + 53 + 28 + 50 + 31 + 53 + 47 + 167 + 47 + 215 + 47 + 3 + 47 + 0 + 44 + 38 + 71 + 0 + 6 + 42 + 19 + 0 + 42 + 50 + 25 + 44 + 26 + 31 + 56 + 97 + 13 + 25 + 62 +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rnd,grey] +MIRP[min,rnd,black] +MDRP[rp0,rnd,grey] +MIRP[min,rnd,black] +SRP0 +MDRP[rp0,min,rnd,grey] +MIRP[min,rnd,black] +MDRP[rp0,rnd,grey] +MIRP[rp0,min,rnd,black] +SRP0 +SROUND +MDRP[rnd,grey] +RTG +SRP1 +SRP2 +IP +DELTAP1 +SRP1 +SRP2 +IP +IP +SRP1 +SRP1 +IP +IP +DELTAP1 +SRP1 +SRP2 +IP +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +SROUND +MDRP[rnd,grey] +RTG +MIAP[rnd] +MDRP[rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +SROUND +MDRP[rp0,rnd,grey] +RTG +SRP1 +SRP2 +IP +IP +RTDG +MDAP[rnd] +MDAP[rnd] +RTG +MDRP[grey] +SRP0 +MDRP[grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +956 772 m 0,0,1 + 936 684 936 684 891 617.5 c 128,-1,2 + 846 551 846 551 781 508 c 1,3,4 + 813 483 813 483 833 431 c 128,-1,5 + 853 379 853 379 833 289 c 0,6,7 + 799 143 799 143 669.5 59 c 128,-1,8 + 540 -25 540 -25 364 -25 c 0,9,10 + 269 -25 269 -25 187 -2.5 c 128,-1,11 + 105 20 105 20 74 37 c 1,12,-1 + 121 240 l 1,13,14 + 156 219 156 219 228.5 190.5 c 128,-1,15 + 301 162 301 162 371 162 c 0,16,17 + 469 162 469 162 523.5 199 c 128,-1,18 + 578 236 578 236 591 289 c 0,19,20 + 600 330 600 330 577.5 371 c 128,-1,21 + 555 412 555 412 439 475 c 0,22,23 + 352 524 352 524 293 586.5 c 128,-1,24 + 234 649 234 649 263 776 c 0,25,26 + 278 840 278 840 325 903.5 c 128,-1,27 + 372 967 372 967 422 1001 c 1,28,29 + 373 1032 373 1032 358.5 1086.5 c 128,-1,30 + 344 1141 344 1141 361 1214 c 0,31,32 + 391 1343 391 1343 502 1436.5 c 128,-1,33 + 613 1530 613 1530 786 1530 c 0,34,35 + 883 1530 883 1530 954 1507.5 c 128,-1,36 + 1025 1485 1025 1485 1053 1466 c 1,37,-1 + 1004 1253 l 1,38,39 + 966 1286 966 1286 896.5 1314.5 c 128,-1,40 + 827 1343 827 1343 745 1343 c 0,41,42 + 672 1343 672 1343 632.5 1316.5 c 128,-1,43 + 593 1290 593 1290 582 1241 c 0,44,45 + 566 1171 566 1171 621.5 1134.5 c 128,-1,46 + 677 1098 677 1098 747 1063 c 0,47,48 + 880 997 880 997 926.5 921.5 c 128,-1,49 + 973 846 973 846 956 772 c 0,0,1 +713 725 m 0,50,51 + 721 758 721 758 688.5 803 c 128,-1,52 + 656 848 656 848 551 905 c 1,53,54 + 525 891 525 891 503 859 c 128,-1,55 + 481 827 481 827 476 807 c 0,56,57 + 459 735 459 735 505.5 703.5 c 128,-1,58 + 552 672 552 672 649 616 c 1,59,60 + 672 633 672 633 689 664.5 c 128,-1,61 + 706 696 706 696 713 725 c 0,50,51 +EndSplineSet +EndChar + +StartChar: bullet +Encoding: 358 8226 135 +Width: 1104 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_5 + 3 + 9 + 0 + 6 + 12 +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rp0,min,rnd,black] +SVTCA[y-axis] +MDAP[rnd] +MDRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +333 768 m 0,0,1 + 374 944 374 944 520.5 1069.5 c 128,-1,2 + 667 1195 667 1195 829 1195 c 256,3,4 + 991 1195 991 1195 1079 1069 c 128,-1,5 + 1167 943 1167 943 1126 768 c 0,6,7 + 1086 595 1086 595 941 468 c 128,-1,8 + 796 341 796 341 631 341 c 256,9,10 + 466 341 466 341 379.5 467.5 c 128,-1,11 + 293 594 293 594 333 768 c 0,0,1 +EndSplineSet +EndChar + +StartChar: paragraph +Encoding: 140 182 136 +Width: 1105 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 54 + 8 + 7 + 24 + 7 + 105 + 8 + 121 + 8 + 4 + 15 + 35 + 6 + 6 + 5 + 13 + 16 + 3 + 34 + 13 + 10 + 1 + 5 + 0 + 0 + 115 + 111 + 1 + 127 + 1 + 2 + 1 + 4 + 34 + 15 + 224 + 5 + 1 + 5 + 63 + 19 + 78 + 19 + 86 + 19 + 3 + 19 + 42 + 79 + 9 + 95 + 9 + 2 + 9 + 22 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MIRP[min,rnd,black] +DELTAP1 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MDRP[rnd,grey] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP1 +SRP2 +IP +MDAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +995 0 m 1,0,-1 + 812 0 l 1,1,-1 + 1108 1284 l 1,2,-1 + 1015 1284 l 1,3,-1 + 719 0 l 1,4,-1 + 519 0 l 1,5,-1 + 661 616 l 1,6,7 + 446 616 446 616 341 715.5 c 128,-1,8 + 236 815 236 815 287 1034 c 2,9,-1 + 299 1085 l 2,10,11 + 343 1274 343 1274 499.5 1391.5 c 128,-1,12 + 656 1509 656 1509 890 1509 c 2,13,-1 + 1343 1509 l 1,14,-1 + 995 0 l 1,0,-1 +710 852 m 1,15,-1 + 809 1280 l 1,16,17 + 706 1280 706 1280 631 1223.5 c 128,-1,18 + 556 1167 556 1167 534 1071 c 0,19,20 + 511 971 511 971 550.5 911.5 c 128,-1,21 + 590 852 590 852 710 852 c 1,15,-1 +EndSplineSet +EndChar + +StartChar: germandbls +Encoding: 181 223 137 +Width: 1214 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 130 + 9 + 29 + 3 + 33 + 23 + 1 + 20 + 33 + 36 + 33 + 54 + 33 + 89 + 14 + 90 + 39 + 121 + 14 + 169 + 14 + 213 + 11 + 213 + 41 + 215 + 42 + 232 + 18 + 232 + 36 + 248 + 13 + 246 + 14 + 17 + 22 + 41 + 137 + 14 + 2 + 40 + 37 + 0 + 34 + 13 + 16 + 10 + 19 + 19 + 34 + 10 + 0 + 4 + 3 + 31 + 7 + 47 + 6 + 3 + 3 + 27 + 2 + 22 + 47 + 31 + 10 + 16 + 40 + 133 + 40 + 182 + 40 + 3 + 40 + 0 + 37 + 123 + 13 + 1 + 13 + 10 + 16 + 16 + 6 + 32 + 6 + 2 + 6 + 6 + 0 + 16 + 32 + 37 + 48 + 37 + 64 + 37 + 3 + 37 + 49 + 14 + 16 + 1 + 16 + 16 + 34 + 26 + 19 + 49 + 47 + 34 + 1 + 34 + 34 + 0 + 27 + 10 + 23 + 6 + 0 + 32 + 0 + 224 + 0 + 3 + 0 + 26 + 22 + 27 + 43 +SRP0 +MDRP[rp0,rnd,white] +MIRP[rp0,min,rnd,black] +MDRP[rp0,min,rnd,black] +DELTAP1 +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +MDAP[rnd] +DELTAP1 +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +MDAP[rnd] +DELTAP1 +MIRP[rp0,min,rnd,black] +DELTAP1 +SRP1 +SRP2 +IP +MDAP[rnd] +DELTAP1 +SRP1 +SRP2 +IP +DELTAP1 +SRP1 +SRP2 +IP +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIAP[rnd] +MDRP[rp0,rnd,grey] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +SLOOP +IP +SRP1 +SRP2 +IP +IP +SRP1 +SRP2 +IP +IP +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP2 +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1249 326 m 0,0,1 + 1208 147 1208 147 1093.5 62 c 128,-1,2 + 979 -23 979 -23 784 -23 c 0,3,4 + 770 -22 770 -22 735 -20 c 0,5,6 + 735 -20 735 -20 700 -18 c 1,7,-1 + 750 199 l 1,8,9 + 861 199 861 199 899 226.5 c 128,-1,10 + 937 254 937 254 950 313 c 0,11,12 + 966 383 966 383 921.5 437.5 c 128,-1,13 + 877 492 877 492 835 528 c 0,14,15 + 733 614 733 614 706 686 c 128,-1,16 + 679 758 679 758 709 887 c 0,17,18 + 729 975 729 975 811 1060 c 128,-1,19 + 893 1145 893 1145 906 1198 c 0,20,21 + 916 1241 916 1241 900 1266.5 c 128,-1,22 + 884 1292 884 1292 834 1292 c 0,23,24 + 784 1292 784 1292 738 1250 c 128,-1,25 + 692 1208 692 1208 674 1130 c 2,26,-1 + 414 0 l 1,27,-1 + 130 0 l 1,28,-1 + 384 1102 l 2,29,30 + 421 1264 421 1264 558 1386.5 c 128,-1,31 + 695 1509 695 1509 870 1509 c 0,32,33 + 1066 1509 1066 1509 1136 1435.5 c 128,-1,34 + 1206 1362 1206 1362 1174 1225 c 0,35,36 + 1151 1126 1151 1126 1066.5 1056.5 c 128,-1,37 + 982 987 982 987 971 938 c 0,38,39 + 958 881 958 881 979.5 839 c 128,-1,40 + 1001 797 1001 797 1099 709 c 0,41,42 + 1189 627 1189 627 1232 533.5 c 128,-1,43 + 1275 440 1275 440 1249 326 c 0,0,1 +EndSplineSet +EndChar + +StartChar: registered +Encoding: 132 174 138 +Width: 1413 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 75 + 4 + 16 + 4 + 20 + 50 + 46 + 94 + 14 + 82 + 16 + 82 + 20 + 94 + 22 + 101 + 46 + 117 + 46 + 148 + 46 + 10 + 53 + 34 + 69 + 69 + 40 + 41 + 68 + 212 + 41 + 25 + 40 + 40 + 41 + 3 + 9 + 21 + 75 + 3 + 74 + 15 + 75 + 15 + 9 + 1 + 9 + 24 + 114 + 25 + 50 + 114 + 53 + 63 + 59 + 1 + 59 + 40 + 69 + 37 + 114 + 48 + 40 + 1 + 40 + 40 + 0 + 6 + 12 + 76 + 0 + 74 + 18 + 76 + 6 + 75 +SRP0 +MDRP[rp0,rnd,white] +MIRP[min,rnd,black] +MIRP[rp0,min,rnd,black] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +MDAP[rnd] +DELTAP1 +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MDRP[rnd,grey] +MIRP[min,rnd,black] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MDAP[rnd] +DELTAP1 +MIRP[min,rnd,black] +MIRP[rp0,min,rnd,black] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +IP +MDAP[rnd] +MDRP[rnd,grey] +MDAP[rnd] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +MDAP[rnd] +MDRP[rp0,min,rnd,black] +IP +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1483 690 m 0,0,1 + 1419 414 1419 414 1193.5 220.5 c 128,-1,2 + 968 27 968 27 713 27 c 0,3,4 + 457 27 457 27 321 220.5 c 128,-1,5 + 185 414 185 414 248 690 c 0,6,7 + 312 967 312 967 538 1162.5 c 128,-1,8 + 764 1358 764 1358 1020 1358 c 0,9,10 + 1275 1358 1275 1358 1411 1162.5 c 128,-1,11 + 1547 967 1547 967 1483 690 c 0,0,1 +1337 690 m 256,12,13 + 1385 899 1385 899 1283 1056.5 c 128,-1,14 + 1181 1214 1181 1214 987 1214 c 256,15,16 + 793 1214 793 1214 619.5 1056.5 c 128,-1,17 + 446 899 446 899 397 690 c 256,18,19 + 349 481 349 481 450 323.5 c 128,-1,20 + 551 166 551 166 745 166 c 256,21,22 + 939 166 939 166 1114 323.5 c 128,-1,23 + 1289 481 1289 481 1337 690 c 256,12,13 +1040 342 m 1,24,-1 + 864 342 l 1,25,26 + 864 342 864 342 870.5 371.5 c 128,-1,27 + 877 401 877 401 893 469 c 0,28,29 + 911 545 911 545 896.5 599 c 128,-1,30 + 882 653 882 653 836 653 c 0,31,32 + 825 653 825 653 807.5 653 c 128,-1,33 + 790 653 790 653 790 653 c 257,34,35 + 790 653 790 653 768.5 559 c 128,-1,36 + 747 465 747 465 719 342 c 1,37,38 + 686 342 686 342 640.5 342 c 128,-1,39 + 595 342 595 342 561 342 c 1,40,-1 + 725 1053 l 1,41,42 + 725 1053 725 1053 826.5 1053 c 128,-1,43 + 928 1053 928 1053 982 1053 c 0,44,45 + 1086 1053 1086 1053 1128 1010 c 128,-1,46 + 1170 967 1170 967 1154 899 c 0,47,48 + 1152 891 1152 891 1146.5 867.5 c 128,-1,49 + 1141 844 1141 844 1140 838 c 0,50,51 + 1127 780 1127 780 1088.5 742.5 c 128,-1,52 + 1050 705 1050 705 1002 694 c 1,53,54 + 1062 655 1062 655 1072 587.5 c 128,-1,55 + 1082 520 1082 520 1067 459 c 0,56,57 + 1063 442 1063 442 1040 342 c 1,24,-1 +992 831 m 0,58,59 + 994 838 994 838 995 844 c 128,-1,60 + 996 850 996 850 998 858 c 0,61,62 + 1006 891 1006 891 993 914.5 c 128,-1,63 + 980 938 980 938 952 938 c 0,64,65 + 944 938 944 938 908 938 c 128,-1,66 + 872 938 872 938 856 938 c 1,67,-1 + 812 748 l 1,68,69 + 812 748 812 748 847.5 748 c 128,-1,70 + 883 748 883 748 908 748 c 0,71,72 + 941 748 941 748 961.5 768 c 128,-1,73 + 982 788 982 788 992 831 c 0,58,59 +EndSplineSet +EndChar + +StartChar: copyright +Encoding: 127 169 139 +Width: 1413 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 84 + 4 + 16 + 4 + 20 + 58 + 28 + 94 + 14 + 82 + 16 + 82 + 20 + 94 + 22 + 187 + 28 + 186 + 29 + 202 + 28 + 217 + 29 + 233 + 29 + 12 + 51 + 48 + 212 + 24 + 27 + 48 + 38 + 1 + 38 + 41 + 212 + 48 + 37 + 1 + 37 + 34 + 27 + 34 + 3 + 9 + 21 + 75 + 3 + 74 + 15 + 75 + 15 + 9 + 1 + 9 + 61 + 38 + 1 + 38 + 15 + 24 + 1 + 24 + 44 + 113 + 9 + 30 + 25 + 30 + 41 + 30 + 3 + 30 + 30 + 0 + 6 + 12 + 76 + 0 + 74 + 18 + 76 + 6 + 52 +SRP0 +MDRP[rp0,rnd,white] +MIRP[min,rnd,black] +MIRP[rp0,min,rnd,black] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +MDAP[rnd] +DELTAP1 +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +DELTAP1 +MDRP[rnd,grey] +DELTAP1 +SVTCA[y-axis] +MDAP[rnd] +DELTAP1 +MIRP[min,rnd,black] +MIRP[rp0,min,rnd,black] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +IP +MDAP[rnd] +MDRP[rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +DELTAP1 +MDAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1483 690 m 0,0,1 + 1419 414 1419 414 1193.5 220.5 c 128,-1,2 + 968 27 968 27 713 27 c 0,3,4 + 457 27 457 27 321 220.5 c 128,-1,5 + 185 414 185 414 248 690 c 0,6,7 + 312 967 312 967 538 1162.5 c 128,-1,8 + 764 1358 764 1358 1020 1358 c 0,9,10 + 1275 1358 1275 1358 1411 1162.5 c 128,-1,11 + 1547 967 1547 967 1483 690 c 0,0,1 +1337 690 m 256,12,13 + 1385 899 1385 899 1283 1056.5 c 128,-1,14 + 1181 1214 1181 1214 987 1214 c 256,15,16 + 793 1214 793 1214 619.5 1056.5 c 128,-1,17 + 446 899 446 899 397 690 c 256,18,19 + 349 481 349 481 450 323.5 c 128,-1,20 + 551 166 551 166 745 166 c 256,21,22 + 939 166 939 166 1114 323.5 c 128,-1,23 + 1289 481 1289 481 1337 690 c 256,12,13 +1025 379 m 1,24,25 + 992 360 992 360 933.5 344 c 128,-1,26 + 875 328 875 328 815 328 c 0,27,28 + 657 328 657 328 589.5 404.5 c 128,-1,29 + 522 481 522 481 563 657 c 2,30,-1 + 571 692 l 2,31,32 + 607 850 607 850 720 938 c 128,-1,33 + 833 1026 833 1026 980 1026 c 0,34,35 + 1054 1026 1054 1026 1097 1009.5 c 128,-1,36 + 1140 993 1140 993 1163 977 c 1,37,-1 + 1134 850 l 1,38,39 + 1112 870 1112 870 1069 886.5 c 128,-1,40 + 1026 903 1026 903 979 903 c 0,41,42 + 897 903 897 903 844 854 c 128,-1,43 + 791 805 791 805 763 682 c 2,44,-1 + 760 668 l 2,45,46 + 731 541 731 541 759 502 c 128,-1,47 + 787 463 787 463 868 463 c 0,48,49 + 926 463 926 463 976.5 479.5 c 128,-1,50 + 1027 496 1027 496 1056 514 c 1,51,-1 + 1025 379 l 1,24,25 +EndSplineSet +EndChar + +StartChar: trademark +Encoding: 369 8482 140 +Width: 1339 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 69 + 169 + 5 + 185 + 5 + 200 + 5 + 217 + 5 + 4 + 63 + 2 + 63 + 5 + 2 + 5 + 10 + 2 + 3 + 9 + 3 + 51 + 4 + 1 + 4 + 4 + 7 + 9 + 1 + 7 + 16 + 19 + 14 + 17 + 11 + 9 + 19 + 10 + 3 + 4 + 4 + 3 + 10 + 3 + 2 + 5 + 11 + 0 + 63 + 2 + 144 + 2 + 2 + 2 + 7 + 9 + 5 + 48 + 7 + 1 + 7 + 16 + 13 + 15 + 18 + 47 + 16 + 1 + 16 + 21 +SRP0 +MDRP[rp0,min,rnd,white] +DELTAP1 +MDRP[min,rnd,grey] +MDRP[rp0,min,rnd,black] +MDRP[rp0,min,rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MDRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MDRP[rp0,min,rnd,black] +MDRP[grey] +SRP1 +SRP2 +SLOOP +IP +MDAP[rnd] +MDRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[rnd,grey] +SRP1 +SRP2 +IP +MDAP[rnd] +DELTAP1 +MDRP[rnd,grey] +SRP2 +SLOOP +IP +DELTAP1 +IUP[x] +IUP[y] +SVTCA[y-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1505 885 m 1,0,-1 + 1343 885 l 1,1,-1 + 1418 1208 l 1,2,-1 + 1194 1016 l 1,3,-1 + 1150 1016 l 1,4,-1 + 1017 1206 l 1,5,-1 + 943 885 l 1,6,-1 + 817 885 l 1,7,-1 + 957 1493 l 1,8,-1 + 1047 1493 l 1,9,-1 + 1251 1202 l 1,10,-1 + 1586 1493 l 1,11,-1 + 1645 1493 l 1,12,-1 + 1505 885 l 1,0,-1 +854 1372 m 1,13,-1 + 669 1372 l 1,14,-1 + 556 885 l 1,15,-1 + 411 885 l 1,16,-1 + 524 1372 l 1,17,-1 + 340 1372 l 1,18,-1 + 367 1493 l 1,19,-1 + 882 1493 l 1,20,-1 + 854 1372 l 1,13,-1 +EndSplineSet +EndChar + +StartChar: acute +Encoding: 138 180 141 +Width: 942 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 16 + 3 + 91 + 1 + 36 + 3 + 1 + 0 + 2 + 0 + 47 + 2 + 63 + 2 + 2 + 2 + 4 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MDRP[rp0,min,rnd,grey] +SRP1 +SRP2 +IP +IP +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1118 1677 m 1,0,-1 + 731 1323 l 1,1,-1 + 529 1323 l 1,2,-1 + 844 1677 l 1,3,-1 + 1118 1677 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: dieresis +Encoding: 126 168 142 +Width: 946 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 25 + 9 + 21 + 96 + 3 + 63 + 15 + 1 + 15 + 39 + 48 + 0 + 1 + 0 + 96 + 6 + 48 + 12 + 1 + 12 + 96 + 47 + 18 + 1 + 18 + 24 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MIRP[min,rnd,black] +DELTAP1 +MDRP[rp0,min,rnd,grey] +MIRP[min,rnd,black] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +DELTAP1 +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1160 1479 m 0,0,1 + 1147 1421 1147 1421 1100 1380 c 128,-1,2 + 1053 1339 1053 1339 999 1339 c 0,3,4 + 948 1339 948 1339 920 1380 c 128,-1,5 + 892 1421 892 1421 906 1479 c 0,6,7 + 919 1534 919 1534 965.5 1575 c 128,-1,8 + 1012 1616 1012 1616 1063 1616 c 0,9,10 + 1117 1616 1117 1616 1145 1575 c 128,-1,11 + 1173 1534 1173 1534 1160 1479 c 0,0,1 +733 1479 m 0,12,13 + 720 1421 720 1421 673 1380 c 128,-1,14 + 626 1339 626 1339 573 1339 c 0,15,16 + 521 1339 521 1339 493.5 1380 c 128,-1,17 + 466 1421 466 1421 480 1479 c 0,18,19 + 493 1534 493 1534 539 1575 c 128,-1,20 + 585 1616 585 1616 637 1616 c 0,21,22 + 690 1616 690 1616 718 1575 c 128,-1,23 + 746 1534 746 1534 733 1479 c 0,12,13 +EndSplineSet +EndChar + +StartChar: notequal +Encoding: 397 8800 143 +Width: 1188 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 89 + 83 + 1 + 89 + 12 + 129 + 2 + 144 + 2 + 150 + 11 + 200 + 4 + 6 + 37 + 18 + 1 + 1 + 18 + 17 + 14 + 4 + 99 + 2 + 115 + 2 + 2 + 13 + 2 + 4 + 7 + 8 + 11 + 4 + 12 + 3 + 13 + 6 + 12 + 22 + 12 + 54 + 12 + 3 + 12 + 2 + 71 + 3 + 1 + 3 + 3 + 12 + 1 + 4 + 69 + 18 + 7 + 11 + 17 + 8 + 69 + 14 + 11 + 4 + 7 + 8 + 11 + 12 + 14 + 17 + 18 + 1 + 2 + 10 + 13 + 3 + 48 + 13 + 1 + 13 + 15 + 0 + 21 + 63 + 3 + 1 + 3 + 9 + 5 + 20 +SRP0 +MDRP[rp0,rnd,white] +MDRP[rnd,grey] +MDRP[rnd,grey] +DELTAP1 +SRP0 +MDRP[rp0,rnd,white] +MDRP[rnd,grey] +MDRP[rnd,grey] +DELTAP1 +SRP1 +SRP2 +SLOOP +IP +SVTCA[y-axis] +MDAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SPVTL[orthog] +SFVTCA[x-axis] +SRP0 +DELTAP1 +MDRP[rp0,min,black] +MDAP[rnd] +DELTAP1 +MDRP[rp0,min,black] +SPVTCA[x-axis] +SRP1 +SRP2 +SLOOP +IP +SRP1 +SRP2 +DELTAP1 +SLOOP +IP +DELTAP1 +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1065 236 m 1,0,-1 + 469 236 l 1,1,-1 + 334 25 l 1,2,-1 + 184 117 l 1,3,-1 + 260 236 l 1,4,-1 + 125 236 l 1,5,-1 + 125 473 l 1,6,-1 + 412 473 l 1,7,-1 + 567 715 l 1,8,-1 + 125 715 l 1,9,-1 + 125 952 l 1,10,-1 + 721 952 l 1,11,-1 + 870 1186 l 1,12,-1 + 1022 1094 l 1,13,-1 + 930 952 l 1,14,-1 + 1065 952 l 1,15,-1 + 1065 715 l 1,16,-1 + 778 715 l 1,17,-1 + 623 473 l 1,18,-1 + 1065 473 l 1,19,-1 + 1065 236 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: AE +Encoding: 156 198 144 +Width: 1660 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 70 + 39 + 18 + 1 + 59 + 17 + 1 + 17 + 42 + 6 + 18 + 3 + 59 + 4 + 1 + 4 + 42 + 54 + 5 + 1 + 5 + 5 + 6 + 6 + 4 + 3 + 18 + 4 + 1 + 5 + 8 + 12 + 0 + 1 + 10 + 14 + 23 + 17 + 1 + 47 + 5 + 1 + 47 + 5 + 239 + 5 + 2 + 5 + 19 + 13 + 30 + 10 + 73 + 3 + 30 + 18 + 18 + 10 + 1 + 6 + 17 + 9 + 30 + 6 + 1 + 14 + 30 + 1 + 4 + 5 + 0 +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MDRP[grey] +SRP1 +SRP2 +IP +IP +MDAP[rnd] +MIRP[rp0,min,rnd,black] +SROUND +MDAP[rnd] +RTG +MIRP[rp0,min,rnd,black] +SVTCA[x-axis] +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +DELTAP2 +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[rnd,grey] +SRP1 +SRP2 +SLOOP +IP +SPVTL[orthog] +SRP0 +DELTAP1 +MIRP[rp0,min,rnd,black] +DELTAP1 +MDRP[rnd,grey] +MDRP[rnd,grey] +SFVTCA[y-axis] +SRP0 +MIRP[rp0,min,rnd,black] +DELTAP1 +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1593 0 m 1,0,-1 + 725 0 l 1,1,-1 + 808 362 l 1,2,-1 + 458 362 l 1,3,-1 + 224 0 l 1,4,-1 + -50 0 l 1,5,-1 + 942 1509 l 1,6,-1 + 1905 1509 l 1,7,-1 + 1853 1284 l 1,8,-1 + 1330 1284 l 1,9,-1 + 1236 879 l 1,10,-1 + 1640 879 l 1,11,-1 + 1587 651 l 1,12,-1 + 1184 651 l 1,13,-1 + 1085 225 l 1,14,-1 + 1645 225 l 1,15,-1 + 1593 0 l 1,0,-1 +859 580 m 1,16,-1 + 1007 1223 l 1,17,-1 + 600 580 l 1,18,-1 + 859 580 l 1,16,-1 +EndSplineSet +EndChar + +StartChar: Oslash +Encoding: 174 216 145 +Width: 1365 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 14 + 0 + 0 + 22 + 0 + 24 + 3 + 32 + 0 + 40 + 2 + 5 + 63 + 33 + 1 +SVTCA[x-axis] +DELTAP1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 381 -1 N 1 0 0 1 -20 -36 3 +Refer: 50 79 N 1 0 0 1 0 0 2 +EndChar + +StartChar: infinity +Encoding: 394 8734 146 +Width: 1544 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 86 + 4 + 1 + 10 + 11 + 10 + 13 + 4 + 23 + 20 + 1 + 27 + 11 + 27 + 13 + 20 + 23 + 36 + 1 + 42 + 11 + 42 + 13 + 36 + 23 + 58 + 31 + 54 + 37 + 71 + 7 + 71 + 17 + 81 + 7 + 93 + 19 + 101 + 29 + 106 + 47 + 118 + 29 + 120 + 47 + 22 + 30 + 18 + 33 + 15 + 6 + 36 + 3 + 39 + 33 + 45 + 32 + 3 + 0 + 9 + 1 + 9 + 27 + 39 + 32 + 21 + 15 + 36 + 30 + 6 + 18 + 18 + 0 + 12 + 24 + 105 + 31 + 0 + 47 + 0 + 2 + 0 + 49 + 42 + 105 + 12 + 48 +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[min,rnd,black] +SRP0 +MDRP[rp0,min,rnd,white] +DELTAP1 +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +RTDG +MDAP[rnd] +RTG +MDRP[rnd,grey] +MDRP[grey] +MDRP[grey] +SVTCA[y-axis] +MDAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDAP[rnd] +DELTAP1 +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP1 +SRP2 +IP +IP +SRP1 +SRP2 +IP +IP +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1460 682 m 0,0,1 + 1460 512 1460 512 1360 425 c 128,-1,2 + 1260 338 1260 338 1100 338 c 0,3,4 + 987 338 987 338 906 385 c 128,-1,5 + 825 432 825 432 766 512 c 1,6,7 + 707 426 707 426 634 384 c 128,-1,8 + 561 342 561 342 440 342 c 0,9,10 + 281 342 281 342 178.5 428 c 128,-1,11 + 76 514 76 514 76 682 c 256,12,13 + 76 850 76 850 180.5 936 c 128,-1,14 + 285 1022 285 1022 436 1022 c 0,15,16 + 553 1022 553 1022 634 973 c 128,-1,17 + 715 924 715 924 770 842 c 1,18,19 + 827 922 827 922 906 970 c 128,-1,20 + 985 1018 985 1018 1094 1018 c 0,21,22 + 1262 1018 1262 1018 1361 928 c 128,-1,23 + 1460 838 1460 838 1460 682 c 0,0,1 +1274 682 m 0,24,25 + 1274 758 1274 758 1220.5 805 c 128,-1,26 + 1167 852 1167 852 1096 852 c 0,27,28 + 1038 852 1038 852 991 820 c 128,-1,29 + 944 788 944 788 877 684 c 1,30,31 + 934 586 934 586 983 549 c 128,-1,32 + 1032 512 1032 512 1100 512 c 0,33,34 + 1169 512 1169 512 1221.5 560 c 128,-1,35 + 1274 608 1274 608 1274 682 c 0,24,25 +657 676 m 1,36,37 + 606 770 606 770 556 809 c 128,-1,38 + 506 848 506 848 436 848 c 0,39,40 + 369 848 369 848 315.5 799 c 128,-1,41 + 262 750 262 750 262 680 c 0,42,43 + 262 602 262 602 314.5 555 c 128,-1,44 + 367 508 367 508 436 508 c 0,45,46 + 506 508 506 508 557 546 c 128,-1,47 + 608 584 608 584 657 676 c 1,36,37 +EndSplineSet +EndChar + +StartChar: plusminus +Encoding: 135 177 147 +Width: 1097 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 38 + 14 + 69 + 48 + 13 + 1 + 13 + 0 + 3 + 106 + 1 + 4 + 68 + 10 + 7 + 106 + 63 + 8 + 1 + 8 + 12 + 12 + 0 + 106 + 10 + 1 + 69 + 4 + 13 + 5 + 106 + 7 + 47 + 4 + 1 + 4 + 4 + 17 + 16 +SRP1 +SRP2 +IP +MDAP[rnd] +DELTAP1 +MDRP[rnd,grey] +MIRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +SRP0 +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +SVTCA[y-axis] +MIAP[rnd] +DELTAP1 +MIRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MIRP[min,rnd,grey] +MIAP[rnd] +DELTAP1 +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1071 647 m 1,0,-1 + 803 647 l 1,1,-1 + 740 375 l 1,2,-1 + 530 375 l 1,3,-1 + 593 647 l 1,4,-1 + 327 647 l 1,5,-1 + 381 883 l 1,6,-1 + 647 883 l 1,7,-1 + 710 1155 l 1,8,-1 + 920 1155 l 1,9,-1 + 857 883 l 1,10,-1 + 1125 883 l 1,11,-1 + 1071 647 l 1,0,-1 +932 0 m 1,12,-1 + 177 0 l 1,13,-1 + 228 217 l 1,14,-1 + 982 217 l 1,15,-1 + 932 0 l 1,12,-1 +EndSplineSet +EndChar + +StartChar: lessequal +Encoding: 398 8804 148 +Width: 1024 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 65 + 6 + 5 + 50 + 0 + 50 + 3 + 54 + 5 + 118 + 5 + 5 + 60 + 4 + 51 + 6 + 2 + 9 + 69 + 48 + 8 + 1 + 8 + 0 + 4 + 3 + 2 + 6 + 0 + 1 + 14 + 0 + 30 + 0 + 2 + 0 + 1 + 1 + 3 + 17 + 3 + 2 + 3 + 5 + 48 + 1 + 1 + 1 + 92 + 63 + 2 + 1 + 2 + 5 + 0 + 1 + 63 + 7 + 1 + 7 + 4 + 0 + 12 + 9 + 1 + 11 +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rnd,grey] +MDRP[rnd,grey] +DELTAP1 +SRP1 +SRP2 +IP +SVTCA[y-axis] +MDAP[rnd] +DELTAP1 +MIRP[min,rnd,black] +DELTAP1 +IP +MDRP[rp0,min,rnd,grey] +DELTAP1 +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +SRP1 +SRP2 +IP +SRP1 +SRP2 +IP +MIAP[rnd] +DELTAP1 +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +DELTAP1 +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +925 311 m 1,0,-1 + 75 657 l 1,1,-1 + 75 866 l 1,2,-1 + 925 1214 l 1,3,-1 + 925 950 l 1,4,-1 + 450 766 l 1,5,-1 + 925 573 l 1,6,-1 + 925 311 l 1,0,-1 +925 0 m 1,7,-1 + 75 0 l 1,8,-1 + 75 221 l 1,9,-1 + 925 221 l 1,10,-1 + 925 0 l 1,7,-1 +EndSplineSet +EndChar + +StartChar: greaterequal +Encoding: 399 8805 149 +Width: 1024 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_8 + 121 + 5 + 1 + 63 + 1 + 63 + 8 + 2 +SVTCA[x-axis] +DELTAP1 +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +Refer: 148 8804 N -1 0 0 1 1024 0 2 +EndChar + +StartChar: yen +Encoding: 123 165 150 +Width: 1226 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 108 + 54 + 24 + 71 + 0 + 2 + 100 + 23 + 1 + 23 + 31 + 22 + 23 + 1 + 24 + 31 + 0 + 0 + 1 + 55 + 23 + 88 + 23 + 2 + 23 + 22 + 20 + 22 + 22 + 21 + 21 + 20 + 24 + 22 + 0 + 21 + 23 + 1 + 20 + 16 + 0 + 80 + 0 + 2 + 0 + 9 + 5 + 8 + 4 + 1 + 20 + 21 + 12 + 16 + 13 + 17 + 80 + 20 + 1 + 15 + 20 + 31 + 20 + 144 + 20 + 3 + 20 + 20 + 26 + 25 + 9 + 12 + 89 + 6 + 15 + 15 + 31 + 15 + 32 + 15 + 3 + 15 + 19 + 5 + 16 + 89 + 2 + 19 + 1 + 20 + 0 + 24 + 22 + 53 + 1 + 54 + 20 + 51 + 23 + 3 + 20 + 1 + 23 + 3 + 11 + 21 + 10 + 11 + 0 +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +SRP2 +SLOOP +IP +DELTAP1 +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDAP[rnd] +MDRP[rnd,grey] +MDRP[rp0,rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SVTCA[x-axis] +SRP1 +SRP2 +IP +MDAP[rnd] +DELTAP1 +DELTAP3 +MDRP[min,rnd,grey] +MDRP[min,rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[min,rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[min,rnd,grey] +MDRP[min,rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[rp0,min,rnd,grey] +DELTAP1 +SRP1 +SRP2 +IP +SRP1 +SRP2 +IP +IP +SPVTL[orthog] +SRP0 +MIRP[rp0,min,rnd,black] +SRP0 +MIRP[rp0,min,rnd,black] +DELTAP1 +SPVTL[orthog] +SRP0 +MIRP[rp0,min,rnd,black] +SRP0 +SFVTL[parallel] +MIRP[rp0,min,rnd,black] +DELTAP1 +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1583 1509 m 1,0,-1 + 916 725 l 1,1,-1 + 897 643 l 1,2,-1 + 1196 643 l 1,3,-1 + 1156 471 l 1,4,-1 + 857 471 l 1,5,-1 + 835 373 l 1,6,-1 + 1133 373 l 1,7,-1 + 1094 201 l 1,8,-1 + 795 201 l 1,9,-1 + 749 0 l 1,10,-1 + 471 0 l 1,11,-1 + 517 201 l 1,12,-1 + 218 201 l 1,13,-1 + 257 373 l 1,14,-1 + 557 373 l 1,15,-1 + 579 471 l 1,16,-1 + 280 471 l 1,17,-1 + 320 643 l 1,18,-1 + 619 643 l 1,19,-1 + 630 690 l 1,20,-1 + 345 1509 l 1,21,-1 + 695 1509 l 1,22,-1 + 883 969 l 1,23,-1 + 1331 1509 l 1,24,-1 + 1583 1509 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: mu +Encoding: 139 181 151 +Width: 1131 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 45 + 6 + 3 + 22 + 3 + 38 + 3 + 3 + 19 + 2 + 96 + 5 + 10 + 18 + 16 + 34 + 1 + 57 + 5 + 72 + 5 + 2 + 5 + 2 + 20 + 11 + 12 + 0 + 22 + 2 + 1 + 64 + 19 + 160 + 19 + 2 + 175 + 19 + 1 + 19 + 10 + 13 + 8 + 22 + 10 + 22 +SRP0 +MDRP[rp0,rnd,white] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +DELTAP2 +MDRP[rnd,grey] +MDRP[grey] +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIAP[rnd] +DELTAP1 +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +SRP0 +SMD +MDRP[rp0,min,grey] +MDRP[min,rnd,grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1082 327 m 1,0,1 + 1078 309 1078 309 1072 264.5 c 128,-1,2 + 1066 220 1066 220 1068 212 c 0,3,4 + 1068 197 1068 197 1078 186.5 c 128,-1,5 + 1088 176 1088 176 1101 176 c 0,6,7 + 1151 176 1151 176 1253 348 c 1,8,-1 + 1267 372 l 1,9,-1 + 1320 337 l 1,10,11 + 1198 126 1198 126 1119.5 52.5 c 128,-1,12 + 1041 -21 1041 -21 935 -21 c 0,13,14 + 898 -21 898 -21 870.5 -11.5 c 128,-1,15 + 843 -2 843 -2 824.5 10.5 c 128,-1,16 + 806 23 806 23 794.5 46.5 c 128,-1,17 + 783 70 783 70 777 88.5 c 128,-1,18 + 771 107 771 107 771 138.5 c 128,-1,19 + 771 170 771 170 771.5 186 c 128,-1,20 + 772 202 772 202 778 235 c 128,-1,21 + 784 268 784 268 785.5 276 c 128,-1,22 + 787 284 787 284 793 309 c 1,23,24 + 744 228 744 228 642 42 c 0,25,26 + 574 0 574 0 488 0 c 0,27,28 + 460 0 460 0 437.5 0 c 128,-1,29 + 415 0 415 0 392 8 c 1,30,-1 + 311 -342 l 1,31,-1 + 51 -342 l 1,32,-1 + 382 1093 l 1,33,-1 + 666 1093 l 1,34,-1 + 501 381 l 2,35,36 + 477 278 477 278 506.5 251.5 c 128,-1,37 + 536 225 536 225 581 225 c 0,38,39 + 641 225 641 225 702.5 251.5 c 128,-1,40 + 764 278 764 278 798 329 c 1,41,-1 + 974 1093 l 1,42,-1 + 1259 1093 l 1,43,-1 + 1082 327 l 1,0,1 +EndSplineSet +EndChar + +StartChar: partialdiff +Encoding: 386 8706 152 +Width: 1190 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 59 + 38 + 23 + 71 + 5 + 86 + 5 + 104 + 8 + 121 + 8 + 5 + 72 + 31 + 88 + 31 + 2 + 31 + 34 + 12 + 9 + 9 + 3 + 22 + 18 + 40 + 15 + 34 + 19 + 40 + 22 + 15 + 70 + 37 + 86 + 37 + 2 + 37 + 46 + 3 + 5 + 63 + 41 + 1 + 0 + 25 + 12 + 16 + 28 + 32 + 28 + 112 + 28 + 3 + 28 + 18 + 34 + 25 + 6 + 40 +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[min,rnd,black] +MDRP[rnd,grey] +MDRP[rp0,min,rnd,grey] +DELTAP1 +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +DELTAP1 +MIAP[rnd] +MIRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MIRP[rnd,grey] +SRP1 +SRP2 +IP +MDAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +DELTAP1 +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1114 573 m 0,0,1 + 1096 287 1096 287 955.5 131 c 128,-1,2 + 815 -25 815 -25 543 -25 c 0,3,4 + 315 -25 315 -25 189 137 c 128,-1,5 + 63 299 63 299 63 528 c 0,6,7 + 63 778 63 778 186 924.5 c 128,-1,8 + 309 1071 309 1071 543 1071 c 0,9,10 + 621 1071 621 1071 686.5 1033 c 128,-1,11 + 752 995 752 995 788 954 c 1,12,13 + 770 1137 770 1137 686 1231 c 128,-1,14 + 602 1325 602 1325 461 1325 c 0,15,16 + 401 1325 401 1325 348 1312.5 c 128,-1,17 + 295 1300 295 1300 231 1268 c 1,18,-1 + 231 1499 l 1,19,20 + 266 1513 266 1513 323.5 1529.5 c 128,-1,21 + 381 1546 381 1546 461 1546 c 0,22,23 + 764 1546 764 1546 930 1306.5 c 128,-1,24 + 1096 1067 1096 1067 1114 705 c 0,25,26 + 1114 692 1114 692 1115 647 c 128,-1,27 + 1116 602 1116 602 1114 573 c 0,0,1 +780 532 m 0,28,29 + 780 657 780 657 734 756.5 c 128,-1,30 + 688 856 688 856 584 856 c 0,31,32 + 496 856 496 856 446.5 788.5 c 128,-1,33 + 397 721 397 721 397 528 c 0,34,35 + 397 369 397 369 440 266.5 c 128,-1,36 + 483 164 483 164 584 164 c 0,37,38 + 682 164 682 164 731 266.5 c 128,-1,39 + 780 369 780 369 780 532 c 0,28,29 +EndSplineSet +EndChar + +StartChar: summation +Encoding: 389 8721 153 +Width: 1276 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 76 + 40 + 3 + 73 + 3 + 87 + 9 + 184 + 3 + 4 + 10 + 2 + 55 + 10 + 2 + 9 + 10 + 9 + 134 + 8 + 1 + 8 + 4 + 4 + 3 + 23 + 9 + 1 + 9 + 8 + 3 + 1 + 3 + 71 + 10 + 84 + 10 + 2 + 10 + 2 + 2 + 3 + 9 + 3 + 10 + 8 + 4 + 8 + 34 + 5 + 14 + 2 + 10 + 68 + 1 + 0 + 9 + 3 + 8 + 10 + 4 + 0 + 2 + 7 + 48 + 0 + 1 + 0 + 54 + 4 + 1 + 4 + 47 + 2 + 1 + 2 + 12 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MDRP[rnd,grey] +DELTAP1 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MDRP[rnd,grey] +SRP1 +SRP2 +SLOOP +IP +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +IP +MIAP[rnd] +MIRP[rp0,min,rnd,black] +IP +SRP1 +SRP2 +IP +IP +SPVTL[orthog] +SFVTCA[x-axis] +SRP0 +MDRP[rp0,min,rnd,black] +DELTAP1 +SRP0 +DELTAP1 +MDRP[grey] +DELTAP1 +SPVTL[orthog] +SRP0 +MDRP[rp0,min,rnd,black] +DELTAP1 +SFVTL[parallel] +MDRP[grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP2 +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1214 0 m 1,0,-1 + 86 0 l 1,1,-1 + 86 199 l 1,2,-1 + 651 770 l 1,3,-1 + 106 1315 l 1,4,-1 + 106 1509 l 1,5,-1 + 1133 1509 l 1,6,-1 + 1133 1284 l 1,7,-1 + 545 1284 l 1,8,-1 + 1042 786 l 1,9,-1 + 516 264 l 1,10,-1 + 1214 264 l 1,11,-1 + 1214 0 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: product +Encoding: 388 8719 154 +Width: 1360 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 31 + 1 + 4 + 8 + 34 + 10 + 14 + 3 + 7 + 4 + 0 + 2 + 25 + 32 + 3 + 1 + 3 + 6 + 25 + 176 + 9 + 1 + 9 + 31 + 7 + 79 + 7 + 144 + 7 + 3 + 7 + 12 +SRP0 +MDRP[rp0,min,rnd,white] +DELTAP1 +MDRP[min,rnd,grey] +DELTAP1 +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +MDRP[rp0,min,rnd,grey] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[rnd,grey] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1335 1284 m 1,0,-1 + 1182 1284 l 1,1,-1 + 1182 0 l 1,2,-1 + 844 0 l 1,3,-1 + 844 1284 l 1,4,-1 + 516 1284 l 1,5,-1 + 516 0 l 1,6,-1 + 178 0 l 1,7,-1 + 178 1284 l 1,8,-1 + 27 1284 l 1,9,-1 + 27 1509 l 1,10,-1 + 1335 1509 l 1,11,-1 + 1335 1284 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: pi +Encoding: 346 960 155 +Width: 1191 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 23 + 1 + 4 + 8 + 33 + 10 + 12 + 3 + 7 + 0 + 0 + 2 + 22 + 3 + 6 + 22 + 9 + 31 + 7 + 79 + 7 + 2 + 7 + 12 +SRP0 +MDRP[rp0,min,rnd,white] +DELTAP1 +MDRP[min,rnd,grey] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rp0,min,rnd,grey] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[rnd,grey] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1354 894 m 1,0,-1 + 1224 894 l 1,1,-1 + 1017 0 l 1,2,-1 + 733 0 l 1,3,-1 + 940 894 l 1,4,-1 + 659 894 l 1,5,-1 + 453 0 l 1,6,-1 + 169 0 l 1,7,-1 + 375 894 l 1,8,-1 + 245 894 l 1,9,-1 + 291 1093 l 1,10,-1 + 1400 1093 l 1,11,-1 + 1354 894 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: integral +Encoding: 395 8747 156 +Width: 967 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 32 + 5 + 9 + 9 + 23 + 22 + 9 + 25 + 23 + 4 + 1 + 4 + 52 + 0 + 25 + 15 + 15 + 18 + 52 + 14 + 11 + 19 + 1 + 0 + 8 + 25 + 21 + 15 + 14 + 21 + 21 + 29 + 28 +SRP1 +SRP2 +IP +MDAP[rnd] +MDRP[rp0,min,grey] +MDRP[grey] +SRP0 +MIRP[rp0,min,rnd,black] +MDRP[rp0,min,grey] +MDRP[grey] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +995 1460 m 1,0,-1 + 895 1251 l 1,1,2 + 870 1272 870 1272 837.5 1290.5 c 128,-1,3 + 805 1309 805 1309 758 1309 c 0,4,5 + 700 1309 700 1309 674.5 1269 c 128,-1,6 + 649 1229 649 1229 649 1151 c 2,7,-1 + 649 57 l 2,8,9 + 649 -127 649 -127 534.5 -240.5 c 128,-1,10 + 420 -354 420 -354 254 -354 c 0,11,12 + 137 -354 137 -354 74.5 -335.5 c 128,-1,13 + 12 -317 12 -317 -37 -285 c 1,14,-1 + 57 -86 l 1,15,16 + 82 -106 82 -106 118 -123.5 c 128,-1,17 + 154 -141 154 -141 201 -141 c 0,18,19 + 258 -141 258 -141 283.5 -97 c 128,-1,20 + 309 -53 309 -53 309 25 c 2,21,-1 + 309 1118 l 2,22,23 + 309 1303 309 1303 424 1416.5 c 128,-1,24 + 539 1530 539 1530 705 1530 c 0,25,26 + 821 1530 821 1530 883.5 1511.5 c 128,-1,27 + 946 1493 946 1493 995 1460 c 1,0,-1 +EndSplineSet +EndChar + +StartChar: ordfeminine +Encoding: 128 170 157 +Width: 855 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 82 + 11 + 7 + 11 + 10 + 26 + 7 + 26 + 10 + 42 + 7 + 42 + 10 + 36 + 28 + 59 + 7 + 59 + 10 + 52 + 28 + 74 + 7 + 70 + 27 + 173 + 10 + 201 + 10 + 14 + 31 + 110 + 32 + 79 + 38 + 111 + 15 + 12 + 12 + 5 + 26 + 1 + 77 + 45 + 110 + 2 + 5 + 78 + 15 + 22 + 1 + 22 + 69 + 19 + 111 + 15 + 23 + 1 + 23 + 70 + 26 + 11 + 22 + 22 + 0 + 8 + 30 + 0 + 107 + 2 + 15 + 64 + 34 + 128 + 34 + 2 + 34 + 42 + 108 + 31 + 67 + 8 + 48 +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[rnd,grey] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +DELTAP1 +MDRP[rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP1 +SRP2 +IP +MDAP[rnd] +SVTCA[y-axis] +MIAP[rnd] +SROUND +MDRP[rnd,grey] +RTG +DELTAP2 +MIRP[rp0,min,rnd,black] +SROUND +MDRP[rnd,grey] +RTG +DELTAP2 +MIAP[rnd] +MDRP[min,grey] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +SRP1 +SRP2 +IP +MDAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +903 566 m 1,0,-1 + 670 566 l 1,1,-1 + 691 666 l 1,2,3 + 653 613 653 613 594 583 c 128,-1,4 + 535 553 535 553 455 553 c 0,5,6 + 316 553 316 553 262.5 627 c 128,-1,7 + 209 701 209 701 243 846 c 2,8,-1 + 246 861 l 2,9,10 + 280 1009 280 1009 393 1093 c 128,-1,11 + 506 1177 506 1177 664 1177 c 0,12,13 + 712 1177 712 1177 743 1160.5 c 128,-1,14 + 774 1144 774 1144 791 1122 c 1,15,-1 + 815 1229 l 2,16,17 + 830 1294 830 1294 808.5 1325 c 128,-1,18 + 787 1356 787 1356 717 1356 c 0,19,20 + 642 1356 642 1356 576.5 1324.5 c 128,-1,21 + 511 1293 511 1293 459 1250 c 1,22,-1 + 509 1468 l 1,23,24 + 570 1493 570 1493 643.5 1511.5 c 128,-1,25 + 717 1530 717 1530 815 1530 c 0,26,27 + 975 1530 975 1530 1033 1454 c 128,-1,28 + 1091 1378 1091 1378 1049 1197 c 2,29,-1 + 903 566 l 1,0,-1 +834 272 m 1,30,-1 + 127 272 l 1,31,-1 + 171 463 l 1,32,-1 + 878 463 l 1,33,-1 + 834 272 l 1,30,-1 +736 861 m 2,34,-1 + 740 877 l 2,35,36 + 756 944 756 944 733 978.5 c 128,-1,37 + 710 1013 710 1013 649 1013 c 0,38,39 + 595 1013 595 1013 555 981 c 128,-1,40 + 515 949 515 949 498 877 c 2,41,-1 + 494 860 l 2,42,43 + 476 784 476 784 504 760 c 128,-1,44 + 532 736 532 736 586 736 c 0,45,46 + 633 736 633 736 677 765.5 c 128,-1,47 + 721 795 721 795 736 861 c 2,34,-1 +EndSplineSet +EndChar + +StartChar: ordmasculine +Encoding: 144 186 158 +Width: 970 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 30 + 15 + 110 + 16 + 79 + 22 + 110 + 10 + 15 + 29 + 110 + 3 + 78 + 17 + 67 + 0 + 109 + 48 + 18 + 96 + 18 + 144 + 18 + 3 + 18 + 25 + 109 + 15 + 67 + 6 + 32 +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[rnd,grey] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +MIRP[rnd,grey] +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1132 1014 m 2,0,1 + 1082 799 1082 799 945 678 c 128,-1,2 + 808 557 808 557 619 557 c 256,3,4 + 430 557 430 557 347 681 c 128,-1,5 + 264 805 264 805 312 1014 c 2,6,-1 + 326 1075 l 2,7,8 + 373 1280 373 1280 513.5 1406 c 128,-1,9 + 654 1532 654 1532 843 1532 c 0,10,11 + 1034 1532 1034 1532 1113.5 1405 c 128,-1,12 + 1193 1278 1193 1278 1146 1075 c 2,13,-1 + 1132 1014 l 2,0,1 +925 272 m 1,14,-1 + 159 272 l 1,15,-1 + 203 463 l 1,16,-1 + 969 463 l 1,17,-1 + 925 272 l 1,14,-1 +883 1038 m 2,18,-1 + 886 1051 l 2,19,20 + 925 1221 925 1221 894.5 1282 c 128,-1,21 + 864 1343 864 1343 797 1343 c 0,22,23 + 730 1343 730 1343 673 1276.5 c 128,-1,24 + 616 1210 616 1210 579 1051 c 2,25,-1 + 576 1038 l 2,26,27 + 535 862 535 862 563 803.5 c 128,-1,28 + 591 745 591 745 661 745 c 0,29,30 + 724 745 724 745 782.5 801.5 c 128,-1,31 + 841 858 841 858 883 1038 c 2,18,-1 +EndSplineSet +EndChar + +StartChar: Omega +Encoding: 345 937 159 +AltUni2: 002126.ffffffff.0 002126.ffffffff.0 +Width: 1421 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 50 + 15 + 46 + 38 + 34 + 1 + 34 + 15 + 8 + 22 + 3 + 8 + 27 + 24 + 27 + 2 + 27 + 4 + 25 + 30 + 7 + 24 + 0 + 3 + 30 + 8 + 27 + 30 + 22 + 8 + 22 + 11 + 19 + 5 + 0 + 28 + 11 + 19 + 28 + 25 + 15 + 30 + 31 + 30 + 63 + 30 + 79 + 30 + 4 + 30 + 38 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MDRP[rnd,grey] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP1 +SRP2 +IP +IP +MDAP[rnd] +MIRP[rp0,min,rnd,black] +MDAP[rnd] +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[rp0,min,rnd,white] +DELTAP1 +MDRP[rnd,grey] +MDRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MIAP[rnd] +DELTAP1 +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1544 831 m 2,0,1 + 1504 657 1504 657 1386 535.5 c 128,-1,2 + 1268 414 1268 414 1118 354 c 1,3,-1 + 1088 225 l 1,4,-1 + 1393 225 l 1,5,-1 + 1341 0 l 1,6,-1 + 822 0 l 1,7,-1 + 926 453 l 1,8,9 + 1050 494 1050 494 1122 613.5 c 128,-1,10 + 1194 733 1194 733 1226 870 c 2,11,-1 + 1236 911 l 2,12,13 + 1289 1141 1289 1141 1232 1241 c 128,-1,14 + 1175 1341 1175 1341 1020 1341 c 0,15,16 + 869 1341 869 1341 765 1243 c 128,-1,17 + 661 1145 661 1145 607 911 c 2,18,-1 + 597 870 l 2,19,20 + 560 711 560 711 585 598 c 128,-1,21 + 610 485 610 485 705 453 c 1,22,-1 + 600 0 l 1,23,-1 + 80 0 l 1,24,-1 + 132 225 l 1,25,-1 + 437 225 l 1,26,-1 + 467 354 l 1,27,28 + 365 391 365 391 290.5 513 c 128,-1,29 + 216 635 216 635 265 850 c 2,30,-1 + 288 946 l 2,31,32 + 343 1186 343 1186 541.5 1358 c 128,-1,33 + 740 1530 740 1530 1072 1530 c 0,34,35 + 1377 1530 1377 1530 1504 1368 c 128,-1,36 + 1631 1206 1631 1206 1571 946 c 2,37,-1 + 1544 831 l 2,0,1 +EndSplineSet +EndChar + +StartChar: ae +Encoding: 188 230 160 +Width: 1626 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 122 + 11 + 19 + 11 + 22 + 26 + 19 + 26 + 22 + 48 + 15 + 50 + 70 + 82 + 45 + 100 + 45 + 153 + 46 + 185 + 19 + 10 + 1 + 32 + 56 + 73 + 61 + 32 + 27 + 24 + 56 + 24 + 17 + 38 + 38 + 41 + 54 + 41 + 70 + 41 + 3 + 41 + 53 + 33 + 44 + 38 + 34 + 51 + 255 + 31 + 1 + 31 + 33 + 35 + 51 + 38 + 13 + 64 + 7 + 80 + 7 + 96 + 7 + 3 + 7 + 58 + 14 + 1 + 14 + 4 + 68 + 33 + 96 + 8 + 1 + 8 + 71 + 11 + 17 + 3 + 63 + 72 + 1 + 34 + 34 + 1 + 20 + 8 + 0 + 49 + 64 + 50 + 1 + 16 + 50 + 64 + 50 + 2 + 50 + 57 + 41 + 14 + 1 + 57 + 56 + 1 + 49 + 27 + 80 + 57 + 1 + 57 + 65 + 22 + 32 + 20 + 144 + 20 + 176 + 20 + 3 + 20 + 71 +SRP0 +MDRP[rp0,min,rnd,white] +DELTAP1 +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +DELTAP1 +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP1 +SRP2 +IP +IP +SRP0 +MDRP[rp0,min,rnd,black] +DELTAP1 +DELTAP2 +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP1 +SRP2 +IP +MDAP[rnd] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +SROUND +MDRP[rnd,grey] +RTG +DELTAP1 +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[grey] +DELTAP1 +MDRP[rnd,grey] +DELTAP1 +MIAP[rnd] +MIRP[rnd,grey] +MIRP[rp0,min,rnd,black] +DELTAP1 +MIRP[rnd,grey] +SRP0 +MDRP[rp0,rnd,grey] +MIRP[min,rnd,black] +MDRP[grey] +DELTAP1 +SRP1 +SRP2 +IP +IP +MDAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +SROUND +MDAP[rnd] +RTG +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1651 477 m 1,0,-1 + 1015 477 l 1,1,2 + 975 306 975 306 1030.5 239 c 128,-1,3 + 1086 172 1086 172 1200 172 c 0,4,5 + 1275 172 1275 172 1358 193.5 c 128,-1,6 + 1441 215 1441 215 1533 257 c 1,7,-1 + 1491 74 l 1,8,9 + 1407 25 1407 25 1314.5 0 c 128,-1,10 + 1222 -25 1222 -25 1107 -25 c 0,11,12 + 978 -25 978 -25 896 21.5 c 128,-1,13 + 814 68 814 68 764 174 c 1,14,15 + 697 78 697 78 611 32 c 128,-1,16 + 525 -14 525 -14 404 -14 c 0,17,18 + 245 -14 245 -14 169.5 77.5 c 128,-1,19 + 94 169 94 169 137 355 c 2,20,-1 + 141 375 l 2,21,22 + 184 559 184 559 317.5 658.5 c 128,-1,23 + 451 758 451 758 636 758 c 0,24,25 + 690 758 690 758 738 738.5 c 128,-1,26 + 786 719 786 719 803 690 c 1,27,-1 + 827 794 l 2,28,29 + 839 848 839 848 804 885.5 c 128,-1,30 + 769 923 769 923 687 923 c 0,31,32 + 600 923 600 923 527 902.5 c 128,-1,33 + 454 882 454 882 395 851 c 1,34,-1 + 438 1038 l 1,35,36 + 512 1070 512 1070 597.5 1093 c 128,-1,37 + 683 1116 683 1116 799 1116 c 0,38,39 + 896 1116 896 1116 952.5 1091 c 128,-1,40 + 1009 1066 1009 1066 1046 1019 c 1,41,42 + 1096 1067 1096 1067 1178.5 1123.5 c 128,-1,43 + 1261 1180 1261 1180 1373 1180 c 0,44,45 + 1585 1180 1585 1180 1661 1016.5 c 128,-1,46 + 1737 853 1737 853 1677 590 c 2,47,48 + 1677 590 1677 590 1665.5 540 c 128,-1,49 + 1654 490 1654 490 1651 477 c 1,0,-1 +1440 641 m 1,50,51 + 1468 800 1468 800 1435.5 868.5 c 128,-1,52 + 1403 937 1403 937 1319 937 c 0,53,54 + 1245 937 1245 937 1170.5 869.5 c 128,-1,55 + 1096 802 1096 802 1054 641 c 1,56,-1 + 1440 641 l 1,50,51 +731 369 m 2,57,-1 + 737 397 l 2,58,59 + 759 492 759 492 730.5 542 c 128,-1,60 + 702 592 702 592 625 592 c 0,61,62 + 551 592 551 592 501.5 545 c 128,-1,63 + 452 498 452 498 429 395 c 2,64,-1 + 422 367 l 2,65,66 + 397 260 397 260 430 224 c 128,-1,67 + 463 188 463 188 533 188 c 0,68,69 + 593 188 593 188 651 231.5 c 128,-1,70 + 709 275 709 275 731 369 c 2,57,-1 +EndSplineSet +EndChar + +StartChar: oslash +Encoding: 206 248 161 +Width: 1360 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 382 -1 N 1 0 0 1 0 0 2 +Refer: 82 111 N 1 0 0 1 0 0 2 +EndChar + +StartChar: questiondown +Encoding: 149 191 162 +Width: 853 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 51 + 10 + 9 + 10 + 12 + 26 + 9 + 165 + 25 + 165 + 26 + 230 + 25 + 230 + 26 + 7 + 35 + 69 + 32 + 34 + 4 + 69 + 7 + 15 + 19 + 117 + 1 + 72 + 2 + 10 + 63 + 37 + 1 + 10 + 22 + 127 + 29 + 239 + 29 + 2 + 29 + 3 + 19 + 2 + 65 + 20 + 3 + 63 + 35 + 1 + 35 + 37 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIRP[rp0,min,rnd,white] +MIAP[rnd] +SROUND +MDRP[rnd,grey] +RTG +MIRP[rp0,min,rnd,black] +SROUND +MDRP[rnd,grey] +RTG +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +901 1190 m 1,0,-1 + 609 1190 l 1,1,-1 + 683 1511 l 1,2,-1 + 975 1511 l 1,3,-1 + 901 1190 l 1,0,-1 +862 113 m 1,4,5 + 774 55 774 55 669.5 18.5 c 128,-1,6 + 565 -18 565 -18 419 -18 c 0,7,8 + 196 -18 196 -18 125 80 c 128,-1,9 + 54 178 54 178 83 303 c 0,10,11 + 108 410 108 410 166.5 484.5 c 128,-1,12 + 225 559 225 559 329 623 c 0,13,14 + 424 682 424 682 484.5 749.5 c 128,-1,15 + 545 817 545 817 563 899 c 0,16,17 + 573 944 573 944 579.5 970 c 128,-1,18 + 586 996 586 996 598 1047 c 1,19,-1 + 851 1047 l 1,20,21 + 851 1047 851 1047 834.5 977 c 128,-1,22 + 818 907 818 907 807 858 c 0,23,24 + 786 768 786 768 722.5 686 c 128,-1,25 + 659 604 659 604 581 555 c 0,26,27 + 478 489 478 489 440 444 c 128,-1,28 + 402 399 402 399 388 338 c 0,29,30 + 374 276 374 276 406.5 239.5 c 128,-1,31 + 439 203 439 203 546 203 c 0,32,33 + 641 203 641 203 743.5 246 c 128,-1,34 + 846 289 846 289 913 336 c 1,35,-1 + 862 113 l 1,4,5 +EndSplineSet +EndChar + +StartChar: exclamdown +Encoding: 119 161 163 +Width: 652 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 32 + 5 + 0 + 6 + 116 + 1 + 66 + 2 + 10 + 63 + 9 + 1 + 7 + 6 + 4 + 5 + 4 + 67 + 3 + 65 + 2 + 67 + 63 + 5 + 79 + 5 + 128 + 5 + 144 + 5 + 4 + 5 + 8 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MIRP[rnd,black] +MIRP[rp0,min,rnd,black] +MIRP[rnd,black] +SRP1 +SRP2 +IP +IP +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIRP[rp0,min,rnd,white] +MIAP[rnd] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +757 1190 m 1,0,-1 + 460 1190 l 1,1,-1 + 534 1509 l 1,2,-1 + 830 1509 l 1,3,-1 + 757 1190 l 1,0,-1 +503 0 m 1,4,-1 + 164 0 l 1,5,-1 + 430 995 l 1,6,-1 + 697 995 l 1,7,-1 + 503 0 l 1,4,-1 +EndSplineSet +EndChar + +StartChar: logicalnot +Encoding: 130 172 164 +Width: 1106 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 20 + 2 + 68 + 1 + 4 + 70 + 48 + 5 + 64 + 5 + 2 + 5 + 69 + 16 + 1 + 112 + 1 + 2 + 1 + 3 + 6 +SRP0 +MDRP[rp0,rnd,white] +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MDRP[min,rnd,grey] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1074 358 m 1,0,-1 + 868 358 l 1,1,-1 + 919 580 l 1,2,-1 + 230 580 l 1,3,-1 + 280 797 l 1,4,-1 + 1175 797 l 1,5,-1 + 1074 358 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: radical +Encoding: 393 8730 165 +Width: 1251 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 65 + 59 + 4 + 1 + 59 + 5 + 1 + 5 + 42 + 4 + 5 + 51 + 1 + 1 + 1 + 57 + 6 + 87 + 6 + 101 + 6 + 3 + 6 + 42 + 0 + 0 + 1 + 48 + 5 + 1 + 5 + 42 + 2 + 53 + 4 + 1 + 4 + 42 + 3 + 3 + 2 + 4 + 4 + 3 + 5 + 3 + 2 + 6 + 10 + 2 + 6 + 0 + 1 + 4 + 2 + 3 + 59 + 0 + 1 + 0 + 5 + 1 + 33 + 2 + 3 + 7 +SRP0 +MDRP[rp0,rnd,white] +MDRP[rp0,min,rnd,grey] +MIRP[rp0,min,rnd,black] +IP +MDRP[rp0,min,rnd,grey] +DELTAP1 +SRP1 +SRP2 +IP +SRP1 +SRP2 +IP +SVTCA[y-axis] +MDAP[rnd] +MIAP[rnd] +SRP2 +SLOOP +IP +MDAP[rnd] +SPVTL[orthog] +SFVTCA[x-axis] +SRP0 +MIRP[rp0,min,rnd,black] +DELTAP1 +SRP0 +MIRP[rp0,min,rnd,black] +DELTAP1 +SPVTL[orthog] +SRP0 +MIRP[rp0,min,rnd,black] +DELTAP1 +SRP0 +DELTAP1 +SFVTL[parallel] +MIRP[rp0,min,rnd,black] +DELTAP1 +IUP[x] +IUP[y] +SVTCA[y-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1350 1509 m 1,0,-1 + 580 -49 l 1,1,-1 + 381 -49 l 1,2,-1 + 41 414 l 1,3,-1 + 287 569 l 1,4,-1 + 475 283 l 1,5,-1 + 1051 1509 l 1,6,-1 + 1350 1509 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: florin +Encoding: 334 402 166 +Width: 814 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 49 + 9 + 32 + 233 + 31 + 248 + 31 + 3 + 8 + 31 + 1 + 11 + 26 + 32 + 8 + 29 + 29 + 15 + 33 + 1 + 4 + 52 + 0 + 33 + 15 + 19 + 22 + 52 + 18 + 15 + 19 + 1 + 0 + 32 + 10 + 48 + 10 + 2 + 10 + 12 + 22 + 25 + 15 + 14 + 27 + 29 + 25 + 25 + 37 + 36 +SRP1 +SRP2 +IP +MDAP[rnd] +MDRP[rnd,grey] +MDRP[min,rnd,grey] +MDRP[rp0,min,grey] +MDRP[grey] +SRP0 +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +DELTAP1 +MDRP[rp0,min,grey] +MDRP[grey] +SVTCA[y-axis] +MIAP[rnd] +MDRP[grey] +MIRP[rp0,min,rnd,black] +MDRP[grey] +MIAP[rnd] +MDRP[grey] +MIRP[rp0,min,rnd,black] +MDRP[grey] +SRP1 +SRP2 +IP +MDAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP2 +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1199 1460 m 1,0,-1 + 1065 1251 l 1,1,2 + 1051 1270 1051 1270 1026 1289.5 c 128,-1,3 + 1001 1309 1001 1309 961 1309 c 0,4,5 + 913 1309 913 1309 874.5 1276 c 128,-1,6 + 836 1243 836 1243 814 1151 c 2,7,-1 + 757 903 l 1,8,-1 + 1000 903 l 1,9,-1 + 959 727 l 1,10,-1 + 716 727 l 1,11,-1 + 562 57 l 2,12,13 + 520 -127 520 -127 402.5 -240.5 c 128,-1,14 + 285 -354 285 -354 140 -354 c 0,15,16 + 41 -354 41 -354 -10.5 -335.5 c 128,-1,17 + -62 -317 -62 -317 -98 -285 c 1,18,-1 + 29 -86 l 1,19,20 + 46 -106 46 -106 73.5 -123.5 c 128,-1,21 + 101 -141 101 -141 139 -141 c 0,22,23 + 188 -141 188 -141 220 -97 c 128,-1,24 + 252 -53 252 -53 270 25 c 2,25,-1 + 432 729 l 1,26,-1 + 305 729 l 1,27,-1 + 345 903 l 1,28,-1 + 472 903 l 1,29,-1 + 522 1118 l 2,30,31 + 564 1298 564 1298 692.5 1414 c 128,-1,32 + 821 1530 821 1530 962 1530 c 0,33,34 + 1063 1530 1063 1530 1115 1510.5 c 128,-1,35 + 1167 1491 1167 1491 1199 1460 c 1,0,-1 +EndSplineSet +EndChar + +StartChar: approxequal +Encoding: 396 8776 167 +Width: 1188 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 67 + 24 + 1 + 24 + 9 + 106 + 1 + 106 + 9 + 4 + 8 + 11 + 70 + 9 + 43 + 8 + 14 + 110 + 14 + 126 + 14 + 3 + 14 + 10 + 43 + 15 + 70 + 12 + 13 + 5 + 3 + 70 + 0 + 1 + 43 + 8 + 6 + 110 + 6 + 126 + 6 + 3 + 6 + 2 + 43 + 7 + 70 + 4 + 5 + 5 + 2 + 6 + 1 + 13 + 10 + 14 + 9 + 8 + 8 + 11 + 0 + 48 + 8 + 1 + 8 + 3 + 11 + 16 +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rnd,grey] +MDRP[rp0,min,rnd,black] +DELTAP1 +MDRP[rnd,grey] +SRP1 +SRP2 +SLOOP +IP +SVTCA[y-axis] +MDAP[rnd] +MDRP[rnd,grey] +SROUND +MDRP[rnd,grey] +RTG +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SROUND +MDRP[rnd,grey] +RTG +SRP0 +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +SROUND +MDRP[rnd,grey] +RTG +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +SROUND +MDRP[rnd,grey] +RTG +MDRP[rnd,grey] +IUP[x] +IUP[y] +SVTCA[y-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1065 715 m 1,0,-1 + 819 602 l 1,1,-1 + 348 807 l 1,2,-1 + 125 715 l 1,3,-1 + 125 952 l 1,4,-1 + 369 1065 l 1,5,-1 + 840 860 l 1,6,-1 + 1065 952 l 1,7,-1 + 1065 715 l 1,0,-1 +1065 236 m 1,8,-1 + 819 123 l 1,9,-1 + 348 328 l 1,10,-1 + 125 236 l 1,11,-1 + 125 473 l 1,12,-1 + 369 586 l 1,13,-1 + 840 381 l 1,14,-1 + 1065 473 l 1,15,-1 + 1065 236 l 1,8,-1 +EndSplineSet +EndChar + +StartChar: Delta +Encoding: 387 8710 168 +Width: 1360 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 23 + 5 + 6 + 4 + 5 + 2 + 3 + 5 + 1 + 0 + 55 + 1 + 1 + 1 + 4 + 6 + 45 + 0 + 1 + 0 + 5 + 3 + 2 + 10 +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MDRP[rnd,grey] +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SVTCA[x-axis] +MDAP[rnd] +DELTAP1 +MDAP[rnd] +SRP0 +SLOOP +IP +RTHG +MDAP[rnd] +RTG +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1346 0 m 1,0,-1 + -2 0 l 1,1,-1 + 559 1509 l 1,2,-1 + 772 1509 l 1,3,-1 + 1346 0 l 1,0,-1 +891 246 m 1,4,-1 + 618 1008 l 1,5,-1 + 342 246 l 1,6,-1 + 891 246 l 1,4,-1 +EndSplineSet +EndChar + +StartChar: guillemotleft +Encoding: 129 171 169 +Width: 1262 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 47 + 11 + 4 + 3 + 2 + 13 + 6 + 0 + 1 + 7 + 0 + 1 + 10 + 3 + 12 + 5 + 8 + 48 + 1 + 1 + 1 + 52 + 9 + 2 + 5 + 0 + 1 + 4 + 0 + 95 + 48 + 1 + 1 + 1 + 8 + 12 + 7 + 8 + 11 + 7 + 95 + 63 + 8 + 159 + 8 + 2 + 8 + 14 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP1 +SRP2 +IP +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP1 +SRP2 +IP +SVTCA[y-axis] +MDAP[rnd] +MDRP[rnd,grey] +MIRP[min,rnd,black] +DELTAP1 +SHP[rp2] +IP +IP +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +SRP1 +SRP2 +IP +IP +SRP1 +SRP2 +IP +IP +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1229 123 m 1,0,-1 + 740 496 l 1,1,-1 + 789 709 l 1,2,-1 + 1452 1090 l 1,3,-1 + 1389 815 l 1,4,-1 + 1018 602 l 1,5,-1 + 1292 397 l 1,6,-1 + 1229 123 l 1,0,-1 +654 123 m 1,7,-1 + 165 496 l 1,8,-1 + 214 709 l 1,9,-1 + 877 1090 l 1,10,-1 + 814 815 l 1,11,-1 + 443 602 l 1,12,-1 + 717 397 l 1,13,-1 + 654 123 l 1,7,-1 +EndSplineSet +EndChar + +StartChar: guillemotright +Encoding: 145 187 170 +Width: 1360 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 169 171 N -1 0 0 1 1360 0 2 +EndChar + +StartChar: ellipsis +Encoding: 359 8230 171 +Width: 1621 +GlyphClass: 1 +Flags: W +CounterMasks: 1 70 +TtInstrs: +NPUSHB + 24 + 2 + 6 + 10 + 72 + 128 + 1 + 5 + 9 + 0 + 0 + 71 + 1 + 4 + 71 + 5 + 8 + 71 + 223 + 9 + 224 + 9 + 2 + 9 + 12 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +MIRP[min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MDRP[rnd,grey] +SMD +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[rnd,grey] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1506 0 m 1,0,-1 + 1203 0 l 1,1,-1 + 1278 326 l 1,2,-1 + 1581 326 l 1,3,-1 + 1506 0 l 1,0,-1 +965 0 m 1,4,-1 + 662 0 l 1,5,-1 + 738 326 l 1,6,-1 + 1040 326 l 1,7,-1 + 965 0 l 1,4,-1 +424 0 m 1,8,-1 + 121 0 l 1,9,-1 + 196 326 l 1,10,-1 + 499 326 l 1,11,-1 + 424 0 l 1,8,-1 +EndSplineSet +EndChar + +StartChar: nonbreakingspace +Encoding: 118 160 172 +Width: 1242 +GlyphClass: 1 +Flags: W +LayerCount: 2 +EndChar + +StartChar: Agrave +Encoding: 150 192 173 +Width: 1358 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 48 + 13 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 36 65 N 1 0 0 1 0 0 2 +Refer: 67 96 N 1 0 0 1 214 264 2 +EndChar + +StartChar: Atilde +Encoding: 153 195 174 +Width: 1358 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 48 + 24 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 36 65 N 1 0 0 1 0 0 2 +Refer: 217 732 N 1 0 0 1 208 246 2 +EndChar + +StartChar: Otilde +Encoding: 171 213 175 +Width: 1365 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_6 + 63 + 41 + 1 + 63 + 55 + 1 +DELTAP1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 50 79 N 1 0 0 1 0 0 3 +Refer: 217 732 N 1 0 0 1 252 246 2 +EndChar + +StartChar: OE +Encoding: 289 338 176 +Width: 1743 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 59 + 23 + 30 + 233 + 5 + 228 + 36 + 3 + 23 + 30 + 20 + 73 + 20 + 1 + 16 + 29 + 31 + 13 + 11 + 26 + 19 + 30 + 16 + 10 + 38 + 31 + 4 + 1 + 41 + 24 + 30 + 1 + 0 + 18 + 22 + 63 + 0 + 1 + 0 + 20 + 24 + 23 + 16 + 1 + 26 + 0 + 41 + 80 + 41 + 160 + 41 + 3 + 41 + 35 + 21 + 48 + 7 + 1 + 7 + 44 +SRP0 +MDRP[rp0,min,rnd,white] +DELTAP1 +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +DELTAP1 +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[rp0,min,rnd,grey] +DELTAP1 +MDRP[rnd,grey] +MDRP[rnd,grey] +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +SROUND +MDAP[rnd] +RTG +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1688 0 m 1,0,-1 + 877 0 l 1,1,2 + 828 -16 828 -16 795 -20.5 c 128,-1,3 + 762 -25 762 -25 680 -25 c 0,4,5 + 364 -25 364 -25 252.5 156.5 c 128,-1,6 + 141 338 141 338 225 702 c 0,7,8 + 229 719 229 719 234 740.5 c 128,-1,9 + 239 762 239 762 245 788 c 0,10,11 + 326 1139 326 1139 538 1334.5 c 128,-1,12 + 750 1530 750 1530 1039 1530 c 0,13,14 + 1113 1530 1113 1530 1163 1519.5 c 128,-1,15 + 1213 1509 1213 1509 1251 1509 c 2,16,-1 + 1998 1509 l 1,17,-1 + 1946 1284 l 1,18,-1 + 1437 1284 l 1,19,-1 + 1344 879 l 1,20,-1 + 1743 879 l 1,21,-1 + 1690 651 l 1,22,-1 + 1291 651 l 1,23,-1 + 1193 225 l 1,24,-1 + 1739 225 l 1,25,-1 + 1688 0 l 1,0,-1 +1134 1296 m 1,26,27 + 1100 1305 1100 1305 1060.5 1310 c 128,-1,28 + 1021 1315 1021 1315 970 1315 c 0,29,30 + 823 1315 823 1315 725.5 1192 c 128,-1,31 + 628 1069 628 1069 558 768 c 0,32,33 + 554 752 554 752 553 746.5 c 128,-1,34 + 552 741 552 741 549 727 c 0,35,36 + 484 445 484 445 514.5 316.5 c 128,-1,37 + 545 188 545 188 717 188 c 0,38,39 + 777 188 777 188 815.5 198.5 c 128,-1,40 + 854 209 854 209 884 223 c 1,41,42 + 884 223 884 223 1006 747.5 c 128,-1,43 + 1128 1272 1128 1272 1134 1296 c 1,26,27 +EndSplineSet +EndChar + +StartChar: oe +Encoding: 290 339 177 +Width: 1777 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 103 + 70 + 18 + 70 + 25 + 73 + 50 + 89 + 13 + 86 + 28 + 86 + 33 + 89 + 50 + 105 + 13 + 101 + 33 + 105 + 50 + 121 + 50 + 150 + 18 + 150 + 25 + 154 + 33 + 14 + 1 + 46 + 127 + 44 + 1 + 44 + 44 + 29 + 14 + 3 + 11 + 32 + 29 + 41 + 51 + 33 + 32 + 26 + 13 + 7 + 70 + 14 + 4 + 60 + 33 + 8 + 70 + 11 + 17 + 3 + 8 + 0 + 49 + 16 + 38 + 64 + 38 + 96 + 38 + 3 + 38 + 45 + 57 + 14 + 57 + 29 + 2 + 14 + 29 + 1 + 45 + 44 + 1 + 49 + 48 + 45 + 1 + 45 + 57 + 23 + 15 + 20 + 96 + 20 + 128 + 20 + 144 + 20 + 4 + 255 + 20 + 1 + 20 + 63 +SRP0 +MDRP[rp0,min,rnd,white] +DELTAP1 +DELTAP2 +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP1 +SRP2 +IP +IP +DELTAP1 +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +SROUND +MDRP[rnd,grey] +RTG +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[grey] +SROUND +MDRP[rnd,grey] +RTG +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[rnd,grey] +SRP1 +SRP2 +SLOOP +IP +MDAP[rnd] +DELTAP1 +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1798 484 m 1,0,-1 + 1128 484 l 1,1,2 + 1088 309 1088 309 1146.5 240.5 c 128,-1,3 + 1205 172 1205 172 1325 172 c 0,4,5 + 1404 172 1404 172 1490.5 193.5 c 128,-1,6 + 1577 215 1577 215 1675 258 c 1,7,-1 + 1633 74 l 1,8,9 + 1546 25 1546 25 1448.5 0 c 128,-1,10 + 1351 -25 1351 -25 1230 -25 c 0,11,12 + 1135 -25 1135 -25 1048.5 7 c 128,-1,13 + 962 39 962 39 910 129 c 1,14,15 + 841 63 841 63 752 19 c 128,-1,16 + 663 -25 663 -25 561 -25 c 0,17,18 + 355 -25 355 -25 252.5 120.5 c 128,-1,19 + 150 266 150 266 207 516 c 0,20,21 + 211 535 211 535 216 555.5 c 128,-1,22 + 221 576 221 576 225 594 c 0,23,24 + 280 831 280 831 452.5 976.5 c 128,-1,25 + 625 1122 625 1122 824 1122 c 0,26,27 + 955 1122 955 1122 1022.5 1087 c 128,-1,28 + 1090 1052 1090 1052 1121 986 c 1,29,30 + 1195 1052 1195 1052 1287 1087 c 128,-1,31 + 1379 1122 1379 1122 1480 1122 c 0,32,33 + 1705 1122 1705 1122 1791 973.5 c 128,-1,34 + 1877 825 1877 825 1821 586 c 2,35,36 + 1821 586 1821 586 1811 541 c 128,-1,37 + 1801 496 1801 496 1798 484 c 1,0,-1 +1582 648 m 1,38,39 + 1608 803 1608 803 1568.5 870 c 128,-1,40 + 1529 937 1529 937 1440 937 c 0,41,42 + 1363 937 1363 937 1286 871 c 128,-1,43 + 1209 805 1209 805 1168 648 c 1,44,-1 + 1582 648 l 1,38,39 +896 532 m 0,45,46 + 900 550 900 550 901.5 556 c 128,-1,47 + 903 562 903 562 906 577 c 0,48,49 + 942 731 942 731 917.5 831 c 128,-1,50 + 893 931 893 931 794 931 c 0,51,52 + 692 931 692 931 625 844 c 128,-1,53 + 558 757 558 757 517 577 c 0,54,55 + 514 562 514 562 511.5 552.5 c 128,-1,56 + 509 543 509 543 506 532 c 0,57,58 + 463 347 463 347 490 257.5 c 128,-1,59 + 517 168 517 168 617 168 c 0,60,61 + 716 168 716 168 784 254.5 c 128,-1,62 + 852 341 852 341 896 532 c 0,45,46 +EndSplineSet +EndChar + +StartChar: endash +Encoding: 348 8211 178 +Width: 949 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 16 + 1 + 68 + 2 + 70 + 48 + 0 + 1 + 0 + 47 + 1 + 63 + 1 + 2 + 1 + 73 + 4 +SRP0 +SROUND +MDRP[rp0,rnd,white] +RTG +DELTAP1 +MDRP[rp0,min,rnd,black] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1025 561 m 1,0,-1 + 184 561 l 1,1,-1 + 239 799 l 1,2,-1 + 1080 799 l 1,3,-1 + 1025 561 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: emdash +Encoding: 349 8212 179 +Width: 1584 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 11 + 1 + 68 + 2 + 70 + 48 + 0 + 1 + 0 + 1 + 73 + 4 +SRP0 +SROUND +MDRP[rp0,rnd,white] +RTG +MDRP[rp0,min,rnd,black] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1641 561 m 1,0,-1 + 202 561 l 1,1,-1 + 257 799 l 1,2,-1 + 1696 799 l 1,3,-1 + 1641 561 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: quotedblleft +Encoding: 353 8220 180 +Width: 1017 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 49 + 11 + 3 + 11 + 14 + 26 + 3 + 26 + 14 + 4 + 11 + 1 + 85 + 20 + 9 + 86 + 5 + 17 + 6 + 75 + 16 + 5 + 11 + 63 + 23 + 1 + 15 + 6 + 31 + 6 + 2 + 6 + 10 + 84 + 9 + 89 + 1 + 12 + 15 + 16 + 31 + 16 + 2 + 16 + 21 + 84 + 20 + 89 + 12 + 22 +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[min,rnd,black] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +DELTAP1 +SRP0 +MDRP[rp0,min,rnd,grey] +MIRP[min,rnd,black] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +DELTAP1 +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MIRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1101 854 m 1,0,-1 + 785 854 l 1,1,-1 + 864 1194 l 2,2,3 + 905 1372 905 1372 1019.5 1451 c 128,-1,4 + 1134 1530 1134 1530 1257 1530 c 1,5,-1 + 1223 1382 l 1,6,7 + 1145 1382 1145 1382 1099 1343.5 c 128,-1,8 + 1053 1305 1053 1305 1027 1194 c 1,9,-1 + 1180 1194 l 1,10,-1 + 1101 854 l 1,0,-1 +634 854 m 1,11,-1 + 318 854 l 1,12,-1 + 397 1194 l 2,13,14 + 438 1372 438 1372 552.5 1451 c 128,-1,15 + 667 1530 667 1530 790 1530 c 1,16,-1 + 756 1382 l 1,17,18 + 678 1382 678 1382 632 1343.5 c 128,-1,19 + 586 1305 586 1305 560 1194 c 1,20,-1 + 713 1194 l 1,21,-1 + 634 854 l 1,11,-1 +EndSplineSet +EndChar + +StartChar: quotedblright +Encoding: 354 8221 181 +Width: 1017 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 48 + 5 + 1 + 5 + 12 + 20 + 1 + 20 + 12 + 4 + 4 + 15 + 75 + 3 + 14 + 86 + 7 + 18 + 85 + 9 + 20 + 10 + 63 + 23 + 1 + 7 + 89 + 10 + 84 + 0 + 4 + 16 + 4 + 2 + 4 + 9 + 20 + 18 + 89 + 21 + 84 + 0 + 15 + 16 + 15 + 2 + 15 + 20 + 22 +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +MIRP[rp0,min,rnd,black] +SRP0 +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +MIRP[rp0,min,rnd,black] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1178 1169 m 2,0,1 + 1137 991 1137 991 1022.5 912.5 c 128,-1,2 + 908 834 908 834 784 834 c 1,3,-1 + 818 981 l 1,4,5 + 896 981 896 981 942.5 1020 c 128,-1,6 + 989 1059 989 1059 1014 1169 c 1,7,-1 + 861 1169 l 1,8,-1 + 940 1509 l 1,9,-1 + 1256 1509 l 1,10,-1 + 1178 1169 l 2,0,1 +711 1169 m 2,11,12 + 670 991 670 991 555.5 912.5 c 128,-1,13 + 441 834 441 834 317 834 c 1,14,-1 + 351 981 l 1,15,16 + 429 981 429 981 475.5 1020 c 128,-1,17 + 522 1059 522 1059 547 1169 c 1,18,-1 + 395 1169 l 1,19,-1 + 473 1509 l 1,20,-1 + 789 1509 l 1,21,-1 + 711 1169 l 2,11,12 +EndSplineSet +EndChar + +StartChar: quoteleft +Encoding: 350 8216 182 +Width: 625 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 25 + 10 + 3 + 25 + 3 + 2 + 1 + 85 + 9 + 86 + 6 + 75 + 5 + 11 + 15 + 6 + 31 + 6 + 2 + 6 + 10 + 84 + 9 + 89 + 1 + 11 +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[min,rnd,black] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIRP[min,rnd,black] +MIRP[rp0,min,rnd,grey] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +672 854 m 1,0,-1 + 356 854 l 1,1,-1 + 434 1194 l 2,2,3 + 475 1372 475 1372 589.5 1451 c 128,-1,4 + 704 1530 704 1530 828 1530 c 1,5,-1 + 794 1382 l 1,6,7 + 716 1382 716 1382 669.5 1343.5 c 128,-1,8 + 623 1305 623 1305 598 1194 c 1,9,-1 + 750 1194 l 1,10,-1 + 672 854 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: quoteright +Encoding: 351 8217 183 +Width: 625 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 25 + 6 + 1 + 21 + 1 + 2 + 4 + 75 + 3 + 86 + 7 + 85 + 9 + 10 + 7 + 89 + 10 + 84 + 0 + 4 + 16 + 4 + 2 + 4 + 9 + 11 +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIRP[rp0,min,rnd,grey] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +748 1169 m 2,0,1 + 707 991 707 991 592.5 912.5 c 128,-1,2 + 478 834 478 834 355 834 c 1,3,-1 + 388 981 l 1,4,5 + 466 981 466 981 512.5 1020 c 128,-1,6 + 559 1059 559 1059 584 1169 c 1,7,-1 + 432 1169 l 1,8,-1 + 510 1509 l 1,9,-1 + 827 1509 l 1,10,-1 + 748 1169 l 2,0,1 +EndSplineSet +EndChar + +StartChar: divide +Encoding: 205 247 184 +Width: 1100 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 24 + 19 + 56 + 25 + 13 + 69 + 14 + 9 + 56 + 3 + 14 + 70 + 48 + 12 + 1 + 12 + 0 + 16 + 55 + 13 + 6 + 22 + 22 + 29 + 28 +SRP1 +SRP2 +IP +MDAP[rnd] +MDRP[rnd,grey] +MDRP[min,rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[rp0,min,rnd,grey] +DELTAP2 +SVTCA[y-axis] +MIAP[rnd] +MDRP[rp0,min,rnd,white] +MIRP[rp0,min,rnd,black] +SRP0 +MIRP[rp0,min,rnd,black] +MDRP[rp0,min,rnd,white] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +933 1036 m 0,0,1 + 918 973 918 973 865.5 928 c 128,-1,2 + 813 883 813 883 754 883 c 0,3,4 + 694 883 694 883 663 928 c 128,-1,5 + 632 973 632 973 646 1036 c 0,6,7 + 661 1100 661 1100 713 1145 c 128,-1,8 + 765 1190 765 1190 825 1190 c 0,9,10 + 884 1190 884 1190 916 1145 c 128,-1,11 + 948 1100 948 1100 933 1036 c 0,0,1 +1134 582 m 1,12,-1 + 237 582 l 1,13,-1 + 287 799 l 1,14,-1 + 1184 799 l 1,15,-1 + 1134 582 l 1,12,-1 +770 332 m 0,16,17 + 755 268 755 268 702.5 223 c 128,-1,18 + 650 178 650 178 591 178 c 0,19,20 + 531 178 531 178 500 223 c 128,-1,21 + 469 268 469 268 484 332 c 0,22,23 + 499 395 499 395 551 440 c 128,-1,24 + 603 485 603 485 662 485 c 0,25,26 + 721 485 721 485 753 440 c 128,-1,27 + 785 395 785 395 770 332 c 0,16,17 +EndSplineSet +EndChar + +StartChar: lozenge +Encoding: 403 9674 185 +Width: 1372 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 65 + 42 + 1 + 37 + 3 + 59 + 5 + 52 + 7 + 89 + 1 + 86 + 3 + 202 + 1 + 198 + 3 + 8 + 105 + 1 + 170 + 5 + 165 + 7 + 3 + 37 + 0 + 42 + 2 + 59 + 4 + 52 + 6 + 86 + 0 + 89 + 2 + 198 + 0 + 201 + 2 + 8 + 105 + 2 + 170 + 4 + 166 + 6 + 3 + 7 + 5 + 6 + 4 + 0 + 3 + 48 + 2 + 1 + 2 + 4 + 6 + 5 + 7 + 1 + 3 + 3 +RTHG +MDAP[rnd] +SLOOP +ALIGNRP +MDRP[grey] +MDRP[grey] +SVTCA[y-axis] +MDAP[rnd] +DELTAP1 +SLOOP +ALIGNRP +MDRP[grey] +MDRP[grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP2 +DELTAP1 +SVTCA[y-axis] +DELTAP2 +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1280 682 m 1,0,-1 + 686 88 l 1,1,-1 + 92 682 l 1,2,-1 + 686 1276 l 1,3,-1 + 1280 682 l 1,0,-1 +973 682 m 1,4,-1 + 686 969 l 1,5,-1 + 399 682 l 1,6,-1 + 686 395 l 1,7,-1 + 973 682 l 1,4,-1 +EndSplineSet +EndChar + +StartChar: ydieresis +Encoding: 213 255 186 +Width: 1190 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 16 + 0 + 11 + 0 + 23 + 16 + 11 + 16 + 23 + 4 + 31 + 14 + 31 + 26 + 63 + 14 + 3 +DELTAP1 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 92 121 N 1 0 0 1 0 0 2 +Refer: 142 168 N 1 0 0 1 61 -88 2 +EndChar + +StartChar: Ydieresis +Encoding: 327 376 187 +Width: 1327 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 63 + 15 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 60 89 N 1 0 0 1 0 0 2 +Refer: 142 168 N 1 0 0 1 261 247 2 +EndChar + +StartChar: fraction +Encoding: 363 8260 188 +AltUni2: 002215.ffffffff.0 002215.ffffffff.0 +Width: 489 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 45 + 6 + 1 + 22 + 1 + 24 + 3 + 39 + 1 + 40 + 3 + 53 + 0 + 57 + 1 + 57 + 2 + 72 + 3 + 88 + 3 + 10 + 0 + 57 + 3 + 1 + 3 + 55 + 1 + 1 + 1 + 2 + 2 + 3 + 3 + 57 + 2 + 1 + 2 + 4 + 1 + 2 + 0 + 0 + 3 + 10 +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIAP[rnd] +MDRP[rnd,grey] +SVTCA[x-axis] +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MDRP[rp0,min,rnd,grey] +SPVTL[orthog] +SRP0 +MDRP[rp0,min,rnd,black] +DELTAP1 +SRP0 +DELTAP1 +MDRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1133 1509 m 1,0,-1 + -94 0 l 1,1,-1 + -296 0 l 1,2,-1 + 931 1509 l 1,3,-1 + 1133 1509 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: Euro +Encoding: 368 8364 189 +Width: 1413 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 41 + 5 + 227 + 6 + 227 + 1 + 227 + 2 + 16 + 18 + 227 + 19 + 21 + 227 + 16 + 8 + 29 + 27 + 227 + 10 + 4 + 228 + 7 + 228 + 0 + 228 + 3 + 29 + 18 + 228 + 19 + 29 + 24 + 2 + 6 + 228 + 1 + 5 + 24 + 227 + 13 + 30 +SRP0 +MDRP[rp0,rnd,white] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,black] +ALIGNRP +MIRP[rp0,rnd,black] +ALIGNRP +SRP0 +MDRP[rp0,min,rnd,black] +ALIGNRP +MIRP[rp0,rnd,black] +SRP0 +MDRP[rp0,rnd,black] +MIRP[rp0,rnd,black] +MIRP[rp0,min,rnd,black] +MIRP[rp0,rnd,black] +SVTCA[y-axis] +MDAP[rnd] +MIRP[rp0,min,rnd,black] +MDRP[rp0,rnd,black] +MDRP[rnd,black] +MDAP[rnd] +MIRP[rp0,min,rnd,black] +MDRP[rp0,rnd,black] +MIRP[min,rnd,black] +SRP0 +MDRP[rp0,min,rnd,black] +MIRP[rp0,min,rnd,black] +MIRP[rp0,rnd,black] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1315 715 m 1,0,-1 + 254 715 l 1,1,-1 + 324 825 l 1,2,-1 + 1384 825 l 1,3,-1 + 1315 715 l 1,0,-1 +1177 495 m 1,4,-1 + 203 495 l 1,5,-1 + 273 605 l 1,6,-1 + 1247 605 l 1,7,-1 + 1177 495 l 1,4,-1 +1301 149 m 1,8,9 + 1106 0 1106 0 869 0 c 0,10,11 + 609 0 609 0 475.5 190 c 128,-1,12 + 342 380 342 380 407 660 c 256,13,14 + 472 940 472 940 692.5 1130 c 128,-1,15 + 913 1320 913 1320 1173 1320 c 256,16,17 + 1433 1320 1433 1320 1571 1124 c 1,18,-1 + 1500 1013 l 1,19,20 + 1375 1210 1375 1210 1148 1210 c 0,21,22 + 934 1210 934 1210 748 1050 c 128,-1,23 + 562 890 562 890 509 660 c 256,24,25 + 456 430 456 430 568 270 c 128,-1,26 + 680 110 680 110 894 110 c 0,27,28 + 1126 110 1126 110 1338 311 c 1,29,-1 + 1301 149 l 1,8,9 +EndSplineSet +EndChar + +StartChar: guilsinglleft +Encoding: 361 8249 190 +Width: 688 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 27 + 4 + 3 + 2 + 6 + 0 + 1 + 0 + 1 + 3 + 5 + 48 + 1 + 1 + 1 + 52 + 2 + 63 + 8 + 1 + 5 + 0 + 1 + 4 + 0 + 95 + 1 + 7 +SRP0 +MDRP[rp0,rnd,white] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP1 +SRP2 +IP +DELTAP1 +SVTCA[y-axis] +MDAP[rnd] +MIRP[min,rnd,black] +DELTAP1 +IP +MDRP[rp0,min,rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +SRP1 +SRP2 +IP +SRP1 +SRP2 +IP +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +656 123 m 1,0,-1 + 166 496 l 1,1,-1 + 216 709 l 1,2,-1 + 879 1090 l 1,3,-1 + 816 815 l 1,4,-1 + 445 602 l 1,5,-1 + 719 397 l 1,6,-1 + 656 123 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: guilsinglright +Encoding: 362 8250 191 +Width: 743 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 190 8249 N -1 0 0 1 744 0 2 +EndChar + +StartChar: fi +Encoding: 407 64257 192 +Width: 1294 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 38 + 9 + 11 + 26 + 11 + 2 + 3 + 6 + 32 + 24 + 9 + 12 + 1 + 5 + 2 + 17 + 20 + 33 + 16 + 13 + 17 + 0 + 22 + 143 + 1 + 1 + 1 + 16 + 5 + 24 + 4 + 22 + 9 + 7 + 144 + 5 + 1 + 5 + 26 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MDRP[min,rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[min,rnd,grey] +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MDRP[grey] +MIRP[rp0,min,rnd,black] +MDRP[grey] +MIAP[rnd] +MDRP[rnd,grey] +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1149 0 m 1,0,-1 + 829 0 l 1,1,-1 + 829 961 l 1,2,-1 + 489 961 l 1,3,-1 + 489 0 l 1,4,-1 + 170 0 l 1,5,-1 + 170 963 l 1,6,-1 + 18 963 l 1,7,-1 + 18 1151 l 1,8,-1 + 176 1151 l 1,9,-1 + 176 1194 l 2,10,11 + 176 1362 176 1362 276.5 1462.5 c 128,-1,12 + 377 1563 377 1563 545 1563 c 0,13,14 + 625 1563 625 1563 701.5 1541.5 c 128,-1,15 + 778 1520 778 1520 829 1487 c 1,16,-1 + 745 1317 l 1,17,18 + 719 1337 719 1337 687 1351.5 c 128,-1,19 + 655 1366 655 1366 608 1366 c 0,20,21 + 541 1366 541 1366 516.5 1325 c 128,-1,22 + 492 1284 492 1284 492 1204 c 2,23,-1 + 492 1151 l 1,24,-1 + 1149 1151 l 1,25,-1 + 1149 0 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: fl +Encoding: 408 64258 193 +Width: 1276 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 48 + 9 + 20 + 25 + 20 + 40 + 20 + 3 + 12 + 15 + 32 + 9 + 18 + 12 + 1 + 14 + 2 + 2 + 5 + 33 + 22 + 17 + 25 + 16 + 11 + 1 + 13 + 0 + 22 + 25 + 47 + 1 + 95 + 1 + 143 + 1 + 3 + 1 + 14 + 9 + 13 + 22 + 18 + 16 + 144 + 14 + 1 + 14 + 27 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MDRP[min,rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MDRP[grey] +MIAP[rnd] +MDRP[rnd,grey] +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1135 0 m 1,0,-1 + 815 0 l 1,1,-1 + 815 1290 l 1,2,3 + 786 1331 786 1331 742 1348.5 c 128,-1,4 + 698 1366 698 1366 653 1366 c 0,5,6 + 561 1366 561 1366 526.5 1315 c 128,-1,7 + 492 1264 492 1264 492 1184 c 2,8,-1 + 492 1151 l 1,9,-1 + 707 1151 l 1,10,-1 + 707 961 l 1,11,-1 + 489 961 l 1,12,-1 + 489 0 l 1,13,-1 + 170 0 l 1,14,-1 + 170 963 l 1,15,-1 + 18 963 l 1,16,-1 + 18 1151 l 1,17,-1 + 176 1151 l 1,18,-1 + 176 1161 l 2,19,20 + 176 1333 176 1333 271.5 1448 c 128,-1,21 + 367 1563 367 1563 545 1563 c 0,22,23 + 627 1563 627 1563 699.5 1549.5 c 128,-1,24 + 772 1536 772 1536 815 1509 c 1,25,-1 + 1135 1509 l 1,26,-1 + 1135 0 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: daggerdbl +Encoding: 357 8225 194 +Width: 974 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 49 + 1 + 4 + 89 + 18 + 64 + 7 + 128 + 11 + 17 + 8 + 89 + 14 + 11 + 12 + 3 + 18 + 12 + 16 + 48 + 21 + 1 + 2 + 64 + 18 + 1 + 17 + 14 + 19 + 48 + 15 + 1 + 15 + 101 + 13 + 22 + 4 + 7 + 8 + 11 + 3 + 64 + 6 + 63 + 10 + 1 + 10 + 101 + 12 + 20 +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[min,rnd,grey] +DELTAP1 +SHP[rp2] +MIRP[rnd,black] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MIRP[min,rnd,grey] +DELTAP1 +SHP[rp2] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[grey] +MIRP[rp0,rnd,black] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +SMD +MDRP[rp0,min,rnd,grey] +SMD +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1120 672 m 1,0,-1 + 756 672 l 1,1,-1 + 515 -342 l 1,2,-1 + 292 -342 l 1,3,-1 + 518 672 l 1,4,-1 + 172 672 l 1,5,-1 + 212 848 l 1,6,-1 + 552 848 l 1,7,-1 + 577 975 l 1,8,-1 + 241 975 l 1,9,-1 + 282 1151 l 1,10,-1 + 610 1151 l 1,11,-1 + 681 1509 l 1,12,-1 + 980 1509 l 1,13,-1 + 887 1151 l 1,14,-1 + 1231 1151 l 1,15,-1 + 1190 975 l 1,16,-1 + 838 975 l 1,17,-1 + 805 848 l 1,18,-1 + 1161 848 l 1,19,-1 + 1120 672 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: periodcentered +Encoding: 141 183 195 +AltUni2: 002219.ffffffff.0 002219.ffffffff.0 +Width: 625 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_7 + 1 + 71 + 2 + 0 + 71 + 1 + 4 +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MDAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +586 524 m 1,0,-1 + 283 524 l 1,1,-1 + 358 850 l 1,2,-1 + 661 850 l 1,3,-1 + 586 524 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: quotesinglbase +Encoding: 352 8218 196 +Width: 625 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +SplineSet +478 0 m 2,0,1 + 437 -178 437 -178 322.5 -257 c 128,-1,2 + 208 -336 208 -336 84 -336 c 1,3,-1 + 119 -188 l 1,4,5 + 197 -188 197 -188 243 -149.5 c 0,6,7 + 288 -111 288 -111 314 0 c 1,8,-1 + 162 0 l 1,9,-1 + 240 340 l 1,10,-1 + 557 340 l 1,11,-1 + 478 0 l 2,0,1 +EndSplineSet +EndChar + +StartChar: quotedblbase +Encoding: 355 8222 197 +Width: 1017 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 45 + 5 + 1 + 5 + 12 + 21 + 1 + 21 + 12 + 4 + 15 + 4 + 14 + 3 + 86 + 21 + 9 + 85 + 18 + 7 + 0 + 63 + 23 + 1 + 7 + 0 + 84 + 0 + 4 + 16 + 4 + 2 + 4 + 8 + 19 + 18 + 11 + 84 + 0 + 14 + 16 + 14 + 2 + 14 + 19 + 22 +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +MDRP[rp0,min,rnd,black] +SRP0 +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +MDRP[rp0,min,rnd,black] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIRP[min,rnd,black] +SHP[rp2] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[rp0,min,rnd,black] +MDRP[rnd,grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +908 0 m 2,0,1 + 867 -178 867 -178 752.5 -257 c 128,-1,2 + 638 -336 638 -336 514 -336 c 1,3,-1 + 548 -188 l 1,4,5 + 626 -188 626 -188 672 -149.5 c 128,-1,6 + 718 -111 718 -111 744 0 c 1,7,-1 + 592 0 l 1,8,-1 + 670 340 l 1,9,-1 + 986 340 l 1,10,-1 + 908 0 l 2,0,1 +441 0 m 2,11,12 + 400 -178 400 -178 285.5 -257 c 128,-1,13 + 171 -336 171 -336 47 -336 c 1,14,-1 + 81 -188 l 1,15,16 + 159 -188 159 -188 205.5 -149.5 c 128,-1,17 + 252 -111 252 -111 277 0 c 1,18,-1 + 125 0 l 1,19,-1 + 203 340 l 1,20,-1 + 519 340 l 1,21,-1 + 441 0 l 2,11,12 +EndSplineSet +EndChar + +StartChar: perthousand +Encoding: 360 8240 198 +Width: 2378 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 129 + 4 + 1 + 11 + 5 + 11 + 8 + 4 + 12 + 4 + 19 + 11 + 23 + 11 + 26 + 4 + 30 + 4 + 33 + 11 + 37 + 11 + 40 + 4 + 44 + 20 + 1 + 27 + 5 + 27 + 8 + 20 + 12 + 20 + 19 + 27 + 23 + 27 + 26 + 20 + 30 + 20 + 33 + 27 + 37 + 27 + 40 + 20 + 44 + 55 + 15 + 25 + 14 + 14 + 16 + 4 + 57 + 89 + 3 + 87 + 50 + 89 + 10 + 15 + 78 + 64 + 89 + 42 + 28 + 87 + 21 + 85 + 71 + 89 + 35 + 21 + 5 + 17 + 53 + 14 + 1 + 48 + 14 + 1 + 14 + 15 + 61 + 16 + 1 + 63 + 16 + 1 + 16 + 6 + 18 + 88 + 60 + 90 + 68 + 88 + 24 + 38 + 32 + 88 + 74 + 90 + 81 + 88 + 15 + 38 + 1 + 38 + 6 + 0 + 88 + 46 + 90 + 54 + 88 + 0 + 6 + 48 + 6 + 64 + 6 + 176 + 6 + 4 + 6 + 88 +SRP0 +MDRP[rp0,min,rnd,white] +DELTAP1 +MIRP[min,rnd,black] +MIRP[rp0,min,rnd,grey] +MIRP[rp0,min,rnd,black] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[min,rnd,black] +MIRP[rp0,min,rnd,grey] +MIRP[rp0,min,rnd,black] +SRP0 +MDRP[rp0,min,rnd,grey] +MIRP[min,rnd,black] +MIRP[rp0,min,rnd,grey] +MIRP[rp0,min,rnd,black] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +DELTAP1 +MDRP[min,rnd,grey] +MDRP[rp0,min,rnd,black] +DELTAP1 +DELTAP1 +MDRP[rp0,min,rnd,grey] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MIAP[rnd] +MIRP[min,rnd,black] +MIRP[rp0,min,rnd,black] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIAP[rnd] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1049 1153 m 2,0,1 + 1015 1006 1015 1006 897 915.5 c 128,-1,2 + 779 825 779 825 619 825 c 0,3,4 + 461 825 461 825 379 919.5 c 128,-1,5 + 297 1014 297 1014 329 1151 c 2,6,-1 + 341 1206 l 2,7,8 + 370 1333 370 1333 487.5 1432.5 c 128,-1,9 + 605 1532 605 1532 782 1532 c 0,10,11 + 949 1532 949 1532 1021 1439 c 128,-1,12 + 1093 1346 1093 1346 1061 1206 c 2,13,-1 + 1049 1153 l 2,0,1 +1686 1509 m 1,14,-1 + 375 0 l 1,15,-1 + 212 0 l 1,16,-1 + 1523 1509 l 1,17,-1 + 1686 1509 l 1,14,-1 +2385 305 m 2,18,19 + 2347 139 2347 139 2230 58 c 128,-1,20 + 2113 -23 2113 -23 1957 -23 c 0,21,22 + 1789 -23 1789 -23 1711 69.5 c 128,-1,23 + 1633 162 1633 162 1665 303 c 2,24,-1 + 1678 358 l 2,25,26 + 1706 481 1706 481 1825 582.5 c 128,-1,27 + 1944 684 1944 684 2120 684 c 0,28,29 + 2287 684 2287 684 2358.5 592 c 128,-1,30 + 2430 500 2430 500 2397 358 c 2,31,-1 + 2385 305 l 2,18,19 +1574 305 m 2,32,33 + 1536 139 1536 139 1419 58 c 128,-1,34 + 1302 -23 1302 -23 1147 -23 c 0,35,36 + 979 -23 979 -23 901 69.5 c 128,-1,37 + 823 162 823 162 855 303 c 2,38,-1 + 868 358 l 2,39,40 + 896 481 896 481 1015 582.5 c 128,-1,41 + 1134 684 1134 684 1310 684 c 0,42,43 + 1477 684 1477 684 1548 592 c 128,-1,44 + 1619 500 1619 500 1586 358 c 2,45,-1 + 1574 305 l 2,32,33 +813 1163 m 2,46,-1 + 817 1180 l 2,47,48 + 836 1262 836 1262 825.5 1312 c 128,-1,49 + 815 1362 815 1362 743 1362 c 0,50,51 + 672 1362 672 1362 633.5 1314 c 128,-1,52 + 595 1266 595 1266 575 1180 c 2,53,-1 + 571 1163 l 2,54,55 + 555 1094 555 1094 568.5 1037.5 c 128,-1,56 + 582 981 582 981 657 981 c 0,57,58 + 724 981 724 981 759.5 1033 c 128,-1,59 + 795 1085 795 1085 813 1163 c 2,46,-1 +2151 315 m 2,60,-1 + 2155 332 l 2,61,62 + 2174 414 2174 414 2163 464 c 128,-1,63 + 2152 514 2152 514 2081 514 c 0,64,65 + 2010 514 2010 514 1971 466 c 128,-1,66 + 1932 418 1932 418 1912 332 c 2,67,-1 + 1908 315 l 2,68,69 + 1892 246 1892 246 1906 189.5 c 128,-1,70 + 1920 133 1920 133 1994 133 c 0,71,72 + 2060 133 2060 133 2096.5 185.5 c 128,-1,73 + 2133 238 2133 238 2151 315 c 2,60,-1 +1340 315 m 2,74,-1 + 1344 332 l 2,75,76 + 1363 414 1363 414 1352.5 464 c 128,-1,77 + 1342 514 1342 514 1270 514 c 0,78,79 + 1199 514 1199 514 1160.5 466 c 128,-1,80 + 1122 418 1122 418 1102 332 c 2,81,-1 + 1098 315 l 2,82,83 + 1082 246 1082 246 1096 189.5 c 128,-1,84 + 1110 133 1110 133 1184 133 c 0,85,86 + 1250 133 1250 133 1286 185.5 c 128,-1,87 + 1322 238 1322 238 1340 315 c 2,74,-1 +EndSplineSet +EndChar + +StartChar: Acircumflex +Encoding: 152 194 199 +Width: 1358 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 36 65 N 1 0 0 1 0 0 2 +Refer: 216 710 N 1 0 0 1 212 260 2 +EndChar + +StartChar: Ecircumflex +Encoding: 160 202 200 +Width: 1167 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 40 69 N 1 0 0 1 0 0 2 +Refer: 216 710 N 1 0 0 1 174 260 2 +EndChar + +StartChar: Aacute +Encoding: 151 193 201 +Width: 1358 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 36 65 N 1 0 0 1 0 0 2 +Refer: 141 180 N 1 0 0 1 302 260 2 +EndChar + +StartChar: Edieresis +Encoding: 161 203 202 +Width: 1167 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 142 168 N 1 0 0 1 176 246 2 +Refer: 40 69 N 1 0 0 1 0 0 2 +EndChar + +StartChar: Egrave +Encoding: 158 200 203 +Width: 1167 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 48 + 14 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 40 69 N 1 0 0 1 0 0 2 +Refer: 67 96 N 1 0 0 1 166 260 2 +EndChar + +StartChar: Iacute +Encoding: 163 205 204 +Width: 750 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 63 + 9 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 44 73 N 1 0 0 1 0 0 2 +Refer: 141 180 N 1 0 0 1 2 260 2 +EndChar + +StartChar: Icircumflex +Encoding: 164 206 205 +Width: 750 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_6 + 63 + 8 + 1 + 63 + 12 + 1 +DELTAP1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 44 73 N 1 0 0 1 0 0 2 +Refer: 216 710 N 1 0 0 1 -67 260 2 +EndChar + +StartChar: Idieresis +Encoding: 165 207 206 +Width: 750 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_8 + 63 + 10 + 63 + 22 + 2 + 63 + 29 + 1 +DELTAP1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 44 73 N 1 0 0 1 0 0 2 +Refer: 142 168 N 1 0 0 1 -80 247 2 +EndChar + +StartChar: Igrave +Encoding: 162 204 207 +Width: 750 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 63 + 9 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 44 73 N 1 0 0 1 0 0 2 +Refer: 67 96 N 1 0 0 1 -129 260 2 +EndChar + +StartChar: Oacute +Encoding: 169 211 208 +Width: 1365 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_6 + 63 + 30 + 1 + 63 + 33 + 1 +DELTAP1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 50 79 N 1 0 0 1 0 0 3 +Refer: 141 180 N 1 0 0 1 292 260 2 +EndChar + +StartChar: Ocircumflex +Encoding: 170 212 209 +Width: 1365 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_6 + 63 + 32 + 1 + 63 + 36 + 1 +DELTAP1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 50 79 N 1 0 0 1 0 0 3 +Refer: 216 710 N 1 0 0 1 250 260 2 +EndChar + +StartChar: uniF8FF +Encoding: 406 63743 210 +Width: 1585 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 14 + 48 + 3 + 1 + 3 + 9 + 57 + 0 + 1 + 0 + 62 + 6 + 1 + 6 + 12 +SRP0 +MDRP[rp0,min,rnd,white] +DELTAP1 +MDRP[rp0,min,rnd,black] +DELTAP1 +SVTCA[y-axis] +MDAP[rnd] +MDRP[rp0,min,rnd,black] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1346 762 m 0,0,1 + 1346 532 1346 532 1184 370.5 c 128,-1,2 + 1022 209 1022 209 793 209 c 0,3,4 + 563 209 563 209 401.5 370.5 c 128,-1,5 + 240 532 240 532 240 762 c 0,6,7 + 240 991 240 991 401.5 1153 c 128,-1,8 + 563 1315 563 1315 793 1315 c 0,9,10 + 1022 1315 1022 1315 1184 1153 c 128,-1,11 + 1346 991 1346 991 1346 762 c 0,0,1 +EndSplineSet +EndChar + +StartChar: Ograve +Encoding: 168 210 211 +Width: 1365 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 63 + 33 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 50 79 N 1 0 0 1 0 0 3 +Refer: 67 96 N 1 0 0 1 166 260 2 +EndChar + +StartChar: Uacute +Encoding: 176 218 212 +Width: 1343 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 63 + 20 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 56 85 N 1 0 0 1 0 0 2 +Refer: 141 180 N 1 0 0 1 279 260 2 +EndChar + +StartChar: Ucircumflex +Encoding: 177 219 213 +Width: 1343 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 63 + 22 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 56 85 N 1 0 0 1 0 0 2 +Refer: 216 710 N 1 0 0 1 227 260 2 +EndChar + +StartChar: Ugrave +Encoding: 175 217 214 +Width: 1343 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 56 85 N 1 0 0 1 0 0 2 +Refer: 67 96 N 1 0 0 1 193 260 2 +EndChar + +StartChar: dotlessi +Encoding: 261 305 215 +Width: 532 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 11 + 2 + 12 + 1 + 0 + 0 + 22 + 160 + 1 + 1 + 1 + 4 +SRP0 +MDRP[rp0,min,rnd,white] +DELTAP2 +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +487 327 m 1,0,1 + 483 309 483 309 477.5 264.5 c 128,-1,2 + 472 220 472 220 474 212 c 0,3,4 + 474 197 474 197 484 186.5 c 128,-1,5 + 494 176 494 176 506 176 c 0,6,7 + 556 176 556 176 658 345 c 1,8,-1 + 672 370 l 1,9,-1 + 725 334 l 1,10,11 + 603 125 603 125 524.5 52 c 128,-1,12 + 446 -21 446 -21 341 -21 c 0,13,14 + 304 -21 304 -21 276 -12 c 128,-1,15 + 248 -3 248 -3 229.5 9.5 c 128,-1,16 + 211 22 211 22 199.5 44.5 c 128,-1,17 + 188 67 188 67 181.5 85 c 128,-1,18 + 175 103 175 103 175 133 c 128,-1,19 + 175 163 175 163 175.5 179 c 128,-1,20 + 176 195 176 195 181.5 227 c 128,-1,21 + 187 259 187 259 188.5 266.5 c 128,-1,22 + 190 274 190 274 196 299 c 1,23,-1 + 380 1093 l 1,24,-1 + 664 1093 l 1,25,-1 + 487 327 l 1,0,1 +EndSplineSet +EndChar + +StartChar: circumflex +Encoding: 335 710 216 +Width: 950 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 16 + 2 + 5 + 91 + 0 + 4 + 36 + 52 + 0 + 1 + 0 + 8 + 59 + 4 + 1 + 4 + 7 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +PUSHB_2 + 4 + 0 +MD[grid] +EVEN +IF +PUSHB_2 + 0 + 64 +SHPIX +EIF +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rp0,min,rnd,grey] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1217 1323 m 1,0,-1 + 1015 1323 l 1,1,-1 + 825 1516 l 1,2,-1 + 546 1323 l 1,3,-1 + 345 1323 l 1,4,-1 + 754 1677 l 1,5,-1 + 971 1677 l 1,6,-1 + 1217 1323 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: tilde +Encoding: 342 732 217 +Width: 949 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 36 + 137 + 25 + 153 + 25 + 169 + 25 + 185 + 25 + 199 + 12 + 201 + 25 + 214 + 12 + 7 + 9 + 46 + 25 + 16 + 22 + 46 + 12 + 3 + 37 + 16 + 22 + 9 + 3 + 4 + 0 + 13 + 25 + 0 + 12 + 13 + 26 +SRP0 +MDRP[rp0,rnd,white] +MDRP[min,grey] +MDRP[rp0,min,rnd,grey] +MDRP[rp0,min,grey] +SRP1 +SRP2 +SLOOP +IP +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rp0,min,rnd,grey] +MDRP[rp0,rnd,grey] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[y-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1227 1520 m 1,0,1 + 1185 1456 1185 1456 1102.5 1397.5 c 128,-1,2 + 1020 1339 1020 1339 929 1339 c 0,3,4 + 868 1339 868 1339 840.5 1355.5 c 128,-1,5 + 813 1372 813 1372 776 1397 c 0,6,7 + 740 1421 740 1421 713 1435.5 c 128,-1,8 + 686 1450 686 1450 646 1450 c 256,9,10 + 606 1450 606 1450 571 1435.5 c 128,-1,11 + 536 1421 536 1421 487 1368 c 1,12,-1 + 412 1452 l 1,13,14 + 452 1507 452 1507 530.5 1568.5 c 128,-1,15 + 609 1630 609 1630 722 1630 c 0,16,17 + 773 1630 773 1630 803 1612.5 c 128,-1,18 + 833 1595 833 1595 861 1573 c 0,19,20 + 889 1550 889 1550 918 1534 c 128,-1,21 + 947 1518 947 1518 1000 1518 c 0,22,23 + 1048 1518 1048 1518 1080.5 1543.5 c 128,-1,24 + 1113 1569 1113 1569 1146 1612 c 1,25,-1 + 1227 1520 l 1,0,1 +EndSplineSet +EndChar + +StartChar: macron +Encoding: 133 175 218 +AltUni2: 0002c9.ffffffff.0 0002c9.ffffffff.0 +Width: 946 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_7 + 2 + 46 + 1 + 39 + 0 + 1 + 4 +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1114 1335 m 1,0,-1 + 447 1335 l 1,1,-1 + 489 1518 l 1,2,-1 + 1157 1518 l 1,3,-1 + 1114 1335 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: breve +Encoding: 338 728 219 +Width: 945 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 25 + 8 + 4 + 24 + 4 + 2 + 0 + 6 + 10 + 94 + 167 + 3 + 183 + 3 + 199 + 3 + 3 + 3 + 38 + 13 + 76 + 0 + 7 + 76 + 6 + 14 +SRP0 +MDRP[rp0,rnd,white] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,black] +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +DELTAP1 +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,black] +MDRP[rnd,grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1183 1657 m 1,0,1 + 1149 1509 1149 1509 1022 1413 c 128,-1,2 + 895 1317 895 1317 777 1317 c 0,3,4 + 642 1317 642 1317 568 1413 c 128,-1,5 + 494 1509 494 1509 528 1657 c 1,6,-1 + 684 1657 l 1,7,8 + 673 1579 673 1579 707.5 1530 c 128,-1,9 + 742 1481 742 1481 815 1481 c 256,10,11 + 888 1481 888 1481 944.5 1530 c 128,-1,12 + 1001 1579 1001 1579 1027 1657 c 1,13,-1 + 1183 1657 l 1,0,1 +EndSplineSet +EndChar + +StartChar: dotaccent +Encoding: 339 729 220 +Width: 937 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_8 + 9 + 56 + 3 + 38 + 0 + 55 + 6 + 12 +SRP0 +MDRP[rp0,rnd,white] +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +955 1483 m 0,0,1 + 940 1419 940 1419 887 1374 c 128,-1,2 + 834 1329 834 1329 776 1329 c 256,3,4 + 717 1329 717 1329 685 1374 c 128,-1,5 + 653 1419 653 1419 668 1483 c 0,6,7 + 683 1546 683 1546 735.5 1591 c 128,-1,8 + 788 1636 788 1636 847 1636 c 256,9,10 + 906 1636 906 1636 938 1591 c 128,-1,11 + 970 1546 970 1546 955 1483 c 0,0,1 +EndSplineSet +EndChar + +StartChar: ring +Encoding: 340 730 221 +Width: 942 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 61 + 4 + 1 + 11 + 5 + 11 + 7 + 4 + 11 + 20 + 1 + 27 + 5 + 27 + 7 + 20 + 11 + 37 + 1 + 42 + 5 + 42 + 7 + 37 + 11 + 12 + 9 + 210 + 15 + 21 + 210 + 3 + 38 + 0 + 75 + 112 + 12 + 1 + 48 + 12 + 128 + 12 + 208 + 12 + 3 + 61 + 12 + 144 + 12 + 224 + 12 + 3 + 12 + 18 + 75 + 47 + 6 + 56 + 6 + 2 + 6 + 24 +SRP0 +MDRP[rp0,min,rnd,white] +DELTAP1 +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +DELTAP1 +DELTAP2 +DELTAP3 +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1093 1581 m 256,0,1 + 1067 1468 1067 1468 976.5 1392.5 c 128,-1,2 + 886 1317 886 1317 775 1317 c 256,3,4 + 664 1317 664 1317 609 1392.5 c 128,-1,5 + 554 1468 554 1468 580 1581 c 256,6,7 + 606 1694 606 1694 696.5 1769.5 c 128,-1,8 + 787 1845 787 1845 897 1845 c 256,9,10 + 1008 1845 1008 1845 1063.5 1769.5 c 128,-1,11 + 1119 1694 1119 1694 1093 1581 c 256,0,1 +958 1581 m 256,12,13 + 970 1634 970 1634 943 1673 c 128,-1,14 + 916 1712 916 1712 867 1712 c 256,15,16 + 818 1712 818 1712 772.5 1673 c 128,-1,17 + 727 1634 727 1634 715 1581 c 256,18,19 + 703 1528 703 1528 730 1489 c 128,-1,20 + 757 1450 757 1450 806 1450 c 256,21,22 + 855 1450 855 1450 900.5 1489 c 128,-1,23 + 946 1528 946 1528 958 1581 c 256,12,13 +EndSplineSet +EndChar + +StartChar: cedilla +Encoding: 142 184 222 +Width: 940 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 47 + 35 + 1 + 35 + 22 + 51 + 22 + 67 + 1 + 67 + 22 + 83 + 1 + 83 + 22 + 99 + 1 + 99 + 22 + 9 + 19 + 20 + 17 + 73 + 3 + 7 + 70 + 10 + 210 + 6 + 3 + 0 + 92 + 111 + 13 + 1 + 13 + 19 + 76 + 48 + 6 + 1 + 6 + 63 + 18 + 1 + 18 + 23 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MDRP[rnd,grey] +DELTAP1 +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MDAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +SROUND +MDRP[rnd,grey] +RTG +SRP0 +SROUND +MDRP[rp0,min,rnd,grey] +RTG +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +684 -256 m 0,0,1 + 657 -371 657 -371 564 -431.5 c 128,-1,2 + 471 -492 471 -492 378 -492 c 0,3,4 + 309 -492 309 -492 290.5 -486.5 c 128,-1,5 + 272 -481 272 -481 246 -469 c 1,6,-1 + 276 -336 l 1,7,8 + 298 -350 298 -350 323.5 -356 c 128,-1,9 + 349 -362 349 -362 383 -362 c 0,10,11 + 421 -362 421 -362 453 -337.5 c 128,-1,12 + 485 -313 485 -313 495 -270 c 0,13,14 + 508 -215 508 -215 487.5 -192.5 c 128,-1,15 + 467 -170 467 -170 424 -170 c 2,16,-1 + 336 -170 l 1,17,-1 + 381 25 l 1,18,-1 + 531 25 l 1,19,-1 + 512 -57 l 1,20,21 + 612 -57 612 -57 659 -108.5 c 128,-1,22 + 706 -160 706 -160 684 -256 c 0,0,1 +EndSplineSet +EndChar + +StartChar: hungarumlaut +Encoding: 343 733 223 +Width: 950 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 11 + 3 + 7 + 91 + 2 + 6 + 36 + 0 + 2 + 4 + 6 + 8 +SRP0 +MDRP[rp0,rnd,white] +MDRP[rp0,min,rnd,grey] +MDRP[rp0,rnd,grey] +MDRP[rp0,min,rnd,grey] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1354 1677 m 1,0,-1 + 987 1323 l 1,1,-1 + 804 1323 l 1,2,-1 + 1118 1677 l 1,3,-1 + 1354 1677 l 1,0,-1 +964 1677 m 1,4,-1 + 596 1323 l 1,5,-1 + 414 1323 l 1,6,-1 + 728 1677 l 1,7,-1 + 964 1677 l 1,4,-1 +EndSplineSet +EndChar + +StartChar: ogonek +Encoding: 341 731 224 +Width: 939 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 26 + 10 + 7 + 26 + 7 + 42 + 7 + 58 + 7 + 74 + 7 + 91 + 7 + 6 + 0 + 3 + 18 + 11 + 10 + 0 + 0 + 21 + 11 + 15 + 89 + 6 + 22 +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +MDRP[rp0,rnd,grey] +MDRP[rnd,grey] +SVTCA[y-axis] +MIAP[rnd] +MDRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +MDRP[rp0,min,rnd,black] +MDRP[rnd,grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +499 -471 m 1,0,1 + 456 -483 456 -483 432 -487.5 c 128,-1,2 + 408 -492 408 -492 386 -492 c 0,3,4 + 273 -492 273 -492 219.5 -435.5 c 128,-1,5 + 166 -379 166 -379 197 -246 c 0,6,7 + 223 -135 223 -135 312.5 -67.5 c 128,-1,8 + 402 0 402 0 509 0 c 2,9,-1 + 589 0 l 1,10,-1 + 554 -150 l 1,11,-1 + 474 -150 l 2,12,13 + 430 -150 430 -150 398 -175.5 c 128,-1,14 + 366 -201 366 -201 355 -246 c 0,15,16 + 344 -295 344 -295 366 -318.5 c 128,-1,17 + 388 -342 388 -342 426 -342 c 0,18,19 + 433 -342 433 -342 453 -339 c 128,-1,20 + 473 -336 473 -336 489 -332 c 1,21,-1 + 499 -471 l 1,0,1 +EndSplineSet +EndChar + +StartChar: caron +Encoding: 336 711 225 +Width: 950 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 16 + 0 + 3 + 91 + 5 + 2 + 36 + 52 + 0 + 1 + 0 + 8 + 59 + 3 + 1 + 3 + 7 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +PUSHB_2 + 3 + 0 +MD[grid] +EVEN +IF +PUSHB_2 + 0 + 64 +SHPIX +EIF +SVTCA[y-axis] +MIAP[rnd] +MDRP[min,rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1299 1677 m 1,0,-1 + 890 1323 l 1,1,-1 + 672 1323 l 1,2,-1 + 426 1677 l 1,3,-1 + 628 1677 l 1,4,-1 + 818 1485 l 1,5,-1 + 1097 1677 l 1,6,-1 + 1299 1677 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: Lslash +Encoding: 274 321 226 +Width: 1126 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 48 + 11 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 47 76 N 1 0 0 1 0 0 2 +Refer: 375 -1 N 1 0 0 1 0 0 2 +EndChar + +StartChar: lslash +Encoding: 275 322 227 +Width: 635 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 79 108 N 1 0 0 1 0 0 2 +Refer: 376 -1 N 1 0 0 1 -18 0 2 +EndChar + +StartChar: Scaron +Encoding: 303 352 228 +Width: 1155 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 48 + 41 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 54 83 N 1 0 0 1 0 0 2 +Refer: 225 711 N 1 0 0 1 149 260 2 +EndChar + +StartChar: scaron +Encoding: 304 353 229 +Width: 932 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 18 + 0 + 40 + 16 + 40 + 2 + 0 + 41 + 63 + 41 + 64 + 41 + 3 + 63 + 46 + 1 + 63 + 46 + 1 +DELTAP1 +DELTAP1 +DELTAP1 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 86 115 N 1 0 0 1 0 0 2 +Refer: 225 711 N 1 0 0 1 -44 -75 2 +EndChar + +StartChar: Zcaron +Encoding: 332 381 230 +Width: 1108 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 63 + 13 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 61 90 N 1 0 0 1 0 0 2 +Refer: 225 711 N 1 0 0 1 99 260 2 +EndChar + +StartChar: zcaron +Encoding: 333 382 231 +Width: 1020 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 11 + 0 + 12 + 16 + 12 + 2 + 63 + 13 + 1 + 63 + 18 + 1 +DELTAP1 +DELTAP1 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 93 122 N 1 0 0 1 0 0 2 +Refer: 225 711 N 1 0 0 1 -13 -75 2 +EndChar + +StartChar: brokenbar +Encoding: 124 166 232 +Width: 778 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 14 + 6 + 5 + 1 + 2 + 16 + 7 + 63 + 0 + 1 + 0 + 58 + 5 + 1 + 8 +SRP0 +MDRP[rp0,rnd,white] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +DELTAP1 +MDRP[rnd,grey] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rp0,min,rnd,black] +MDAP[rnd] +MDRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +671 690 m 1,0,-1 + 431 690 l 1,1,-1 + 620 1509 l 1,2,-1 + 860 1509 l 1,3,-1 + 671 690 l 1,0,-1 +418 -408 m 1,4,-1 + 178 -408 l 1,5,-1 + 367 412 l 1,6,-1 + 607 412 l 1,7,-1 + 418 -408 l 1,4,-1 +EndSplineSet +EndChar + +StartChar: Eth +Encoding: 166 208 233 +Width: 1360 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 377 -1 N 1 0 0 1 0 0 2 +Refer: 39 68 N 1 0 0 1 0 0 2 +EndChar + +StartChar: eth +Encoding: 198 240 234 +Width: 1047 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 56 + 52 + 31 + 1 + 17 + 16 + 18 + 15 + 31 + 34 + 33 + 7 + 9 + 32 + 43 + 31 + 12 + 9 + 9 + 3 + 28 + 24 + 21 + 34 + 25 + 32 + 28 + 17 + 49 + 33 + 3 + 3 + 61 + 31 + 63 + 33 + 2 + 17 + 16 + 18 + 15 + 31 + 34 + 32 + 33 + 8 + 0 + 24 + 0 + 23 + 12 + 40 + 46 + 23 + 24 + 6 + 52 +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rnd,grey] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +MDRP[grey] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +SLOOP +IP +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MDRP[grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP1 +SRP2 +IP +MDAP[rnd] +MDRP[grey] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +SLOOP +IP +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1109 573 m 0,0,1 + 1030 287 1030 287 868.5 131 c 128,-1,2 + 707 -25 707 -25 471 -25 c 0,3,4 + 275 -25 275 -25 199.5 137 c 128,-1,5 + 124 299 124 299 176 528 c 0,6,7 + 233 776 233 776 376.5 923.5 c 128,-1,8 + 520 1071 520 1071 720 1071 c 0,9,10 + 788 1071 788 1071 838 1032 c 128,-1,11 + 888 993 888 993 907 954 c 1,12,13 + 920 1034 920 1034 916 1099.5 c 128,-1,14 + 912 1165 912 1165 891 1212 c 1,15,-1 + 780 1104 l 1,16,-1 + 713 1206 l 1,17,-1 + 812 1303 l 1,18,19 + 793 1315 793 1315 767 1320 c 128,-1,20 + 741 1325 741 1325 710 1325 c 0,21,22 + 659 1325 659 1325 609 1311.5 c 128,-1,23 + 559 1298 559 1298 497 1268 c 1,24,-1 + 550 1499 l 1,25,26 + 592 1518 592 1518 643 1532 c 128,-1,27 + 694 1546 694 1546 760 1546 c 0,28,29 + 833 1546 833 1546 894.5 1528.5 c 128,-1,30 + 956 1511 956 1511 993 1479 c 1,31,-1 + 1122 1599 l 1,32,-1 + 1207 1509 l 1,33,-1 + 1079 1391 l 1,34,35 + 1150 1286 1150 1286 1166 1107 c 128,-1,36 + 1182 928 1182 928 1139 705 c 0,37,38 + 1136 692 1136 692 1125.5 646 c 128,-1,39 + 1115 600 1115 600 1109 573 c 0,0,1 +802 532 m 0,40,41 + 830 655 830 655 814 755.5 c 128,-1,42 + 798 856 798 856 709 856 c 0,43,44 + 635 856 635 856 576.5 787.5 c 128,-1,45 + 518 719 518 719 474 528 c 0,46,47 + 437 369 437 369 451 266.5 c 128,-1,48 + 465 164 465 164 550 164 c 0,49,50 + 634 164 634 164 699 266.5 c 128,-1,51 + 764 369 764 369 802 532 c 0,40,41 +EndSplineSet +EndChar + +StartChar: Yacute +Encoding: 179 221 235 +Width: 1327 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 60 89 N 1 0 0 1 0 0 2 +Refer: 141 180 N 1 0 0 1 311 260 2 +EndChar + +StartChar: yacute +Encoding: 211 253 236 +Width: 1190 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_5 + 0 + 10 + 16 + 10 + 2 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 92 121 N 1 0 0 1 0 0 2 +Refer: 141 180 N 1 0 0 1 151 -71 2 +EndChar + +StartChar: Thorn +Encoding: 180 222 237 +Width: 1104 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 40 + 170 + 17 + 1 + 0 + 19 + 31 + 9 + 20 + 31 + 4 + 9 + 4 + 6 + 7 + 7 + 10 + 6 + 0 + 151 + 0 + 167 + 0 + 246 + 0 + 3 + 0 + 21 + 32 + 14 + 80 + 14 + 2 + 14 + 6 + 9 + 19 + 4 + 20 + 6 + 24 +SRP0 +MDRP[rp0,rnd,white] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[min,rnd,black] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +SRP1 +SRP2 +IP +IP +MDAP[rnd] +MIRP[rp0,min,rnd,black] +MDAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +DELTAP1 +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1227 705 m 2,0,1 + 1186 528 1186 528 1042 421.5 c 128,-1,2 + 898 315 898 315 708 315 c 2,3,-1 + 523 315 l 1,4,-1 + 450 0 l 1,5,-1 + 134 0 l 1,6,-1 + 482 1509 l 1,7,-1 + 798 1509 l 1,8,-1 + 726 1194 l 1,9,-1 + 886 1194 l 2,10,11 + 1102 1194 1102 1194 1197 1090.5 c 128,-1,12 + 1292 987 1292 987 1241 768 c 2,13,-1 + 1227 705 l 2,0,1 +914 713 m 2,14,-1 + 929 776 l 2,15,16 + 952 877 952 877 915 936 c 128,-1,17 + 878 995 878 995 752 995 c 2,18,-1 + 680 995 l 1,19,-1 + 570 522 l 1,20,-1 + 643 522 l 2,21,22 + 750 522 750 522 820 565 c 128,-1,23 + 890 608 890 608 914 713 c 2,14,-1 +EndSplineSet +EndChar + +StartChar: thorn +Encoding: 212 254 238 +Width: 1145 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 49 + 6 + 16 + 22 + 16 + 122 + 1 + 3 + 9 + 16 + 26 + 25 + 6 + 3 + 3 + 14 + 64 + 11 + 96 + 14 + 22 + 47 + 14 + 13 + 29 + 32 + 3 + 3 + 8 + 18 + 96 + 33 + 1 + 0 + 23 + 16 + 18 + 64 + 18 + 112 + 18 + 3 + 18 + 8 + 11 + 26 + 6 + 22 + 8 + 32 +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MIRP[min,rnd,black] +SRP0 +SMD +MDRP[min,grey] +SMD +SRP1 +SRP2 +SLOOP +IP +MIAP[rnd] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1170 520 m 2,0,1 + 1111 265 1111 265 969 124.5 c 128,-1,2 + 827 -16 827 -16 623 -16 c 0,3,4 + 572 -16 572 -16 515.5 0 c 128,-1,5 + 459 16 459 16 427 59 c 1,6,-1 + 335 -342 l 1,7,-1 + 51 -342 l 1,8,-1 + 479 1511 l 1,9,-1 + 762 1511 l 1,10,-1 + 629 933 l 1,11,12 + 715 1042 715 1042 796 1079 c 128,-1,13 + 877 1116 877 1116 955 1116 c 0,14,15 + 1122 1116 1122 1116 1185.5 989 c 128,-1,16 + 1249 862 1249 862 1199 644 c 2,17,-1 + 1170 520 l 2,0,1 +880 533 m 2,18,-1 + 896 602 l 2,19,20 + 936 775 936 775 902 832.5 c 128,-1,21 + 868 890 868 890 806 890 c 0,22,23 + 741 890 741 890 681.5 853 c 128,-1,24 + 622 816 622 816 590 765 c 1,25,-1 + 473 258 l 1,26,27 + 471 222 471 222 509 188 c 128,-1,28 + 547 154 547 154 616 154 c 0,29,30 + 684 154 684 154 754 222 c 128,-1,31 + 824 290 824 290 880 533 c 2,18,-1 +EndSplineSet +EndChar + +StartChar: minus +Encoding: 390 8722 239 +Width: 1188 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 10 + 1 + 68 + 2 + 70 + 48 + 0 + 1 + 0 + 1 + 4 +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rp0,min,rnd,black] +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1085 569 m 1,0,-1 + 104 569 l 1,1,-1 + 104 807 l 1,2,-1 + 1085 807 l 1,3,-1 + 1085 569 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: multiply +Encoding: 173 215 240 +Width: 1096 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 11 + 80 + 2 + 1 + 2 + 8 + 11 + 5 + 11 + 5 + 2 + 8 +RTHG +MDAP[rnd] +ALIGNRP +MDRP[rnd,grey] +MDRP[rnd,grey] +SVTCA[y-axis] +MDAP[rnd] +ALIGNRP +MDRP[rnd,grey] +MDRP[rnd,grey] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1019 438 m 1,0,-1 + 850 293 l 1,1,-1 + 674 545 l 1,2,-1 + 382 293 l 1,3,-1 + 280 438 l 1,4,-1 + 573 690 l 1,5,-1 + 397 942 l 1,6,-1 + 565 1087 l 1,7,-1 + 741 836 l 1,8,-1 + 1034 1087 l 1,9,-1 + 1135 942 l 1,10,-1 + 843 690 l 1,11,-1 + 1019 438 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: onesuperior +Encoding: 143 185 241 +Width: 697 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 18 + 3 + 210 + 4 + 7 + 14 + 1 + 81 + 0 + 99 + 7 + 3 + 144 + 1 + 1 + 1 + 1 + 10 + 9 +SRP1 +SRP2 +IP +MDAP[rnd] +DELTAP1 +MDRP[min,rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +MDRP[rp0,min,rnd,grey] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +615 590 m 1,0,-1 + 404 590 l 1,1,-1 + 557 1251 l 1,2,-1 + 406 1251 l 1,3,-1 + 436 1380 l 1,4,5 + 552 1393 552 1393 608.5 1438 c 128,-1,6 + 665 1483 665 1483 677 1511 c 1,7,-1 + 828 1511 l 1,8,-1 + 615 590 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: twosuperior +Encoding: 136 178 242 +Width: 849 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 12 + 51 + 17 + 70 + 16 + 102 + 20 + 136 + 4 + 153 + 4 + 5 + 2 +PUSHW_1 + -16 +NPUSHB + 20 + 12 + 8 + 111 + 15 + 15 + 2 + 21 + 111 + 1 + 81 + 21 + 0 + 18 + 99 + 5 + 2 + 11 + 12 + 2 + 23 +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rp0,rnd,grey] +MDRP[rp0,min,rnd,black] +SRP0 +MDRP[rp0,min,rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[grey] +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MDRP[grey] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MDRP[rp0,rnd,grey] +SHPIX +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +886 590 m 1,0,-1 + 267 590 l 1,1,-1 + 308 770 l 1,2,3 + 574 989 574 989 686.5 1075 c 128,-1,4 + 799 1161 799 1161 818 1247 c 0,5,6 + 831 1305 831 1305 809.5 1329.5 c 128,-1,7 + 788 1354 788 1354 737 1354 c 0,8,9 + 674 1354 674 1354 641.5 1325 c 128,-1,10 + 609 1296 609 1296 595 1268 c 1,11,-1 + 403 1268 l 1,12,13 + 452 1391 452 1391 563 1459.5 c 128,-1,14 + 674 1528 674 1528 790 1528 c 0,15,16 + 952 1528 952 1528 1007.5 1455 c 128,-1,17 + 1063 1382 1063 1382 1038 1270 c 0,18,19 + 1004 1124 1004 1124 793 962.5 c 128,-1,20 + 582 801 582 801 514 760 c 1,21,-1 + 925 760 l 1,22,-1 + 886 590 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: threesuperior +Encoding: 137 179 243 +Width: 849 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 72 + 36 + 1 + 36 + 34 + 51 + 1 + 51 + 34 + 68 + 34 + 86 + 34 + 169 + 31 + 7 + 38 + 17 + 75 + 15 + 18 + 31 + 18 + 2 + 18 + 73 + 18 + 3 + 32 + 29 + 71 + 25 + 94 + 32 + 15 + 64 + 7 + 1 + 7 + 71 + 10 + 94 + 6 + 3 + 82 + 36 + 38 + 52 + 38 + 2 + 38 + 18 + 18 + 0 + 6 + 35 + 93 + 95 + 22 + 1 + 22 + 0 + 99 + 48 + 13 + 1 + 13 + 6 + 28 + 29 + 7 + 6 + 41 +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[grey] +MDRP[rp0,rnd,grey] +MDRP[rp0,min,rnd,black] +SRP0 +MDRP[rp0,min,rnd,black] +DELTAP1 +MIRP[min,rnd,black] +MDRP[rp0,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +MDAP[rnd] +IP +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MDRP[grey] +MIRP[rp0,min,rnd,black] +SROUND +MDRP[rnd,grey] +RTG +DELTAP1 +MIAP[rnd] +MIRP[rp0,min,rnd,black] +SROUND +MDRP[rnd,grey] +RTG +SRP1 +SRP2 +IP +SROUND +MDAP[rnd] +RTG +DELTAP1 +MIRP[rp0,min,rnd,black] +IP +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +946 856 m 0,0,1 + 917 731 917 731 812.5 650 c 128,-1,2 + 708 569 708 569 559 569 c 0,3,4 + 468 569 468 569 394 608 c 128,-1,5 + 320 647 320 647 283 733 c 1,6,-1 + 434 827 l 1,7,8 + 461 772 461 772 498.5 748.5 c 128,-1,9 + 536 725 536 725 567 725 c 0,10,11 + 643 725 643 725 685.5 761 c 128,-1,12 + 728 797 728 797 744 866 c 0,13,14 + 762 942 762 942 718.5 965.5 c 128,-1,15 + 675 989 675 989 622 989 c 2,16,-1 + 535 989 l 1,17,-1 + 568 1133 l 1,18,-1 + 661 1133 l 2,19,20 + 735 1133 735 1133 775.5 1163.5 c 128,-1,21 + 816 1194 816 1194 830 1253 c 0,22,23 + 842 1305 842 1305 817 1334.5 c 128,-1,24 + 792 1364 792 1364 741 1364 c 0,25,26 + 676 1364 676 1364 644 1335.5 c 128,-1,27 + 612 1307 612 1307 592 1268 c 1,28,-1 + 406 1268 l 1,29,30 + 459 1395 459 1395 565.5 1461.5 c 128,-1,31 + 672 1528 672 1528 786 1528 c 0,32,33 + 929 1528 929 1528 987.5 1460.5 c 128,-1,34 + 1046 1393 1046 1393 1022 1286 c 0,35,36 + 1001 1194 1001 1194 955 1146 c 128,-1,37 + 909 1098 909 1098 861 1075 c 1,38,39 + 924 1044 924 1044 944 989 c 128,-1,40 + 964 934 964 934 946 856 c 0,0,1 +EndSplineSet +EndChar + +StartChar: onehalf +Encoding: 147 189 244 +Width: 1647 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 48 + 1 + 1 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 242 178 N 1 0 0 1 685 -591 2 +Refer: 241 185 N 1 0 0 1 -67 0 2 +Refer: 188 8260 N 1 0 0 1 476 0 2 +EndChar + +StartChar: onequarter +Encoding: 146 188 245 +Width: 1647 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_5 + 48 + 3 + 48 + 13 + 2 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 374 8308 N 1 0 0 1 638 -590 2 +Refer: 241 185 N 1 0 0 1 -67 0 2 +Refer: 188 8260 N 1 0 0 1 518 0 2 +EndChar + +StartChar: threequarters +Encoding: 148 190 246 +Width: 1764 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 10 + 48 + 3 + 48 + 13 + 2 + 63 + 20 + 48 + 57 + 2 +SVTCA[x-axis] +DELTAP1 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 374 8308 N 1 0 0 1 738 -593 2 +Refer: 243 179 N 1 0 0 1 -108 0 2 +Refer: 188 8260 N 1 0 0 1 628 0 2 +EndChar + +StartChar: franc +Encoding: 366 8355 247 +Width: 1110 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 9 + 80 + 12 + 1 + 0 + 12 + 1 + 48 + 15 + 1 +DELTAP1 +SVTCA[y-axis] +DELTAP2 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 41 70 N 1 0 0 1 0 0 2 +Refer: 377 -1 N 1 0 0 1 -83 -360 2 +EndChar + +StartChar: Gbreve +Encoding: 244 286 248 +Width: 1360 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 63 + 38 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 42 71 N 1 0 0 1 0 0 2 +Refer: 219 728 N 1 0 0 1 283 254 2 +EndChar + +StartChar: gbreve +Encoding: 245 287 249 +Width: 1337 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 9 + 0 + 47 + 16 + 47 + 32 + 47 + 48 + 47 + 4 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 74 103 N 1 0 0 1 0 0 2 +Refer: 219 728 N 1 0 0 1 113 -71 2 +EndChar + +StartChar: Idotaccent +Encoding: 260 304 250 +Width: 750 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 9 + 48 + 7 + 1 + 63 + 17 + 1 + 63 + 10 + 1 +DELTAP1 +DELTAP1 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 44 73 N 1 0 0 1 0 0 2 +Refer: 220 729 N 1 0 0 1 -73 256 2 +EndChar + +StartChar: Scedilla +Encoding: 301 350 251 +Width: 1155 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 54 83 N 1 0 0 1 0 0 2 +Refer: 222 184 N 1 0 0 1 23 -3 2 +EndChar + +StartChar: scedilla +Encoding: 302 351 252 +Width: 932 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 11 + 0 + 44 + 63 + 44 + 2 + 63 + 62 + 1 + 63 + 62 + 1 +DELTAP1 +DELTAP1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 86 115 N 1 0 0 1 0 0 2 +Refer: 222 184 N 1 0 0 1 -117 9 2 +EndChar + +StartChar: Cacute +Encoding: 220 262 253 +Width: 1190 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 63 + 30 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 38 67 N 1 0 0 1 0 0 2 +Refer: 141 180 N 1 0 0 1 303 260 2 +EndChar + +StartChar: cacute +Encoding: 221 263 254 +Width: 1124 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_5 + 0 + 30 + 16 + 30 + 2 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 70 99 N 1 0 0 1 0 0 2 +Refer: 141 180 N 1 0 0 1 132 -71 2 +EndChar + +StartChar: Ccaron +Encoding: 226 268 255 +Width: 1190 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 63 + 31 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 38 67 N 1 0 0 1 0 0 2 +Refer: 225 711 N 1 0 0 1 201 260 2 +EndChar + +StartChar: ccaron +Encoding: 227 269 256 +Width: 1124 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_8 + 0 + 30 + 16 + 30 + 2 + 63 + 31 + 1 +DELTAP1 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 70 99 N 1 0 0 1 0 0 2 +Refer: 225 711 N 1 0 0 1 66 -71 2 +EndChar + +StartChar: dcroat +Encoding: 231 273 257 +Width: 1298 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 17 + 63 + 34 + 111 + 34 + 143 + 34 + 175 + 34 + 224 + 34 + 5 + 96 + 37 + 1 + 224 + 34 + 1 +DELTAP1 +DELTAP1 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 71 100 N 1 0 0 1 0 0 2 +Refer: 377 -1 N 1 0 0 1 719 579 2 +EndChar + +StartChar: Amacron +Encoding: 214 256 258 +Width: 1358 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 36 65 N 1 0 0 1 0 0 2 +Refer: 218 175 N 1 0 0 1 204 248 2 +EndChar + +StartChar: amacron +Encoding: 215 257 259 +Width: 1238 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 19 + 0 + 45 + 16 + 45 + 2 + 16 + 49 + 63 + 49 + 192 + 49 + 240 + 49 + 4 + 0 + 45 + 16 + 45 + 2 +DELTAP1 +DELTAP1 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 68 97 N 1 0 0 1 0 0 2 +Refer: 218 175 N 1 0 0 1 99 -65 2 +EndChar + +StartChar: Abreve +Encoding: 216 258 260 +Width: 1358 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 36 65 N 1 0 0 1 0 0 2 +Refer: 219 728 N 1 0 0 1 208 247 2 +EndChar + +StartChar: abreve +Encoding: 217 259 261 +Width: 1238 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 23 + 0 + 47 + 16 + 47 + 32 + 47 + 48 + 47 + 4 + 0 + 50 + 16 + 50 + 2 + 16 + 59 + 63 + 59 + 192 + 59 + 240 + 59 + 4 +DELTAP1 +DELTAP1 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 68 97 N 1 0 0 1 0 0 2 +Refer: 219 728 N 1 0 0 1 101 -72 2 +EndChar + +StartChar: Aogonek +Encoding: 218 260 262 +Width: 1358 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 63 + 17 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 36 65 N 1 0 0 1 0 0 2 +Refer: 224 731 N 1 0 0 1 421 149 2 +EndChar + +StartChar: aogonek +Encoding: 219 261 263 +Width: 1238 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 12 + 47 + 50 + 1 + 16 + 67 + 63 + 67 + 192 + 67 + 240 + 67 + 4 +DELTAP1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 68 97 N 1 0 0 1 0 0 2 +Refer: 224 731 N 1 0 0 1 481 148 2 +EndChar + +StartChar: Dcaron +Encoding: 228 270 264 +Width: 1360 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 39 68 N 1 0 0 1 0 0 2 +Refer: 225 711 N 1 0 0 1 147 261 2 +EndChar + +StartChar: dcaron +Encoding: 229 271 265 +Width: 1546 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_6 + 63 + 44 + 1 + 63 + 41 + 1 +DELTAP1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 71 100 N 1 0 0 1 0 0 2 +Refer: 183 8217 N 1 0 0 1 1029 0 2 +EndChar + +StartChar: Dcroat +Encoding: 230 272 266 +Width: 1360 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 377 -1 N 1 0 0 1 0 0 2 +Refer: 39 68 N 1 0 0 1 0 0 2 +EndChar + +StartChar: Emacron +Encoding: 232 274 267 +Width: 1167 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 40 69 N 1 0 0 1 0 0 2 +Refer: 218 175 N 1 0 0 1 163 250 2 +EndChar + +StartChar: emacron +Encoding: 233 275 268 +Width: 1303 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_5 + 0 + 32 + 16 + 32 + 2 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 72 101 N 1 0 0 1 0 0 2 +Refer: 218 175 N 1 0 0 1 144 -54 2 +EndChar + +StartChar: Edotaccent +Encoding: 236 278 269 +Width: 1167 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 40 69 N 1 0 0 1 0 0 2 +Refer: 220 729 N 1 0 0 1 164 231 2 +EndChar + +StartChar: edotaccent +Encoding: 237 279 270 +Width: 1303 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_5 + 0 + 34 + 16 + 34 + 2 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 72 101 N 1 0 0 1 0 0 2 +Refer: 220 729 N 1 0 0 1 111 -90 2 +EndChar + +StartChar: Eogonek +Encoding: 238 280 271 +Width: 1167 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 63 + 18 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 40 69 N 1 0 0 1 0 0 2 +Refer: 224 731 N 1 0 0 1 399 151 2 +EndChar + +StartChar: eogonek +Encoding: 239 281 272 +Width: 1303 +GlyphClass: 1 +Flags: W +TtInstrs: +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 72 101 N 1 0 0 1 0 0 2 +Refer: 224 731 N 1 0 0 1 170 129 2 +EndChar + +StartChar: Ecaron +Encoding: 240 282 273 +Width: 1167 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 40 69 N 1 0 0 1 0 0 2 +Refer: 225 711 N 1 0 0 1 158 260 2 +EndChar + +StartChar: ecaron +Encoding: 241 283 274 +Width: 1303 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_5 + 0 + 32 + 16 + 32 + 2 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 72 101 N 1 0 0 1 0 0 2 +Refer: 225 711 N 1 0 0 1 118 -71 2 +EndChar + +StartChar: Gcedilla +Encoding: 248 290 275 +Width: 1360 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 0 + 38 + 1 +SVTCA[x-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 42 71 N 1 0 0 1 0 0 2 +Refer: 222 184 N 1 0 0 1 240 -6 2 +EndChar + +StartChar: gcedilla +Encoding: 249 291 276 +Width: 1337 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 33 + 15 + 45 + 31 + 45 + 111 + 45 + 127 + 45 + 143 + 45 + 207 + 45 + 239 + 45 + 255 + 45 + 8 + 47 + 45 + 63 + 45 + 64 + 45 + 127 + 45 + 143 + 45 + 223 + 45 + 6 + 15 + 45 + 1 +SVTCA[y-axis] +DELTAP2 +DELTAP1 +SVTCA[x-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 74 103 N 1 0 0 1 0 0 2 +Refer: 182 8216 N 1 0 0 1 382 414 2 +EndChar + +StartChar: Imacron +Encoding: 256 298 277 +Width: 750 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_6 + 63 + 6 + 1 + 63 + 9 + 1 +DELTAP1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 44 73 N 1 0 0 1 0 0 2 +Refer: 218 175 N 1 0 0 1 -78 250 2 +EndChar + +StartChar: imacron +Encoding: 257 299 278 +Width: 610 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_5 + 0 + 5 + 16 + 5 + 2 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 215 305 N 1 0 0 1 0 0 2 +Refer: 218 175 N 1 0 0 1 -228 -53 2 +EndChar + +StartChar: logonek +Encoding: 410 -1 279 +Width: 750 +GlyphClass: 1 +Flags: HW +TtInstrs: +PUSHB_3 + 63 + 27 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 44 73 N 1 0 0 1 0 0 2 +Refer: 224 731 N 1 0 0 1 -94 150 2 +EndChar + +StartChar: iogonek +Encoding: 259 303 280 +Width: 610 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 48 + 22 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 76 105 N 1 0 0 1 0 0 2 +Refer: 224 731 N 1 0 0 1 -139 150 2 +EndChar + +StartChar: IJ +Encoding: 262 306 281 +Width: 1536 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 44 73 N 1 0 0 1 0 0 2 +Refer: 45 74 N 1 0 0 1 562 0 2 +EndChar + +StartChar: ij +Encoding: 263 307 282 +Width: 1249 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_5 + 0 + 22 + 0 + 41 + 2 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 76 105 N 1 0 0 1 0 0 2 +Refer: 77 106 N 1 0 0 1 570 0 2 +EndChar + +StartChar: Inodot +Encoding: 411 -1 283 +Width: 750 +GlyphClass: 1 +Flags: HW +TtInstrs: +PUSHB_3 + 63 + 5 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 44 73 N 1 0 0 1 0 0 2 +EndChar + +StartChar: idot +Encoding: 412 -1 284 +Width: 610 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 76 105 N 1 0 0 1 0 0 2 +EndChar + +StartChar: Kcedilla +Encoding: 266 310 285 +Width: 1339 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_7 + 0 + 18 + 16 + 18 + 48 + 18 + 3 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 46 75 N 1 0 0 1 0 0 2 +Refer: 222 184 N 1 0 0 1 -83 15 2 +EndChar + +StartChar: kcedilla +Encoding: 267 311 286 +Width: 1190 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_5 + 0 + 17 + 16 + 17 + 2 +SVTCA[x-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 78 107 N 1 0 0 1 0 0 2 +Refer: 222 184 N 1 0 0 1 -92 53 2 +EndChar + +StartChar: Lacute +Encoding: 268 313 287 +Width: 1126 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 48 + 11 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 47 76 N 1 0 0 1 0 0 2 +Refer: 141 180 N 1 0 0 1 81 260 2 +EndChar + +StartChar: lacute +Encoding: 269 314 288 +Width: 635 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 79 108 N 1 0 0 1 0 0 2 +Refer: 141 180 N 1 0 0 1 -66 261 2 +EndChar + +StartChar: Lcedilla +Encoding: 270 315 289 +Width: 1126 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 48 + 30 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 47 76 N 1 0 0 1 0 0 2 +Refer: 222 184 N 1 0 0 1 -144 15 2 +EndChar + +StartChar: lcedilla +Encoding: 271 316 290 +Width: 635 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 79 108 N 1 0 0 1 0 0 2 +Refer: 222 184 N 1 0 0 1 -184 9 2 +EndChar + +StartChar: Lcaron +Encoding: 272 317 291 +Width: 1126 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 48 + 18 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 47 76 N 1 0 0 1 0 0 2 +Refer: 183 8217 N 1 0 0 1 441 0 2 +EndChar + +StartChar: lcaron +Encoding: 273 318 292 +Width: 946 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 63 + 13 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 79 108 N 1 0 0 1 0 0 2 +Refer: 183 8217 N 1 0 0 1 378 0 2 +EndChar + +StartChar: Ldotaccent +Encoding: 413 -1 293 +Width: 1126 +GlyphClass: 1 +Flags: HW +TtInstrs: +PUSHB_3 + 48 + 19 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 47 76 N 1 0 0 1 0 0 2 +Refer: 220 729 N 1 0 0 1 297 -726 2 +EndChar + +StartChar: ldotaccent +Encoding: 414 -1 294 +Width: 940 +GlyphClass: 1 +Flags: HW +TtInstrs: +PUSHB_3 + 63 + 10 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 79 108 N 1 0 0 1 0 0 2 +Refer: 220 729 N 1 0 0 1 204 -777 2 +EndChar + +StartChar: Nacute +Encoding: 276 323 295 +Width: 1397 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 48 + 12 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 49 78 N 1 0 0 1 0 0 2 +Refer: 141 180 N 1 0 0 1 337 260 2 +EndChar + +StartChar: nacute +Encoding: 277 324 296 +Width: 1296 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_5 + 0 + 22 + 16 + 22 + 2 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 81 110 N 1 0 0 1 0 0 2 +Refer: 141 180 N 1 0 0 1 170 -71 2 +EndChar + +StartChar: Ncedilla +Encoding: 278 325 297 +Width: 1397 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 48 + 15 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 49 78 N 1 0 0 1 0 0 2 +Refer: 222 184 N 1 0 0 1 -242 2 2 +EndChar + +StartChar: ncedilla +Encoding: 279 326 298 +Width: 1296 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 81 110 N 1 0 0 1 0 0 2 +Refer: 222 184 N 1 0 0 1 -185 11 2 +EndChar + +StartChar: Ncaron +Encoding: 280 327 299 +Width: 1397 +GlyphClass: 1 +Flags: W +TtInstrs: +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 49 78 N 1 0 0 1 0 0 2 +Refer: 225 711 N 1 0 0 1 246 260 2 +EndChar + +StartChar: ncaron +Encoding: 281 328 300 +Width: 1296 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_5 + 0 + 20 + 16 + 20 + 2 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 81 110 N 1 0 0 1 0 0 2 +Refer: 225 711 N 1 0 0 1 121 -72 2 +EndChar + +StartChar: napostrophe +Encoding: 282 329 301 +Width: 1296 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 81 110 N 1 0 0 1 0 0 2 +Refer: 183 8217 N 1 0 0 1 372 332 2 +EndChar + +StartChar: Omacron +Encoding: 283 332 302 +Width: 1365 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_6 + 63 + 30 + 1 + 63 + 33 + 1 +DELTAP1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 50 79 N 1 0 0 1 0 0 3 +Refer: 218 175 N 1 0 0 1 273 267 2 +EndChar + +StartChar: omacron +Encoding: 284 333 303 +Width: 1360 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_5 + 0 + 29 + 16 + 29 + 2 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 82 111 N 1 0 0 1 0 0 2 +Refer: 218 175 N 1 0 0 1 177 -65 2 +EndChar + +StartChar: Ohungarumlaut +Encoding: 287 336 304 +Width: 1365 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 63 + 37 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 50 79 N 1 0 0 1 0 0 3 +Refer: 223 733 N 1 0 0 1 310 260 2 +EndChar + +StartChar: ohungarumlaut +Encoding: 288 337 305 +Width: 1360 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 14 + 0 + 30 + 0 + 34 + 16 + 30 + 16 + 34 + 4 + 48 + 30 + 48 + 34 + 2 +DELTAP1 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 82 111 N 1 0 0 1 0 0 2 +Refer: 223 733 N 1 0 0 1 177 -71 2 +EndChar + +StartChar: Racute +Encoding: 291 340 306 +Width: 1251 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 53 82 N 1 0 0 1 0 0 2 +Refer: 141 180 N 1 0 0 1 221 260 2 +EndChar + +StartChar: racute +Encoding: 292 341 307 +Width: 1020 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 85 114 N 1 0 0 1 0 0 2 +Refer: 141 180 N 1 0 0 1 42 -71 2 +EndChar + +StartChar: Rcedilla +Encoding: 293 342 308 +Width: 1251 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 53 82 N 1 0 0 1 0 0 2 +Refer: 222 184 N 1 0 0 1 -175 -1 2 +EndChar + +StartChar: rcedilla +Encoding: 294 343 309 +Width: 1020 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 85 114 N 1 0 0 1 0 0 2 +Refer: 222 184 N 1 0 0 1 -187 23 2 +EndChar + +StartChar: Rcircumflex +Encoding: 415 -1 310 +Width: 1251 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 53 82 N 1 0 0 1 0 0 2 +Refer: 216 710 N 1 0 0 1 107 260 2 +EndChar + +StartChar: rcircumflex +Encoding: 416 -1 311 +Width: 1020 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 85 114 N 1 0 0 1 0 0 2 +Refer: 216 710 N 1 0 0 1 49 -71 2 +EndChar + +StartChar: Sacute +Encoding: 297 346 312 +Width: 1155 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 54 83 N 1 0 0 1 0 0 2 +Refer: 141 180 N 1 0 0 1 183 260 2 +EndChar + +StartChar: sacute +Encoding: 298 347 313 +Width: 932 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 14 + 0 + 40 + 16 + 40 + 2 + 63 + 40 + 1 + 63 + 43 + 1 + 63 + 43 + 1 +DELTAP1 +DELTAP1 +DELTAP1 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 86 115 N 1 0 0 1 0 0 2 +Refer: 141 180 N 1 0 0 1 24 -72 2 +EndChar + +StartChar: Tcedilla +Encoding: 305 354 314 +Width: 1024 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 55 84 N 1 0 0 1 0 0 2 +Refer: 222 184 N 1 0 0 1 20 17 2 +EndChar + +StartChar: tcedilla +Encoding: 306 355 315 +Width: 850 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_5 + 0 + 30 + 16 + 30 + 2 +SVTCA[x-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 87 116 N 1 0 0 1 0 0 2 +Refer: 222 184 N 1 0 0 1 -10 13 2 +EndChar + +StartChar: Tcaron +Encoding: 307 356 316 +Width: 1024 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 55 84 N 1 0 0 1 0 0 2 +Refer: 225 711 N 1 0 0 1 82 260 2 +EndChar + +StartChar: tcaronaltone +Encoding: 417 -1 317 +Width: 850 +GlyphClass: 1 +Flags: HW +TtInstrs: +PUSHB_3 + 63 + 26 + 1 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 87 116 N 1 0 0 1 0 0 2 +Refer: 225 711 N 1 0 0 1 -90 251 2 +EndChar + +StartChar: Umacron +Encoding: 313 362 318 +Width: 1343 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 63 + 20 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 56 85 N 1 0 0 1 0 0 2 +Refer: 218 175 N 1 0 0 1 267 266 2 +EndChar + +StartChar: umacron +Encoding: 314 363 319 +Width: 1296 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_5 + 0 + 20 + 16 + 20 + 2 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 88 117 N 1 0 0 1 0 0 2 +Refer: 218 175 N 1 0 0 1 108 -56 2 +EndChar + +StartChar: Uring +Encoding: 317 366 320 +Width: 1343 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 31 + 223 + 24 + 223 + 37 + 2 + 144 + 24 + 144 + 37 + 2 + 63 + 24 + 63 + 37 + 80 + 24 + 80 + 37 + 96 + 24 + 96 + 37 + 208 + 24 + 208 + 37 + 224 + 24 + 224 + 37 + 10 +DELTAP1 +DELTAP3 +DELTAP2 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 56 85 N 1 0 0 1 0 0 2 +Refer: 221 730 N 1 0 0 1 276 223 2 +EndChar + +StartChar: uring +Encoding: 318 367 321 +Width: 1296 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 22 + 0 + 23 + 0 + 41 + 16 + 23 + 16 + 41 + 4 + 191 + 20 + 191 + 38 + 2 + 207 + 20 + 207 + 38 + 2 + 96 + 45 + 1 +DELTAP1 +DELTAP1 +DELTAP2 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 88 117 N 1 0 0 1 0 0 2 +Refer: 221 730 N 1 0 0 1 117 -89 2 +EndChar + +StartChar: Uhungarumlaut +Encoding: 319 368 322 +Width: 1343 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 56 85 N 1 0 0 1 0 0 2 +Refer: 223 733 N 1 0 0 1 285 260 2 +EndChar + +StartChar: uhungarumlaut +Encoding: 320 369 323 +Width: 1296 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 12 + 0 + 22 + 0 + 26 + 16 + 22 + 16 + 26 + 4 + 96 + 29 + 1 +DELTAP1 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 88 117 N 1 0 0 1 0 0 2 +Refer: 223 733 N 1 0 0 1 90 -71 2 +EndChar + +StartChar: Ucedilla +Encoding: 418 -1 324 +Width: 1343 +GlyphClass: 1 +Flags: HW +TtInstrs: +PUSHB_3 + 0 + 24 + 1 +SVTCA[x-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 56 85 N 1 0 0 1 0 0 2 +Refer: 222 184 N 1 0 0 1 186 8 2 +EndChar + +StartChar: ucedilla +Encoding: 419 -1 325 +Width: 1296 +GlyphClass: 1 +Flags: HW +TtInstrs: +PUSHB_5 + 48 + 25 + 96 + 44 + 2 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 88 117 N 1 0 0 1 0 0 2 +Refer: 222 184 N 1 0 0 1 470 14 2 +EndChar + +StartChar: Zacute +Encoding: 328 377 326 +Width: 1108 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 61 90 N 1 0 0 1 0 0 2 +Refer: 141 180 N 1 0 0 1 187 260 2 +EndChar + +StartChar: zacute +Encoding: 329 378 327 +Width: 1020 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 11 + 0 + 12 + 16 + 12 + 2 + 63 + 12 + 1 + 63 + 15 + 1 +DELTAP1 +DELTAP1 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 93 122 N 1 0 0 1 0 0 2 +Refer: 141 180 N 1 0 0 1 17 -71 2 +EndChar + +StartChar: Zdotaccent +Encoding: 330 379 328 +Width: 1108 +GlyphClass: 1 +Flags: W +TtInstrs: +PUSHB_3 + 63 + 16 + 1 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 61 90 N 1 0 0 1 0 0 2 +Refer: 220 729 N 1 0 0 1 126 245 2 +EndChar + +StartChar: zdotaccent +Encoding: 331 380 329 +Width: 1020 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 11 + 0 + 13 + 16 + 13 + 2 + 63 + 16 + 1 + 63 + 23 + 1 +DELTAP1 +DELTAP1 +SVTCA[y-axis] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 93 122 N 1 0 0 1 0 0 2 +Refer: 220 729 N 1 0 0 1 -5 -101 2 +EndChar + +StartChar: coloncur +Encoding: 420 -1 330 +Width: 1292 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 75 + 53 + 15 + 1 + 28 + 31 + 35 + 35 + 10 + 29 + 30 + 33 + 34 + 3 + 14 + 40 + 17 + 30 + 13 + 40 + 10 + 11 + 27 + 40 + 24 + 30 + 0 + 40 + 3 + 1 + 29 + 31 + 28 + 16 + 30 + 1 + 0 + 30 + 16 + 30 + 2 + 30 + 34 + 33 + 35 + 32 + 15 + 34 + 31 + 34 + 2 + 34 + 6 + 14 + 0 + 0 + 16 + 0 + 160 + 0 + 192 + 0 + 4 + 0 + 7 + 21 + 21 + 21 + 2 + 21 + 21 + 223 + 6 + 1 + 6 + 36 +SRP0 +MDRP[rp0,min,rnd,white] +DELTAP2 +MIRP[min,rnd,black] +DELTAP1 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MDRP[rnd,grey] +SRP0 +MDRP[rp0,rnd,grey] +DELTAP1 +MDRP[rp0,min,rnd,black] +IP +IP +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +DELTAP1 +MDRP[rp0,min,rnd,black] +IP +IP +SVTCA[y-axis] +MIAP[rnd] +MIRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MIRP[rnd,grey] +MIAP[rnd] +MIRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MIRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1229 90 m 1,0,1 + 1153 39 1153 39 1049.5 7 c 128,-1,2 + 946 -25 946 -25 801 -25 c 0,3,4 + 430 -25 430 -25 254 165.5 c 128,-1,5 + 78 356 78 356 78 717 c 2,6,-1 + 78 778 l 2,7,8 + 78 1118 78 1118 267.5 1324 c 128,-1,9 + 457 1530 457 1530 829 1530 c 0,10,11 + 956 1530 956 1530 1054.5 1495 c 128,-1,12 + 1153 1460 1153 1460 1229 1407 c 1,13,-1 + 1229 1135 l 1,14,15 + 1149 1210 1149 1210 1064 1249 c 128,-1,16 + 979 1288 979 1288 881 1288 c 0,17,18 + 627 1288 627 1288 527.5 1149 c 128,-1,19 + 428 1010 428 1010 428 758 c 2,20,-1 + 428 737 l 2,21,22 + 428 467 428 467 532.5 344 c 128,-1,23 + 637 221 637 221 834 221 c 0,24,25 + 967 221 967 221 1070.5 266 c 128,-1,26 + 1174 311 1174 311 1229 360 c 1,27,-1 + 1229 90 l 1,0,1 +1178 1731 m 1,28,-1 + 854 -188 l 1,29,-1 + 698 -188 l 1,30,-1 + 1022 1731 l 1,31,-1 + 1178 1731 l 1,28,-1 +858 1731 m 1,32,-1 + 535 -188 l 1,33,-1 + 379 -188 l 1,34,-1 + 702 1731 l 1,35,-1 + 858 1731 l 1,32,-1 +EndSplineSet +EndChar + +StartChar: cruzeiro +Encoding: 365 8354 331 +Width: 1389 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 378 -1 N 1 0 0 1 0 0 2 +Refer: 38 67 N 1 0 0 1 0 0 2 +EndChar + +StartChar: pesetas +Encoding: 367 8359 332 +Width: 1622 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 104 + 23 + 38 + 89 + 41 + 44 + 42 + 68 + 30 + 70 + 27 + 33 + 31 + 34 + 3 + 81 + 46 + 78 + 62 + 59 + 65 + 65 + 78 + 59 + 46 + 4 + 49 + 68 + 72 + 75 + 94 + 71 + 68 + 112 + 53 + 70 + 56 + 89 + 52 + 49 + 3 + 18 + 31 + 4 + 4 + 6 + 7 + 17 + 31 + 7 + 10 + 6 + 0 + 81 + 46 + 78 + 62 + 59 + 65 + 72 + 70 + 46 + 93 + 59 + 78 + 93 + 53 + 65 + 37 + 22 + 30 + 37 + 44 + 24 + 88 + 42 + 41 + 39 + 37 + 6 + 6 + 0 + 22 + 0 + 39 + 0 + 3 + 0 + 21 + 16 + 12 + 32 + 12 + 64 + 12 + 112 + 12 + 4 + 12 + 6 + 18 + 4 + 20 + 6 + 84 +SRP0 +MDRP[rp0,rnd,white] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[min,rnd,black] +DELTAP1 +SRP0 +MDRP[rp0,min,rnd,grey] +MDRP[min,rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +MIRP[rp0,min,rnd,black] +SROUND +MDRP[rnd,grey] +RTG +SRP1 +SRP2 +IP +SRP1 +SRP2 +IP +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +MDAP[rnd] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +SROUND +MDRP[rnd,grey] +RTG +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP1 +SRP2 +SLOOP +IP +SRP1 +SRP2 +IP +SRP1 +SRP2 +IP +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +SROUND +MDRP[rnd,grey] +SRP0 +MDRP[min,rnd,grey] +MDRP[rp0,rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1338 1040 m 2,0,1 + 1300 877 1300 877 1179 785.5 c 128,-1,2 + 1058 694 1058 694 835 694 c 2,3,-1 + 610 694 l 1,4,-1 + 450 0 l 1,5,-1 + 134 0 l 1,6,-1 + 482 1509 l 1,7,-1 + 1043 1509 l 2,8,9 + 1247 1509 1247 1509 1325 1415 c 128,-1,10 + 1403 1321 1403 1321 1371 1180 c 2,11,-1 + 1338 1040 l 2,0,1 +1047 1059 m 2,12,-1 + 1066 1143 l 2,13,14 + 1086 1229 1086 1229 1048.5 1259.5 c 128,-1,15 + 1011 1290 1011 1290 908 1290 c 2,16,-1 + 748 1290 l 1,17,-1 + 658 901 l 1,18,-1 + 836 901 l 2,19,20 + 910 901 910 901 969.5 942 c 128,-1,21 + 1029 983 1029 983 1047 1059 c 2,12,-1 +1172 444 m 1,22,-1 + 985 444 l 1,23,-1 + 966 362 l 2,24,25 + 943 264 943 264 951.5 226 c 128,-1,26 + 960 188 960 188 1004 188 c 0,27,28 + 1029 188 1029 188 1053 192.5 c 128,-1,29 + 1077 197 1077 197 1111 211 c 1,30,-1 + 1069 31 l 1,31,32 + 1033 6 1033 6 994.5 -2 c 128,-1,33 + 956 -10 956 -10 902 -10 c 0,34,35 + 755 -10 755 -10 721 91.5 c 128,-1,36 + 687 193 687 193 731 383 c 2,37,-1 + 745 446 l 1,38,-1 + 618 446 l 1,39,-1 + 656 610 l 1,40,-1 + 803 618 l 1,41,-1 + 909 872 l 1,42,-1 + 1084 872 l 1,43,-1 + 1026 621 l 1,44,-1 + 1213 621 l 1,45,-1 + 1172 444 l 1,22,-1 +1624 205 m 0,46,47 + 1602 109 1602 109 1525.5 50.5 c 128,-1,48 + 1449 -8 1449 -8 1300 -8 c 0,49,50 + 1237 -8 1237 -8 1189 3 c 128,-1,51 + 1141 14 1141 14 1122 31 c 1,52,-1 + 1164 211 l 1,53,54 + 1188 197 1188 197 1231 179.5 c 128,-1,55 + 1274 162 1274 162 1323 162 c 0,56,57 + 1361 162 1361 162 1389.5 171 c 128,-1,58 + 1418 180 1418 180 1424 207 c 0,59,60 + 1431 236 1431 236 1396 251 c 128,-1,61 + 1361 266 1361 266 1340 274 c 0,62,63 + 1259 303 1259 303 1242.5 362.5 c 128,-1,64 + 1226 422 1226 422 1245 504 c 0,65,66 + 1264 588 1264 588 1334 645 c 128,-1,67 + 1404 702 1404 702 1526 702 c 0,68,69 + 1584 702 1584 702 1625 691 c 128,-1,70 + 1666 680 1666 680 1690 666 c 1,71,-1 + 1653 506 l 1,72,73 + 1630 522 1630 522 1586.5 532.5 c 128,-1,74 + 1543 543 1543 543 1502 543 c 0,75,76 + 1473 543 1473 543 1450 533.5 c 128,-1,77 + 1427 524 1427 524 1420 494 c 0,78,79 + 1414 467 1414 467 1430.5 456.5 c 128,-1,80 + 1447 446 1447 446 1490 430 c 0,81,82 + 1596 393 1596 393 1618.5 336 c 128,-1,83 + 1641 279 1641 279 1624 205 c 0,46,47 +EndSplineSet +EndChar + +StartChar: rupees +Encoding: 421 -1 333 +Width: 1729 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 81 + 3 + 11 + 20 + 11 + 37 + 11 + 3 + 64 + 29 + 61 + 45 + 42 + 48 + 48 + 61 + 42 + 29 + 4 + 32 + 51 + 55 + 58 + 94 + 54 + 51 + 112 + 36 + 70 + 39 + 89 + 35 + 32 + 3 + 16 + 5 + 31 + 25 + 25 + 7 + 8 + 24 + 31 + 8 + 10 + 0 + 7 + 0 + 64 + 29 + 61 + 45 + 42 + 48 + 55 + 70 + 29 + 93 + 42 + 61 + 93 + 36 + 48 + 7 + 16 + 20 + 13 + 0 + 25 + 1 + 13 + 24 + 32 + 20 + 1 + 20 + 7 + 25 + 5 + 20 + 7 + 67 +SRP0 +MDRP[rp0,rnd,white] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[min,rnd,black] +MDRP[rp0,rnd,grey] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +SRP0 +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +MIRP[rp0,min,rnd,black] +SROUND +MDRP[rnd,grey] +RTG +SRP1 +SRP2 +IP +SRP1 +SRP2 +IP +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +MDAP[rnd] +MIRP[rp0,min,rnd,black] +IP +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +SROUND +MDRP[rnd,grey] +RTG +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP1 +SRP2 +SLOOP +IP +SRP1 +SRP2 +IP +SRP1 +SRP2 +IP +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1180 0 m 1,0,-1 + 844 0 l 1,1,2 + 840 365 840 365 774.5 521.5 c 128,-1,3 + 709 678 709 678 567 678 c 2,4,-1 + 487 678 l 1,5,-1 + 487 0 l 1,6,-1 + 147 0 l 1,7,-1 + 147 1509 l 1,8,-1 + 741 1509 l 2,9,10 + 958 1509 958 1509 1048.5 1415 c 128,-1,11 + 1139 1321 1139 1321 1139 1184 c 2,12,-1 + 1139 1102 l 2,13,14 + 1139 973 1139 973 1060 895 c 128,-1,15 + 981 817 981 817 823 803 c 1,16,17 + 1001 750 1001 750 1090.5 567.5 c 128,-1,18 + 1180 385 1180 385 1180 0 c 1,0,-1 +829 1071 m 2,19,-1 + 829 1112 l 2,20,21 + 829 1210 829 1210 777 1252 c 128,-1,22 + 725 1294 725 1294 623 1294 c 2,23,-1 + 487 1294 l 1,24,-1 + 487 895 l 1,25,-1 + 604 895 l 2,26,27 + 725 895 725 895 777 939 c 128,-1,28 + 829 983 829 983 829 1071 c 2,19,-1 +1698 199 m 0,29,30 + 1698 102 1698 102 1630.5 44 c 128,-1,31 + 1563 -14 1563 -14 1403 -14 c 0,32,33 + 1335 -14 1335 -14 1281 -3 c 128,-1,34 + 1227 8 1227 8 1202 25 c 1,35,-1 + 1202 205 l 1,36,37 + 1231 190 1231 190 1282 173 c 128,-1,38 + 1333 156 1333 156 1386 156 c 0,39,40 + 1427 156 1427 156 1455 165 c 128,-1,41 + 1483 174 1483 174 1483 201 c 0,42,43 + 1483 229 1483 229 1442 244.5 c 128,-1,44 + 1401 260 1401 260 1376 268 c 0,45,46 + 1282 297 1282 297 1249.5 356.5 c 128,-1,47 + 1217 416 1217 416 1217 498 c 0,48,49 + 1217 582 1217 582 1278 639 c 128,-1,50 + 1339 696 1339 696 1470 696 c 0,51,52 + 1532 696 1532 696 1579 685 c 128,-1,53 + 1626 674 1626 674 1655 659 c 1,54,-1 + 1655 500 l 1,55,56 + 1626 516 1626 516 1577 526.5 c 128,-1,57 + 1528 537 1528 537 1483 537 c 0,58,59 + 1452 537 1452 537 1429.5 527.5 c 128,-1,60 + 1407 518 1407 518 1407 487 c 0,61,62 + 1407 461 1407 461 1427.5 450.5 c 128,-1,63 + 1448 440 1448 440 1499 424 c 0,64,65 + 1622 387 1622 387 1660 329.5 c 128,-1,66 + 1698 272 1698 272 1698 199 c 0,29,30 +EndSplineSet +EndChar + +StartChar: pesoph +Encoding: 422 -1 334 +Width: 1374 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 43 + 217 + 20 + 249 + 20 + 2 + 24 + 32 + 23 + 23 + 7 + 18 + 31 + 4 + 4 + 6 + 7 + 17 + 31 + 7 + 10 + 6 + 0 + 22 + 39 + 0 + 1 + 0 + 21 + 6 + 12 + 22 + 12 + 32 + 12 + 3 + 12 + 6 + 18 + 4 + 20 + 23 + 6 + 26 +SRP0 +MDRP[rp0,rnd,white] +MDRP[min,rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +DELTAP1 +MDRP[rp0,min,rnd,grey] +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +MDAP[rnd] +MIRP[min,rnd,black] +SRP2 +IP +MDAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1188 877 m 2,0,1 + 1188 713 1188 713 1080.5 621.5 c 128,-1,2 + 973 530 973 530 733 530 c 2,3,-1 + 487 530 l 1,4,-1 + 487 0 l 1,5,-1 + 147 0 l 1,6,-1 + 147 1509 l 1,7,-1 + 754 1509 l 2,8,9 + 973 1509 973 1509 1080.5 1417 c 128,-1,10 + 1188 1325 1188 1325 1188 1159 c 2,11,-1 + 1188 877 l 2,0,1 +870 895 m 2,12,-1 + 870 1122 l 2,13,14 + 870 1208 870 1208 822 1249 c 128,-1,15 + 774 1290 774 1290 664 1290 c 2,16,-1 + 487 1290 l 1,17,-1 + 487 737 l 1,18,-1 + 682 737 l 2,19,20 + 762 737 762 737 816 778 c 128,-1,21 + 870 819 870 819 870 895 c 2,12,-1 +1307 924 m 1,22,-1 + 94 924 l 1,23,-1 + 94 1100 l 1,24,-1 + 1307 1100 l 1,25,-1 + 1307 924 l 1,22,-1 +EndSplineSet +EndChar + +StartChar: rupiah +Encoding: 405 63197 335 +Width: 2183 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 77 + 3 + 11 + 5 + 45 + 20 + 11 + 22 + 45 + 37 + 11 + 5 + 54 + 55 + 35 + 3 + 32 + 43 + 64 + 40 + 96 + 43 + 51 + 34 + 43 + 38 + 38 + 7 + 8 + 58 + 33 + 32 + 3 + 37 + 18 + 16 + 5 + 31 + 25 + 25 + 7 + 8 + 24 + 31 + 8 + 10 + 0 + 7 + 0 + 29 + 23 + 47 + 37 + 39 + 40 + 55 + 35 + 24 + 192 + 37 + 1 + 37 + 7 + 16 + 20 + 13 + 0 + 25 + 1 + 13 + 24 + 20 + 7 + 25 + 5 + 20 + 7 + 61 +SRP0 +MDRP[rp0,rnd,white] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +MIRP[min,rnd,black] +MDRP[rp0,rnd,grey] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +MDAP[rnd] +MIRP[rp0,min,rnd,black] +IP +MIAP[rnd] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +MDAP[rnd] +MDRP[rp0,rnd,grey] +MIRP[min,rnd,black] +SRP0 +SMD +MDRP[min,grey] +SMD +SRP1 +SRP2 +SLOOP +IP +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1118 0 m 1,0,-1 + 782 0 l 1,1,2 + 778 365 778 365 723 521.5 c 128,-1,3 + 668 678 668 678 526 678 c 2,4,-1 + 487 678 l 1,5,-1 + 487 0 l 1,6,-1 + 147 0 l 1,7,-1 + 147 1509 l 1,8,-1 + 700 1509 l 2,9,10 + 868 1509 868 1509 972.5 1422 c 128,-1,11 + 1077 1335 1077 1335 1077 1184 c 2,12,-1 + 1077 1102 l 2,13,14 + 1077 973 1077 973 998.5 895 c 128,-1,15 + 920 817 920 817 762 803 c 1,16,17 + 940 750 940 750 1029 567.5 c 128,-1,18 + 1118 385 1118 385 1118 0 c 1,0,-1 +768 1071 m 2,19,-1 + 768 1112 l 2,20,21 + 768 1210 768 1210 726 1252 c 128,-1,22 + 684 1294 684 1294 582 1294 c 2,23,-1 + 487 1294 l 1,24,-1 + 487 895 l 1,25,-1 + 563 895 l 2,26,27 + 684 895 684 895 726 939 c 128,-1,28 + 768 983 768 983 768 1071 c 2,19,-1 +2142 434 m 2,29,30 + 2142 207 2142 207 2018 96.5 c 128,-1,31 + 1894 -14 1894 -14 1722 -14 c 0,32,33 + 1663 -14 1663 -14 1592.5 2 c 128,-1,34 + 1522 18 1522 18 1473 61 c 1,35,-1 + 1473 -342 l 1,36,-1 + 1174 -342 l 1,37,-1 + 1174 926 l 1,38,-1 + 1460 926 l 1,39,-1 + 1468 813 l 1,40,41 + 1513 874 1513 874 1575.5 911 c 128,-1,42 + 1638 948 1638 948 1753 948 c 0,43,44 + 1939 948 1939 948 2040.5 824 c 128,-1,45 + 2142 700 2142 700 2142 526 c 2,46,-1 + 2142 434 l 2,29,30 +1812 446 m 2,47,-1 + 1812 481 l 2,48,49 + 1812 608 1812 608 1766 665.5 c 128,-1,50 + 1720 723 1720 723 1626 723 c 0,51,52 + 1559 723 1559 723 1529 696.5 c 128,-1,53 + 1499 670 1499 670 1473 625 c 1,54,-1 + 1473 248 l 1,55,56 + 1489 221 1489 221 1526 198.5 c 128,-1,57 + 1563 176 1563 176 1606 176 c 0,58,59 + 1704 176 1704 176 1758 233.5 c 128,-1,60 + 1812 291 1812 291 1812 446 c 2,47,-1 +EndSplineSet +EndChar + +StartChar: blank +Encoding: 423 -1 336 +Width: 1364 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 15 + 4 + 217 + 1 + 18 + 6 + 2 + 0 + 5 + 217 + 0 + 9 + 3 + 217 + 1 + 8 +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[rp0,min,rnd,black] +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1206 -350 m 1,0,-1 + 162 -350 l 1,1,-1 + 162 0 l 1,2,-1 + 352 0 l 1,3,-1 + 352 -162 l 1,4,-1 + 1014 -162 l 1,5,-1 + 1014 0 l 1,6,-1 + 1206 0 l 1,7,-1 + 1206 -350 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: pageup +Encoding: 377 8670 337 +Width: 1536 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 40 + 9 + 12 + 201 + 6 + 15 + 19 + 5 + 16 + 201 + 2 + 19 + 19 + 11 + 20 + 1 + 20 + 22 + 15 + 11 + 218 + 22 + 10 + 11 + 3 + 8 + 2 + 6 + 0 + 10 + 200 + 11 + 17 + 13 + 21 + 19 + 15 + 11 + 11 + 24 + 23 +SRP1 +SRP2 +IP +MDAP[rnd] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[min,rnd,grey] +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +SRP0 +MIRP[rp0,min,rnd,black] +MDRP[min,rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +SRP1 +SRP2 +IP +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +SRP1 +SRP2 +IP +MDAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1259 1043 m 1,0,-1 + 904 1043 l 1,1,-1 + 904 822 l 1,2,-1 + 1199 822 l 1,3,-1 + 1199 614 l 1,4,-1 + 904 614 l 1,5,-1 + 904 442 l 1,6,-1 + 1199 442 l 1,7,-1 + 1199 234 l 1,8,-1 + 904 234 l 1,9,-1 + 904 -171 l 1,10,-1 + 636 -171 l 1,11,-1 + 636 234 l 1,12,-1 + 341 234 l 1,13,-1 + 341 442 l 1,14,-1 + 636 442 l 1,15,-1 + 636 614 l 1,16,-1 + 341 614 l 1,17,-1 + 341 822 l 1,18,-1 + 636 822 l 1,19,-1 + 636 1043 l 1,20,-1 + 281 1043 l 1,21,-1 + 770 1536 l 1,22,-1 + 1259 1043 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: capslock +Encoding: 385 8682 338 +Width: 2388 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 66 + 56 + 6 + 88 + 6 + 200 + 6 + 249 + 6 + 4 + 73 + 6 + 185 + 6 + 2 + 55 + 5 + 1 + 7 + 9 + 204 + 1 + 4 + 70 + 4 + 6 + 11 + 204 + 3 + 16 + 25 + 8 + 221 + 8 + 2 + 8 + 6 + 15 + 15 + 204 + 21 + 20 + 204 + 16 + 0 + 8 + 6 + 12 + 11 + 7 + 0 + 2 + 19 + 12 + 202 + 17 + 2 + 23 + 9 + 5 + 3 + 20 + 11 + 202 + 16 + 3 + 22 +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +MDRP[rp0,min,rnd,grey] +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +MDRP[rp0,min,rnd,grey] +SRP1 +SRP2 +IP +IP +SVTCA[y-axis] +MIAP[rnd] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +MDRP[grey] +DELTAP1 +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[min,rnd,black] +SRP2 +IP +SROUND +MDAP[rnd] +RTG +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +SVTCA[y-axis] +DELTAP2 +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +2220 512 m 1,0,-1 + 1708 512 l 1,1,-1 + 1708 172 l 1,2,-1 + 684 172 l 1,3,-1 + 684 512 l 1,4,-1 + 172 512 l 1,5,-1 + 1196 1536 l 1,6,-1 + 2220 512 l 1,0,-1 +1806 684 m 1,7,-1 + 1196 1294 l 1,8,-1 + 584 683 l 1,9,-1 + 852 684 l 1,10,-1 + 852 340 l 1,11,-1 + 1536 340 l 1,12,-1 + 1536 684 l 1,13,-1 + 1806 684 l 1,7,-1 +1708 -512 m 1,14,-1 + 684 -512 l 1,15,-1 + 684 0 l 1,16,-1 + 1708 0 l 1,17,-1 + 1708 -512 l 1,14,-1 +1536 -340 m 1,18,-1 + 1536 -172 l 1,19,-1 + 852 -172 l 1,20,-1 + 852 -340 l 1,21,-1 + 1536 -340 l 1,18,-1 +EndSplineSet +EndChar + +StartChar: arrowupwhite +Encoding: 2 5 339 +Width: 2188 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 54 + 56 + 6 + 88 + 6 + 170 + 8 + 200 + 6 + 249 + 6 + 5 + 73 + 6 + 185 + 6 + 2 + 55 + 5 + 1 + 7 + 9 + 204 + 1 + 4 + 4 + 6 + 11 + 204 + 3 + 0 + 25 + 8 + 221 + 8 + 2 + 8 + 6 + 15 + 8 + 6 + 12 + 11 + 7 + 0 + 12 + 202 + 2 + 15 + 9 + 5 + 11 + 202 + 3 + 14 +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +MDRP[rp0,min,rnd,grey] +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +MDRP[rp0,min,rnd,grey] +SRP1 +SRP2 +IP +IP +SVTCA[y-axis] +MIAP[rnd] +MDRP[grey] +DELTAP1 +MIAP[rnd] +MIRP[min,rnd,black] +SRP2 +IP +MDAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +SVTCA[y-axis] +DELTAP2 +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +2118 512 m 1,0,-1 + 1606 512 l 1,1,-1 + 1606 0 l 1,2,-1 + 582 0 l 1,3,-1 + 582 512 l 1,4,-1 + 70 512 l 1,5,-1 + 1094 1536 l 1,6,-1 + 2118 512 l 1,0,-1 +1704 684 m 1,7,-1 + 1094 1294 l 1,8,-1 + 482 683 l 1,9,-1 + 750 684 l 1,10,-1 + 750 172 l 1,11,-1 + 1434 172 l 1,12,-1 + 1434 684 l 1,13,-1 + 1704 684 l 1,7,-1 +EndSplineSet +EndChar + +StartChar: propellor +Encoding: 12 17 340 +Width: 1876 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 78 + 55 + 31 + 59 + 204 + 43 + 72 + 47 + 20 + 95 + 20 + 2 + 20 + 27 + 51 + 63 + 204 + 36 + 27 + 15 + 78 + 8 + 86 + 204 + 44 + 73 + 32 + 19 + 80 + 19 + 2 + 19 + 12 + 82 + 93 + 204 + 3 + 12 + 0 + 55 + 44 + 78 + 205 + 32 + 70 + 47 + 7 + 1 + 7 + 0 + 48 + 74 + 205 + 39 + 15 + 0 + 1 + 0 + 97 + 59 + 19 + 86 + 205 + 31 + 73 + 32 + 8 + 1 + 8 + 15 + 66 + 90 + 205 + 24 + 176 + 15 + 1 + 15 + 96 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,black] +DELTAP1 +MDRP[rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,black] +DELTAP1 +MDRP[rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[rnd,grey] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,black] +DELTAP1 +MDRP[rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[rnd,grey] +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,black] +DELTAP1 +MDRP[rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[rnd,grey] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1708 340 m 0,0,1 + 1708 203 1708 203 1607.5 101.5 c 128,-1,2 + 1507 0 1507 0 1366 0 c 0,3,4 + 1227 0 1227 0 1125.5 100.5 c 128,-1,5 + 1024 201 1024 201 1024 340 c 2,6,-1 + 1024 512 l 1,7,-1 + 854 512 l 1,8,-1 + 854 340 l 2,9,10 + 854 203 854 203 753.5 101.5 c 128,-1,11 + 653 0 653 0 512 0 c 0,12,13 + 373 0 373 0 272.5 100.5 c 128,-1,14 + 172 201 172 201 172 340 c 0,15,16 + 172 483 172 483 272.5 583.5 c 128,-1,17 + 373 684 373 684 512 684 c 2,18,-1 + 684 684 l 1,19,-1 + 684 852 l 1,20,-1 + 512 852 l 2,21,22 + 371 852 371 852 271.5 954.5 c 128,-1,23 + 172 1057 172 1057 172 1194 c 0,24,25 + 172 1335 172 1335 272.5 1435.5 c 128,-1,26 + 373 1536 373 1536 512 1536 c 256,27,28 + 651 1536 651 1536 752.5 1435.5 c 128,-1,29 + 854 1335 854 1335 854 1196 c 2,30,-1 + 854 1024 l 1,31,-1 + 1024 1024 l 1,32,-1 + 1024 1196 l 2,33,34 + 1024 1335 1024 1335 1125.5 1435.5 c 128,-1,35 + 1227 1536 1227 1536 1364 1536 c 0,36,37 + 1505 1536 1505 1536 1606.5 1435.5 c 128,-1,38 + 1708 1335 1708 1335 1708 1196 c 256,39,40 + 1708 1057 1708 1057 1607.5 954.5 c 128,-1,41 + 1507 852 1507 852 1364 852 c 2,42,-1 + 1196 852 l 1,43,-1 + 1196 684 l 1,44,-1 + 1366 684 l 2,45,46 + 1505 684 1505 684 1606.5 582.5 c 128,-1,47 + 1708 481 1708 481 1708 340 c 0,0,1 +1536 1196 m 256,48,49 + 1536 1266 1536 1266 1486 1315 c 128,-1,50 + 1436 1364 1436 1364 1366 1364 c 256,51,52 + 1296 1364 1296 1364 1246 1315 c 128,-1,53 + 1196 1266 1196 1266 1196 1196 c 2,54,-1 + 1196 1024 l 1,55,-1 + 1366 1024 l 2,56,57 + 1438 1024 1438 1024 1487 1075 c 128,-1,58 + 1536 1126 1536 1126 1536 1196 c 256,48,49 +684 1024 m 1,59,-1 + 684 1196 l 2,60,61 + 684 1266 684 1266 633 1315 c 128,-1,62 + 582 1364 582 1364 512 1364 c 256,63,64 + 442 1364 442 1364 392 1315 c 128,-1,65 + 342 1266 342 1266 342 1196 c 0,66,67 + 342 1124 342 1124 392 1074 c 128,-1,68 + 442 1024 442 1024 512 1024 c 2,69,-1 + 684 1024 l 1,59,-1 +1024 684 m 1,70,-1 + 1024 852 l 1,71,-1 + 854 852 l 1,72,-1 + 854 684 l 1,73,-1 + 1024 684 l 1,70,-1 +1536 340 m 0,74,75 + 1536 412 1536 412 1486 462 c 128,-1,76 + 1436 512 1436 512 1366 512 c 2,77,-1 + 1196 512 l 1,78,-1 + 1196 340 l 2,79,80 + 1196 270 1196 270 1246 221 c 128,-1,81 + 1296 172 1296 172 1366 172 c 256,82,83 + 1436 172 1436 172 1486 222 c 128,-1,84 + 1536 272 1536 272 1536 340 c 0,74,75 +684 340 m 2,85,-1 + 684 512 l 1,86,-1 + 512 512 l 2,87,88 + 440 512 440 512 391 461 c 128,-1,89 + 342 410 342 410 342 340 c 256,90,91 + 342 270 342 270 392 221 c 128,-1,92 + 442 172 442 172 512 172 c 256,93,94 + 582 172 582 172 633 221 c 128,-1,95 + 684 270 684 270 684 340 c 2,85,-1 +EndSplineSet +EndChar + +StartChar: clear +Encoding: 21 28 341 +Width: 2220 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 93 + 38 + 22 + 41 + 34 + 2 + 28 + 22 + 34 + 26 + 4 + 32 + 29 + 17 + 12 + 16 + 13 + 11 + 25 + 20 + 21 + 29 + 204 + 18 + 15 + 14 + 11 + 15 + 2 + 7 + 3 + 6 + 18 + 27 + 33 + 35 + 32 + 204 + 1 + 4 + 5 + 8 + 0 + 11 + 14 + 15 + 18 + 1 + 4 + 5 + 8 + 29 + 21 + 20 + 25 + 27 + 33 + 35 + 32 + 16 + 12 + 13 + 16 + 17 + 3 + 6 + 28 + 22 + 26 + 34 + 10 + 7 + 6 + 2 + 22 + 2 + 2 + 2 + 11 + 7 + 25 + 7 + 2 + 7 + 0 + 9 + 23 + 205 + 0 + 37 + 31 + 205 + 9 + 36 +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[rp0,min,rnd,black] +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +MDAP[rnd] +DELTAP1 +MDAP[rnd] +DELTAP1 +SRP2 +SLOOP +IP +SLOOP +IP +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +MIAP[rnd] +MDRP[rnd,grey] +MDRP[grey] +MDRP[grey] +MIAP[rnd] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +MDRP[grey] +MDRP[grey] +SRP1 +SRP2 +SLOOP +IP +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +2048 0 m 1,0,-1 + 1694 0 l 1,1,-1 + 1851 -252 l 1,2,-1 + 1708 -340 l 1,3,-1 + 1493 0 l 1,4,-1 + 727 0 l 1,5,-1 + 512 -340 l 1,6,-1 + 367 -252 l 1,7,-1 + 524 0 l 1,8,-1 + 172 0 l 1,9,-1 + 172 1536 l 1,10,-1 + 524 1536 l 1,11,-1 + 367 1788 l 1,12,-1 + 512 1876 l 1,13,-1 + 727 1536 l 1,14,-1 + 1493 1536 l 1,15,-1 + 1708 1876 l 1,16,-1 + 1851 1788 l 1,17,-1 + 1694 1536 l 1,18,-1 + 2048 1536 l 1,19,-1 + 2048 0 l 1,0,-1 +1384 1364 m 1,20,-1 + 834 1364 l 1,21,-1 + 1110 928 l 1,22,-1 + 1384 1364 l 1,20,-1 +1878 172 m 1,23,-1 + 1878 1364 l 1,24,-1 + 1587 1364 l 1,25,-1 + 1210 768 l 1,26,-1 + 1587 172 l 1,27,-1 + 1878 172 l 1,23,-1 +1010 768 m 1,28,-1 + 633 1364 l 1,29,-1 + 342 1364 l 1,30,-1 + 342 172 l 1,31,-1 + 633 172 l 1,32,-1 + 1010 768 l 1,28,-1 +1384 172 m 1,33,-1 + 1110 608 l 1,34,-1 + 834 172 l 1,35,-1 + 1384 172 l 1,33,-1 +EndSplineSet +EndChar + +StartChar: arrowleft +Encoding: 371 8592 342 +Width: 2048 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 15 + 3 + 1 + 5 + 2 + 1 + 219 + 4 + 5 + 220 + 0 + 8 + 5 + 1 + 3 + 7 +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,white] +SVTCA[y-axis] +MIAP[rnd] +MDRP[min,rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rp0,min,rnd,grey] +SRP1 +SRP2 +IP +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1880 624 m 1,0,-1 + 668 624 l 1,1,-1 + 668 270 l 1,2,-1 + 172 768 l 1,3,-1 + 668 1266 l 1,4,-1 + 668 912 l 1,5,-1 + 1880 912 l 1,6,-1 + 1880 624 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: arrowright +Encoding: 373 8594 343 +Width: 2048 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 342 8592 N -1 0 0 1 2051 0 2 +EndChar + +StartChar: enter +Encoding: 424 -1 344 +Width: 2220 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 37 + 57 + 5 + 54 + 7 + 73 + 5 + 70 + 7 + 4 + 4 + 8 + 5 + 7 + 43 + 6 + 1 + 6 + 9 + 9 + 1 + 9 + 1 + 221 + 2 + 10 + 6 + 9 + 4 + 8 + 5 + 4 + 0 + 11 + 7 + 8 + 1 + 10 +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rp0,rnd,grey] +MDRP[grey] +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rp0,rnd,grey] +MDRP[grey] +SRP1 +SRP2 +IP +IP +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MDRP[rp0,grey] +DELTAP1 +MDRP[grey] +DELTAP1 +MDAP[rnd] +MDRP[rnd,grey] +MDRP[rp0,min,grey] +MDRP[grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +2048 1234 m 1,0,-1 + 172 1234 l 1,1,-1 + 172 1506 l 1,2,-1 + 2048 1506 l 1,3,-1 + 2048 1234 l 1,0,-1 +1922 424 m 1,4,-1 + 1725 227 l 1,5,-1 + 1110 844 l 1,6,-1 + 495 227 l 1,7,-1 + 296 424 l 1,8,-1 + 1110 1234 l 1,9,-1 + 1922 424 l 1,4,-1 +EndSplineSet +EndChar + +StartChar: arrowupleft +Encoding: 375 8624 345 +Width: 1876 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 33 + 88 + 2 + 1 + 6 + 2 + 4 + 3 + 3 + 1 + 5 + 0 + 87 + 1 + 103 + 1 + 119 + 1 + 3 + 1 + 5 + 10 + 6 + 2 + 3 + 5 + 5 + 0 + 3 + 1 + 0 + 8 + 3 + 7 +SRP0 +MDRP[rp0,min,rnd,white] +SRP0 +MDRP[rp0,white] +RTDG +MDRP[min,grey] +RTG +SRP1 +SRP2 +IP +RTDG +MDAP[rnd] +SRP2 +IP +IP +RTG +SVTCA[y-axis] +MIAP[rnd] +MDAP[no-rnd] +DELTAP1 +MDRP[grey] +SRP1 +SRP2 +IP +RTDG +MDAP[rnd] +SRP2 +IP +IP +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1681 207 m 1,0,-1 + 1473 -1 l 1,1,-1 + 420 1050 l 1,2,-1 + 172 802 l 1,3,-1 + 172 1506 l 1,4,-1 + 876 1506 l 1,5,-1 + 628 1258 l 1,6,-1 + 1681 207 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: arrowtableft +Encoding: 1 3 346 +AltUni2: 0021e4.ffffffff.0 0021e4.ffffffff.0 +Width: 2048 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 20 + 3 + 1 + 5 + 8 + 2 + 1 + 219 + 9 + 4 + 5 + 220 + 0 + 12 + 5 + 1 + 3 + 7 + 221 + 8 + 11 +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[rp0,min,rnd,black] +MDRP[rp0,rnd,grey] +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,white] +SVTCA[y-axis] +MIAP[rnd] +MDRP[min,rnd,grey] +MDRP[min,rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[min,rnd,grey] +MDRP[min,rnd,grey] +SRP1 +SRP2 +IP +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1956 624 m 1,0,-1 + 1002 624 l 1,1,-1 + 1002 270 l 1,2,-1 + 506 768 l 1,3,-1 + 1002 1266 l 1,4,-1 + 1002 912 l 1,5,-1 + 1956 912 l 1,6,-1 + 1956 624 l 1,0,-1 +451 172 m 1,7,-1 + 171 172 l 1,8,-1 + 171 1364 l 1,9,-1 + 451 1364 l 1,10,-1 + 451 172 l 1,7,-1 +EndSplineSet +EndChar + +StartChar: arrowtabright +Encoding: 0 2 347 +AltUni2: 0021e5.ffffffff.0 0021e5.ffffffff.0 +Width: 2048 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +SplineSet +92 624 m 1,0,-1 + 92 912 l 1,1,-1 + 1046 912 l 1,2,-1 + 1046 1266 l 1,3,-1 + 1542 768 l 1,4,-1 + 1046 270 l 1,5,-1 + 1046 624 l 1,6,-1 + 92 624 l 1,0,-1 +1597 172 m 1,7,-1 + 1597 1364 l 1,8,-1 + 1877 1364 l 1,9,-1 + 1877 172 l 1,10,-1 + 1597 172 l 1,7,-1 +EndSplineSet +EndChar + +StartChar: help +Encoding: 425 -1 348 +Width: 1734 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 84 + 44 + 14 + 34 + 16 + 34 + 20 + 44 + 22 + 58 + 35 + 53 + 49 + 102 + 49 + 7 + 95 + 43 + 1 + 43 + 70 + 40 + 203 + 44 + 70 + 63 + 47 + 1 + 47 + 31 + 52 + 217 + 0 + 51 + 16 + 51 + 48 + 51 + 3 + 51 + 47 + 51 + 3 + 9 + 21 + 201 + 3 + 211 + 15 + 201 + 9 + 24 + 201 + 94 + 37 + 143 + 37 + 2 + 37 + 51 + 112 + 30 + 1 + 30 + 31 + 50 + 201 + 43 + 16 + 51 + 144 + 51 + 2 + 51 + 51 + 0 + 6 + 12 + 201 + 0 + 211 + 18 + 201 + 6 + 54 +SRP0 +MDRP[rp0,rnd,white] +MIRP[min,rnd,black] +MIRP[rp0,min,rnd,black] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +MDAP[rnd] +DELTAP1 +MDRP[min,rnd,grey] +MIRP[min,rnd,black] +MDRP[rp0,rnd,grey] +MDRP[rp0,min,rnd,black] +DELTAP1 +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MDAP[rnd] +MIRP[min,rnd,black] +MIRP[rp0,min,rnd,black] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +IP +MDAP[rnd] +DELTAP1 +MIRP[rp0,min,rnd,black] +MDRP[rp0,min,rnd,white] +MDAP[rnd] +DELTAP1 +SROUND +MDRP[rnd,grey] +RTG +MIRP[rp0,min,rnd,black] +SROUND +MDRP[rnd,grey] +RTG +DELTAP1 +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1717 765 m 256,0,1 + 1717 415 1717 415 1469.5 167.5 c 128,-1,2 + 1222 -80 1222 -80 872 -80 c 256,3,4 + 522 -80 522 -80 274.5 167.5 c 128,-1,5 + 27 415 27 415 27 765 c 256,6,7 + 27 1115 27 1115 274.5 1362.5 c 128,-1,8 + 522 1610 522 1610 872 1610 c 256,9,10 + 1222 1610 1222 1610 1469.5 1362.5 c 128,-1,11 + 1717 1115 1717 1115 1717 765 c 256,0,1 +1505 765 m 0,12,13 + 1505 1030 1505 1030 1319 1217 c 128,-1,14 + 1133 1404 1133 1404 872 1404 c 0,15,16 + 603 1404 603 1404 421 1219 c 128,-1,17 + 239 1034 239 1034 239 765 c 0,18,19 + 239 497 239 497 421 312 c 128,-1,20 + 603 127 603 127 872 127 c 0,21,22 + 1149 127 1149 127 1327 310.5 c 128,-1,23 + 1505 494 1505 494 1505 765 c 0,12,13 +1160 982 m 0,24,25 + 1160 894 1160 894 1059 824.5 c 128,-1,26 + 958 755 958 755 958 704 c 0,27,28 + 958 647 958 647 958 624 c 128,-1,29 + 958 601 958 601 958 601 c 1,30,-1 + 766 601 l 1,31,32 + 766 601 766 601 767 629 c 128,-1,33 + 768 657 768 657 768 710 c 0,34,35 + 768 771 768 771 857 841.5 c 128,-1,36 + 946 912 946 912 946 967 c 0,37,38 + 946 1025 946 1025 918 1038.5 c 128,-1,39 + 890 1052 890 1052 847 1052 c 0,40,41 + 778 1052 778 1052 704.5 1013 c 128,-1,42 + 631 974 631 974 587 930 c 1,43,-1 + 587 1106 l 1,44,45 + 631 1148 631 1148 702.5 1183.5 c 128,-1,46 + 774 1219 774 1219 868 1219 c 0,47,48 + 989 1219 989 1219 1074.5 1157 c 128,-1,49 + 1160 1095 1160 1095 1160 982 c 0,24,25 +982 304 m 1,50,-1 + 774 304 l 1,51,-1 + 774 497 l 1,52,-1 + 982 497 l 1,53,-1 + 982 304 l 1,50,-1 +EndSplineSet +EndChar + +StartChar: arrowup +Encoding: 372 8593 349 +Width: 1536 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 17 + 1 + 4 + 6 + 15 + 3 + 218 + 6 + 2 + 3 + 0 + 2 + 219 + 5 + 3 + 3 + 8 + 7 +SRP1 +SRP2 +IP +MDAP[rnd] +MDRP[min,rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[min,rnd,grey] +SRP1 +SRP2 +IP +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1266 1040 m 1,0,-1 + 912 1040 l 1,1,-1 + 912 -171 l 1,2,-1 + 624 -171 l 1,3,-1 + 624 1040 l 1,4,-1 + 270 1040 l 1,5,-1 + 768 1536 l 1,6,-1 + 1266 1040 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: arrowdownright +Encoding: 376 8627 350 +Width: 1876 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 30 + 41 + 6 + 1 + 5 + 2 + 1 + 6 + 6 + 4 + 1 + 3 + 4 + 10 + 5 + 2 + 0 + 1 + 1 + 0 + 3 + 4 + 121 + 3 + 140 + 3 + 2 + 3 + 7 + 0 + 8 +SRP0 +MDRP[rp0,min,rnd,white] +SRP0 +MDRP[rp0,white] +DELTAP1 +RTDG +MDRP[min,grey] +RTG +SRP1 +SRP2 +IP +RTDG +MDAP[rnd] +SRP2 +IP +IP +RTG +SVTCA[y-axis] +MIAP[no-rnd] +MDRP[grey] +MDAP[rnd] +SRP2 +IP +RTDG +MDAP[rnd] +SRP2 +IP +IP +IUP[x] +IUP[y] +SVTCA[y-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1710 0 m 1,0,-1 + 1006 0 l 1,1,-1 + 1254 248 l 1,2,-1 + 201 1299 l 1,3,-1 + 409 1507 l 1,4,-1 + 1462 456 l 1,5,-1 + 1710 704 l 1,6,-1 + 1710 0 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: arrowdown +Encoding: 374 8595 351 +Width: 1536 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +SplineSet +1267 323 m 1,0,-1 + 769 -173 l 1,1,-1 + 271 323 l 1,2,-1 + 625 323 l 1,3,-1 + 625 1534 l 1,4,-1 + 913 1534 l 1,5,-1 + 913 323 l 1,6,-1 + 1267 323 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: control +Encoding: 3 6 352 +Width: 1558 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 21 + 4 + 2 + 0 + 3 + 5 + 1 + 3 + 5 + 15 + 48 + 0 + 96 + 0 + 2 + 0 + 7 + 47 + 4 + 1 + 4 + 6 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +PUSHB_2 + 4 + 0 +MD[grid] +EVEN +IF +PUSHB_2 + 0 + 64 +SHPIX +EIF +RTG +SVTCA[y-axis] +MIAP[rnd] +MDAP[rnd] +MDRP[rnd,grey] +SRP2 +SLOOP +IP +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1456 852 m 1,0,-1 + 1238 634 l 1,1,-1 + 772 1100 l 1,2,-1 + 306 634 l 1,3,-1 + 88 852 l 1,4,-1 + 772 1536 l 1,5,-1 + 1456 852 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: deleteleft +Encoding: 16 23 353 +Width: 2644 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 99 + 9 + 14 + 9 + 16 + 70 + 8 + 72 + 16 + 84 + 8 + 135 + 10 + 136 + 14 + 136 + 15 + 136 + 16 + 135 + 20 + 135 + 21 + 11 + 10 + 11 + 10 + 13 + 5 + 17 + 5 + 19 + 71 + 17 + 71 + 19 + 85 + 17 + 85 + 19 + 136 + 11 + 136 + 12 + 136 + 13 + 135 + 17 + 135 + 18 + 135 + 19 + 14 + 21 + 18 + 8 + 15 + 21 + 3 + 2 + 2 + 1 + 3 + 7 + 203 + 47 + 3 + 79 + 3 + 95 + 3 + 127 + 3 + 4 + 3 + 15 + 9 + 203 + 1 + 0 + 21 + 15 + 12 + 18 + 8 + 2 + 9 + 7 + 1 + 3 + 18 + 3 + 0 + 2 + 5 + 205 + 0 + 23 + 2 + 22 +SRP0 +MDRP[rp0,min,rnd,white] +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +IP +MDAP[rnd] +MDRP[rnd,grey] +MDRP[grey] +MDRP[grey] +SRP2 +IP +RTHG +MDAP[rnd] +ALIGNRP +MDRP[grey] +MDRP[grey] +RTG +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MIAP[rnd] +DELTAP1 +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +RTHG +MDAP[rnd] +SLOOP +ALIGNRP +MDRP[grey] +MDRP[grey] +IUP[x] +IUP[y] +SVTCA[y-axis] +DELTAP1 +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +2476 0 m 1,0,-1 + 940 0 l 1,1,-1 + 172 768 l 1,2,-1 + 940 1536 l 1,3,-1 + 2476 1536 l 1,4,-1 + 2476 0 l 1,0,-1 +2304 167 m 1,5,-1 + 2304 1369 l 1,6,-1 + 1024 1369 l 1,7,-1 + 423 768 l 1,8,-1 + 1024 167 l 1,9,-1 + 2304 167 l 1,5,-1 +2019 406 m 1,10,-1 + 1898 285 l 1,11,-1 + 1536 647 l 1,12,-1 + 1176 285 l 1,13,-1 + 1055 406 l 1,14,-1 + 1415 768 l 1,15,-1 + 1055 1130 l 1,16,-1 + 1176 1251 l 1,17,-1 + 1536 889 l 1,18,-1 + 1898 1251 l 1,19,-1 + 2019 1130 l 1,20,-1 + 1657 768 l 1,21,-1 + 2019 406 l 1,10,-1 +EndSplineSet +EndChar + +StartChar: deleteright +Encoding: 7 10 354 +Width: 2644 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +SplineSet +172 0 m 1,0,-1 + 172 1536 l 1,1,-1 + 1708 1536 l 1,2,-1 + 2476 768 l 1,3,-1 + 1708 0 l 1,4,-1 + 172 0 l 1,0,-1 +344 167 m 1,5,-1 + 1624 167 l 1,6,-1 + 2225 768 l 1,7,-1 + 1624 1369 l 1,8,-1 + 344 1369 l 1,9,-1 + 344 167 l 1,5,-1 +629 406 m 1,10,-1 + 991 768 l 1,11,-1 + 629 1130 l 1,12,-1 + 750 1251 l 1,13,-1 + 1112 889 l 1,14,-1 + 1472 1251 l 1,15,-1 + 1593 1130 l 1,16,-1 + 1233 768 l 1,17,-1 + 1593 406 l 1,18,-1 + 1472 285 l 1,19,-1 + 1112 647 l 1,20,-1 + 750 285 l 1,21,-1 + 629 406 l 1,10,-1 +EndSplineSet +EndChar + +StartChar: option +Encoding: 4 7 355 +Width: 2328 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 23 + 1 + 6 + 200 + 2 + 8 + 10 + 200 + 5 + 0 + 1 + 1 + 9 + 6 + 5 + 10 + 5 + 4 + 7 + 0 + 4 + 13 + 7 + 12 +SRP0 +MDRP[rp0,min,rnd,white] +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rnd,grey] +SRP1 +SRP2 +SLOOP +IP +MDAP[rnd] +SVTCA[y-axis] +MIAP[rnd] +MIRP[rp0,min,rnd,black] +MDAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +2188 1204 m 1,0,-1 + 1334 1204 l 1,1,-1 + 1334 1476 l 1,2,-1 + 2188 1476 l 1,3,-1 + 2188 1204 l 1,0,-1 +2188 0 m 1,4,-1 + 1375 0 l 1,5,-1 + 695 1204 l 1,6,-1 + 140 1204 l 1,7,-1 + 140 1476 l 1,8,-1 + 845 1476 l 1,9,-1 + 1525 272 l 1,10,-1 + 2188 272 l 1,11,-1 + 2188 0 l 1,4,-1 +EndSplineSet +EndChar + +StartChar: escape +Encoding: 20 27 356 +AltUni2: 00241b.ffffffff.0 00241b.ffffffff.0 +Width: 1876 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 77 + 8 + 5 + 6 + 25 + 40 + 5 + 51 + 1 + 56 + 3 + 72 + 3 + 88 + 3 + 105 + 3 + 120 + 5 + 136 + 5 + 152 + 3 + 11 + 6 + 2 + 3 + 3 + 0 + 4 + 1 + 1 + 13 + 48 + 13 + 2 + 13 + 1 + 13 + 10 + 24 + 54 + 17 + 1 + 17 + 223 + 10 + 23 + 223 + 4 + 24 + 15 + 1 + 6 + 2 + 5 + 5 + 4 + 0 + 24 + 0 + 24 + 7 + 13 + 57 + 20 + 1 + 20 + 223 + 7 + 14 + 223 + 4 + 144 + 13 + 1 + 13 + 27 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MDRP[rnd,grey] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,black] +MIRP[rp0,min,rnd,black] +DELTAP1 +SRP1 +SRP2 +IP +IP +MDAP[rnd] +MDAP[rnd] +SRP2 +IP +MDAP[rnd] +IP +IP +MDRP[grey] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,black] +MIRP[rp0,min,rnd,black] +DELTAP1 +SRP1 +SRP2 +IP +IP +MDAP[rnd] +DELTAP1 +MDAP[rnd] +SRP2 +IP +IP +MDAP[rnd] +IP +IP +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1029 857 m 1,0,-1 + 849 677 l 1,1,-1 + 382 1144 l 1,2,-1 + 172 932 l 1,3,-1 + 172 1536 l 1,4,-1 + 774 1536 l 1,5,-1 + 562 1324 l 1,6,-1 + 1029 857 l 1,0,-1 +1708 768 m 0,7,8 + 1708 453 1708 453 1485.5 226.5 c 128,-1,9 + 1263 0 1263 0 940 0 c 0,10,11 + 637 0 637 0 414 203.5 c 128,-1,12 + 191 407 191 407 172 768 c 1,13,-1 + 402 768 l 1,14,15 + 415 512 415 512 574 371 c 128,-1,16 + 733 230 733 230 940 230 c 0,17,18 + 1153 230 1153 230 1314.5 379.5 c 128,-1,19 + 1476 529 1476 529 1476 768 c 0,20,21 + 1476 979 1476 979 1333 1135.5 c 128,-1,22 + 1190 1292 1190 1292 940 1305 c 1,23,-1 + 940 1536 l 1,24,25 + 1300 1526 1300 1526 1504 1288 c 128,-1,26 + 1708 1050 1708 1050 1708 768 c 0,7,8 +EndSplineSet +EndChar + +StartChar: glyph357 +Encoding: 426 -1 357 +Width: 2220 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 64 + 55 + 4 + 71 + 4 + 87 + 4 + 103 + 4 + 119 + 4 + 135 + 4 + 151 + 4 + 166 + 4 + 8 + 39 + 4 + 55 + 4 + 71 + 4 + 86 + 4 + 118 + 4 + 134 + 4 + 6 + 0 + 7 + 98 + 1 + 8 + 98 + 0 + 0 + 1 + 4 + 2 + 6 + 3 + 2 + 200 + 5 + 6 + 8 + 200 + 10 + 10 + 7 + 1 + 8 + 10 + 4 + 0 + 4 + 0 + 12 + 2 + 6 + 4 + 11 +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,white] +SRP1 +SRP2 +SLOOP +IP +MDAP[rnd] +SVTCA[y-axis] +MDAP[rnd] +MIRP[rp0,min,rnd,black] +MDAP[rnd] +MDRP[min,rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rp0,min,rnd,grey] +SRP1 +SRP2 +IP +SPVTL[orthog] +SFVTCA[x-axis] +SRP0 +MIRP[rp0,min,rnd,black] +MDAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +DELTAP1 +SVTCA[x-axis] +DELTAP2 +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +2147 1416 m 1,0,-1 + 1343 382 l 1,1,-1 + 628 382 l 1,2,-1 + 628 22 l 1,3,-1 + 132 516 l 1,4,-1 + 628 1014 l 1,5,-1 + 628 654 l 1,6,-1 + 1229 654 l 1,7,-1 + 1604 1152 l 1,8,-1 + 1062 1152 l 1,9,-1 + 1062 1416 l 1,10,-1 + 2147 1416 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: glyph358 +Encoding: 427 -1 358 +Width: 2220 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +SplineSet +99 1416 m 1,0,-1 + 1184 1416 l 1,1,-1 + 1184 1152 l 1,2,-1 + 642 1152 l 1,3,-1 + 1017 654 l 1,4,-1 + 1618 654 l 1,5,-1 + 1618 1014 l 1,6,-1 + 2114 516 l 1,7,-1 + 1618 22 l 1,8,-1 + 1618 382 l 1,9,-1 + 903 382 l 1,10,-1 + 99 1416 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: apple +Encoding: 15 20 359 +Width: 1810 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 44 + 22 + 30 + 56 + 18 + 69 + 19 + 70 + 30 + 4 + 15 + 48 + 54 + 48 + 87 + 48 + 2 + 15 + 48 + 27 + 36 + 12 + 9 + 39 + 42 + 36 + 24 + 21 + 27 + 1 + 0 + 9 + 9 + 54 + 33 + 54 + 48 + 15 + 86 + 33 + 102 + 33 + 2 + 33 + 57 +SRP0 +MDRP[rp0,min,rnd,white] +DELTAP1 +MDRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[min,rnd,grey] +SRP1 +SRP2 +IP +MDAP[rnd] +MDRP[rnd,grey] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDAP[rnd] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[rp0,rnd,grey] +MDRP[rp0,min,rnd,grey] +SRP1 +SRP2 +IP +IP +DELTAP1 +RTHG +MDAP[rnd] +MDAP[rnd] +RTG +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1217 1642 m 0,0,1 + 1217 1579 1217 1579 1187 1503 c 128,-1,2 + 1157 1427 1157 1427 1092 1362 c 0,3,4 + 1065 1335 1065 1335 1037.5 1316.5 c 128,-1,5 + 1010 1298 1010 1298 981 1288 c 0,6,7 + 965 1282 965 1282 938 1278 c 128,-1,8 + 911 1274 911 1274 874 1272 c 1,9,10 + 877 1423 877 1423 954.5 1535 c 128,-1,11 + 1032 1647 1032 1647 1210 1688 c 1,12,13 + 1214 1673 1214 1673 1215.5 1664 c 128,-1,14 + 1217 1655 1217 1655 1217 1642 c 0,0,1 +1604 440 m 1,15,16 + 1585 379 1585 379 1552.5 314.5 c 128,-1,17 + 1520 250 1520 250 1477 184 c 0,18,19 + 1411 82 1411 82 1344.5 33 c 128,-1,20 + 1278 -16 1278 -16 1212 -16 c 0,21,22 + 1161 -16 1161 -16 1070 16.5 c 128,-1,23 + 979 49 979 49 915 49 c 0,24,25 + 852 49 852 49 769 14.5 c 128,-1,26 + 686 -20 686 -20 633 -20 c 0,27,28 + 555 -20 555 -20 478 46.5 c 128,-1,29 + 401 113 401 113 326 246 c 0,30,31 + 250 377 250 377 212 506 c 128,-1,32 + 174 635 174 635 174 762 c 0,33,34 + 174 997 174 997 290 1144.5 c 128,-1,35 + 406 1292 406 1292 582 1292 c 0,36,37 + 657 1292 657 1292 762.5 1261.5 c 128,-1,38 + 868 1231 868 1231 905 1231 c 0,39,40 + 950 1231 950 1231 1051.5 1265.5 c 128,-1,41 + 1153 1300 1153 1300 1229 1300 c 0,42,43 + 1290 1300 1290 1300 1344.5 1284 c 128,-1,44 + 1399 1268 1399 1268 1446 1235 c 0,45,46 + 1470 1217 1470 1217 1499 1190 c 128,-1,47 + 1528 1163 1528 1163 1554 1130 c 1,48,49 + 1513 1098 1513 1098 1484.5 1066 c 128,-1,50 + 1456 1034 1456 1034 1438 1010 c 0,51,52 + 1405 963 1405 963 1387.5 909.5 c 128,-1,53 + 1370 856 1370 856 1370 799 c 0,54,55 + 1370 672 1370 672 1441.5 569.5 c 128,-1,56 + 1513 467 1513 467 1604 440 c 1,15,16 +EndSplineSet +EndChar + +StartChar: glyph360 +Encoding: 428 -1 360 +Width: 1708 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 89 + 27 + 15 + 85 + 16 + 121 + 6 + 136 + 6 + 152 + 6 + 168 + 6 + 6 + 20 + 17 + 17 + 15 + 18 + 15 + 18 + 15 + 13 + 4 + 5 + 217 + 10 + 10 + 1 + 2 + 6 + 11 + 225 + 3 + 13 + 225 + 1 + 3 + 8 + 6 + 24 + 6 + 40 + 6 + 56 + 6 + 88 + 6 + 5 + 6 + 217 + 10 + 14 + 32 + 20 + 48 + 20 + 2 + 20 + 15 + 16 + 15 + 17 + 31 + 17 + 47 + 17 + 63 + 17 + 4 + 17 + 4 + 15 + 20 + 15 + 2 + 15 + 10 + 15 + 0 + 1 + 5 + 8 + 225 + 0 + 22 + 13 + 225 + 1 + 21 +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[rp0,min,rnd,black] +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP1 +SRP2 +IP +IP +RTHG +MDAP[rnd] +DELTAP1 +MDRP[rp0,min,rnd,grey] +DELTAP1 +RTG +MDRP[rp0,min,rnd,grey] +RTHG +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +RTG +MDRP[rp0,min,rnd,grey] +MDAP[rnd] +MIRP[rp0,min,rnd,black] +DELTAP1 +MDRP[grey] +SVTCA[y-axis] +MDAP[rnd] +MIRP[rp0,min,rnd,black] +MDAP[rnd] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP1 +SRP2 +IP +MDAP[rnd] +MIRP[min,rnd,black] +SHP[rp2] +SRP2 +IP +IP +MDAP[rnd] +MDAP[rnd] +SRP2 +IP +MDAP[rnd] +MDRP[rnd,grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1586 -50 m 1,0,-1 + 122 -50 l 1,1,-1 + 122 1586 l 1,2,-1 + 1110 1586 l 1,3,-1 + 1586 1108 l 1,4,-1 + 1586 -50 l 1,0,-1 +1300 1108 m 1,5,-1 + 1024 1384 l 1,6,-1 + 1024 1108 l 1,7,-1 + 1300 1108 l 1,5,-1 +1366 170 m 1,8,-1 + 1366 915 l 1,9,-1 + 819 915 l 1,10,-1 + 819 1364 l 1,11,-1 + 342 1364 l 1,12,-1 + 342 170 l 1,13,-1 + 1366 170 l 1,8,-1 +1216 599 m 1,14,-1 + 854 239 l 1,15,-1 + 492 599 l 1,16,-1 + 664 599 l 1,17,-1 + 664 771 l 1,18,-1 + 1044 771 l 1,19,-1 + 1044 599 l 1,20,-1 + 1216 599 l 1,14,-1 +EndSplineSet +EndChar + +StartChar: pagedown +Encoding: 378 8671 361 +Width: 1536 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 32 + 55 + 22 + 71 + 22 + 87 + 22 + 103 + 22 + 119 + 22 + 135 + 22 + 151 + 22 + 7 + 39 + 22 + 54 + 22 + 71 + 22 + 86 + 22 + 103 + 22 + 119 + 22 + 134 + 22 + 151 + 22 + 8 +SVTCA[y-axis] +DELTAP2 +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +Refer: 337 8670 N 1 0 0 -1 0 1325 2 +EndChar + +StartChar: checkmark +Encoding: 13 18 362 +Width: 1886 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 59 + 138 + 0 + 185 + 5 + 2 + 52 + 4 + 1 + 4 + 200 + 3 + 4 + 1 + 50 + 5 + 134 + 5 + 2 + 5 + 200 + 183 + 0 + 1 + 0 + 0 + 1 + 4 + 200 + 1 + 7 + 3 + 1 + 3 + 200 + 2 + 2 + 1 + 3 + 2 + 4 + 0 + 4 + 1 + 5 + 1 + 0 + 5 + 0 + 3 + 2 + 1 + 4 + 4 + 0 + 2 + 0 + 7 + 2 + 6 +SRP0 +MDRP[rp0,rnd,white] +SRP0 +MDRP[rp0,rnd,white] +SRP1 +SRP2 +IP +RTHG +MDAP[rnd] +ALIGNRP +RTG +SRP2 +IP +SRP2 +IP +SVTCA[y-axis] +MIAP[rnd] +MDAP[rnd] +SRP2 +SLOOP +IP +SPVTL[orthog] +SRP0 +MIRP[rp0,min,rnd,black] +DELTAP1 +SRP0 +MIRP[rp0,min,rnd,black] +SPVTL[orthog] +SFVTCA[x-axis] +SRP0 +DELTAP1 +MIRP[rp0,min,rnd,black] +DELTAP1 +SRP0 +SFVTL[parallel] +MIRP[rp0,min,rnd,black] +DELTAP1 +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1792 1196 m 1,0,-1 + 596 0 l 1,1,-1 + 84 512 l 1,2,-1 + 279 707 l 1,3,-1 + 596 394 l 1,4,-1 + 1596 1389 l 1,5,-1 + 1792 1196 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: glyph363 +Encoding: 429 -1 363 +Width: 2220 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 44 + 7 + 5 + 9 + 9 + 0 + 8 + 176 + 8 + 2 + 8 + 17 + 200 + 5 + 6 + 16 + 12 + 1 + 200 + 13 + 2 + 0 + 15 + 32 + 15 + 2 + 15 + 15 + 7 + 16 + 7 + 5 + 9 + 0 + 0 + 4 + 16 + 12 + 10 + 226 + 4 + 20 + 1 + 16 + 19 +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP1 +SRP2 +IP +MDAP[rnd] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +SRP1 +SRP2 +IP +MDAP[rnd] +DELTAP1 +SVTCA[y-axis] +MDAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDAP[rnd] +MDRP[min,rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[min,rnd,grey] +DELTAP1 +MDRP[rnd,grey] +SRP1 +SRP2 +IP +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1115 1214 m 1,0,-1 + 91 1214 l 1,1,-1 + 91 1476 l 1,2,-1 + 1115 1476 l 1,3,-1 + 1115 1214 l 1,0,-1 +2050 385 m 1,4,-1 + 1119 385 l 1,5,-1 + 1119 26 l 1,6,-1 + 632 516 l 1,7,-1 + 1119 1006 l 1,8,-1 + 1119 647 l 1,9,-1 + 1800 647 l 1,10,-1 + 1800 1214 l 1,11,-1 + 1456 1214 l 1,12,-1 + 1456 1476 l 1,13,-1 + 2050 1476 l 1,14,-1 + 2050 385 l 1,4,-1 +517 385 m 1,15,-1 + 91 385 l 1,16,-1 + 91 647 l 1,17,-1 + 517 647 l 1,18,-1 + 517 385 l 1,15,-1 +EndSplineSet +EndChar + +StartChar: glyph364 +Encoding: 430 -1 364 +Width: 2220 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 17 + 56 + 7 + 72 + 7 + 88 + 7 + 105 + 7 + 120 + 7 + 136 + 7 + 152 + 7 + 168 + 7 + 8 +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +Refer: 363 -1 N -1 0 0 1 2221 0 2 +EndChar + +StartChar: glyph365 +Encoding: 431 -1 365 +Width: 1364 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 18 + 3 + 224 + 4 + 1 + 9 + 10 + 7 + 0 + 9 + 223 + 1 + 0 + 12 + 8 + 223 + 4 + 5 + 12 +SRP0 +MDRP[rp0,rnd,white] +MDRP[rp0,min,rnd,grey] +MIRP[rp0,min,rnd,black] +SRP0 +MDRP[rp0,rnd,white] +MDRP[rp0,min,rnd,grey] +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1366 -241 m 1,0,-1 + 1196 -241 l 1,1,-1 + 1196 -485 l 1,2,-1 + 172 -485 l 1,3,-1 + 172 -241 l 1,4,-1 + 0 -241 l 1,5,-1 + 0 0 l 1,6,-1 + 398 0 l 1,7,-1 + 398 -241 l 1,8,-1 + 970 -241 l 1,9,-1 + 970 0 l 1,10,-1 + 1366 0 l 1,11,-1 + 1366 -241 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: blackdiamond +Encoding: 14 19 366 +Width: 1536 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 12 + 1 + 222 + 3 + 222 + 0 + 2 + 0 + 222 + 2 + 222 + 1 + 3 +RTHG +MDAP[rnd] +ALIGNRP +MIRP[rnd,grey] +MIRP[rnd,grey] +SVTCA[y-axis] +MDAP[rnd] +ALIGNRP +MIRP[rnd,grey] +MIRP[rnd,grey] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1366 596 m 1,0,-1 + 768 0 l 1,1,-1 + 172 596 l 1,2,-1 + 768 1196 l 1,3,-1 + 1366 596 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: uniE12E +Encoding: 432 57646 367 +Width: 1836 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 99 + 5 + 8 + 22 + 8 + 20 + 52 + 20 + 56 + 37 + 16 + 37 + 17 + 41 + 20 + 42 + 26 + 36 + 55 + 37 + 56 + 58 + 16 + 73 + 37 + 70 + 55 + 85 + 17 + 84 + 55 + 118 + 34 + 16 + 0 + 3 + 30 + 42 + 36 + 6 + 4 + 15 + 21 + 48 + 204 + 38 + 24 + 1 + 24 + 21 + 45 + 51 + 204 + 3 + 27 + 21 + 55 + 60 + 1 + 60 + 204 + 12 + 15 + 63 + 57 + 204 + 9 + 15 + 1 + 0 + 3 + 3 + 33 + 18 + 36 + 205 + 7 + 6 + 1 + 6 + 42 + 205 + 30 + 39 + 205 + 73 + 33 + 103 + 33 + 2 + 33 + 30 + 6 + 102 + 54 + 1 + 54 + 205 + 18 + 66 +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[min,rnd,black] +DELTAP1 +MDRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +SRP0 +MIRP[rp0,min,grey] +SRP0 +DELTAP1 +MIRP[rp0,min,grey] +SRP1 +SRP2 +IP +MDAP[rnd] +MDRP[rnd,grey] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,rnd,grey] +MIRP[rp0,min,rnd,black] +DELTAP1 +MDAP[rnd] +MDRP[rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +SLOOP +IP +SRP0 +MDRP[rnd,grey] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1250.87011719 1690.28027344 m 1,0,1 + 1264.63964844 1529.89941406 1264.63964844 1529.89941406 1155.29003906 1399.48925781 c 128,-1,2 + 1045.93945312 1269.08007812 1045.93945312 1269.08007812 894.469726562 1275.56054688 c 1,3,4 + 883.129882812 1428.64941406 883.129882812 1428.64941406 986 1548.53027344 c 128,-1,5 + 1088.87011719 1668.41015625 1088.87011719 1668.41015625 1250.87011719 1690.28027344 c 1,0,1 +1642.09960938 435.58984375 m 1,6,7 + 1574.87011719 250.91015625 1574.87011719 250.91015625 1456.20507812 113.614257812 c 128,-1,8 + 1337.54003906 -23.6806640625 1337.54003906 -23.6806640625 1237.09960938 -23.6806640625 c 0,9,10 + 1187.68945312 -23.6806640625 1187.68945312 -23.6806640625 1091.70507812 13.1748046875 c 128,-1,11 + 995.719726562 50.0302734375 995.719726562 50.0302734375 936.58984375 50.0302734375 c 0,12,13 + 878.26953125 50.0302734375 878.26953125 50.0302734375 782.28515625 13.1748046875 c 128,-1,14 + 686.299804688 -23.6806640625 686.299804688 -23.6806640625 636.889648438 -23.6806640625 c 0,15,16 + 488.66015625 -23.6806640625 488.66015625 -23.6806640625 328.280273438 235.51953125 c 128,-1,17 + 167.900390625 494.719726562 167.900390625 494.719726562 167.900390625 781.459960938 c 0,18,19 + 167.900390625 1000.96972656 167.900390625 1000.96972656 295.474609375 1149.20019531 c 128,-1,20 + 423.049804688 1297.4296875 423.049804688 1297.4296875 624.739257812 1297.4296875 c 0,21,22 + 718.700195312 1297.4296875 718.700195312 1297.4296875 798.485351562 1260.16992188 c 128,-1,23 + 878.26953125 1222.91015625 878.26953125 1222.91015625 936.58984375 1222.91015625 c 0,24,25 + 970.610351562 1222.91015625 970.610351562 1222.91015625 1065.37988281 1260.16992188 c 128,-1,26 + 1160.15039062 1297.4296875 1160.15039062 1297.4296875 1249.25 1297.4296875 c 0,27,28 + 1358.59960938 1297.4296875 1358.59960938 1297.4296875 1448.10546875 1251.25976562 c 128,-1,29 + 1537.61035156 1205.08984375 1537.61035156 1205.08984375 1598.36035156 1124.89941406 c 1,30,31 + 1496.29980469 1062.53027344 1496.29980469 1062.53027344 1450.53515625 971.810546875 c 128,-1,32 + 1404.76953125 881.08984375 1404.76953125 881.08984375 1404.76953125 781.459960938 c 0,33,34 + 1404.76953125 667.25 1404.76953125 667.25 1478.88574219 572.075195312 c 128,-1,35 + 1553 476.899414062 1553 476.899414062 1642.09960938 435.58984375 c 1,6,7 +1468.76074219 379.700195312 m 1,36,37 + 1377.23046875 444.5 1377.23046875 444.5 1320.93554688 559.924804688 c 128,-1,38 + 1264.63964844 675.349609375 1264.63964844 675.349609375 1264.63964844 781.459960938 c 0,39,40 + 1264.63964844 900.530273438 1264.63964844 900.530273438 1300.28027344 983.5546875 c 128,-1,41 + 1335.91992188 1066.58007812 1335.91992188 1066.58007812 1384.51953125 1127.33007812 c 1,42,43 + 1355.36035156 1141.91015625 1355.36035156 1141.91015625 1320.53027344 1149.60546875 c 128,-1,44 + 1285.70019531 1157.29980469 1285.70019531 1157.29980469 1247.62988281 1157.29980469 c 0,45,46 + 1172.29980469 1157.29980469 1172.29980469 1157.29980469 1080.36425781 1120.0390625 c 128,-1,47 + 988.4296875 1082.78027344 988.4296875 1082.78027344 936.58984375 1082.78027344 c 0,48,49 + 856.400390625 1082.78027344 856.400390625 1082.78027344 778.639648438 1120.0390625 c 128,-1,50 + 700.879882812 1157.29980469 700.879882812 1157.29980469 625.549804688 1157.29980469 c 0,51,52 + 473.26953125 1157.29980469 473.26953125 1157.29980469 389.83984375 1049.1640625 c 128,-1,53 + 306.41015625 941.030273438 306.41015625 941.030273438 306.41015625 781.459960938 c 0,54,55 + 306.41015625 530.360351562 306.41015625 530.360351562 438.439453125 322.594726562 c 128,-1,56 + 570.469726562 114.830078125 570.469726562 114.830078125 635.26953125 114.830078125 c 0,57,58 + 669.290039062 114.830078125 669.290039062 114.830078125 757.174804688 151.280273438 c 128,-1,59 + 845.060546875 187.73046875 845.060546875 187.73046875 936.58984375 187.73046875 c 0,60,61 + 1028.9296875 187.73046875 1028.9296875 187.73046875 1116.41015625 151.280273438 c 128,-1,62 + 1203.88964844 114.830078125 1203.88964844 114.830078125 1237.91015625 114.830078125 c 0,63,64 + 1274.36035156 114.830078125 1274.36035156 114.830078125 1344.01953125 194.614257812 c 128,-1,65 + 1413.6796875 274.399414062 1413.6796875 274.399414062 1468.76074219 379.700195312 c 1,36,37 +EndSplineSet +EndChar + +StartChar: carriagereturn +Encoding: 8 11 368 +AltUni2: 00240d.ffffffff.0 00240d.ffffffff.0 +Width: 2388 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 48 + 23 + 21 + 55 + 7 + 71 + 7 + 86 + 7 + 103 + 7 + 119 + 7 + 135 + 7 + 151 + 7 + 8 + 39 + 6 + 55 + 6 + 71 + 6 + 134 + 6 + 150 + 6 + 5 + 6 + 4 + 8 + 5 + 4 + 200 + 7 + 8 + 17 + 200 + 18 + 17 + 12 + 219 + 0 + 24 + 8 + 4 + 6 + 23 +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +SVTCA[y-axis] +MDAP[rnd] +MIRP[min,rnd,black] +MDAP[rnd] +MDRP[min,rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rp0,min,rnd,grey] +SRP1 +SRP2 +IP +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP2 +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +2182 726 m 2,0,1 + 2182 587 2182 587 2081.5 481.5 c 128,-1,2 + 1981 376 1981 376 1838 376 c 2,3,-1 + 630 376 l 1,4,-1 + 630 16 l 1,5,-1 + 134 510 l 1,6,-1 + 630 1008 l 1,7,-1 + 630 648 l 1,8,-1 + 1722 648 l 2,9,10 + 1792 648 1792 648 1842 692 c 128,-1,11 + 1892 736 1892 736 1892 806 c 2,12,-1 + 1892 1030 l 2,13,14 + 1892 1100 1892 1100 1842 1139 c 128,-1,15 + 1792 1178 1792 1178 1722 1178 c 2,16,-1 + 1408 1178 l 1,17,-1 + 1408 1450 l 1,18,-1 + 1840 1450 l 2,19,20 + 1981 1450 1981 1450 2081.5 1349.5 c 128,-1,21 + 2182 1249 2182 1249 2182 1110 c 2,22,-1 + 2182 726 l 2,0,1 +EndSplineSet +EndChar + +StartChar: arrowdashleft +Encoding: 17 24 369 +AltUni2: 0021e0.ffffffff.0 0021e0.ffffffff.0 +Width: 2048 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 23 + 10 + 1 + 5 + 9 + 219 + 12 + 2 + 6 + 13 + 220 + 4 + 15 + 1 + 1 + 1 + 0 + 16 + 5 + 8 + 13 + 9 + 11 + 15 +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +MDRP[rp0,min,rnd,grey] +MDRP[rp0,min,rnd,white] +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rp0,min,rnd,black] +DELTAP1 +MDRP[rp0,min,rnd,white] +SVTCA[y-axis] +MIAP[rnd] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[min,rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[rp0,min,rnd,grey] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1899 624 m 1,0,-1 + 1559 624 l 1,1,-1 + 1559 912 l 1,2,-1 + 1899 912 l 1,3,-1 + 1899 624 l 1,0,-1 +1387 624 m 1,4,-1 + 1047 624 l 1,5,-1 + 1047 912 l 1,6,-1 + 1387 912 l 1,7,-1 + 1387 624 l 1,4,-1 +875 624 m 1,8,-1 + 619 624 l 1,9,-1 + 619 270 l 1,10,-1 + 121 768 l 1,11,-1 + 619 1266 l 1,12,-1 + 619 912 l 1,13,-1 + 875 912 l 1,14,-1 + 875 624 l 1,8,-1 +EndSplineSet +EndChar + +StartChar: arrowdashup +Encoding: 18 25 370 +AltUni2: 0021e1.ffffffff.0 0021e1.ffffffff.0 +Width: 1536 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 26 + 8 + 0 + 13 + 1 + 13 + 12 + 9 + 3 + 1 + 4 + 6 + 6 + 1 + 4 + 0 + 14 + 10 + 1 + 219 + 5 + 13 + 9 + 4 + 4 + 16 + 15 +SRP1 +SRP2 +IP +MDAP[rnd] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[min,rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[rnd,grey] +MDRP[rp0,min,rnd,grey] +SRP1 +SRP2 +IP +SVTCA[y-axis] +MDAP[rnd] +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +MDRP[rp0,min,rnd,grey] +MDRP[rp0,min,rnd,white] +MDAP[rnd] +MDRP[rp0,min,rnd,black] +DELTAP1 +MDRP[rp0,min,rnd,white] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1267 1168 m 1,0,-1 + 913 1168 l 1,1,-1 + 913 912 l 1,2,-1 + 625 912 l 1,3,-1 + 625 1168 l 1,4,-1 + 271 1168 l 1,5,-1 + 769 1666 l 1,6,-1 + 1267 1168 l 1,0,-1 +913 400 m 1,7,-1 + 625 400 l 1,8,-1 + 625 740 l 1,9,-1 + 913 740 l 1,10,-1 + 913 400 l 1,7,-1 +913 -112 m 1,11,-1 + 625 -112 l 1,12,-1 + 625 228 l 1,13,-1 + 913 228 l 1,14,-1 + 913 -112 l 1,11,-1 +EndSplineSet +EndChar + +StartChar: arrowdashright +Encoding: 19 26 371 +AltUni2: 0021e2.ffffffff.0 0021e2.ffffffff.0 +Width: 2048 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +SplineSet +159 624 m 1,0,-1 + 159 912 l 1,1,-1 + 499 912 l 1,2,-1 + 499 624 l 1,3,-1 + 159 624 l 1,0,-1 +671 624 m 1,4,-1 + 671 912 l 1,5,-1 + 1011 912 l 1,6,-1 + 1011 624 l 1,7,-1 + 671 624 l 1,4,-1 +1183 624 m 1,8,-1 + 1183 912 l 1,9,-1 + 1439 912 l 1,10,-1 + 1439 1266 l 1,11,-1 + 1937 768 l 1,12,-1 + 1439 270 l 1,13,-1 + 1439 624 l 1,14,-1 + 1183 624 l 1,8,-1 +EndSplineSet +EndChar + +StartChar: arrowdashdown +Encoding: 11 16 372 +AltUni2: 0021e3.ffffffff.0 0021e3.ffffffff.0 +Width: 1536 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +SplineSet +1267 256 m 1,0,-1 + 769 -242 l 1,1,-1 + 271 256 l 1,2,-1 + 625 256 l 1,3,-1 + 625 512 l 1,4,-1 + 913 512 l 1,5,-1 + 913 256 l 1,6,-1 + 1267 256 l 1,0,-1 +913 1024 m 1,7,-1 + 913 684 l 1,8,-1 + 625 684 l 1,9,-1 + 625 1024 l 1,10,-1 + 913 1024 l 1,7,-1 +913 1536 m 1,11,-1 + 913 1196 l 1,12,-1 + 625 1196 l 1,13,-1 + 625 1536 l 1,14,-1 + 913 1536 l 1,11,-1 +EndSplineSet +EndChar + +StartChar: glyph373 +Encoding: 433 -1 373 +Width: 1536 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 25 + 5 + 36 + 1 + 1 + 1 + 1 + 4 + 3 + 3 + 2 + 0 + 0 + 2 + 15 + 38 + 5 + 1 + 5 + 3 + 1 + 3 + 0 + 7 + 1 + 6 +SRP0 +MDRP[rp0,min,rnd,white] +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +DELTAP1 +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +SRP2 +SLOOP +IP +RTHG +MDAP[rnd] +DELTAP1 +ALIGNRP +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1386 0 m 1,0,-1 + 152 768 l 1,1,-1 + 1386 1536 l 1,2,-1 + 1386 0 l 1,0,-1 +1196 353 m 1,3,-1 + 1196 1183 l 1,4,-1 + 527 768 l 1,5,-1 + 1196 353 l 1,3,-1 +EndSplineSet +EndChar + +StartChar: foursuperior +Encoding: 364 8308 374 +Width: 849 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 39 + 6 + 9 + 13 + 110 + 1 + 4 + 4 + 3 + 137 + 12 + 1 + 9 + 12 + 29 + 12 + 121 + 12 + 3 + 12 + 7 + 14 + 3 + 81 + 0 + 9 + 2 + 99 + 7 + 12 + 3 + 13 + 6 + 6 + 22 + 6 + 2 + 6 + 5 + 14 +SRP0 +MDRP[rp0,rnd,white] +MDRP[rnd,grey] +DELTAP1 +MDRP[grey] +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[rp0,min,rnd,grey] +SVTCA[y-axis] +MIAP[rnd] +MIAP[rnd] +MDRP[grey] +DELTAP1 +DELTAP2 +SRP2 +IP +MDAP[rnd] +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDRP[grey] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +986 840 m 1,0,-1 + 841 840 l 1,1,-1 + 783 590 l 1,2,-1 + 581 590 l 1,3,-1 + 639 840 l 1,4,-1 + 222 840 l 1,5,-1 + 265 1026 l 1,6,-1 + 863 1538 l 1,7,-1 + 1002 1538 l 1,8,-1 + 884 1030 l 1,9,-1 + 1030 1030 l 1,10,-1 + 986 840 l 1,0,-1 +683 1030 m 1,11,-1 + 738 1268 l 1,12,-1 + 459 1030 l 1,13,-1 + 683 1030 l 1,11,-1 +EndSplineSet +EndChar + +StartChar: glyph375 +Encoding: 434 -1 375 +Width: 730 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 19 + 61 + 0 + 1 + 0 + 2 + 1 + 63 + 3 + 1 + 3 + 61 + 1 + 1 + 1 + 63 + 0 + 1 + 0 + 1 +MDAP[rnd] +MDAP[rnd] +DELTAP1 +SVTCA[y-axis] +MDAP[rnd] +DELTAP1 +MDAP[rnd] +DELTAP1 +SRP2 +IP +IP +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +721 823 m 1,0,-1 + 4 401 l 1,1,-1 + 4 602 l 1,2,-1 + 721 1026 l 1,3,-1 + 721 823 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: glyph376 +Encoding: 435 -1 376 +Width: 631 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 10 + 0 + 2 + 1 + 3 + 1 + 47 + 0 + 1 + 0 + 1 +MDAP[rnd] +MDAP[rnd] +DELTAP1 +SVTCA[y-axis] +MDAP[rnd] +MDAP[rnd] +SRP2 +IP +IP +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +635 805 m 1,0,-1 + 29 346 l 1,1,-1 + 29 526 l 1,2,-1 + 635 987 l 1,3,-1 + 635 805 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: glyph377 +Encoding: 436 -1 377 +Width: 735 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 13 + 1 + 31 + 2 + 47 + 0 + 1 + 0 + 207 + 1 + 1 + 1 + 71 + 4 +SRP0 +SROUND +MDRP[rp0,rnd,white] +RTG +DELTAP1 +MDRP[rp0,min,rnd,black] +DELTAP1 +SVTCA[y-axis] +MDAP[rnd] +MIRP[rp0,min,rnd,black] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +663 648 m 1,0,-1 + 60 648 l 1,1,-1 + 60 861 l 1,2,-1 + 663 861 l 1,3,-1 + 663 648 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: glyph378 +Encoding: 437 -1 378 +Width: 1389 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 52 + 7 + 36 + 12 + 53 + 12 + 2 + 64 + 12 + 96 + 15 + 15 + 9 + 95 + 9 + 111 + 9 + 223 + 9 + 4 + 9 + 15 + 1 + 31 + 1 + 2 + 1 + 4 + 10 + 71 + 15 + 87 + 15 + 103 + 15 + 3 + 15 + 12 + 0 + 32 + 17 + 1 + 17 + 1 + 9 + 11 + 12 + 8 + 47 + 9 + 1 + 9 + 18 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MDRP[rp0,min,rnd,black] +MDRP[grey] +MDRP[grey] +SRP0 +MDRP[rp0,min,rnd,grey] +MDRP[rp0,min,rnd,black] +DELTAP1 +MDRP[rnd,grey] +SVTCA[y-axis] +MIAP[rnd] +DELTAP1 +MDRP[rnd,grey] +MDRP[rp0,min,rnd,black] +MDRP[rnd,grey] +DELTAP1 +MDAP[rnd] +DELTAP1 +SRP0 +SMD +MDRP[rp0,min,grey] +SMD +DELTAP1 +MDRP[min,grey] +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1341 776 m 1,0,-1 + 1087 776 l 1,1,2 + 1085 834 1085 834 1067 864.5 c 128,-1,3 + 1049 895 1049 895 1004 895 c 0,4,5 + 967 895 967 895 940 876.5 c 128,-1,6 + 913 858 913 858 899 829 c 1,7,-1 + 899 324 l 1,8,-1 + 621 324 l 1,9,-1 + 621 1126 l 1,10,-1 + 887 1126 l 1,11,-1 + 893 975 l 1,12,13 + 930 1071 930 1071 990.5 1110 c 128,-1,14 + 1051 1149 1051 1149 1147 1149 c 0,15,16 + 1264 1149 1264 1149 1315 1053.5 c 128,-1,17 + 1366 958 1366 958 1341 776 c 1,0,-1 +EndSplineSet +EndChar + +StartChar: contextmenu +Encoding: 438 -1 379 +Width: 1881 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 147 + 25 + 22 + 25 + 23 + 39 + 11 + 70 + 10 + 85 + 8 + 82 + 10 + 84 + 28 + 117 + 8 + 8 + 103 + 4 + 121 + 11 + 2 + 10 + 7 + 10 + 11 + 4 + 28 + 6 + 29 + 59 + 27 + 58 + 30 + 6 + 27 + 30 + 9 + 8 + 6 + 5 + 6 + 25 + 1 + 57 + 24 + 1 + 21 + 24 + 20 + 3 + 26 + 48 + 25 + 1 + 25 + 25 + 23 + 32 + 26 + 57 + 26 + 2 + 26 + 13 + 16 + 204 + 19 + 15 + 73 + 121 + 28 + 1 + 11 + 7 + 4 + 15 + 28 + 29 + 10 + 26 + 8 + 2 + 23 + 1 + 2 + 9 + 8 + 6 + 5 + 30 + 27 + 6 + 13 + 16 + 28 + 29 + 4 + 7 + 10 + 11 + 6 + 12 + 17 + 61 + 21 + 63 + 22 + 63 + 23 + 61 + 24 + 4 + 24 + 21 + 23 + 22 + 4 + 20 + 26 + 20 + 26 + 0 + 205 + 1 + 16 + 26 + 1 + 26 + 26 + 12 + 17 + 13 + 15 + 12 + 31 + 12 + 47 + 12 + 159 + 12 + 4 + 12 + 32 + 16 + 17 + 31 +SRP0 +MDRP[rp0,rnd,white] +MDRP[min,rnd,black] +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MDRP[min,rnd,black] +SRP1 +SRP2 +IP +MDAP[rnd] +DELTAP1 +MDRP[rp0,rnd,grey] +MIRP[rp0,min,rnd,black] +SRP0 +MDRP[grey] +SRP1 +SRP2 +SLOOP +IP +DELTAP1 +SRP1 +SRP2 +SLOOP +IP +SRP1 +SRP2 +SLOOP +IP +SVTCA[y-axis] +MDAP[rnd] +MDRP[rp0,min,rnd,grey] +MDAP[rnd] +SRP2 +SLOOP +IP +DELTAP1 +SROUND +MDAP[rnd] +RTG +MDRP[rnd,grey] +MIRP[rp0,min,rnd,black] +MDRP[rnd,grey] +MDAP[rnd] +DELTAP1 +SRP2 +IP +MDAP[rnd] +DELTAP1 +SRP2 +SLOOP +IP +DELTAP1 +SRP1 +SRP2 +SLOOP +IP +DELTAP1 +IUP[x] +IUP[y] +SVTCA[y-axis] +DELTAP1 +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1026 1170 m 1,0,-1 + 855 1170 l 1,1,-1 + 855 1476 l 1,2,-1 + 1026 1476 l 1,3,-1 + 1026 1170 l 1,0,-1 +1546 1246 m 1,4,-1 + 1304 1003 l 1,5,-1 + 1183 1124 l 1,6,-1 + 1426 1366 l 1,7,-1 + 1546 1246 l 1,4,-1 +715 1107 m 1,8,-1 + 594 986 l 1,9,-1 + 379 1201 l 1,10,-1 + 500 1322 l 1,11,-1 + 715 1107 l 1,8,-1 +1710 681 m 1,12,-1 + 1368 681 l 1,13,-1 + 1368 852 l 1,14,-1 + 1710 852 l 1,15,-1 + 1710 681 l 1,12,-1 +513 681 m 1,16,-1 + 171 681 l 1,17,-1 + 171 852 l 1,18,-1 + 513 852 l 1,19,-1 + 513 681 l 1,16,-1 +1347 519 m 1,20,-1 + 1180 450 l 1,21,-1 + 1344 57 l 1,22,-1 + 1186 -8 l 1,23,-1 + 1023 386 l 1,24,-1 + 856 317 l 1,25,-1 + 855 1013 l 1,26,-1 + 1347 519 l 1,20,-1 +700 399 m 1,27,-1 + 457 157 l 1,28,-1 + 337 277 l 1,29,-1 + 580 520 l 1,30,-1 + 700 399 l 1,27,-1 +EndSplineSet +EndChar + +StartChar: power +Encoding: 439 -1 380 +Width: 1704 +GlyphClass: 1 +Flags: W +CounterMasks: 1 1c +TtInstrs: +NPUSHB + 38 + 45 + 14 + 34 + 16 + 34 + 20 + 45 + 22 + 4 + 25 + 26 + 25 + 26 + 3 + 9 + 21 + 201 + 3 + 211 + 15 + 201 + 9 + 24 + 25 + 25 + 0 + 6 + 12 + 201 + 0 + 211 + 18 + 201 + 48 + 6 + 1 + 6 + 28 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MIRP[min,rnd,black] +MIRP[rp0,min,rnd,black] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +MDAP[rnd] +MDRP[rp0,min,rnd,black] +SVTCA[y-axis] +MDAP[rnd] +MIRP[min,rnd,black] +MIRP[rp0,min,rnd,black] +MIRP[rp0,min,rnd,black] +SRP1 +SRP2 +IP +IP +MDAP[rnd] +MDAP[rnd] +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1717 765 m 256,0,1 + 1717 415 1717 415 1469.5 167.5 c 128,-1,2 + 1222 -80 1222 -80 872 -80 c 256,3,4 + 522 -80 522 -80 274.5 167.5 c 128,-1,5 + 27 415 27 415 27 765 c 256,6,7 + 27 1115 27 1115 274.5 1362.5 c 128,-1,8 + 522 1610 522 1610 872 1610 c 256,9,10 + 1222 1610 1222 1610 1469.5 1362.5 c 128,-1,11 + 1717 1115 1717 1115 1717 765 c 256,0,1 +1505 765 m 0,12,13 + 1505 1030 1505 1030 1319 1217 c 128,-1,14 + 1133 1404 1133 1404 872 1404 c 0,15,16 + 603 1404 603 1404 421 1219 c 128,-1,17 + 239 1034 239 1034 239 765 c 0,18,19 + 239 497 239 497 421 312 c 128,-1,20 + 603 127 603 127 872 127 c 0,21,22 + 1149 127 1149 127 1327 310.5 c 128,-1,23 + 1505 494 1505 494 1505 765 c 0,12,13 +992 358 m 1,24,-1 + 752 358 l 1,25,-1 + 752 1172 l 1,26,-1 + 992 1172 l 1,27,-1 + 992 358 l 1,24,-1 +EndSplineSet +EndChar + +StartChar: glyph381 +Encoding: 440 -1 381 +Width: 1365 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 35 + 51 + 0 + 1 + 0 + 100 + 63 + 3 + 1 + 3 + 54 + 1 + 1 + 1 + 100 + 2 + 2 + 3 + 0 + 3 + 1 + 56 + 2 + 1 + 2 + 4 + 2 + 63 + 1 + 1 + 1 + 0 + 48 + 3 + 1 + 3 +SVTCA[y-axis] +MDAP[rnd] +DELTAP1 +MDRP[grey] +MDAP[rnd] +DELTAP1 +MDRP[grey] +SVTCA[x-axis] +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +MDRP[rnd,grey] +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +SPVTL[orthog] +SRP0 +MIRP[rp0,min,black] +DELTAP1 +SRP0 +DELTAP1 +MIRP[rp0,min,black] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1341 1569 m 1,0,-1 + 281 -168 l 1,1,-1 + 117 -45 l 1,2,-1 + 1165 1669 l 1,3,-1 + 1341 1569 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: glyph382 +Encoding: 441 -1 382 +Width: 1360 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 30 + 7 + 0 + 22 + 0 + 2 + 0 + 118 + 8 + 3 + 50 + 3 + 2 + 3 + 55 + 1 + 1 + 1 + 118 + 2 + 2 + 3 + 0 + 3 + 1 + 2 + 4 + 2 + 1 + 0 + 3 +SVTCA[y-axis] +MDAP[rnd] +MDRP[grey] +MDAP[rnd] +MDRP[grey] +SVTCA[x-axis] +SRP0 +MDRP[rp0,rnd,white] +MDRP[rnd,grey] +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +SPVTL[orthog] +SRP0 +MIRP[rp0,min,black] +DELTAP1 +SRP0 +DELTAP1 +MIRP[rp0,min,black] +DELTAP1 +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1235 1239 m 1,0,-1 + 252 -135 l 1,1,-1 + 141 -45 l 1,2,-1 + 1126 1323 l 1,3,-1 + 1235 1239 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: glyph383 +Encoding: 442 -1 383 +Width: 0 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +SplineSet +24 2048 m 1,0,-1 + 24 -512 l 1,1,-1 + 24 0 l 1,2,-1 + 1048 0 l 1,3,-1 + -1000 0 l 1,4,-1 + 24 2048 l 1,0,-1 +EndSplineSet +EndChar + +StartChar: Ccircumflex +Encoding: 222 264 384 +Width: 1190 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 38 67 N 1 0 0 1 0 0 2 +Refer: 216 710 N 1 0 0 1 152 309 2 +EndChar + +StartChar: ccircumflex +Encoding: 223 265 385 +Width: 1124 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 70 99 N 1 0 0 1 0 0 2 +Refer: 216 710 N 1 0 0 1 40 -41 2 +EndChar + +StartChar: Cdotaccent +Encoding: 224 266 386 +Width: 1190 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 38 67 N 1 0 0 1 0 0 2 +Refer: 220 729 N 1 0 0 1 251 303 2 +EndChar + +StartChar: cdotaccent +Encoding: 225 267 387 +Width: 1124 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 70 99 N 1 0 0 1 0 0 2 +Refer: 220 729 N 1 0 0 1 128 -47 2 +EndChar + +StartChar: Ebreve +Encoding: 234 276 388 +Width: 1167 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 40 69 N 1 0 0 1 0 0 2 +Refer: 219 728 N 1 0 0 1 182 294 2 +EndChar + +StartChar: ebreve +Encoding: 235 277 389 +Width: 1303 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 72 101 N 1 0 0 1 0 0 2 +Refer: 219 728 N 1 0 0 1 134 -35 2 +EndChar + +StartChar: Gcircumflex +Encoding: 242 284 390 +Width: 1360 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 42 71 N 1 0 0 1 0 0 2 +Refer: 216 710 N 1 0 0 1 238 309 2 +EndChar + +StartChar: gcircumflex +Encoding: 243 285 391 +Width: 1337 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 74 103 N 1 0 0 1 0 0 2 +Refer: 216 710 N 1 0 0 1 117 -56 2 +EndChar + +StartChar: Gdotaccent +Encoding: 246 288 392 +Width: 1360 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 42 71 N 1 0 0 1 0 0 2 +Refer: 220 729 N 1 0 0 1 317 303 2 +EndChar + +StartChar: gdotaccent +Encoding: 247 289 393 +Width: 1337 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 74 103 N 1 0 0 1 0 0 2 +Refer: 220 729 N 1 0 0 1 116 -62 2 +EndChar + +StartChar: gacute +Encoding: 443 -1 394 +Width: 1337 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 74 103 N 1 0 0 1 0 0 2 +Refer: 141 180 N 1 0 0 1 257 -56 2 +EndChar + +StartChar: Hcircumflex +Encoding: 250 292 395 +Width: 1360 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 43 72 N 1 0 0 1 0 0 2 +Refer: 216 710 N 1 0 0 1 233 288 2 +EndChar + +StartChar: hcircumflex +Encoding: 251 293 396 +Width: 1296 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 75 104 N 1 0 0 1 0 0 2 +Refer: 216 710 N 1 0 0 1 188 288 2 +EndChar + +StartChar: Hbar +Encoding: 252 294 397 +Width: 1360 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 43 72 N 1 0 0 1 0 0 2 +Refer: 178 8211 N 1 0 0 1 258 392 2 +EndChar + +StartChar: hbar +Encoding: 253 295 398 +Width: 1296 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 75 104 N 1 0 0 1 0 0 2 +Refer: 218 175 N 1 0 0 1 -231 -165 2 +EndChar + +StartChar: Itilde +Encoding: 254 296 399 +Width: 750 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 44 73 N 1 0 0 1 0 0 2 +Refer: 217 732 N 1 0 0 1 -76 272 2 +EndChar + +StartChar: itilde +Encoding: 255 297 400 +Width: 610 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 215 305 N 1 0 0 1 0 0 2 +Refer: 217 732 N 1 0 0 1 -227 -86 2 +EndChar + +StartChar: Iogonek +Encoding: 258 302 401 +Width: 750 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 44 73 N 1 0 0 1 0 0 2 +Refer: 224 731 N 1 0 0 1 -94 0 2 +EndChar + +StartChar: Jcircumflex +Encoding: 264 308 402 +Width: 954 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 45 74 N 1 0 0 1 0 0 2 +Refer: 216 710 N 1 0 0 1 153 288 2 +EndChar + +StartChar: jcircumflex +Encoding: 265 309 403 +Width: 633 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 77 106 N 1 0 0 1 0 0 2 +Refer: 216 710 N 1 0 0 1 -212 -102 2 +EndChar + +StartChar: ncommaaccentaltone +Encoding: 444 -1 404 +Width: 1296 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 81 110 N 1 0 0 1 0 0 2 +Refer: 15 44 N 1 0 0 1 25 1612 2 +EndChar + +StartChar: Obreve +Encoding: 285 334 405 +Width: 1365 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 50 79 N 1 0 0 1 0 0 2 +Refer: 219 728 N 1 0 0 1 248 315 2 +EndChar + +StartChar: obreve +Encoding: 286 335 406 +Width: 1360 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 82 111 N 1 0 0 1 0 0 2 +Refer: 219 728 N 1 0 0 1 159 -35 2 +EndChar + +StartChar: Rcaron +Encoding: 295 344 407 +Width: 1251 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 53 82 N 1 0 0 1 0 0 2 +Refer: 225 711 N 1 0 0 1 193 288 2 +EndChar + +StartChar: rcaron +Encoding: 296 345 408 +Width: 1020 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 85 114 N 1 0 0 1 0 0 2 +Refer: 225 711 N 1 0 0 1 70 -47 2 +EndChar + +StartChar: Scircumflex +Encoding: 299 348 409 +Width: 1155 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 54 83 N 1 0 0 1 0 0 2 +Refer: 216 710 N 1 0 0 1 118 309 2 +EndChar + +StartChar: scircumflex +Encoding: 300 349 410 +Width: 932 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 86 115 N 1 0 0 1 0 0 2 +Refer: 216 710 N 1 0 0 1 -63 -52 2 +EndChar + +StartChar: Tbar +Encoding: 309 358 411 +Width: 1024 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 55 84 N 1 0 0 1 0 0 2 +Refer: 218 175 N 1 0 0 1 -154 -672 2 +EndChar + +StartChar: tbar +Encoding: 310 359 412 +Width: 850 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 87 116 N 1 0 0 1 0 0 2 +Refer: 218 175 N 1 0 0 1 -254 -677 2 +EndChar + +StartChar: Utilde +Encoding: 311 360 413 +Width: 1343 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 56 85 N 1 0 0 1 0 0 2 +Refer: 217 732 N 1 0 0 1 229 272 2 +EndChar + +StartChar: utilde +Encoding: 312 361 414 +Width: 1296 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 88 117 N 1 0 0 1 0 0 2 +Refer: 217 732 N 1 0 0 1 115 -86 2 +EndChar + +StartChar: Ubreve +Encoding: 315 364 415 +Width: 1343 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 56 85 N 1 0 0 1 0 0 2 +Refer: 219 728 N 1 0 0 1 236 294 2 +EndChar + +StartChar: ubreve +Encoding: 316 365 416 +Width: 1296 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 88 117 N 1 0 0 1 0 0 2 +Refer: 219 728 N 1 0 0 1 122 -64 2 +EndChar + +StartChar: Uogonek +Encoding: 321 370 417 +Width: 1343 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 56 85 N 1 0 0 1 0 0 2 +Refer: 224 731 N 1 0 0 1 207 -18 2 +EndChar + +StartChar: uogonek +Encoding: 322 371 418 +Width: 1296 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 88 117 N 1 0 0 1 0 0 2 +Refer: 224 731 N 1 0 0 1 492 10 2 +EndChar + +StartChar: Wcircumflex +Encoding: 323 372 419 +Width: 1931 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 58 87 N 1 0 0 1 0 0 2 +Refer: 216 710 N 1 0 0 1 518 288 2 +EndChar + +StartChar: wcircumflex +Encoding: 324 373 420 +Width: 1774 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 90 119 N 1 0 0 1 0 0 2 +Refer: 216 710 N 1 0 0 1 358 -70 2 +EndChar + +StartChar: Ycircumflex +Encoding: 325 374 421 +Width: 1327 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 60 89 N 1 0 0 1 0 0 2 +Refer: 216 710 N 1 0 0 1 219 288 2 +EndChar + +StartChar: ycircumflex +Encoding: 326 375 422 +Width: 1190 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 92 121 N 1 0 0 1 0 0 2 +Refer: 216 710 N 1 0 0 1 62 -70 2 +EndChar + +StartChar: Scommaaccent +Encoding: 445 -1 423 +Width: 1155 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 54 83 N 1 0 0 1 0 0 2 +Refer: 15 44 N 1 0 0 1 264 -462 2 +EndChar + +StartChar: scommaaccent +Encoding: 446 -1 424 +Width: 932 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 86 115 N 1 0 0 1 0 0 2 +Refer: 15 44 N 1 0 0 1 166 -467 2 +EndChar + +StartChar: Tcommaaccent +Encoding: 447 -1 425 +Width: 1024 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 55 84 N 1 0 0 1 0 0 2 +Refer: 15 44 N 1 0 0 1 217 -442 2 +EndChar + +StartChar: tcommaaccent +Encoding: 448 -1 426 +Width: 850 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 87 116 N 1 0 0 1 0 0 2 +Refer: 15 44 N 1 0 0 1 118 -452 2 +EndChar + +StartChar: Gcommaaccent +Encoding: 449 -1 427 +Width: 1360 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 42 71 N 1 0 0 1 0 0 2 +Refer: 15 44 N 1 0 0 1 384 -467 2 +EndChar + +StartChar: gcommaaccent +Encoding: 450 -1 428 +Width: 1337 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 74 103 N 1 0 0 1 0 0 2 +Refer: 182 8216 N 1 0 0 1 298 413 2 +EndChar + +StartChar: Lcommaaccent +Encoding: 451 -1 429 +Width: 1126 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 47 76 N 1 0 0 1 0 0 2 +Refer: 15 44 N 1 0 0 1 302 -442 2 +EndChar + +StartChar: lcommaaccent +Encoding: 452 -1 430 +Width: 635 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 79 108 N 1 0 0 1 0 0 2 +Refer: 15 44 N 1 0 0 1 22 -442 2 +EndChar + +StartChar: Kcommaaccent +Encoding: 453 -1 431 +Width: 1339 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 46 75 N 1 0 0 1 0 0 2 +Refer: 15 44 N 1 0 0 1 467 -442 2 +EndChar + +StartChar: kcommaaccent +Encoding: 454 -1 432 +Width: 1190 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 78 107 N 1 0 0 1 0 0 2 +Refer: 15 44 N 1 0 0 1 386 -442 2 +EndChar + +StartChar: Ncommaaccent +Encoding: 455 -1 433 +Width: 1397 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 49 78 N 1 0 0 1 0 0 2 +Refer: 15 44 N 1 0 0 1 403 -442 2 +EndChar + +StartChar: Rcommaaccent +Encoding: 456 -1 434 +Width: 1251 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 53 82 N 1 0 0 1 0 0 2 +Refer: 15 44 N 1 0 0 1 368 -442 2 +EndChar + +StartChar: rcommaaccent +Encoding: 457 -1 435 +Width: 1020 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 85 114 N 1 0 0 1 0 0 2 +Refer: 15 44 N 1 0 0 1 23 -442 2 +EndChar + +StartChar: commaaccent +Encoding: 404 61444 436 +Width: 584 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 15 44 N 1 0 0 1 0 0 2 +EndChar + +StartChar: carriagereturnleft +Encoding: 9 12 437 +Width: 2388 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 35 + 16 + 14 + 18 + 17 + 18 + 200 + 56 + 15 + 72 + 15 + 88 + 15 + 104 + 15 + 120 + 15 + 136 + 15 + 152 + 15 + 7 + 15 + 14 + 5 + 200 + 4 + 4 + 10 + 219 + 0 + 23 + 18 + 15 + 16 + 24 +SRP0 +MDRP[rp0,min,rnd,white] +MDRP[rp0,min,rnd,grey] +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,white] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +SVTCA[y-axis] +MDAP[rnd] +MIRP[min,rnd,black] +MDAP[rnd] +MDRP[min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +MDRP[rp0,min,rnd,grey] +SRP1 +SRP2 +IP +IUP[x] +IUP[y] +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +206 1110 m 2,0,1 + 206 1249 206 1249 306.5 1349.5 c 128,-1,2 + 407 1450 407 1450 550 1450 c 2,3,-1 + 980 1450 l 1,4,-1 + 980 1178 l 1,5,-1 + 666 1178 l 2,6,7 + 596 1178 596 1178 546 1139 c 128,-1,8 + 496 1100 496 1100 496 1030 c 2,9,-1 + 496 806 l 2,10,11 + 496 736 496 736 546 692 c 128,-1,12 + 596 648 596 648 666 648 c 2,13,-1 + 1758 648 l 1,14,-1 + 1758 1008 l 1,15,-1 + 2254 510 l 1,16,-1 + 1758 16 l 1,17,-1 + 1758 376 l 1,18,-1 + 550 376 l 2,19,20 + 407 376 407 376 306.5 481.5 c 128,-1,21 + 206 587 206 587 206 726 c 2,22,-1 + 206 1110 l 2,0,1 +EndSplineSet +EndChar + +StartChar: tcaron +Encoding: 308 357 438 +Width: 850 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 87 116 N 1 0 0 1 0 0 2 +Refer: 15 44 N 1 0 0 1 1008 1174 2 +EndChar + +StartChar: currency +Encoding: 122 164 439 +Width: 1230 +GlyphClass: 1 +Flags: W +TtInstrs: +NPUSHB + 54 + 57 + 0 + 57 + 10 + 53 + 28 + 3 + 53 + 1 + 58 + 9 + 54 + 18 + 57 + 19 + 53 + 27 + 5 + 14 + 23 + 92 + 39 + 45 + 92 + 5 + 32 + 98 + 153 + 36 + 175 + 36 + 217 + 36 + 3 + 36 + 150 + 42 + 214 + 42 + 2 + 42 + 98 + 80 + 14 + 1 + 47 + 14 + 64 + 14 + 176 + 14 + 3 + 14 + 48 +SRP0 +MDRP[rp0,rnd,white] +DELTAP1 +DELTAP2 +MIRP[min,rnd,black] +DELTAP1 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MIRP[rp0,min,rnd,black] +SVTCA[y-axis] +MDAP[rnd] +MIRP[min,rnd,black] +MDRP[rp0,min,rnd,grey] +MIRP[rp0,min,rnd,black] +MPPEM +GTEQ +IF +NPUSHB + 24 + 0 + 63 + 28 + 1 + 28 + 32 + 10 + 48 + 18 + 1 + 18 + 14 + 1 + 48 + 9 + 1 + 9 + 5 + 27 + 63 + 19 + 1 + 19 + 23 +SVTCA[y-axis] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MDRP[rnd,grey] +SVTCA[x-axis] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MDRP[rnd,grey] +SRP0 +MDRP[rp0,min,rnd,grey] +DELTAP1 +MDRP[rnd,grey] +EIF +IUP[x] +IUP[y] +SVTCA[x-axis] +DELTAP1 +SVTCA[y-axis] +DELTAP1 +EndTTInstrs +LayerCount: 2 +Fore +SplineSet +1207 236 m 1,0,-1 + 1043 94 l 1,1,-1 + 885 319 l 1,2,3 + 842 295 842 295 782 282.5 c 128,-1,4 + 722 270 722 270 682 270 c 0,5,6 + 640 270 640 270 586.5 282.5 c 128,-1,7 + 533 295 533 295 500 319 c 1,8,-1 + 239 94 l 1,9,-1 + 140 236 l 1,10,-1 + 401 461 l 1,11,12 + 384 508 384 508 382 566.5 c 128,-1,13 + 380 625 380 625 395 688 c 0,14,15 + 410 752 410 752 438.5 810 c 128,-1,16 + 467 868 467 868 506 915 c 1,17,-1 + 349 1141 l 1,18,-1 + 514 1282 l 1,19,-1 + 671 1057 l 1,20,21 + 715 1081 715 1081 774 1093.5 c 128,-1,22 + 833 1106 833 1106 875 1106 c 0,23,24 + 915 1106 915 1106 969 1093.5 c 128,-1,25 + 1023 1081 1023 1081 1055 1057 c 1,26,-1 + 1317 1282 l 1,27,-1 + 1416 1141 l 1,28,-1 + 1154 915 l 1,29,30 + 1172 868 1172 868 1172.5 804.5 c 128,-1,31 + 1173 741 1173 741 1160 688 c 0,32,33 + 1145 625 1145 625 1117 566.5 c 128,-1,34 + 1089 508 1089 508 1049 461 c 1,35,-1 + 1207 236 l 1,0,-1 +939 698 m 0,36,37 + 958 782 958 782 932 842.5 c 128,-1,38 + 906 903 906 903 828 903 c 0,39,40 + 754 903 754 903 698 842.5 c 128,-1,41 + 642 782 642 782 622 698 c 256,42,43 + 603 614 603 614 631 554 c 128,-1,44 + 659 494 659 494 733 494 c 0,45,46 + 811 494 811 494 864.5 550 c 128,-1,47 + 918 606 918 606 939 698 c 0,36,37 +EndSplineSet +EndChar + +StartChar: gravecmb +Encoding: 458 -1 440 +Width: 1024 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 67 96 N 1 0 0 1 0 0 2 +EndChar + +StartChar: acutecmb +Encoding: 459 -1 441 +Width: 1024 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 141 180 N 1 0 0 1 0 0 2 +EndChar + +StartChar: circumflexcmb +Encoding: 460 -1 442 +Width: 1024 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 216 710 N 1 0 0 1 0 0 2 +EndChar + +StartChar: tildecmb +Encoding: 461 -1 443 +Width: 1024 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 217 732 N 1 0 0 1 0 0 2 +EndChar + +StartChar: macroncmb +Encoding: 462 -1 444 +Width: 1024 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 218 175 N 1 0 0 1 0 0 2 +EndChar + +StartChar: brevecmb +Encoding: 463 -1 445 +Width: 1024 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 219 728 N 1 0 0 1 0 0 2 +EndChar + +StartChar: dotaccentcmb +Encoding: 464 -1 446 +Width: 1024 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 220 729 N 1 0 0 1 0 0 2 +EndChar + +StartChar: dieresiscmb +Encoding: 465 -1 447 +Width: 1024 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 142 168 N 1 0 0 1 0 0 2 +EndChar + +StartChar: ringcmb +Encoding: 466 -1 448 +Width: 1024 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 221 730 N 1 0 0 1 0 0 2 +EndChar + +StartChar: hungarumlautcmb +Encoding: 467 -1 449 +Width: 1024 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 223 733 N 1 0 0 1 0 0 2 +EndChar + +StartChar: caroncmb +Encoding: 468 -1 450 +Width: 1024 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 225 711 N 1 0 0 1 0 0 2 +EndChar + +StartChar: cedillacmb +Encoding: 469 -1 451 +Width: 1024 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 222 184 N 1 0 0 1 0 0 2 +EndChar + +StartChar: ogonekcmb +Encoding: 470 -1 452 +Width: 1024 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 224 731 N 1 0 0 1 0 0 2 +EndChar + +StartChar: soliduslongoverlaycmb +Encoding: 471 -1 453 +Width: 510 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 3 9 N 1 0 0 1 0 0 2 +EndChar + +StartChar: Omegagreek +Encoding: 472 -1 454 +Width: 510 +GlyphClass: 1 +Flags: W +LayerCount: 2 +Fore +Refer: 3 9 N 1 0 0 1 0 0 2 +EndChar + +StartChar: ncommaaccent +Encoding: 473 -1 455 +Width: 1296 +GlyphClass: 1 +Flags: HW +LayerCount: 2 +Fore +Refer: 81 110 N 1 0 0 1 0 0 2 +Refer: 15 44 N 1 0 0 1 350 -451 2 +EndChar + +StartChar: uniE0B0 +Encoding: 474 57520 456 +Width: 1051 +VWidth: 1000 +Flags: W +LayerCount: 2 +Fore +SplineSet +354.5 59 m 0,0,1 + 294.5 -1 294.5 -1 206.900390625 -2.2001953125 c 0,2,3 + 122.900390625 -2.2001953125 122.900390625 -2.2001953125 61.7001953125 59 c 128,-1,4 + 0.5 120.200195312 0.5 120.200195312 0.5 207.799804688 c 2,5,-1 + 0.5 1242.20019531 l 2,6,7 + 0.5 1328.59960938 0.5 1328.59960938 61.7001953125 1390.40039062 c 128,-1,8 + 122.900390625 1452.20019531 122.900390625 1452.20019531 206.900390625 1452.20019531 c 0,9,10 + 294.5 1452.20019531 294.5 1452.20019531 354.5 1391 c 2,11,-1 + 1038.5 725 l 1,12,13 + 628.099609375 325.400390625 628.099609375 325.400390625 354.5 59 c 0,0,1 +EndSplineSet +EndChar + +StartChar: F12 +Encoding: 475 59916 457 +Width: 2220 +Flags: W +LayerCount: 2 +Fore +SplineSet +1778 405 m 1,0,-1 + 1264 405 l 1,1,-1 + 1264 532 l 1,2,3 + 1458.5 700 1458.5 700 1530.75 783 c 128,-1,4 + 1603 866 1603 866 1603 932.5 c 0,5,6 + 1603 982.5 1603 982.5 1579.5 1018.75 c 128,-1,7 + 1556 1055 1556 1055 1505.5 1055 c 0,8,9 + 1449.5 1055 1449.5 1055 1431.5 1034.25 c 128,-1,10 + 1413.5 1013.5 1413.5 1013.5 1404 984.5 c 1,11,-1 + 1259 984.5 l 1,12,13 + 1281.5 1081 1281.5 1081 1351 1125.5 c 128,-1,14 + 1420.5 1170 1420.5 1170 1529 1170 c 0,15,16 + 1646 1170 1646 1170 1709 1108 c 128,-1,17 + 1772 1046 1772 1046 1772 954 c 0,18,19 + 1772 855.5 1772 855.5 1685.5 753.75 c 128,-1,20 + 1599 652 1599 652 1404 517.5 c 1,21,-1 + 1778 517.5 l 1,22,-1 + 1778 405 l 1,0,-1 +1214.5 404 m 1,23,-1 + 1060 404 l 1,24,-1 + 1060 978.5 l 1,25,-1 + 954.5 978.5 l 1,26,-1 + 954.5 1069.5 l 1,27,28 + 1026 1075.5 1026 1075.5 1064 1108 c 128,-1,29 + 1102 1140.5 1102 1140.5 1113 1158.5 c 1,30,-1 + 1214.5 1158.5 l 1,31,-1 + 1214.5 404 l 1,23,-1 +911 1046.5 m 1,32,-1 + 622 1046.5 l 1,33,-1 + 622 844 l 1,34,-1 + 846.5 844 l 1,35,-1 + 846.5 730 l 1,36,-1 + 622 730 l 1,37,-1 + 622 404.5 l 1,38,-1 + 452 404.5 l 1,39,-1 + 452 1159 l 1,40,-1 + 911 1159 l 1,41,-1 + 911 1046.5 l 1,32,-1 +1694 1536 m 1,42,-1 + 2048 1536 l 1,43,-1 + 2048 0 l 1,44,-1 + 1694 0 l 1,45,-1 + 1493 0 l 1,46,-1 + 727 0 l 1,47,-1 + 524 0 l 1,48,-1 + 172 0 l 1,49,-1 + 172 1536 l 1,50,-1 + 524 1536 l 1,51,-1 + 727 1536 l 1,52,-1 + 1493 1536 l 1,53,-1 + 1694 1536 l 1,42,-1 +1384 1364 m 1,54,-1 + 834 1364 l 1,55,-1 + 633 1364 l 1,56,-1 + 342 1364 l 1,57,-1 + 342 172 l 1,58,-1 + 633 172 l 1,59,-1 + 834 172 l 1,60,-1 + 1384 172 l 1,61,-1 + 1587 172 l 1,62,-1 + 1878 172 l 1,63,-1 + 1878 1364 l 1,64,-1 + 1587 1364 l 1,65,-1 + 1384 1364 l 1,54,-1 +EndSplineSet +EndChar + +StartChar: F11 +Encoding: 476 59915 458 +Width: 2220 +Flags: W +LayerCount: 2 +Fore +SplineSet +1685 404 m 5,0,-1 + 1530.5 404 l 5,1,-1 + 1530.5 978.5 l 5,2,-1 + 1425 978.5 l 5,3,-1 + 1425 1069.5 l 5,4,5 + 1496.5 1075.5 1496.5 1075.5 1534.5 1108 c 132,-1,6 + 1572.5 1140.5 1572.5 1140.5 1583.5 1158.5 c 5,7,-1 + 1685 1158.5 l 5,8,-1 + 1685 404 l 5,0,-1 +1298.5 404 m 5,9,-1 + 1144 404 l 5,10,-1 + 1144 978.5 l 5,11,-1 + 1038.5 978.5 l 5,12,-1 + 1038.5 1069.5 l 5,13,14 + 1110 1075.5 1110 1075.5 1148 1108 c 132,-1,15 + 1186 1140.5 1186 1140.5 1197 1158.5 c 5,16,-1 + 1298.5 1158.5 l 5,17,-1 + 1298.5 404 l 5,9,-1 +945 1046.5 m 5,18,-1 + 656 1046.5 l 5,19,-1 + 656 844 l 5,20,-1 + 880.5 844 l 5,21,-1 + 880.5 730 l 5,22,-1 + 656 730 l 5,23,-1 + 656 404.5 l 5,24,-1 + 486 404.5 l 5,25,-1 + 486 1159 l 5,26,-1 + 945 1159 l 5,27,-1 + 945 1046.5 l 5,18,-1 +1694 1536 m 1,28,-1 + 2048 1536 l 1,29,-1 + 2048 0 l 1,30,-1 + 1694 0 l 1,31,-1 + 1493 0 l 1,32,-1 + 727 0 l 1,33,-1 + 524 0 l 1,34,-1 + 172 0 l 1,35,-1 + 172 1536 l 1,36,-1 + 524 1536 l 1,37,-1 + 727 1536 l 1,38,-1 + 1493 1536 l 1,39,-1 + 1694 1536 l 1,28,-1 +1384 1364 m 1,40,-1 + 834 1364 l 1,41,-1 + 633 1364 l 1,42,-1 + 342 1364 l 1,43,-1 + 342 172 l 1,44,-1 + 633 172 l 1,45,-1 + 834 172 l 1,46,-1 + 1384 172 l 1,47,-1 + 1587 172 l 1,48,-1 + 1878 172 l 1,49,-1 + 1878 1364 l 1,50,-1 + 1587 1364 l 1,51,-1 + 1384 1364 l 1,40,-1 +EndSplineSet +EndChar + +StartChar: F10 +Encoding: 477 59914 459 +Width: 2220 +Flags: W +LayerCount: 2 +Fore +SplineSet +1799.5 756.5 m 2,0,1 + 1799.5 576.5 1799.5 576.5 1727.75 488.25 c 128,-1,2 + 1656 400 1656 400 1531 400 c 256,3,4 + 1406 400 1406 400 1334.25 488.25 c 128,-1,5 + 1262.5 576.5 1262.5 576.5 1262.5 756.5 c 2,6,-1 + 1262.5 818 l 2,7,8 + 1262.5 990 1262.5 990 1330.75 1084.25 c 128,-1,9 + 1399 1178.5 1399 1178.5 1531 1178.5 c 256,10,11 + 1663 1178.5 1663 1178.5 1731.25 1084.25 c 128,-1,12 + 1799.5 990 1799.5 990 1799.5 818 c 2,13,-1 + 1799.5 756.5 l 2,0,1 +1631 777 m 2,14,-1 + 1631 797.5 l 2,15,16 + 1631 950 1631 950 1610.75 1007 c 128,-1,17 + 1590.5 1064 1590.5 1064 1531 1064 c 256,18,19 + 1471.5 1064 1471.5 1064 1451.5 1007 c 128,-1,20 + 1431.5 950 1431.5 950 1431.5 797.5 c 2,21,-1 + 1431.5 777 l 2,22,23 + 1431.5 624.5 1431.5 624.5 1452 569.75 c 128,-1,24 + 1472.5 515 1472.5 515 1531 515 c 0,25,26 + 1590.5 515 1590.5 515 1610.75 569.75 c 128,-1,27 + 1631 624.5 1631 624.5 1631 777 c 2,14,-1 +1214.5 404 m 1,28,-1 + 1060 404 l 1,29,-1 + 1060 978.5 l 1,30,-1 + 954.5 978.5 l 1,31,-1 + 954.5 1069.5 l 1,32,33 + 1026 1075.5 1026 1075.5 1064 1108 c 128,-1,34 + 1102 1140.5 1102 1140.5 1113 1158.5 c 1,35,-1 + 1214.5 1158.5 l 1,36,-1 + 1214.5 404 l 1,28,-1 +911 1046.5 m 1,37,-1 + 622 1046.5 l 1,38,-1 + 622 844 l 1,39,-1 + 846.5 844 l 1,40,-1 + 846.5 730 l 1,41,-1 + 622 730 l 1,42,-1 + 622 404.5 l 1,43,-1 + 452 404.5 l 1,44,-1 + 452 1159 l 1,45,-1 + 911 1159 l 1,46,-1 + 911 1046.5 l 1,37,-1 +1694 1536 m 1,47,-1 + 2048 1536 l 1,48,-1 + 2048 0 l 1,49,-1 + 1694 0 l 1,50,-1 + 1493 0 l 1,51,-1 + 727 0 l 1,52,-1 + 524 0 l 1,53,-1 + 172 0 l 1,54,-1 + 172 1536 l 1,55,-1 + 524 1536 l 1,56,-1 + 727 1536 l 1,57,-1 + 1493 1536 l 1,58,-1 + 1694 1536 l 1,47,-1 +1384 1364 m 1,59,-1 + 834 1364 l 1,60,-1 + 633 1364 l 1,61,-1 + 342 1364 l 1,62,-1 + 342 172 l 1,63,-1 + 633 172 l 1,64,-1 + 834 172 l 1,65,-1 + 1384 172 l 1,66,-1 + 1587 172 l 1,67,-1 + 1878 172 l 1,68,-1 + 1878 1364 l 1,69,-1 + 1587 1364 l 1,70,-1 + 1384 1364 l 1,59,-1 +EndSplineSet +EndChar + +StartChar: F9 +Encoding: 478 59913 460 +Width: 2220 +Flags: W +LayerCount: 2 +Fore +SplineSet +1693.5 806.5 m 0,0,1 + 1678.5 612.5 1678.5 612.5 1602.5 508 c 0,2,3 + 1521.5 396 1521.5 396 1367 396 c 0,4,5 + 1327 396 1327 396 1296.75 401.25 c 128,-1,6 + 1266.5 406.5 1266.5 406.5 1241 411.5 c 1,7,-1 + 1241 541.5 l 1,8,9 + 1275 533.5 1275 533.5 1306.25 528.25 c 128,-1,10 + 1337.5 523 1337.5 523 1367 523 c 0,11,12 + 1439.5 523 1439.5 523 1478.5 569.5 c 128,-1,13 + 1517.5 616 1517.5 616 1520.5 696 c 1,14,15 + 1506 670.5 1506 670.5 1461.5 650 c 128,-1,16 + 1417 629.5 1417 629.5 1378 629.5 c 0,17,18 + 1261.5 629.5 1261.5 629.5 1197.25 697.25 c 128,-1,19 + 1133 765 1133 765 1133 889.5 c 0,20,21 + 1133 1018.5 1133 1018.5 1205.5 1095 c 128,-1,22 + 1278 1171.5 1278 1171.5 1408 1171.5 c 0,23,24 + 1547 1171.5 1547 1171.5 1615.5 1098 c 0,25,26 + 1680 1030 1680 1030 1693.5 872.5 c 0,27,28 + 1694.5 861 1694.5 861 1694.5 844 c 0,29,30 + 1694.5 840 1694.5 840 1694.5 835.5 c 0,31,32 + 1694 813 1694 813 1693.5 806.5 c 0,0,1 +1529 900.5 m 0,33,34 + 1529 983.5 1529 983.5 1498.75 1019.25 c 128,-1,35 + 1468.5 1055 1468.5 1055 1415 1055 c 0,36,37 + 1363 1055 1363 1055 1333 1019.75 c 128,-1,38 + 1303 984.5 1303 984.5 1303 900.5 c 0,39,40 + 1303 820.5 1303 820.5 1333 782.75 c 128,-1,41 + 1363 745 1363 745 1415 745 c 0,42,43 + 1461 745 1461 745 1495 782.25 c 128,-1,44 + 1529 819.5 1529 819.5 1529 900.5 c 0,33,34 +1015 1046.5 m 1,45,-1 + 726 1046.5 l 1,46,-1 + 726 844 l 1,47,-1 + 950.5 844 l 1,48,-1 + 950.5 730 l 1,49,-1 + 726 730 l 1,50,-1 + 726 404.5 l 1,51,-1 + 556 404.5 l 1,52,-1 + 556 1159 l 1,53,-1 + 1015 1159 l 1,54,-1 + 1015 1046.5 l 1,45,-1 +1694 1536 m 1,55,-1 + 2048 1536 l 1,56,-1 + 2048 0 l 1,57,-1 + 1694 0 l 1,58,-1 + 1493 0 l 1,59,-1 + 727 0 l 1,60,-1 + 524 0 l 1,61,-1 + 172 0 l 1,62,-1 + 172 1536 l 1,63,-1 + 524 1536 l 1,64,-1 + 727 1536 l 1,65,-1 + 1493 1536 l 1,66,-1 + 1694 1536 l 1,55,-1 +1384 1364 m 1,67,-1 + 834 1364 l 1,68,-1 + 633 1364 l 1,69,-1 + 342 1364 l 1,70,-1 + 342 172 l 1,71,-1 + 633 172 l 1,72,-1 + 834 172 l 1,73,-1 + 1384 172 l 1,74,-1 + 1587 172 l 1,75,-1 + 1878 172 l 1,76,-1 + 1878 1364 l 1,77,-1 + 1587 1364 l 1,78,-1 + 1384 1364 l 1,67,-1 +EndSplineSet +EndChar + +StartChar: F8 +Encoding: 479 59912 461 +Width: 2220 +Flags: W +LayerCount: 2 +Fore +SplineSet +1626 627.5 m 0,0,1 + 1626 518 1626 518 1556.25 455 c 128,-1,2 + 1486.5 392 1486.5 392 1373 392 c 0,3,4 + 1251 392 1251 392 1186.5 449.5 c 128,-1,5 + 1122 507 1122 507 1122 615 c 0,6,7 + 1122 671.5 1122 671.5 1154.75 724.5 c 128,-1,8 + 1187.5 777.5 1187.5 777.5 1248 800 c 1,9,-1 + 1248 811 l 1,10,11 + 1196 828.5 1196 828.5 1168.75 870.5 c 128,-1,12 + 1141.5 912.5 1141.5 912.5 1141.5 974 c 0,13,14 + 1141.5 1054.5 1141.5 1054.5 1202.5 1112 c 128,-1,15 + 1263.5 1169.5 1263.5 1169.5 1373 1169.5 c 0,16,17 + 1484.5 1169.5 1484.5 1169.5 1542.5 1112.75 c 128,-1,18 + 1600.5 1056 1600.5 1056 1600.5 967.5 c 0,19,20 + 1600.5 910.5 1600.5 910.5 1574.25 869 c 128,-1,21 + 1548 827.5 1548 827.5 1502 811 c 1,22,-1 + 1502 800 l 1,23,24 + 1558.5 782.5 1558.5 782.5 1592.25 740.5 c 128,-1,25 + 1626 698.5 1626 698.5 1626 627.5 c 0,0,1 +1456 954.5 m 0,26,27 + 1456 1005.5 1456 1005.5 1435 1030 c 128,-1,28 + 1414 1054.5 1414 1054.5 1373 1054.5 c 256,29,30 + 1332 1054.5 1332 1054.5 1311.5 1029.5 c 128,-1,31 + 1291 1004.5 1291 1004.5 1291 954.5 c 0,32,33 + 1291 901 1291 901 1312.5 877.5 c 128,-1,34 + 1334 854 1334 854 1373 854 c 0,35,36 + 1407 854 1407 854 1431.5 876.5 c 128,-1,37 + 1456 899 1456 899 1456 954.5 c 0,26,27 +1476.5 623.5 m 0,38,39 + 1476.5 688 1476.5 688 1449.75 717.75 c 128,-1,40 + 1423 747.5 1423 747.5 1373 747.5 c 0,41,42 + 1324.5 747.5 1324.5 747.5 1297.5 719.25 c 128,-1,43 + 1270.5 691 1270.5 691 1270.5 623.5 c 0,44,45 + 1270.5 562 1270.5 562 1297.25 534.5 c 128,-1,46 + 1324 507 1324 507 1373 507 c 0,47,48 + 1418 507 1418 507 1447.25 534.5 c 128,-1,49 + 1476.5 562 1476.5 562 1476.5 623.5 c 0,38,39 +1041 1046.5 m 1,50,-1 + 752 1046.5 l 1,51,-1 + 752 844 l 1,52,-1 + 976.5 844 l 1,53,-1 + 976.5 730 l 1,54,-1 + 752 730 l 1,55,-1 + 752 404.5 l 1,56,-1 + 582 404.5 l 1,57,-1 + 582 1159 l 1,58,-1 + 1041 1159 l 1,59,-1 + 1041 1046.5 l 1,50,-1 +1694 1536 m 1,60,-1 + 2048 1536 l 1,61,-1 + 2048 0 l 1,62,-1 + 1694 0 l 1,63,-1 + 1493 0 l 1,64,-1 + 727 0 l 1,65,-1 + 524 0 l 1,66,-1 + 172 0 l 1,67,-1 + 172 1536 l 1,68,-1 + 524 1536 l 1,69,-1 + 727 1536 l 1,70,-1 + 1493 1536 l 1,71,-1 + 1694 1536 l 1,60,-1 +1384 1364 m 1,72,-1 + 834 1364 l 1,73,-1 + 633 1364 l 1,74,-1 + 342 1364 l 1,75,-1 + 342 172 l 1,76,-1 + 633 172 l 1,77,-1 + 834 172 l 1,78,-1 + 1384 172 l 1,79,-1 + 1587 172 l 1,80,-1 + 1878 172 l 1,81,-1 + 1878 1364 l 1,82,-1 + 1587 1364 l 1,83,-1 + 1384 1364 l 1,72,-1 +EndSplineSet +EndChar + +StartChar: F7 +Encoding: 480 59911 462 +Width: 2220 +Flags: W +LayerCount: 2 +Fore +SplineSet +1665 1030 m 1,0,1 + 1545 956 1545 956 1470 805 c 0,2,3 + 1404 673 1404 673 1404 464 c 0,4,5 + 1405 433 1405 433 1406 404 c 1,6,-1 + 1241 404 l 1,7,8 + 1241 577 1241 577 1315 760 c 0,9,10 + 1388 944 1388 944 1537 1046 c 1,11,-1 + 1142 1046 l 1,12,-1 + 1142 1158 l 1,13,-1 + 1665 1158 l 1,14,-1 + 1665 1030 l 1,0,1 +1041 1046 m 1,15,-1 + 752 1046 l 1,16,-1 + 752 844 l 1,17,-1 + 976 844 l 1,18,-1 + 976 730 l 1,19,-1 + 752 730 l 1,20,-1 + 752 404 l 1,21,-1 + 582 404 l 1,22,-1 + 582 1159 l 1,23,-1 + 1041 1159 l 1,24,-1 + 1041 1046 l 1,15,-1 +1694 1536 m 1,25,-1 + 2048 1536 l 1,26,-1 + 2048 0 l 1,27,-1 + 1694 0 l 1,28,-1 + 1493 0 l 1,29,-1 + 727 0 l 1,30,-1 + 524 0 l 1,31,-1 + 172 0 l 1,32,-1 + 172 1536 l 1,33,-1 + 524 1536 l 1,34,-1 + 727 1536 l 1,35,-1 + 1493 1536 l 1,36,-1 + 1694 1536 l 1,25,-1 +1384 1364 m 1,37,-1 + 834 1364 l 1,38,-1 + 633 1364 l 1,39,-1 + 342 1364 l 1,40,-1 + 342 172 l 1,41,-1 + 633 172 l 1,42,-1 + 834 172 l 1,43,-1 + 1384 172 l 1,44,-1 + 1587 172 l 1,45,-1 + 1878 172 l 1,46,-1 + 1878 1364 l 1,47,-1 + 1587 1364 l 1,48,-1 + 1384 1364 l 1,37,-1 +EndSplineSet +EndChar + +StartChar: F6 +Encoding: 481 59910 463 +Width: 2220 +Flags: W +LayerCount: 2 +Fore +SplineSet +1694 676 m 0,0,1 + 1694 547 1694 547 1620 470 c 0,2,3 + 1547 394 1547 394 1417 394 c 0,4,5 + 1280 394 1280 394 1210 468 c 128,-1,6 + 1140 542 1140 542 1132 693 c 0,7,8 + 1131 702 1131 702 1131 716 c 128,-1,9 + 1131 730 1131 730 1131 730 c 0,10,11 + 1132 759 1132 759 1132 759 c 0,12,13 + 1131 772 1131 772 1131 788 c 0,14,15 + 1131 790 1131 790 1131 791 c 128,-1,16 + 1131 792 1131 792 1131 794 c 0,17,18 + 1130.98947368 795.378947368 1130.98947368 795.378947368 1130.98947368 796.757783934 c 0,19,20 + 1130.98947368 926.368421053 1130.98947368 926.368421053 1224 1055 c 0,21,22 + 1307 1170 1307 1170 1448 1170 c 0,23,24 + 1453 1170 1453 1170 1458 1170 c 0,25,26 + 1493 1170 1493 1170 1528 1164 c 0,27,28 + 1548 1161 1548 1161 1584 1154 c 1,29,-1 + 1584 1024 l 1,30,31 + 1549 1032 1549 1032 1519 1037 c 0,32,33 + 1488 1042 1488 1042 1458 1042 c 0,34,35 + 1456 1042 1456 1042 1453 1042 c 128,-1,36 + 1450 1042 1450 1042 1448 1042 c 128,-1,37 + 1446 1042 1446 1042 1443 1042 c 0,38,39 + 1440.84 1042.08 1440.84 1042.08 1438.704 1042.08 c 0,40,41 + 1387.44 1042.08 1387.44 1042.08 1350 996 c 0,42,43 + 1311 948 1311 948 1304 870 c 1,44,45 + 1319 895 1319 895 1363.5 915.5 c 128,-1,46 + 1408 936 1408 936 1447 936 c 0,47,48 + 1564 936 1564 936 1629 868 c 128,-1,49 + 1694 800 1694 800 1694 676 c 0,0,1 +1524 664 m 0,50,51 + 1524 744 1524 744 1493 781.5 c 128,-1,52 + 1462 819 1462 819 1410 819 c 0,53,54 + 1363 819 1363 819 1330 782 c 0,55,56 + 1296 744 1296 744 1296 664 c 0,57,58 + 1296 580 1296 580 1326 544 c 0,59,60 + 1356 509 1356 509 1410 509 c 0,61,62 + 1462 509 1462 509 1493 544.5 c 128,-1,63 + 1524 580 1524 580 1524 664 c 0,50,51 +1041 1046 m 1,64,-1 + 752 1046 l 1,65,-1 + 752 844 l 1,66,-1 + 976 844 l 1,67,-1 + 976 730 l 1,68,-1 + 752 730 l 1,69,-1 + 752 404 l 1,70,-1 + 582 404 l 1,71,-1 + 582 1159 l 1,72,-1 + 1041 1159 l 1,73,-1 + 1041 1046 l 1,64,-1 +1694 1536 m 1,74,-1 + 2048 1536 l 1,75,-1 + 2048 0 l 1,76,-1 + 1694 0 l 1,77,-1 + 1493 0 l 1,78,-1 + 727 0 l 1,79,-1 + 524 0 l 1,80,-1 + 172 0 l 1,81,-1 + 172 1536 l 1,82,-1 + 524 1536 l 1,83,-1 + 727 1536 l 1,84,-1 + 1493 1536 l 1,85,-1 + 1694 1536 l 1,74,-1 +1384 1364 m 1,86,-1 + 834 1364 l 1,87,-1 + 633 1364 l 1,88,-1 + 342 1364 l 1,89,-1 + 342 172 l 1,90,-1 + 633 172 l 1,91,-1 + 834 172 l 1,92,-1 + 1384 172 l 1,93,-1 + 1587 172 l 1,94,-1 + 1878 172 l 1,95,-1 + 1878 1364 l 1,96,-1 + 1587 1364 l 1,97,-1 + 1384 1364 l 1,86,-1 +EndSplineSet +EndChar + +StartChar: F5 +Encoding: 482 59909 464 +Width: 2220 +Flags: WO +LayerCount: 2 +Fore +SplineSet +1661 683 m 0,0,1 + 1661 530 1661 530 1579.5 462 c 128,-1,2 + 1498 394 1498 394 1369 394 c 0,3,4 + 1272 394 1272 394 1206 433 c 128,-1,5 + 1140 472 1140 472 1100 556 c 1,6,-1 + 1212 620 l 1,7,8 + 1235 562 1235 562 1273 537 c 0,9,10 + 1312 511 1312 511 1361 511 c 0,11,12 + 1424 511 1424 511 1456.5 550.5 c 128,-1,13 + 1489 590 1489 590 1489 668 c 0,14,15 + 1489 742 1489 742 1457 780 c 0,16,17 + 1424 817 1424 817 1365 817 c 0,18,19 + 1335 817 1335 817 1303 805 c 0,20,21 + 1272 794 1272 794 1241 768 c 1,22,-1 + 1142 824 l 1,23,-1 + 1157 1162 l 1,24,-1 + 1586 1162 l 1,25,-1 + 1586 1050 l 1,26,-1 + 1281 1050 l 1,27,-1 + 1274 910 l 1,28,29 + 1300 929 1300 929 1346 936 c 0,30,31 + 1390 943 1390 943 1409 943 c 0,32,33 + 1520 943 1520 943 1590.5 869.5 c 128,-1,34 + 1661 796 1661 796 1661 683 c 0,0,1 +1041 1046 m 1,35,-1 + 752 1046 l 1,36,-1 + 752 844 l 1,37,-1 + 976 844 l 1,38,-1 + 976 730 l 1,39,-1 + 752 730 l 1,40,-1 + 752 404 l 1,41,-1 + 582 404 l 1,42,-1 + 582 1159 l 1,43,-1 + 1041 1159 l 1,44,-1 + 1041 1046 l 1,35,-1 +1694 1536 m 1,45,-1 + 2048 1536 l 1,46,-1 + 2048 0 l 1,47,-1 + 1694 0 l 1,48,-1 + 1493 0 l 1,49,-1 + 727 0 l 1,50,-1 + 524 0 l 1,51,-1 + 172 0 l 1,52,-1 + 172 1536 l 1,53,-1 + 524 1536 l 1,54,-1 + 727 1536 l 1,55,-1 + 1493 1536 l 1,56,-1 + 1694 1536 l 1,45,-1 +1384 1364 m 1,57,-1 + 834 1364 l 1,58,-1 + 633 1364 l 1,59,-1 + 342 1364 l 1,60,-1 + 342 172 l 1,61,-1 + 633 172 l 1,62,-1 + 834 172 l 1,63,-1 + 1384 172 l 1,64,-1 + 1587 172 l 1,65,-1 + 1878 172 l 1,66,-1 + 1878 1364 l 1,67,-1 + 1587 1364 l 1,68,-1 + 1384 1364 l 1,57,-1 +EndSplineSet +EndChar + +StartChar: F4 +Encoding: 483 59908 465 +Width: 2220 +Flags: W +LayerCount: 2 +Fore +SplineSet +1620 625 m 1,0,-1 + 1566 625 l 1,1,-1 + 1566 406 l 1,2,-1 + 1413 406 l 1,3,-1 + 1413 625 l 1,4,-1 + 1082 625 l 1,5,-1 + 1074 770 l 1,6,-1 + 1464 1160 l 1,7,-1 + 1566 1160 l 1,8,-1 + 1566 734 l 1,9,-1 + 1620 734 l 1,10,-1 + 1620 625 l 1,0,-1 +1415 734 m 1,11,-1 + 1415 964 l 1,12,-1 + 1186 734 l 1,13,-1 + 1415 734 l 1,11,-1 +1041 1046 m 1,14,-1 + 752 1046 l 1,15,-1 + 752 844 l 1,16,-1 + 976 844 l 1,17,-1 + 976 730 l 1,18,-1 + 752 730 l 1,19,-1 + 752 404 l 1,20,-1 + 582 404 l 1,21,-1 + 582 1159 l 1,22,-1 + 1041 1159 l 1,23,-1 + 1041 1046 l 1,14,-1 +1694 1536 m 1,24,-1 + 2048 1536 l 1,25,-1 + 2048 0 l 1,26,-1 + 1694 0 l 1,27,-1 + 1493 0 l 1,28,-1 + 727 0 l 1,29,-1 + 524 0 l 1,30,-1 + 172 0 l 1,31,-1 + 172 1536 l 1,32,-1 + 524 1536 l 1,33,-1 + 727 1536 l 1,34,-1 + 1493 1536 l 1,35,-1 + 1694 1536 l 1,24,-1 +1384 1364 m 1,36,-1 + 834 1364 l 1,37,-1 + 633 1364 l 1,38,-1 + 342 1364 l 1,39,-1 + 342 172 l 1,40,-1 + 633 172 l 1,41,-1 + 834 172 l 1,42,-1 + 1384 172 l 1,43,-1 + 1587 172 l 1,44,-1 + 1878 172 l 1,45,-1 + 1878 1364 l 1,46,-1 + 1587 1364 l 1,47,-1 + 1384 1364 l 1,36,-1 +EndSplineSet +EndChar + +StartChar: F3 +Encoding: 484 59907 466 +Width: 2220 +Flags: W +LayerCount: 2 +Fore +SplineSet +1640 624 m 0,0,1 + 1640 519 1640 519 1567.5 455.5 c 128,-1,2 + 1495 392 1495 392 1377 392 c 0,3,4 + 1274 392 1274 392 1208 431 c 128,-1,5 + 1142 470 1142 470 1109 530 c 1,6,-1 + 1212 594 l 1,7,8 + 1226 560 1226 560 1266 534 c 0,9,10 + 1306 509 1306 509 1362 509 c 0,11,12 + 1417 509 1417 509 1449 542 c 0,13,14 + 1482 576 1482 576 1482 634 c 0,15,16 + 1482 635 1482 635 1482 637 c 0,17,18 + 1482 682 1482 682 1452 716 c 0,19,20 + 1424 748 1424 748 1364 748 c 2,21,-1 + 1288 748 l 1,22,-1 + 1288 854 l 1,23,-1 + 1368 854 l 2,24,25 + 1421 854 1421 854 1446 878 c 0,26,27 + 1472 903 1472 903 1472 952 c 0,28,29 + 1472 954 1472 954 1472 956 c 128,-1,30 + 1472 958 1472 958 1472 960 c 128,-1,31 + 1472 962 1472 962 1472 964 c 0,32,33 + 1472.04545455 965.636363636 1472.04545455 965.636363636 1472.04545455 967.256198347 c 0,34,35 + 1472.04545455 1001.27272727 1472.04545455 1001.27272727 1452 1028 c 0,36,37 + 1431 1056 1431 1056 1384 1056 c 0,38,39 + 1342 1056 1342 1056 1318.5 1041 c 128,-1,40 + 1295 1026 1295 1026 1283 1000 c 1,41,-1 + 1148 1000 l 1,42,43 + 1162 1070 1162 1070 1221 1120 c 128,-1,44 + 1280 1170 1280 1170 1384 1170 c 0,45,46 + 1386 1170 1386 1170 1388 1170 c 128,-1,47 + 1390 1170 1390 1170 1392 1170 c 128,-1,48 + 1394 1170 1394 1170 1396 1170 c 128,-1,49 + 1397.88135593 1170.01694915 1397.88135593 1170.01694915 1399.74834818 1170.01694915 c 0,50,51 + 1508.03389831 1170.01694915 1508.03389831 1170.01694915 1568 1113 c 0,52,53 + 1629 1055 1629 1055 1629 974 c 0,54,55 + 1629 901 1629 901 1592 861 c 0,56,57 + 1556 822 1556 822 1512 810 c 1,58,59 + 1566 790 1566 790 1603 744 c 128,-1,60 + 1640 698 1640 698 1640 624 c 0,0,1 +1041 1046 m 1,61,-1 + 752 1046 l 1,62,-1 + 752 844 l 1,63,-1 + 976 844 l 1,64,-1 + 976 730 l 1,65,-1 + 752 730 l 1,66,-1 + 752 404 l 1,67,-1 + 582 404 l 1,68,-1 + 582 1159 l 1,69,-1 + 1041 1159 l 1,70,-1 + 1041 1046 l 1,61,-1 +1694 1536 m 1,71,-1 + 2048 1536 l 1,72,-1 + 2048 0 l 1,73,-1 + 1694 0 l 1,74,-1 + 1493 0 l 1,75,-1 + 727 0 l 1,76,-1 + 524 0 l 1,77,-1 + 172 0 l 1,78,-1 + 172 1536 l 1,79,-1 + 524 1536 l 1,80,-1 + 727 1536 l 1,81,-1 + 1493 1536 l 1,82,-1 + 1694 1536 l 1,71,-1 +1384 1364 m 1,83,-1 + 834 1364 l 1,84,-1 + 633 1364 l 1,85,-1 + 342 1364 l 1,86,-1 + 342 172 l 1,87,-1 + 633 172 l 1,88,-1 + 834 172 l 1,89,-1 + 1384 172 l 1,90,-1 + 1587 172 l 1,91,-1 + 1878 172 l 1,92,-1 + 1878 1364 l 1,93,-1 + 1587 1364 l 1,94,-1 + 1384 1364 l 1,83,-1 +EndSplineSet +EndChar + +StartChar: F2 +Encoding: 485 59906 467 +Width: 2220 +Flags: W +LayerCount: 2 +Fore +SplineSet +1661 404 m 1,0,-1 + 1147 404 l 1,1,-1 + 1147 531 l 1,2,3 + 1342 699 1342 699 1414 782 c 128,-1,4 + 1486 865 1486 865 1486 932 c 0,5,6 + 1486 982 1486 982 1462.5 1018 c 128,-1,7 + 1439 1054 1439 1054 1388 1054 c 0,8,9 + 1332 1054 1332 1054 1314 1033 c 128,-1,10 + 1296 1012 1296 1012 1287 984 c 1,11,-1 + 1142 984 l 1,12,13 + 1164 1080 1164 1080 1234 1124 c 0,14,15 + 1304 1169 1304 1169 1412 1169 c 0,16,17 + 1529 1169 1529 1169 1592 1107 c 128,-1,18 + 1655 1045 1655 1045 1655 953 c 0,19,20 + 1655 854 1655 854 1568.5 752.5 c 128,-1,21 + 1482 651 1482 651 1287 516 c 1,22,-1 + 1661 516 l 1,23,-1 + 1661 404 l 1,0,-1 +1041 1046 m 1,24,-1 + 752 1046 l 1,25,-1 + 752 844 l 1,26,-1 + 976 844 l 1,27,-1 + 976 730 l 1,28,-1 + 752 730 l 1,29,-1 + 752 404 l 1,30,-1 + 582 404 l 1,31,-1 + 582 1159 l 1,32,-1 + 1041 1159 l 1,33,-1 + 1041 1046 l 1,24,-1 +1694 1536 m 1,34,-1 + 2048 1536 l 1,35,-1 + 2048 0 l 1,36,-1 + 1694 0 l 1,37,-1 + 1493 0 l 1,38,-1 + 727 0 l 1,39,-1 + 524 0 l 1,40,-1 + 172 0 l 1,41,-1 + 172 1536 l 1,42,-1 + 524 1536 l 1,43,-1 + 727 1536 l 1,44,-1 + 1493 1536 l 1,45,-1 + 1694 1536 l 1,34,-1 +1384 1364 m 1,46,-1 + 834 1364 l 1,47,-1 + 633 1364 l 1,48,-1 + 342 1364 l 1,49,-1 + 342 172 l 1,50,-1 + 633 172 l 1,51,-1 + 834 172 l 1,52,-1 + 1384 172 l 1,53,-1 + 1587 172 l 1,54,-1 + 1878 172 l 1,55,-1 + 1878 1364 l 1,56,-1 + 1587 1364 l 1,57,-1 + 1384 1364 l 1,46,-1 +EndSplineSet +EndChar + +StartChar: F1 +Encoding: 486 59905 468 +Width: 2220 +Flags: W +LayerCount: 2 +Fore +SplineSet +1534 404 m 1,0,-1 + 1379 404 l 1,1,-1 + 1379 978 l 1,2,-1 + 1274 978 l 1,3,-1 + 1274 1070 l 1,4,5 + 1345 1076 1345 1076 1383 1108 c 128,-1,6 + 1421 1140 1421 1140 1432 1158 c 1,7,-1 + 1534 1158 l 1,8,-1 + 1534 404 l 1,0,-1 +1081 1046 m 1,9,-1 + 792 1046 l 1,10,-1 + 792 844 l 1,11,-1 + 1016 844 l 1,12,-1 + 1016 730 l 1,13,-1 + 792 730 l 1,14,-1 + 792 404 l 1,15,-1 + 622 404 l 1,16,-1 + 622 1159 l 1,17,-1 + 1081 1159 l 1,18,-1 + 1081 1046 l 1,9,-1 +1694 1536 m 1,19,-1 + 2048 1536 l 1,20,-1 + 2048 0 l 1,21,-1 + 1694 0 l 1,22,-1 + 1493 0 l 1,23,-1 + 727 0 l 1,24,-1 + 524 0 l 1,25,-1 + 172 0 l 1,26,-1 + 172 1536 l 1,27,-1 + 524 1536 l 1,28,-1 + 727 1536 l 1,29,-1 + 1493 1536 l 1,30,-1 + 1694 1536 l 1,19,-1 +1384 1364 m 1,31,-1 + 834 1364 l 1,32,-1 + 633 1364 l 1,33,-1 + 342 1364 l 1,34,-1 + 342 172 l 1,35,-1 + 633 172 l 1,36,-1 + 834 172 l 1,37,-1 + 1384 172 l 1,38,-1 + 1587 172 l 1,39,-1 + 1878 172 l 1,40,-1 + 1878 1364 l 1,41,-1 + 1587 1364 l 1,42,-1 + 1384 1364 l 1,31,-1 +EndSplineSet +EndChar +EndChars +EndSplineFont diff --git a/libmui/fonts/Dingbat.ttf b/libmui/fonts/Dingbat.ttf new file mode 100644 index 0000000000000000000000000000000000000000..841012978e0cfd0ff053fbf4a8cff711ee074e6d GIT binary patch literal 21452 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≶ww3^2k+UYm$3%!5wy(>Qz-;-CqQS#by zEbW@lJL94VmZfL^$XwK6xOB8fyLR2Xx06%{(FY$1{*AlMCK?RUSVhaYyu>Z0;@`kzQYJXoPpCYl!MD?D`8GZIyrS{&VCO zxC`%Y+;&g?O8$7-^0${5+3VthiN*OCaNC|k+NY)OzG9U>D*kxEGQb(>ddG6}(#h-B zFLh~UmE)58wwqy0p3!a){(BGN|>~rm{_rSkiby{dG;ZHvcTuXdb$|5t}Yk6||J(?ptHz0#&XiL9Y5_Cu%|_r_k7kbO`p) z8`!pu)2T^bJWJg}J7*TcoyCC-hPLCLqDYAP81VZ(XDFD*)xf}X(aZ!h#t?pkMp4Uy zA8CXY%g4?}iSREb!XKT(-k{xCd-}RWE6KFhj*C=lG{1J4X=$Bd$T8AMv=cCRgXXZ7 zv>O{Ou~Fz!6lwI2;gHBtA*~sDNuVaGlavP7x4rJ=X?5eU0pwVPF)G~bV&@G>ZBmny zx?jAqVbP=CtO&8$dB`c6Rb}yTiuI>q5d98`2FYhno;-b~)A=sc0t~XTZ~_r%3P))? zi;)<&c`Av&%xsj^;-${bQc_QqN{Bj|-siMXw>wcwmx3ZZT=NpHaOyaU9puxmfWyS$ zZ@(&M%h6jq-kxAIJL~84R2Nh#)@4=N)(R((ojn#+AK$*?HzlgzSrN@o2MpYHOp@2n zT(D@J@n<>+j&Ad>u7V&QJld(Wp#+f=0x4jn4QL>=52zp3Iyxf!R@0xPo0MdVgR(bm z-@ReYF6jbWL<1G5stX<=P2i*@un`H+Ey4f;m2he`|3-t9HWu@abm7WVX-9 znt+a#sbr-PUvPc&P27AGtksEbTqM(f$t$O=ow^>v4dIXb&<2oe2>x+(p!41JV(F1HaD$!$ za&8fO-fFBd!a%MG2DpAdu+X03<%|8=BzEYJJ$O&i>$LXnaM4jNh0~q&z!FfrKH-EzP@bNkKAothZ&!$UTe`52aB0& zKMz$-ag(nl)A_u)4rd)h>7nz$9g&Iug^!y16?^;-KI*^tr|&y+u#tBWsIPumTR+>k z`O^au%v=J++zfeGn03Nf&2q=}eK@WksZ4zcObyaML6EzZX9GYzep+?2G8fBr9#{Ua z@xgsD)WB-AdPEybS26v82AKt%fc7#MGRxp@^fi4Xv+SbgY=1pLP@ii#Xl+i zRM7-Sp!Z;d$Fsg0+&vCtG{@~|AEVYJ4mpF=%Db*TFTk-u@MyJG(?Hsnm@9JyKipTa|;>2oilf9F}=!L8$e=v`Gl z@Oo8L-w$`v?Ep9LfHJFETUpBo1%9}IF;Pt-tHJ6vYza5TNM`?DgjF0*8^Mq(LI|}L zVg#<%BAdWEB6L%PQ!eq?{Em^nr}iv3H=vipX!XZmQ>}DE{F94$As@0G>+3)JivMpM z(5Ahe`9)vhuj1@<25ZK}NKN#LF#@oU0%5Z~e1MDELz*q=!+(AE)Uaz7$8r2-h>5pNJlX>6dcz;a5(Zx#ZNX&fXXBd}bka9@R2UxgP7XRfn) z5Il`f{|wN#rT9HdK{#pxtea#&8^u~dI|3RH8<@d?|8C9dvg)=tj$I4x8t%=7X`(K6UOG);nG$^dcDhVFCtuH7-^}Q0 z32E~-pA~E+F%TT*q4o^!k$2oioMsFt_|O|Kb13iMHtb_F=}RF95g0+lB53z{l9?jv zHH;>*#a6aN#^Hg+BUywoPBcHHn+#^EY-BMO#ItO0r`Ki`8Y9#dz%;+WyG*U$L!|^; z{}SHfj?g|3&=N-vrtJ6KE?bWV;}Kb!@)Ui77>{0Z)&Aea2@+wE)W9^w(rVD9@e+idy`I=< z$j+l@cWyc@(S|RE3I0J~gWiK2Dh6d*mOJs`x8Q%;swSlyXA6BxL+1`BPe^AWhbkK6 z)FbV#OpeLRNatM97BvG5C-$5^ZT&Yxu!t7_BA(ga3H!GIJVapsHtdUN?B9-Fk}nux z5jxTgUDLaCvBoHR(KJ|{l7;A;Pl*htY(9c%?fH}_AM5DL;)MgdPaWUYZ-Bk~n8DWe z%zmCv$?2T1`e}J75`2&6MY`fg8mdmnAe@Qt&d8FB>M@C?d_t_O3v1kj4VFwzxX^_E znoqqNF#u=6f8Ca6`+^eaU#kq@gHDDi@?AG>isZw=Q$-y3(ve*f-2@Vp8j5i=utr^4 z9bRRGtN3A?=Kpb1i%3Y5=WL*M?5xSxM5GuJ!h~SSRS)vc->`c9`ZZQ6@gN%feCu9l z^0$GKVjdikWyDHk{r?;+smEmx&M&}|PqRD}=)*_amhVRbi!*bA_^hg)?8q|sjd&gJ|cA}D+sUWl&zFG7B-l9xI(2OZ7lW3ba< zYWOh?hsK~FPpUh@rAjQMh;kDmgnGR3K3}}j$*UsE}`f2J8=R zL~uu?SnjBd1Iu!WIGM*y9em{>AOcc$G9XH9X8yE>6|xmgIV-UFtb*8S_$CCf>Ofzw zEniAmJcg&n@tpb~r8cjNSd7^3z}W7XE5F7x@11J;}!_1twy5DO_kcN zG9FuOOa1>8j#f>TlZNvb`y_Yh(-{XktH4U(r)aR=Jn($MP6W1B@rVq>g(tS11cmhH z*!D^B1xSc(JBNV1SQeH9!nlqZOUIYPRCN+yn)h*t=GRo1=6$ryI|!LaBAQ_I1CC(nIli+qq5ap%|&XQGtn==~G`vweW?tX(M3Qu(gohl<1 zow>|& ze+TUFIyOW+j0hBUK`eX#?GHn!8iS{xq0GYD_JJcVRjDW{E!YFg&9m)Wwd&NRMazyS zZ{9q2@rGm?@q+x|+XEiM-_KX!WEn`i!eo61`<~J-Pw%@VQPFeEz!fY))dt~eAic|K z+-UsMFr$f|A>$M7by>Bi`KMsqJW+6@+k#kH3|470Y@QZ4dSI6*g%dNJ^KLeDXajTMlzTmxTbS=3r@{Jv2WyFoj7y`rwrF8|MfIL*+fvS z6F&{W<|{4Waq8f8aJ7chNy&E-3-x2#Ql^D9plgB1j6W4xVbb(ea})Z96?|*iw=uJS zpAjQo8sOM;@W7TWdvR7O*yHrdVT6*$9-&=a;lL_MROAX6>7Ol!&NIm!2B`lT{QzjFJE zQ}S2*z_Jl%I=WeLNU){YmA2G-gJ^4<8gC5Eso`7&OMDEc^t0vO>}Sh)wrjHQ0MAShb(>Z+*ci@Meu(wcOU^vJ(77Psh{EY=17RCz=r8q2XKB%(uKREqARvCzm zOBmm?sHcdIdvUo!4?6uVI6~>*?+e^W^JB|{d{5t`Vfw~maOE!+;Efe1ms;l4^i8M6 z!dskxZowM{t3&ueff9CB!nJnRzYFCn|S&HlJEGg6lw@S_vhdJ6;??gYW5R1^#J;Xz55g+VZ&c ztA~$UmWI|(jBnWRtN4eKX%D!C?_q%~{%fn|uTfu(I1Udtzf5@Hr8m_yg14i9Y8scn z7z4uo;l-FwJ&)!gkDgWHd}n_23y#B?8^Ha?s)UB%z#EM#KIOb-nr?ux@f17zASIQ$u^?- z+!^!b`SNUee(`K0J;8Iom-Qd--M7k6 z;@lhQrv$sgE#`;i<6sl|S(pYasev=(GiQ75!XH3`;Kr|pfO8P8YN}d>a7=cIkS++v zQ)1W<u<34L zTiFvg*`~w$|JQr&aFYh>Qv%(j|J(iDB=Tt+H|g3Nk5Ropca!AR1>U^m)nnB>^J^t; zO}2(ucw4+UlJxdugL$9ZUAVd}ui4gMe$7qfzG?|_8mvdoJ~yd&_F$0P-N{8bkb}=F zh3B_5ptKUI`ZdOjDUTjb#(OEh5{bi{brfL!0JG6dwfZ@L@$cN$S3TNOefBg|2Xr@i zbsCTY9wF^f=T;eN2PLSfwpI@J{=D8WK zV_p%SXu|Pgd!9ztEX7OgU*NU%X4+obY`nIHjrO6|r3=$#>0Z~p_t!<{S!0s_ zxx@QUoIcRIQ)>SKX`T8WIBWH1>mW%6%XA|Sar#5-yI)W1iNd8|QO^c67@7khy*Aw^d zoWgr1{s8>>eZ%9@uYdmWW?YE1%*an98z5EN(KZM`=Cby1v^cEneS<%%7XE-*etqy8 zoIw5&IbA||xSL1ksyvU*`E$QBc4qvZ3vXlvcu1+>jW__sTzXJM%)$N(Vh%=sL>(^T z+%b~b^tO_#2sO7EV4CWwp4qJa)b?FHSiAM2n&PB(a1>lw z+@h$73>b-vwS&ugfox+FYGiGf6w;sg}<29ywPZ-{Xn9k1t_QVN5e>iD}dgpjB z=J-{3|1-QZoPimhfF)6v2FP?Yy9WJ*y$rw6(O8moxk)&hUZZ$!fYZ?^MK+HZBX&R@ zivElr!w!-uKZfm(w?zmGE*8HzcHo+P)A&=H6A@NKdSM)kI~N!AmgJ$G6PTe37D03I z{)I2Tm*g|UPp5yhefG{H@3sE5k`I!_sAP36!$Rp z$t=h~4oDk__6q#S8rEH@?*G);v!= zuOErm@?Wf9v%Efs>TAK%0{=o9URL+?#NS8v^lv1BcD-6NflS0_N*3W2{grqX|5LIH zGWiPm8Yqj9$@PRtVVLkHUd3OH*Xg(6mH89GU3|Z6oAA3vr!i@2YkFvgYes4EHIp>O znz@=YnqRa;8>FqR4bdiOTWdRO`)CJhhiP-PQ}Of;o>zhnt=%8tLk7*TXuY%-ehF3MwxA`5fAw!^b6N zyRU=#8^{(wsDyk@JwDdWWRkm-%yM5Mb8#(ke@i}cPb16SZ;%q)*SPnQ^|)`sv(5`+ z6Pb^(EpVR&*0TUj0Nm^4J!~THd7pM8e-rxq8a3u3&!l4Kp}`IGdYu&GnuUv(e&k++ zUSTufzQ(;5y)HzrbI51Fxxyb>&UXKV+TWn|FR1+uYW+e!#l^>XR_&8xpjV-Pq}KTb z(0)(-AVo&Z0)ZBvhdh!Ws ztw3rs?mTBJ?!1>9_&$VZPVTC`aHzhXI1E8;A>fl7;BfUrA63;)e1xbU^WdNFi1dEA zBVPSzM-o2M(Lw##S+DXBaZDmp$W-zsnL&!lOfrkihUGPvyoZl)EXBt-mct7B1fS(t zjSqCJ#|Jq!;d2~Y$vy~-C%78pszj{97Gl-*Css{M5~y>Ks@f&^wjj^{%zHh-cLa|j z&GY7a?_1qhwZTYlMY~S82H~28s}Qi7;QlVDpsj&)l(#J z9g%+k_bl9>5|dhIB%T|itp$0I4B8ajap?olz8bL!?zA^;rVN% zmy_!F`ek)tAq4wsPg)m%vJcS*W=wM(X>hi%3isxumT*BW3(_`#nCA7~LjMV9yAbJC zXb&1_Vvv3hZCiWjqPYk>mmvQJt~K7Y7HQC1eV*p6$IE?m<}{8VR^2XAryReR0k<*n z(?205-M8phk9OBluMO_w(YCh-j!N(5$mcX)fxDOHx)QW+k8!m`9=||q-iIFa=kuV_ zI~I4)T%D_E&>FPWm7y2O-#JL+U7Fx!~9yc!%_bT=K5!(dtlt0-Z#+Kx1h&9 z+&P^=b6pzJm=nzs+&AG`g9~Y072M~0>HQwkoaTJ4rl5b&R<{Ld&{=J_0&psO+Gx{J zeg<{<+}=Z8RowafIzflq=o@$K+ep6w{8}IlURLjSamPFWAFE*U&`h1TTFCca!k5VB z*DADYNUUPE+P<>BD#l(H{pbN7JR|%I_JNfvcC2rr8;;-I_>7rB=z|p3k|tGmnN&S# zQZ=PX)mtW2TbdxvZa^w41J~aePnxPf{<(j4$+VWqnS(~Z4wV6kEtNOK>s$rW!YwY%^8U#;b6^W01G;Dx7$H>?F+l2Bf)`nu5~J z_~kU5f;v3SHTD!$XHQYJ_7qibPl1M+hJ3EOr$Ec_G}qr#R1H2w)!|cB_)}FaIaSr; zQ&mm=rmD$jARk&;ty8SlDOR=lOyul_ZsGcTCb0S%T88WF+3L7vtF)c1j&8Q9*JrD` zcebkAXM1aM{XSdO@Uwv%Pjf9lTh;TkRZTxz)%9~!m~&K^b5xjf(ECr&Xf4yoKtXC}q zd%IBhKEU@bQD>71d6NoflL}$8iqB@?a~Har+a{aUI-AuVH>))_t2MW(Ia}47 WeJXra&Jc37^$ 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�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 i++) { + int addit = 1; + c2_rect_p ra = &a->e[i]; + for (unsigned int j = 0; j < b->count && addit; j++) { + c2_rect_p rb = &b->e[j]; + if (c2_rect_equal(rb, ra)) { // already a copy here + addit = 0; + } else if (c2_rect_contains_rect(rb, ra)) { + addit = 0; + } else if (c2_rect_intersect_rect(rb, ra)) { + c2_rect_t o = *ra; + c2_rect_union(&o, rb); + long sa = c2_rect_surface_squared(ra); + long sb = c2_rect_surface_squared(rb); + long so = c2_rect_surface_squared(&o); + if (so <= (sa + sb)) { + *rb = o; + addit = 0; + } + } + } + if (addit) + c2_rect_array_add(b, *ra); + } + return b->count < a->count; +} diff --git a/libmui/mui/c2_arrays.h b/libmui/mui/c2_arrays.h new file mode 100644 index 0000000..57fd888 --- /dev/null +++ b/libmui/mui/c2_arrays.h @@ -0,0 +1,33 @@ +/* + * c2_arrays.h + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ + +#ifndef C2_ARRAYS_H_ +#define C2_ARRAYS_H_ + +#include "c2_geometry.h" +#include "c_array.h" + +DECLARE_C_ARRAY(c2_pt_t, c2_pt_array, 16); +DECLARE_C_ARRAY(c2_coord_t, c2_coord_array, 16); +DECLARE_C_ARRAY(c2_rect_t, c2_rect_array, 16); + +IMPLEMENT_C_ARRAY(c2_pt_array); +IMPLEMENT_C_ARRAY(c2_coord_array); +IMPLEMENT_C_ARRAY(c2_rect_array); + +/*! Simplify array 'a' into 'b', return 1 if it was. + * This takes a list of rectangles 'a', duplicates are skipped. If two rectangles + * overlap, see if the union is 'worth' it. Returns results in array 'b' + * Returns 1 if the b array has been simplified somehow, zero if not + */ +int +c2_rect_array_simplify( + c2_rect_array_p a, + c2_rect_array_p b); + +#endif /* C2_ARRAYS_H_ */ diff --git a/libmui/mui/c2_geometry.c b/libmui/mui/c2_geometry.c new file mode 100644 index 0000000..02ce113 --- /dev/null +++ b/libmui/mui/c2_geometry.c @@ -0,0 +1,819 @@ +/* + * c2_geometry.c + * + * C2DGeometry Implementation + * Created: Monday, March 30, 1998 11:51:47 + * Converted back to C99 May 2013 + * + * Copyright (C) 1998-2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ +#include +#include "c2_geometry.h" +#include "c2_geometry_poly.h" + +const char * +c2_rect_as_str( + const c2_rect_p r ) +{ + static char ret[8][32]; + static int reti = 0; + char *d = ret[reti]; + reti = (reti+1) & 7; + if (r) + sprintf(d, "[%d,%d,%d,%d]",r->v[0],r->v[1],r->v[2],r->v[3]); + else strcpy(d, "[NULL]"); + return d; +} + +uint8_t +c2_rect_get_next_edge( + uint8_t inEdge, + int inCW ) +{ + uint8_t ret = inCW ? (inEdge << 1) & 0xf : (inEdge >> 1) & 0xf; + return ret ? ret : inCW ? out_Left : out_Bottom; +} + +uint8_t +c2_rect_is_on_edge( + const c2_rect_p r, + const c2_pt_p p ) +{ + if (p->v[X] == r->tl.v[X]) + return out_Left; + if (p->v[Y] == r->tl.v[Y]) + return out_Top; + if (p->v[X] == r->br.v[X]) + return out_Right; + if (p->v[Y] == r->br.v[Y]) + return out_Bottom; + return 0; +} + +int +c2_rect_get_edge( + const c2_rect_p r, + uint8_t inEdge, + c2_segment_p outEdge ) +{ + switch (inEdge) { + case out_Left: + c2_segment_set(outEdge, r->tl.v[X], r->tl.v[Y], r->tl.v[X], r->br.v[Y]); + break; + case out_Top: + c2_segment_set(outEdge, r->tl.v[X], r->tl.v[Y], r->br.v[X], r->tl.v[Y]); + break; + case out_Right: + c2_segment_set(outEdge, r->br.v[X], r->tl.v[Y], r->br.v[X], r->br.v[Y]); + break; + case out_Bottom: + c2_segment_set(outEdge, r->tl.v[X], r->br.v[Y], r->br.v[X], r->br.v[Y]); + break; + default: + return -1; + } + return 0; +} + +int +c2_rect_get_corner( + const c2_rect_p r, + uint8_t inCorner, + c2_pt_p outCorner, + int inCW ) +{ + uint8_t corner = inCW ? inCorner : c2_rect_get_next_edge(inCorner, 0); + switch (corner) { + case corner_TopLeft: + *outCorner = r->tl; + break; + case corner_TopRight: + outCorner->v[X] = r->br.v[X]; + outCorner->v[Y] = r->tl.v[Y]; + break; + case corner_BottomRight: + *outCorner = r->br; + break; + case corner_BottomLeft: + outCorner->v[X] = r->tl.v[X]; + outCorner->v[Y] = r->br.v[Y]; + break; + default: + return -1; + } + return 0; +} + + +int +c2_rect_clip_segment( + const c2_rect_p r, + c2_segment_p s, + c2_segment_p o, + char * outEdges ) +{ + int accept = 0, done = 0; + uint8_t outcode0 = c2_rect_get_out_code(r, &s->a); + uint8_t outcode1 = c2_rect_get_out_code(r, &s->b); + *o = *s; + do { + if (!(outcode0 | outcode1)) { + accept = 1; done = 1; + continue; + } else if ((outcode0 & outcode1) != 0) { + if (outEdges) { + outEdges[0] = outcode0; + outEdges[1] = outcode1; // return offending borders + } + done = 1; // don't accept + continue; + } + + // at least one end is outside, pick it + c2_pt_t *p; + uint8_t *outcode; + uint8_t dummy; + uint8_t *edge = &dummy; + if (outcode0) { + outcode = &outcode0; + p = &o->a; + if (outEdges) edge = &edge[0]; + } else { + outcode = &outcode1; + p = &o->b; + if (outEdges) edge = &edge[1]; + } + if (*outcode & out_Top) { + p->v[X] = s->a.v[X] + + (s->b.v[X] - s->a.v[X]) * + (double)(r->tl.v[Y] - s->a.v[Y]) / (s->b.v[Y] - s->a.v[Y]); + p->v[Y] = r->tl.v[Y]; + *edge = out_Top; + *outcode &= ~out_Top; + } else if (*outcode & out_Bottom) { + p->v[X] = s->a.v[X] + (s->b.v[X] - s->a.v[X]) * + (double)(r->br.v[Y] - s->a.v[Y]) / (s->b.v[Y] - s->a.v[Y]); + p->v[Y] = r->br.v[Y]; + *edge = out_Bottom; + *outcode &= ~out_Bottom; + } + if (*outcode & out_Left) { + p->v[Y] = s->a.v[Y] + (s->b.v[Y] - s->a.v[Y]) * + (double)(r->tl.v[X] - s->a.v[X]) / (s->b.v[X] - s->a.v[X]); + p->v[X] = r->tl.v[X]; + *edge = out_Left; + *outcode &= ~out_Left; + } else if (*outcode & out_Right) { + p->v[Y] = s->a.v[Y] + (s->b.v[Y] - s->a.v[Y]) * + (double)(r->br.v[X] - s->a.v[X]) / (s->b.v[X] - s->a.v[X]); + p->v[X] = r->br.v[X]; + *edge = out_Right; + *outcode &= ~out_Right; + } + *outcode = c2_rect_get_out_code(r, p); + *outcode = 0; + } while (!done); + + return accept; +} + +// C2DRectangle::Intersect( +int +c2_rect_clip_rect( + const c2_rect_p r, + const c2_rect_p s, + c2_rect_p o ) +{ + uint8_t outcode0 = c2_rect_get_out_code(r, &s->tl); + uint8_t outcode1 = c2_rect_get_out_code(r, &s->br); + + // if both corners are on the same 'side' as the + // other corner, there can't be any intersection anyway + if (outcode0 & outcode1) { + o->br = o->tl;// make sure result rect is empty + return 0; + } + // ^^ this is equivalent to testing for all corners: + // for (int c = out_Left; c <= out_Bottom; c <<= 1) + // if ((outcode0 & c) && (outcode1 & c)) + // return 0; + + /* here we /know/ the rectangle intersects, or contains the + * other, so just detect where the corners are in relation to + * the rectangle, and copy the 'clipped' coordinates over + */ + *o = *s; + if (outcode0 & out_Left) + o->tl.x = r->tl.x; + if (outcode0 & out_Top) + o->tl.y = r->tl.y; + if (outcode1 & out_Right) + o->br.x = r->br.x; + if (outcode1 & out_Bottom) + o->br.y = r->br.y; + // only return 1 is the result is a true + // rectangle. adjacent rectangles would + // return a 'valid' intersection with with + // width or height being 0 + return c2_rect_width(o) && c2_rect_height(o); +} + +//! return true if all corners of 'r2' are inside 'r1' +int +c2_rect_contains_rect( + const c2_rect_p r1, + const c2_rect_p r2 ) +{ + for (int c = corner_TopLeft; c <= corner_BottomLeft; c <<= 1) { + c2_pt_t p = {}; + c2_rect_get_corner(r2, c, &p, 1); + if (!c2_rect_contains_pt(r1, &p)) + return 0; + } + return 1; +} + +// Return true if rectangle 'r' intersects with rectangle 's' +int +c2_rect_intersect_rect( + const c2_rect_p s, + const c2_rect_p r ) +{ + // if any of the corners are inside, we are intersecting anyway + for (int c = corner_TopLeft; c <= corner_BottomLeft; c <<= 1) { + c2_pt_t p = {}; + c2_rect_get_corner(r, c, &p, 1); + if (c2_rect_contains_pt(s, &p)) + return 1; + } + // if any of the edges of 'r' intersects 's', we intersect + for (int e = out_Left; e <= out_Bottom; e <<= 1) { + c2_segment_t seg, clip; + c2_rect_get_edge(r, e, &seg); + if (c2_rect_clip_segment(s, &seg, &clip, NULL)) + return 1; + } + // if 'r' *contains* 's', we intersect + return c2_rect_contains_rect(r, s); +} + +uint8_t +c2_rect_get_out_code( + const c2_rect_p r, + const c2_pt_p p ) +{ + return ((p->v[X] < r->tl.v[X] ? out_Left : (p->v[X] > r->br.v[X]) ? out_Right : 0) | + (p->v[Y] < r->tl.v[Y] ? out_Top : (p->v[Y] > r->br.v[Y]) ? out_Bottom : 0)); +} + +void +c2_rect_clip_pt( + const c2_rect_p r, + c2_pt_p p ) +{ + uint8_t o = c2_rect_get_out_code(r, p); + if (o & out_Left) + p->v[X] = r->tl.v[X]; + else + p->v[X] = r->br.v[X]; + if (o & out_Top) + p->v[Y] = r->tl.v[Y]; + else + p->v[Y] = r->br.v[Y]; +} + + +void +c2_polyline_clear( + c2_polyline_p pl) +{ + const c2_rect_t z = C2_RECT_ZERO; + pl->bounds = z; + c2_pt_array_clear(&pl->pt); +} + +//c2_polyline_t::GetSegment( +int +c2_polyline_get_segment( + c2_polyline_p pl, + long ind, + c2_segment_p o ) +{ + if (ind > pl->pt.count) + return 0; + o->a = pl->pt.e[ind]; + o->b = pl->pt.e[ind == pl->pt.count ? 1 : ind]; + return 1; +} + +void +c2_polyline_offset( + c2_polyline_p pl, + c2_coord_t inX, + c2_coord_t inY ) +{ + for (unsigned int i = 0; i < pl->pt.count; i++) { + pl->pt.e[i].x += inX; + pl->pt.e[i].y += inY; + } + c2_rect_offset(&pl->bounds, inX, inY); +} + +void +c2_polyline_scale( + c2_polyline_p pl, + double inFactor, + c2_rect_p inSkip /* = NULL */ ) +{ + long pMax = pl->pt.count; + c2_pt_t * c = pl->pt.e; + long count = 0; + long total = pMax; + int testrect = inSkip != NULL; + + c2_rect_scale(&pl->bounds, inFactor); + + if (c2_rect_height(&pl->bounds) > 3 && c2_rect_width(&pl->bounds) > 3 && total >= 8) { + c2_pt_t delta = C2_PT_ZERO; + c2_pt_t o = C2_PT_ZERO; + short skip = 11; + c2_pt_t *d = c; + c2_pt_t orig; + + for (int i = 0; i < total; i++, c++) { + orig = *c; + c2_pt_scale(c, inFactor); + delta.v[X] += c->v[X] - o.v[X]; + delta.v[Y] += c->v[Y] - o.v[Y]; + int add = 0; + + if (testrect) + add = orig.v[X] == inSkip->tl.v[X] || orig.v[X] == inSkip->br.v[X] || + orig.v[Y] == inSkip->tl.v[Y] || orig.v[X] == inSkip->br.v[Y]; + + if (!add) + add = skip > 10 || delta.v[X] > 3 || delta.v[X] < -3 || + delta.v[Y] > 3 || delta.v[Y] < -3; + if (add) { + *d++ = *c; + count++; + delta.v[X] = delta.v[Y] = 0; + skip = 0; + } else + skip++; + o = *c; + } + if (count < 2) { + c = pl->pt.e; + c[0] = pl->bounds.tl; + c[1] = pl->bounds.br; + count = 2; + } + if (count & 1) { + *d++ = o; + count++; + } + } else { + for (int i = 1; i <= total; i++, c++) + c2_pt_scale(c, inFactor); + } + + pl->pt.count = count; + // TODO: trim? +} + +void +c2_polyline_add_pt( + c2_polyline_p pl, + c2_pt_p p ) +{ + if (!pl->pt.count) { + pl->bounds.tl = *p; + pl->bounds.br = *p; + } else { + pl->bounds.tl.v[X] = PMIN(pl->bounds.tl.v[X], p->v[X]); + pl->bounds.tl.v[Y] = PMIN(pl->bounds.tl.v[Y], p->v[Y]); + pl->bounds.br.v[X] = PMAX(pl->bounds.br.v[X], p->v[X]); + pl->bounds.br.v[Y] = PMAX(pl->bounds.br.v[Y], p->v[Y]); + } + c2_pt_array_add(&pl->pt, *p); +} + +#if 0 +void +c2_polyline_t::Frame() +{ + Lock(); + SInt32 pMax = GetCount(); + c2_pt_t *c = (c2_pt_t*)GetItemPtr(1); + MoveTo(Pixel_32k(c->v[X]), Pixel_32k(c->v[Y])); + for (int i = 1; i <= GetCount(); i++, c++) { + if (i == 1) + i = 1; + LineTo(Pixel_32k(c->v[X]), Pixel_32k(c->v[Y])); + } + Unlock(); +} +#endif + +void +c2_polyline_array_break( + c2_polyline_array_p pa) +{ + pa->current = NULL; +} + +void +c2_polyline_array_add_pt( + c2_polyline_array_p pa, + c2_pt_p p ) +{ + if (!pa->current) { + pa->current = malloc(sizeof(c2_polyline_t)); + memset(pa->current, 0, sizeof(*pa->current)); + c2_polyline_array_add(pa, pa->current); + } + c2_polyline_add_pt(pa->current, p); +} + +int +c2_polyline_array_clip( + c2_polyline_array_p pa, + c2_rect_p clip, + c2_polyline_array_p outPoly ) +{ + for (long poly = 0; poly < pa->count; poly++) { + c2_polyline_t *p = pa->e[poly]; + if (!p) + break; + + long pMax = p->pt.count; + c2_pt_t * last = p->pt.e; + int lastIn = c2_rect_contains_pt(clip, last); + c2_pt_t * current = last+1; + int currentIn = c2_rect_contains_pt(clip, current); + + for (long pIndex = 2; pIndex <= pMax; pIndex++) { + if (lastIn && currentIn) { // both points are IN + //outPoly.AddPoint(*last); + c2_polyline_array_add_pt(outPoly, current); + } else if (lastIn && !currentIn) { // line goes OUT + c2_segment_t src = { .a = *last, .b = *current }; + c2_segment_t dst; + c2_rect_clip_segment(clip, &src, &dst, NULL); + c2_polyline_array_add_pt(outPoly, &dst.b); + c2_polyline_array_break(outPoly); + } else if (!lastIn && currentIn) { // line goes IN + c2_segment_t src = { .a = *last, .b = *current }; + c2_segment_t dst; + c2_rect_clip_segment(clip, &src, &dst, NULL); + c2_polyline_array_break(outPoly); + c2_polyline_array_add_pt(outPoly, &dst.a); + c2_polyline_array_add_pt(outPoly, &dst.b); + } else { // line outside + c2_segment_t src = { .a = *last, .b = *current }; + c2_segment_t dst; + if (c2_rect_clip_segment(clip, &src, &dst, NULL)) { // both point are out, but crosses the rectangle + c2_polyline_array_break(outPoly); + c2_polyline_array_add_pt(outPoly, &dst.a); + c2_polyline_array_add_pt(outPoly, &dst.b); + } + } + last++; + current++; + lastIn = c2_rect_contains_pt(clip, last); + currentIn = c2_rect_contains_pt(clip, current); + } + + } + return 0; +} + + +void +c2_polyline_array_scale( + c2_polyline_array_p pa, + float inFactor, + c2_rect_p inSkip /* = NULL */ ) +{ + for (unsigned int ind = 0; ind < pa->count; ind++) { + c2_polyline_scale(pa->e[ind], inFactor, inSkip); + } +} + +void +c2_polyline_array_offset( + c2_polyline_array_p pa, + c2_coord_t inX, + c2_coord_t inY ) +{ + for (unsigned int ind = 0; ind < pa->count; ind++) { + c2_polyline_offset(pa->e[ind], inX, inY); + } +} + +void +c2_scanline_array_proper_alloc( + c2_scanline_array_p a, + c2_coord_t height) +{ + c2_scanline_array_realloc(a, height); + memset(a->e, 0, height * sizeof(a->e[0])); +} + +void +c2_scanline_array_proper_clear( + c2_scanline_array_p a) +{ + for (unsigned int i = 0; i < a->count; i++) + c2_coord_array_free(&a->e[i]); +} + +void +c2_scanline_array_add_coord( + c2_scanline_array_p a, + int inY, + c2_coord_t inX ) +{ + if (inY < 0 || inY >= (int)a->count) return; + + c2_coord_array_p l = &a->e[inY]; + + if (l->count == 0 || inX >= l->e[l->count-1]) { + c2_coord_array_add(l, inX); + return; + } + + int num = l->count; + int ind = 0; + c2_coord_t * cur = l->e; + + while (*cur < inX && num--) { + cur++; + ind++; + } + c2_coord_array_insert(l, ind, &inX, 1); +} + +int +c2_polygon_isempty( + c2_polygon_p pl) +{ + return !pl->pt.count || c2_rect_isempty(&pl->bounds); +} + +c2_coord_t +c2_polygon_get_heigth( + c2_polygon_p pl) +{ + return c2_rect_height(&pl->bounds); +} + +void +c2_polygon_clip( + c2_polygon_p pl, + c2_rect_p clip, + c2_polygon_p outPoly ) +{ + int sect = c2_rect_intersect_rect(&pl->bounds, clip); + int contains = c2_rect_contains_rect(&pl->bounds, clip); + if (!sect && !contains) + return; // don't even bother + + long pMax = pl->pt.count; + c2_pt_t * last = pl->pt.e; + int lastIn = c2_rect_contains_pt(clip, last); + c2_pt_t * current = last+1; + int currentIn = c2_rect_contains_pt(clip, current); + //char outPart = clip.GetOutCode(*last); + uint8_t edgesStack[40]; + short edgesClock[40]; + long edgesCount = 0; + + char edgesIndexes[13] = // Bit valued to quadrant index + { 0, 1, 3, 2, 5, 0, 4, 0, 7, 8, 0, 0, 6 }; + + // if we start outside, remember our quadrant + if (!lastIn) + edgesStack[edgesCount++] = edgesIndexes[c2_rect_get_out_code(clip, last)]; + if (lastIn) + c2_polyline_add_pt(outPoly, last); + for (long pIndex = 2; pIndex <= pMax; pIndex++) { + if (lastIn && currentIn) { // both points are IN + c2_polyline_add_pt(outPoly, current); + } else if (lastIn && !currentIn) { // line goes OUT + c2_segment_t src = { .a = *last, .b = *current }; + c2_segment_t dst; + c2_rect_clip_segment(clip, &src, &dst, NULL); + c2_polyline_add_pt(outPoly, &dst.b); + edgesStack[edgesCount++] = edgesIndexes[c2_rect_is_on_edge(clip, &dst.b)]; + } else if (!lastIn && currentIn) { // line goes IN + // flush corner stack + if (edgesCount > 1) { + for (int c = 0; c < edgesCount; c++) { + c2_pt_t p; + switch (edgesStack[c]) { + case 2: // topleft + p = clip->tl; + c2_polyline_add_pt(outPoly, &p); + break; + case 4: // topright + p.v[X] = clip->br.v[X]; p.v[Y] = clip->tl.v[Y]; + c2_polyline_add_pt(outPoly, &p); + break; + case 6: // botright + p = clip->br; + c2_polyline_add_pt(outPoly, &p); + break; + case 8: // botleft + p.v[X] = clip->tl.v[X]; p.v[Y] = clip->br.v[Y]; + c2_polyline_add_pt(outPoly, &p); + break; + } + } + } + edgesCount = 0; + c2_segment_t src = { .a = *last, .b = *current }; + c2_segment_t dst; + c2_rect_clip_segment(clip, &src, &dst, NULL); + c2_polyline_add_pt(outPoly, &dst.a); + c2_polyline_add_pt(outPoly, &dst.b); + } else { // line outside + c2_segment_t src = { .a = *last, .b = *current }; + c2_segment_t dst; + if (c2_rect_clip_segment(clip, &src, &dst, NULL)) { // both point are out, but crosses the rectangle + // flush corner stack + if (edgesCount > 1) { + for (int c = 0; c < edgesCount; c++) { + c2_pt_t p; + switch (edgesStack[c]) { + case 2: // topleft + p = clip->tl; + c2_polyline_add_pt(outPoly, &p); + break; + case 4: // topright + p.v[X] = clip->br.v[X]; p.v[Y] = clip->tl.v[Y]; + c2_polyline_add_pt(outPoly, &p); + break; + case 6: // botright + p = clip->br; + c2_polyline_add_pt(outPoly, &p); + break; + case 8: // botleft + p.v[X] = clip->tl.v[X]; p.v[Y] = clip->br.v[Y]; + c2_polyline_add_pt(outPoly, &p); + break; + } + } + } + edgesCount = 0; + c2_polyline_add_pt(outPoly, &dst.a); + c2_polyline_add_pt(outPoly, &dst.b); + edgesStack[edgesCount++] = edgesIndexes[c2_rect_is_on_edge(clip, &dst.b)]; + } else { // setup edge stack properly + unsigned char clockTable[9] = + { 0, 0x0e, 0x1c, 0x38, 0x70, 0xe0, 0xc1, 0x83, 0x07 }; + uint8_t o = edgesIndexes[c2_rect_get_out_code(clip, current)]; + if (o != edgesStack[edgesCount - 1]) { + short clock = clockTable[edgesStack[edgesCount - 1]] & (1 << (o-1)) ? 1 : -1; + + do { + if (edgesCount == 1) + edgesClock[edgesCount -1] = clock; + + if (clock == edgesClock[edgesCount -1]) { + int c = edgesStack[edgesCount - 1]; + do { + c += clock; + if (c == 0) c = 8; + else if (c == 9) c = 1; + edgesClock[edgesCount] = clock; + edgesStack[edgesCount++] = c; +#ifdef TRACELIB + { + static long ecount = 0; + if (edgesCount > ecount) { + ecount = edgesCount; + printf("Max edgecount = %d\n", ecount); + } + } + if (edgesCount == 40) { + DebugStr("\p HELP!"); + } +#endif + } while (c != o); + } else { // change direction, remove points + while (edgesCount > 1 && edgesStack[edgesCount - 1] != o) + edgesCount--; + } + } while (o != edgesStack[edgesCount -1]); + } + } + } + last++; + current++; + lastIn = c2_rect_contains_pt(clip, last); + currentIn = c2_rect_contains_pt(clip, current); + } + // flush corner stack + if (edgesCount > 1) { + for (int c = 0; c < edgesCount; c++) { + c2_pt_t p; + switch (edgesStack[c]) { + case 2: // topleft + p = clip->tl; + c2_polyline_add_pt(outPoly, &p); + break; + case 4: // topright + p.v[X] = clip->br.v[X]; p.v[Y] = clip->tl.v[Y]; + c2_polyline_add_pt(outPoly, &p); + break; + case 6: // botright + p = clip->br; + c2_polyline_add_pt(outPoly, &p); + break; + case 8: // botleft + p.v[X] = clip->tl.v[X]; p.v[Y] = clip->br.v[Y]; + c2_polyline_add_pt(outPoly, &p); + break; + } + } + } + edgesCount = 0; +} + + +#define exchange(t, p1, p2) { t e=p1;p1=p2;p2=e; } + +void +c2_polygon_scanline( + c2_polygon_p pl, + c2_scanline_array_p ioList, + c2_coord_t ymin ) +{ + c2_coord_t ysize = ioList->height;//TODO check if we meant the polygon height? + if (ysize <= 0 || c2_rect_width(&pl->bounds) <= 0) + return; + + c2_pt_t *first = pl->pt.e; + c2_pt_t *sp1 = first; + c2_pt_t *sp2 = sp1+1; + long numPoints = pl->pt.count; + // + // Compute list of intersections between the vertex and the horizontal + // lines. The lists are automaticaly sorted + // + do { + if (numPoints == 1) sp2 = first; + c2_pt_t *p1 = sp1; + c2_pt_t *p2 = sp2; + if (p2->v[Y] < p1->v[Y]) + exchange(c2_pt_t *, p1, p2); + c2_coord_t y1 = p1->v[Y]; + c2_coord_t y2 = p2->v[Y]; + c2_coord_t idy = y2 - y1; + if ((y1 < ymin && y2 < ymin) || + ((y1-ymin > ioList->height) && (y2-ymin > ioList->height))) + continue; + if (idy) { + double x, dx; + x = p1->v[X]; + dx = (p2->v[X] - p1->v[X]) / (double)idy; + do { + c2_coord_t ix = x; + c2_scanline_array_add_coord(ioList, y1-ymin, x - ix > 0.5 ? ix+1 : ix); + x += dx; + y1++; + } while (y1 < y2); + } else { + c2_scanline_array_add_coord(ioList, y1-ymin, p1->v[X]); + c2_scanline_array_add_coord(ioList, y1-ymin, p2->v[X]); + } + sp1++; + sp2++; + } while(numPoints-- > 1); +} + +#if 0 +void +c2_polygon_t::Paint() +{ + c2_scanline_list_t scan(Height()+1); + AddIntersections(scan, bounds.tl.v[Y]); + + for (int i = 0; i < Height(); i++) { + SInt32 num = scan.Get(i).GetCount(); + if (num > 1) { + c2_coord_t * cur = scan.Get(i); + for (SInt32 ind = 1; ind <= num - 1; ind += 2, cur += 2) { + if (cur[0] < cur[1]) { + MoveTo(cur[0], bounds.tl.v[Y]+i); + LineTo(cur[1], bounds.tl.v[Y]+i); + } + } + } + } +} +#endif diff --git a/libmui/mui/c2_geometry.h b/libmui/mui/c2_geometry.h new file mode 100644 index 0000000..e47c8ac --- /dev/null +++ b/libmui/mui/c2_geometry.h @@ -0,0 +1,148 @@ +/* + * c2_geometry.h + * + * C2DGeometry Implementation + * Created: Monday, March 30, 1998 11:51:47 + * Converted back to C99 May 2013 + * + * Copyright (C) 1998-2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ + +#ifndef __C2_GEOMETRY_H__ +#define __C2_GEOMETRY_H__ + +#include + +#define PMIN(x1,x2) ((x1 < x2) ? x1 : x2) +#define PMAX(x1,x2) ((x1 > x2) ? x1 : x2) + +typedef int32_t c2_coord_t; + +enum { + X = 0, Y, +// L = 0, T, R, B, +}; + +// +// a basic x,y point +// +typedef union c2_pt_t { + struct { + c2_coord_t x, y; + }; + c2_coord_t v[2]; +} c2_pt_t;//, *c2_pt_p; +#define c2_pt_p union c2_pt_t* + +#define C2_PT(_a, _b) (c2_pt_t){ .x = (_a), .y = (_b) } +#define C2_PT_ZERO C2_PT(0,0) + +// +// a basic segment, a pair of points +// +typedef struct c2_segment_t { + c2_pt_t a, b; +} c2_segment_t;//, *c2_segment_p; +#define c2_segment_p struct c2_segment_t* + +#define C2_SEGMENT_ZERO { C2_PT_ZERO, C2_PT_ZERO } + +// +// a rectangle defined by two points +// +typedef union c2_rect_t { + struct { + c2_pt_t tl, br; + }; + c2_coord_t v[4]; + struct { + c2_coord_t l,t,r,b; + }; +} c2_rect_t;//, *c2_rect_p; +#define c2_rect_p union c2_rect_t* + +#define C2_RECT(_l,_t,_r,_b) (c2_rect_t){ .l = _l, .t = _t, .r = _r, .b = _b } +#define C2_RECT_WH(_l,_t,_w,_h) (c2_rect_t){ \ + .l = _l, .t = _t, .r = (_l)+(_w), .b = (_t)+(_h) } +/* Return the squared surface of the rectangle, for area comparison sake */ +#define c2_rect_surface_squared(_r) \ + ((c2_rect_width(_r)*c2_rect_width(_r))+\ + (c2_rect_height(_r)*c2_rect_height(_r))) + +#define C2_RECT_ZERO C2_RECT(0,0,0,0) + +int +c2_rect_contains_rect( + const c2_rect_p r1, + const c2_rect_p r2 ); + +int +c2_rect_clip_segment( + const c2_rect_p r, + c2_segment_p s, + c2_segment_p o, + char * outEdges ); +int +c2_rect_clip_rect( + const c2_rect_p r, + const c2_rect_p s, + c2_rect_p o ); +int +c2_rect_intersect_rect( + const c2_rect_p s, + const c2_rect_p r ); + +enum { + out_Left = (1 << 0), + out_Top = (1 << 1), + out_Right = (1 << 2), + out_Bottom = (1 << 3), +}; +enum { + corner_TopLeft = out_Left, + corner_TopRight = out_Top, + corner_BottomRight = out_Right, + corner_BottomLeft = out_Bottom, +}; + +uint8_t +c2_rect_get_out_code( + const c2_rect_p r, + const c2_pt_p p ); +void +c2_rect_clip_pt( + const c2_rect_p r, + c2_pt_p p ); + +uint8_t +c2_rect_get_next_edge( + uint8_t inEdge, + int inCW ); +uint8_t +c2_rect_is_on_edge( + const c2_rect_p r, + const c2_pt_p p ); +int +c2_rect_get_edge( + const c2_rect_p r, + uint8_t inEdge, + c2_segment_p outEdge ); +int +c2_rect_get_corner( + const c2_rect_p r, + uint8_t inCorner, + c2_pt_p outCorner, + int inCW ); + +const char * +c2_rect_as_str( + const c2_rect_p r ); + +#ifndef C2_GEOMETRY_INLINE_H_ +#ifndef C2_DECL +#include "c2_geometry_inline.h" +#endif +#endif +#endif diff --git a/libmui/mui/c2_geometry_inline.h b/libmui/mui/c2_geometry_inline.h new file mode 100644 index 0000000..bbdf211 --- /dev/null +++ b/libmui/mui/c2_geometry_inline.h @@ -0,0 +1,244 @@ +/* + * c2_geometry_inline.h + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ + +#ifndef C2_GEOMETRY_INLINE_H_ +#define C2_GEOMETRY_INLINE_H_ + +#include "c2_geometry.h" + +#ifndef C2_DECL +#define C2_DECL static inline __attribute__((unused)) +#endif + +C2_DECL void +c2_pt_offset( + c2_pt_p p, + c2_coord_t inX, + c2_coord_t inY) +{ + p->x += inX; + p->y += inY; +} + +C2_DECL int +c2_pt_equal( + const c2_pt_p p1, + const c2_pt_p p2) +{ + return p1->v[0] == p2->v[0] && p1->v[1] == p2->v[1]; +} + +C2_DECL void +c2_pt_scale( + c2_pt_p p, + float inFactor ) +{// TODO fix rounding + p->v[X] *= inFactor; + p->v[Y] *= inFactor; +} + +C2_DECL c2_segment_p +c2_segment_set( + c2_segment_p s, + c2_coord_t x1, + c2_coord_t y1, + c2_coord_t x2, + c2_coord_t y2 ) +{ + s->a.v[X] = PMIN(x1, x2); + s->a.v[Y] = PMIN(y1, y2); + s->b.v[X] = PMAX(x1, x2); + s->b.v[Y] = PMAX(y1, y2); + return s; +} + + +C2_DECL int +c2_segment_equal( + const c2_segment_p s1, + const c2_segment_p s2) +{ + return (c2_pt_equal(&s1->a, &s2->a) && c2_pt_equal(&s1->b, &s2->b)) || + (c2_pt_equal(&s1->a, &s2->b) && c2_pt_equal(&s1->b, &s2->a)); +} + +C2_DECL int +c2_segment_isempty( + const c2_segment_p s) +{ + return c2_pt_equal(&s->a, &s->b); +} + +C2_DECL void +c2_segment_offset( + c2_segment_p s, + c2_coord_t inX, + c2_coord_t inY ) +{ + c2_pt_offset(&s->a, inX, inY); + c2_pt_offset(&s->b, inX, inY); +} + +C2_DECL void +c2_segment_scale( + c2_segment_p s, + double inFactor ) +{ + c2_pt_scale(&s->a, inFactor); + c2_pt_scale(&s->b, inFactor); +} + + +C2_DECL c2_rect_p +c2_rect_set( + c2_rect_p r, + c2_coord_t x1, + c2_coord_t y1, + c2_coord_t x2, + c2_coord_t y2 ) +{ + r->tl.v[X] = PMIN(x1, x2); + r->tl.v[Y] = PMIN(y1, y2); + r->br.v[X] = PMAX(x1, x2); + r->br.v[Y] = PMAX(y1, y2); + return r; +} + +C2_DECL int +c2_rect_isempty( + const c2_rect_p r) +{ + return c2_pt_equal(&r->tl, &r->br) || + r->tl.x >= r->br.x || r->tl.y >= r->br.y; +} +C2_DECL int +c2_rect_equal( + const c2_rect_p r, + const c2_rect_p o) +{ + return r->v[0] == o->v[0] && r->v[1] == o->v[1] && + r->v[2] == o->v[2] && r->v[3] == o->v[3]; + /* This got miscompiled under gcc 4.8 and 4.9. The above is more straightforward. + * return c2_pt_equal(&r->tl, &o->tl) && c2_pt_equal(&r->br, &o->br); + */ +} +C2_DECL c2_coord_t +c2_rect_height( + const c2_rect_p r) +{ + return r->br.v[Y] - r->tl.v[Y]; +} + +C2_DECL c2_coord_t +c2_rect_width( + const c2_rect_p r) +{ + return r->br.v[X] - r->tl.v[X]; +} + +C2_DECL c2_pt_t +c2_rect_size( + const c2_rect_p r) +{ + return (c2_pt_t){ .x = c2_rect_width(r), .y = c2_rect_height(r) }; +} + +C2_DECL void +c2_rect_offset( + c2_rect_p r, + c2_coord_t inX, + c2_coord_t inY) +{ + c2_pt_offset(&r->tl, inX, inY); + c2_pt_offset(&r->br, inX, inY); +} + +C2_DECL void +c2_rect_inset( + c2_rect_p r, + c2_coord_t inX, + c2_coord_t inY) +{ + c2_pt_offset(&r->tl, inX, inY); + c2_pt_offset(&r->br, -inX, -inY); +} + +C2_DECL void +c2_rect_scale( + c2_rect_p r, + double inFactor ) +{ + c2_pt_scale(&r->tl, inFactor); + c2_pt_scale(&r->br, inFactor); +} + +C2_DECL int +c2_rect_contains_pt( + const c2_rect_p r, + c2_pt_p p ) +{ + return (p->v[X] >= r->tl.v[X] && p->v[X] <= r->br.v[X]) && + (p->v[Y] >= r->tl.v[Y] && p->v[Y] <= r->br.v[Y]); +} + +/* Returns 'r' as the union of 'r' and 'u' (enclosing rectangle) */ +C2_DECL c2_rect_p +c2_rect_union( + c2_rect_p r, + c2_rect_p u ) +{ + if (!r || !u) return r; + if (c2_rect_isempty(r)) { + *r = *u; + return r; + } + r->l = PMIN(r->l, u->l); + r->t = PMIN(r->t, u->t); + r->r = PMAX(r->r, u->r); + r->b = PMAX(r->b, u->b); + return r; +} + +static c2_rect_t +c2_rect_left_of( + c2_rect_t * r, + int mark, + int margin) +{ + c2_rect_offset(r, -r->l + mark - c2_rect_width(r) - margin, 0); + return *r; +} +static c2_rect_t +c2_rect_right_of( + c2_rect_t * r, + int mark, + int margin) +{ + c2_rect_offset(r, -r->l + mark + margin, 0); + return *r; +} +static c2_rect_t +c2_rect_top_of( + c2_rect_t * r, + int mark, + int margin) +{ + c2_rect_offset(r, 0, -r->t + mark - c2_rect_height(r) - margin); + return *r; +} +static c2_rect_t +c2_rect_bottom_of( + c2_rect_t * r, + int mark, + int margin) +{ + c2_rect_offset(r, 0, -r->t + mark + margin); + return *r; +} + +#endif /* C2_GEOMETRY_INLINE_H_ */ diff --git a/libmui/mui/c2_geometry_poly.h b/libmui/mui/c2_geometry_poly.h new file mode 100644 index 0000000..c8b464f --- /dev/null +++ b/libmui/mui/c2_geometry_poly.h @@ -0,0 +1,119 @@ +/* + * c2_geometry_poly.h + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ + +#ifndef C2_GEOMETRY_POLY_H_ +#define C2_GEOMETRY_POLY_H_ + +#include "c2_arrays.h" + +typedef struct c2_polyline_t { + c2_pt_array_t pt; + c2_rect_t bounds; +} c2_polyline_t, *c2_polyline_p; + +DECLARE_C_ARRAY(c2_polyline_p, c2_polyline_array, 16, + c2_polyline_p current; ); +DECLARE_C_ARRAY(c2_coord_array_t, c2_scanline_array, 16, + c2_coord_t height; ); + +IMPLEMENT_C_ARRAY(c2_scanline_array); +IMPLEMENT_C_ARRAY(c2_polyline_array); + + +void +c2_polyline_clear( + c2_polyline_p pl); + +int +c2_polyline_get_segment( + c2_polyline_p pl, + long ind, + c2_segment_p o ); + +void +c2_polyline_offset( + c2_polyline_p pl, + c2_coord_t inX, + c2_coord_t inY ); + +void +c2_polyline_scale( + c2_polyline_p pl, + double inFactor, + c2_rect_p inSkip /* = NULL */ ); + +void +c2_polyline_add_pt( + c2_polyline_p pl, + c2_pt_p p ); + +int +c2_polyline_array_clip( + c2_polyline_array_p pa, + c2_rect_p clip, + c2_polyline_array_p outPoly ); +void +c2_polyline_array_break( + c2_polyline_array_p pa); +void +c2_polyline_array_add_pt( + c2_polyline_array_p pa, + c2_pt_p p ); + +void +c2_polyline_array_scale( + c2_polyline_array_p pa, + float inFactor, + c2_rect_p inSkip /* = NULL */ ); +void +c2_polyline_array_offset( + c2_polyline_array_p pa, + c2_coord_t inX, + c2_coord_t inY ); + + +void +c2_scanline_array_proper_alloc( + c2_scanline_array_p a, + c2_coord_t height); +void +c2_scanline_array_proper_clear( + c2_scanline_array_p a); +void +c2_scanline_array_add_coord( + c2_scanline_array_p a, + int inY, + c2_coord_t inX ); + + +typedef c2_polyline_p c2_polygon_p; + +int +c2_polygon_isempty( + c2_polygon_p pl); + +c2_coord_t +c2_polygon_get_heigth( + c2_polygon_p pl); + + +void +c2_polygon_clip( + c2_polygon_p pl, + c2_rect_p clip, + c2_polygon_p outPoly ); + +void +c2_polygon_scanline( + c2_polygon_p pl, + c2_scanline_array_p ioList, + c2_coord_t ymin ); + + + +#endif /* C2_GEOMETRY_POLY_H_ */ diff --git a/libmui/mui/c_array.h b/libmui/mui/c_array.h new file mode 100644 index 0000000..fd40780 --- /dev/null +++ b/libmui/mui/c_array.h @@ -0,0 +1,161 @@ +/* + * c_array.h + * + * Copyright 2012 Michel Pollet + * + * SPDX-License-Identifier: Apache-2.0 + * vi: ts=4 + */ + + +#ifndef __C_ARRAY_H___ +#define __C_ARRAY_H___ + +#include +#include +#include + +#ifndef C_ARRAY_INLINE +#define C_ARRAY_INLINE inline +#endif +#ifndef C_ARRAY_SIZE_TYPE +#define C_ARRAY_SIZE_TYPE uint32_t +#endif +/* New compilers don't like unused static functions. However, + * we do like 'static inlines' for these small accessors, + * so we mark them as 'unused'. It stops it complaining */ +#ifdef __GNUC__ +#define C_ARRAY_DECL static __attribute__ ((unused)) +#else +#define C_ARRAY_DECL static +#endif + + +#define DECLARE_C_ARRAY(__type, __name, __page, __args...) \ +enum { __name##_page_size = __page }; \ +typedef __type __name##_element_t; \ +typedef C_ARRAY_SIZE_TYPE __name##_count_t; \ +typedef struct __name##_t {\ + volatile __name##_count_t count;\ + volatile __name##_count_t size;\ + __name##_element_t * e;\ + __args \ +} __name##_t, *__name##_p; + +#define C_ARRAY_NULL { 0, 0, NULL } + +#ifndef NO_DECL +#define IMPLEMENT_C_ARRAY(__name) \ +C_ARRAY_DECL C_ARRAY_INLINE \ + void __name##_init(\ + __name##_p a) \ +{\ + static const __name##_t zero = {}; \ + if (!a) return;\ + *a = zero;\ +}\ +C_ARRAY_DECL C_ARRAY_INLINE \ + void __name##_free(\ + __name##_p a) \ +{\ + if (!a) return;\ + if (a->e) free(a->e);\ + a->count = a->size = 0;\ + a->e = NULL;\ +}\ +C_ARRAY_DECL C_ARRAY_INLINE \ + void __name##_clear(\ + __name##_p a) \ +{\ + if (!a) return;\ + a->count = 0;\ +}\ +C_ARRAY_DECL C_ARRAY_INLINE \ + void __name##_realloc(\ + __name##_p a, __name##_count_t size) \ +{\ + if (!a || a->size == size) return; \ + if (size == 0) { if (a->e) free(a->e); a->e = NULL; } \ + else a->e = (__name##_element_t*)realloc(a->e, \ + size * sizeof(__name##_element_t));\ + a->size = size; \ +}\ +C_ARRAY_DECL C_ARRAY_INLINE \ + void __name##_trim(\ + __name##_p a) \ +{\ + if (!a) return;\ + __name##_count_t n = a->count + __name##_page_size;\ + n -= (n % __name##_page_size);\ + if (n != a->size)\ + __name##_realloc(a, n);\ +}\ +C_ARRAY_DECL C_ARRAY_INLINE \ + __name##_element_t * __name##_get_ptr(\ + __name##_p a, __name##_count_t index) \ +{\ + if (!a) return NULL;\ + if (index > a->count) index = a->count;\ + return index < a->count ? a->e + index : NULL;\ +}\ +C_ARRAY_DECL C_ARRAY_INLINE \ + __name##_count_t __name##_add(\ + __name##_p a, __name##_element_t e) \ +{\ + if (!a) return 0;\ + if (a->count + 1 >= a->size)\ + __name##_realloc(a, a->size + __name##_page_size);\ + a->e[a->count++] = e;\ + return a->count;\ +}\ +C_ARRAY_DECL C_ARRAY_INLINE \ + __name##_count_t __name##_push(\ + __name##_p a, __name##_element_t e) \ +{\ + return __name##_add(a, e);\ +}\ +C_ARRAY_DECL C_ARRAY_INLINE \ + int __name##_pop(\ + __name##_p a, __name##_element_t *e) \ +{\ + if (a->count) { *e = a->e[--a->count]; return 1; } \ + return 0;\ +}\ +C_ARRAY_DECL C_ARRAY_INLINE \ + __name##_count_t __name##_insert(\ + __name##_p a, __name##_count_t index, \ + __name##_element_t * e, __name##_count_t count) \ +{\ + if (!a) return 0;\ + if (index > a->count) index = a->count;\ + if (a->count + count >= a->size) \ + __name##_realloc(a, (((a->count + count) / __name##_page_size)+1) * __name##_page_size);\ + if (index < a->count)\ + memmove(&a->e[index + count], &a->e[index], \ + (a->count - index) * sizeof(__name##_element_t));\ + memmove(&a->e[index], e, count * sizeof(__name##_element_t));\ + a->count += count;\ + return a->count;\ +}\ +C_ARRAY_DECL C_ARRAY_INLINE \ + __name##_count_t __name##_delete(\ + __name##_p a, __name##_count_t index, __name##_count_t count) \ +{\ + if (!a) return 0;\ + if (index > a->count) index = a->count;\ + if (index + count > a->count) \ + count = a->count - index;\ + if (count && a->count - index) { \ + memmove(&a->e[index], &a->e[index + count], \ + (a->count - index - count) * sizeof(__name##_element_t));\ + }\ + a->count -= count;\ + return a->count;\ +} +#else /* NO_DECL */ + +#define IMPLEMENT_C_ARRAY(__name) + +#endif + +#endif /* __C_ARRAY_H___ */ diff --git a/libmui/mui/cg.c b/libmui/mui/cg.c new file mode 100644 index 0000000..35fac43 --- /dev/null +++ b/libmui/mui/cg.c @@ -0,0 +1,2512 @@ +/* + * cg.c + * + * Copyright(c) 2007-2023 Jianjun Jiang <8192542@qq.com> + * Mobile phone: +86-18665388956 + * QQ: 8192542 + * + * 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. + * + */ + +#include "cg.h" + +#define cg_array_init(array) \ + do { \ + array.data = NULL; \ + array.size = 0; \ + array.capacity = 0; \ + } while(0) + +#define cg_array_ensure(array, count) \ + do { \ + if(array.size + count > array.capacity) { \ + int capacity = array.size + count; \ + int newcapacity = (array.capacity == 0) ? 8 : array.capacity; \ + while(newcapacity < capacity) { newcapacity <<= 1; } \ + array.data = realloc(array.data, (size_t)newcapacity * sizeof(array.data[0])); \ + array.capacity = newcapacity; \ + } \ + } while(0) + +static inline void cg_color_init_rgba(struct cg_color_t * color, double r, double g, double b, double a) +{ + color->r = r; + color->g = g; + color->b = b; + color->a = a; +} + +static inline void cg_rect_init(struct cg_rect_t * rect, double x, double y, double w, double h) +{ + rect->x = x; + rect->y = y; + rect->w = w; + rect->h = h; +} + +void cg_matrix_init(struct cg_matrix_t * m, double a, double b, double c, double d, double tx, double ty) +{ + m->a = a; m->b = b; + m->c = c; m->d = d; + m->tx = tx; m->ty = ty; +} + +void cg_matrix_init_identity(struct cg_matrix_t * m) +{ + m->a = 1; m->b = 0; + m->c = 0; m->d = 1; + m->tx = 0; m->ty = 0; +} + +void cg_matrix_init_translate(struct cg_matrix_t * m, double tx, double ty) +{ + m->a = 1; m->b = 0; + m->c = 0; m->d = 1; + m->tx = tx; m->ty = ty; +} + +void cg_matrix_init_scale(struct cg_matrix_t * m, double sx, double sy) +{ + m->a = sx; m->b = 0; + m->c = 0; m->d = sy; + m->tx = 0; m->ty = 0; +} + +void cg_matrix_init_rotate(struct cg_matrix_t * m, double r) +{ + double s = sin(r); + double c = cos(r); + + m->a = c; m->b = s; + m->c = -s; m->d = c; + m->tx = 0; m->ty = 0; +} + +void cg_matrix_translate(struct cg_matrix_t * m, double tx, double ty) +{ + m->tx += m->a * tx + m->c * ty; + m->ty += m->b * tx + m->d * ty; +} + +void cg_matrix_scale(struct cg_matrix_t * m, double sx, double sy) +{ + m->a *= sx; + m->b *= sx; + m->c *= sy; + m->d *= sy; +} + +void cg_matrix_rotate(struct cg_matrix_t * m, double r) +{ + double s = sin(r); + double c = cos(r); + double ca = c * m->a; + double cb = c * m->b; + double cc = c * m->c; + double cd = c * m->d; + double sa = s * m->a; + double sb = s * m->b; + double sc = s * m->c; + double sd = s * m->d; + + m->a = ca + sc; + m->b = cb + sd; + m->c = cc - sa; + m->d = cd - sb; +} + +void cg_matrix_multiply(struct cg_matrix_t * m, struct cg_matrix_t * m1, struct cg_matrix_t * m2) +{ + struct cg_matrix_t t; + + t.a = m1->a * m2->a; + t.b = 0.0; + t.c = 0.0; + t.d = m1->d * m2->d; + t.tx = m1->tx * m2->a + m2->tx; + t.ty = m1->ty * m2->d + m2->ty; + if(m1->b != 0.0 || m1->c != 0.0 || m2->b != 0.0 || m2->c != 0.0) + { + t.a += m1->b * m2->c; + t.b += m1->a * m2->b + m1->b * m2->d; + t.c += m1->c * m2->a + m1->d * m2->c; + t.d += m1->c * m2->b; + t.tx += m1->ty * m2->c; + t.ty += m1->tx * m2->b; + } + memcpy(m, &t, sizeof(struct cg_matrix_t)); +} + +void cg_matrix_invert(struct cg_matrix_t * m) +{ + double a, b, c, d, tx, ty; + double det; + + if((m->c == 0.0) && (m->b == 0.0)) + { + m->tx = -m->tx; + m->ty = -m->ty; + if(m->a != 1.0) + { + if(m->a == 0.0) + return; + m->a = 1.0 / m->a; + m->tx *= m->a; + } + if(m->d != 1.0) + { + if(m->d == 0.0) + return; + m->d = 1.0 / m->d; + m->ty *= m->d; + } + } + else + { + det = m->a * m->d - m->b * m->c; + if(det != 0.0) + { + a = m->a; + b = m->b; + c = m->c; + d = m->d; + tx = m->tx; + ty = m->ty; + m->a = d / det; + m->b = -b / det; + m->c = -c / det; + m->d = a / det; + m->tx = (c * ty - d * tx) / det; + m->ty = (b * tx - a * ty) / det; + } + } +} + +void cg_matrix_map_point(struct cg_matrix_t * m, struct cg_point_t * p1, struct cg_point_t * p2) +{ + p2->x = p1->x * m->a + p1->y * m->c + m->tx; + p2->y = p1->x * m->b + p1->y * m->d + m->ty; +} + +struct cg_surface_t * cg_surface_create(int width, int height) +{ + struct cg_surface_t * surface = malloc(sizeof(struct cg_surface_t)); + surface->ref = 1; + surface->width = width; + surface->height = height; + surface->stride = width << 2; + surface->owndata = 1; + surface->pixels = calloc(1, (size_t)(height * surface->stride)); + return surface; +} + +struct cg_surface_t * cg_surface_create_for_data(int width, int height, void * pixels) +{ + struct cg_surface_t * surface = malloc(sizeof(struct cg_surface_t)); + surface->ref = 1; + surface->width = width; + surface->height = height; + surface->stride = width << 2; + surface->owndata = 0; + surface->pixels = pixels; + return surface; +} + +void cg_surface_destroy(struct cg_surface_t * surface) +{ + if(surface) + { + if(--surface->ref == 0) + { + if(surface->owndata) + free(surface->pixels); + free(surface); + } + } +} + +struct cg_surface_t * cg_surface_reference(struct cg_surface_t * surface) +{ + if(surface) + { + ++surface->ref; + return surface; + } + return NULL; +} + +static struct cg_path_t * cg_path_create(void) +{ + struct cg_path_t * path = malloc(sizeof(struct cg_path_t)); + path->contours = 0; + path->start.x = 0.0; + path->start.y = 0.0; + cg_array_init(path->elements); + cg_array_init(path->points); + return path; +} + +static void cg_path_destroy(struct cg_path_t * path) +{ + if(path) + { + if(path->elements.data) + free(path->elements.data); + if(path->points.data) + free(path->points.data); + free(path); + } +} + +static inline void cg_path_get_current_point(struct cg_path_t * path, double * x, double * y) +{ + if(path->points.size == 0) + { + *x = 0.0; + *y = 0.0; + } + else + { + *x = path->points.data[path->points.size - 1].x; + *y = path->points.data[path->points.size - 1].y; + } +} + +static void cg_path_move_to(struct cg_path_t * path, double x, double y) +{ + cg_array_ensure(path->elements, 1); + cg_array_ensure(path->points, 1); + + path->elements.data[path->elements.size] = CG_PATH_ELEMENT_MOVE_TO; + path->elements.size += 1; + path->contours += 1; + path->points.data[path->points.size].x = x; + path->points.data[path->points.size].y = y; + path->points.size += 1; + path->start.x = x; + path->start.y = y; +} + +static void cg_path_line_to(struct cg_path_t * path, double x, double y) +{ + cg_array_ensure(path->elements, 1); + cg_array_ensure(path->points, 1); + + path->elements.data[path->elements.size] = CG_PATH_ELEMENT_LINE_TO; + path->elements.size += 1; + path->points.data[path->points.size].x = x; + path->points.data[path->points.size].y = y; + path->points.size += 1; +} + +static void cg_path_curve_to(struct cg_path_t * path, double x1, double y1, double x2, double y2, double x3, double y3) +{ + cg_array_ensure(path->elements, 1); + cg_array_ensure(path->points, 3); + + path->elements.data[path->elements.size] = CG_PATH_ELEMENT_CURVE_TO; + path->elements.size += 1; + struct cg_point_t * points = path->points.data + path->points.size; + points[0].x = x1; + points[0].y = y1; + points[1].x = x2; + points[1].y = y2; + points[2].x = x3; + points[2].y = y3; + path->points.size += 3; +} + +static void cg_path_quad_to(struct cg_path_t * path, double x1, double y1, double x2, double y2) +{ + double x, y; + cg_path_get_current_point(path, &x, &y); + + double cx = 2.0 / 3.0 * x1 + 1.0 / 3.0 * x; + double cy = 2.0 / 3.0 * y1 + 1.0 / 3.0 * y; + double cx1 = 2.0 / 3.0 * x1 + 1.0 / 3.0 * x2; + double cy1 = 2.0 / 3.0 * y1 + 1.0 / 3.0 * y2; + cg_path_curve_to(path, cx, cy, cx1, cy1, x2, y2); +} + +static void cg_path_close(struct cg_path_t * path) +{ + if(path->elements.size == 0) + return; + if(path->elements.data[path->elements.size - 1] == CG_PATH_ELEMENT_CLOSE) + return; + cg_array_ensure(path->elements, 1); + cg_array_ensure(path->points, 1); + path->elements.data[path->elements.size] = CG_PATH_ELEMENT_CLOSE; + path->elements.size += 1; + path->points.data[path->points.size].x = path->start.x; + path->points.data[path->points.size].y = path->start.y; + path->points.size += 1; +} + +static void cg_path_rel_move_to(struct cg_path_t * path, double dx, double dy) +{ + double x, y; + cg_path_get_current_point(path, &x, &y); + cg_path_move_to(path, dx + x, dy + y); +} + +static void cg_path_rel_line_to(struct cg_path_t * path, double dx, double dy) +{ + double x, y; + cg_path_get_current_point(path, &x, &y); + cg_path_line_to(path, dx + x, dy + y); +} + +static void cg_path_rel_curve_to(struct cg_path_t * path, double dx1, double dy1, double dx2, double dy2, double dx3, double dy3) +{ + double x, y; + cg_path_get_current_point(path, &x, &y); + cg_path_curve_to(path, dx1 + x, dy1 + y, dx2 + x, dy2 + y, dx3 + x, dy3 + y); +} + +static void cg_path_rel_quad_to(struct cg_path_t * path, double dx1, double dy1, double dx2, double dy2) +{ + double x, y; + cg_path_get_current_point(path, &x, &y); + cg_path_quad_to(path, dx1 + x, dy1 + y, dx2 + x, dy2 + y); +} + +static inline void cg_path_add_rectangle(struct cg_path_t * path, double x, double y, double w, double h) +{ + cg_path_move_to(path, x, y); + cg_path_line_to(path, x + w, y); + cg_path_line_to(path, x + w, y + h); + cg_path_line_to(path, x, y + h); + cg_path_line_to(path, x, y); + cg_path_close(path); +} + +static inline void cg_path_add_round_rectangle(struct cg_path_t * path, double x, double y, double w, double h, double rx, double ry) +{ + rx = CG_MIN(rx, w * 0.5); + ry = CG_MIN(ry, h * 0.5); + + double right = x + w; + double bottom = y + h; + double cpx = rx * 0.55228474983079339840; + double cpy = ry * 0.55228474983079339840; + + cg_path_move_to(path, x, y + ry); + cg_path_curve_to(path, x, y + ry - cpy, x + rx - cpx, y, x + rx, y); + cg_path_line_to(path, right - rx, y); + cg_path_curve_to(path, right - rx + cpx, y, right, y + ry - cpy, right, y + ry); + cg_path_line_to(path, right, bottom - ry); + cg_path_curve_to(path, right, bottom - ry + cpy, right - rx + cpx, bottom, right - rx, bottom); + cg_path_line_to(path, x + rx, bottom); + cg_path_curve_to(path, x + rx - cpx, bottom, x, bottom - ry + cpy, x, bottom - ry); + cg_path_line_to(path, x, y + ry); + cg_path_close(path); +} + +static void cg_path_add_ellipse(struct cg_path_t * path, double cx, double cy, double rx, double ry) +{ + double left = cx - rx; + double top = cy - ry; + double right = cx + rx; + double bottom = cy + ry; + double cpx = rx * 0.55228474983079339840; + double cpy = ry * 0.55228474983079339840; + + cg_path_move_to(path, cx, top); + cg_path_curve_to(path, cx + cpx, top, right, cy - cpy, right, cy); + cg_path_curve_to(path, right, cy + cpy, cx + cpx, bottom, cx, bottom); + cg_path_curve_to(path, cx - cpx, bottom, left, cy + cpy, left, cy); + cg_path_curve_to(path, left, cy - cpy, cx - cpx, top, cx, top); + cg_path_close(path); +} + +static void cg_path_add_arc(struct cg_path_t * path, double cx, double cy, double r, double a0, double a1, int ccw) +{ + double da = a1 - a0; + if(fabs(da) > 6.28318530717958647693) + { + da = 6.28318530717958647693; + } + else if(da != 0.0 && ccw != (da < 0.0)) + { + da += 6.28318530717958647693 * (ccw ? -1 : 1); + } + int seg_n = (int)(ceil(fabs(da) / 1.57079632679489661923)); + double seg_a = da / seg_n; + double d = (seg_a / 1.57079632679489661923) * 0.55228474983079339840 * r; + double a = a0; + double ax = cx + cos(a) * r; + double ay = cy + sin(a) * r; + double dx = -sin(a) * d; + double dy = cos(a) * d; + if(path->points.size == 0) + cg_path_move_to(path, ax, ay); + else + cg_path_line_to(path, ax, ay); + for(int i = 0; i < seg_n; i++) + { + double cp1x = ax + dx; + double cp1y = ay + dy; + a += seg_a; + ax = cx + cos(a) * r; + ay = cy + sin(a) * r; + dx = -sin(a) * d; + dy = cos(a) * d; + double cp2x = ax - dx; + double cp2y = ay - dy; + cg_path_curve_to(path, cp1x, cp1y, cp2x, cp2y, ax, ay); + } +} + +static inline void cg_path_clear(struct cg_path_t * path) +{ + path->elements.size = 0; + path->points.size = 0; + path->contours = 0; + path->start.x = 0.0; + path->start.y = 0.0; +} + +struct cg_bezier_t { + double x1; double y1; + double x2; double y2; + double x3; double y3; + double x4; double y4; +}; + +static inline void split(struct cg_bezier_t * b, struct cg_bezier_t * first, struct cg_bezier_t * second) +{ + double c = (b->x2 + b->x3) * 0.5; + first->x2 = (b->x1 + b->x2) * 0.5; + second->x3 = (b->x3 + b->x4) * 0.5; + first->x1 = b->x1; + second->x4 = b->x4; + first->x3 = (first->x2 + c) * 0.5; + second->x2 = (second->x3 + c) * 0.5; + first->x4 = second->x1 = (first->x3 + second->x2) * 0.5; + + c = (b->y2 + b->y3) * 0.5; + first->y2 = (b->y1 + b->y2) * 0.5; + second->y3 = (b->y3 + b->y4) * 0.5; + first->y1 = b->y1; + second->y4 = b->y4; + first->y3 = (first->y2 + c) * 0.5; + second->y2 = (second->y3 + c) * 0.5; + first->y4 = second->y1 = (first->y3 + second->y2) * 0.5; +} + +static inline void flatten(struct cg_path_t * path, struct cg_point_t * p0, struct cg_point_t * p1, struct cg_point_t * p2, struct cg_point_t * p3) +{ + struct cg_bezier_t beziers[32]; + struct cg_bezier_t * b = beziers; + + beziers[0].x1 = p0->x; + beziers[0].y1 = p0->y; + beziers[0].x2 = p1->x; + beziers[0].y2 = p1->y; + beziers[0].x3 = p2->x; + beziers[0].y3 = p2->y; + beziers[0].x4 = p3->x; + beziers[0].y4 = p3->y; + while(b >= beziers) + { + double y4y1 = b->y4 - b->y1; + double x4x1 = b->x4 - b->x1; + double l = fabs(x4x1) + fabs(y4y1); + double d; + if(l > 1.0) + { + d = fabs((x4x1) * (b->y1 - b->y2) - (y4y1) * (b->x1 - b->x2)) + fabs((x4x1) * (b->y1 - b->y3) - (y4y1) * (b->x1 - b->x3)); + } + else + { + d = fabs(b->x1 - b->x2) + fabs(b->y1 - b->y2) + fabs(b->x1 - b->x3) + fabs(b->y1 - b->y3); + l = 1.0; + } + if((d < l * 0.25) || (b == beziers + 31)) + { + cg_path_line_to(path, b->x4, b->y4); + --b; + } + else + { + split(b, b + 1, b); + ++b; + } + } +} + +static inline struct cg_path_t * cg_path_clone(const struct cg_path_t * path) +{ + struct cg_path_t * result = cg_path_create(); + cg_array_ensure(result->elements, path->elements.size); + cg_array_ensure(result->points, path->points.size); + + memcpy(result->elements.data, path->elements.data, (size_t)path->elements.size * sizeof(enum cg_path_element_t)); + memcpy(result->points.data, path->points.data, (size_t)path->points.size * sizeof(struct cg_point_t)); + + result->elements.size = path->elements.size; + result->points.size = path->points.size; + result->contours = path->contours; + result->start = path->start; + return result; +} + +static inline struct cg_path_t * cg_path_clone_flat(struct cg_path_t * path) +{ + struct cg_point_t * points = path->points.data; + struct cg_path_t * result = cg_path_create(); + struct cg_point_t p0; + + cg_array_ensure(result->elements, path->elements.size); + cg_array_ensure(result->points, path->points.size); + for(int i = 0; i < path->elements.size; i++) + { + switch(path->elements.data[i]) + { + case CG_PATH_ELEMENT_MOVE_TO: + cg_path_move_to(result, points[0].x, points[0].y); + points += 1; + break; + case CG_PATH_ELEMENT_LINE_TO: + cg_path_line_to(result, points[0].x, points[0].y); + points += 1; + break; + case CG_PATH_ELEMENT_CURVE_TO: + cg_path_get_current_point(result, &p0.x, &p0.y); + flatten(result, &p0, points, points + 1, points + 2); + points += 3; + break; + case CG_PATH_ELEMENT_CLOSE: + cg_path_line_to(result, points[0].x, points[0].y); + points += 1; + break; + default: + break; + } + } + return result; +} + +static struct cg_dash_t * cg_dash_create(double * dashes, int ndash, double offset) +{ + if(dashes && (ndash > 0)) + { + struct cg_dash_t * dash = malloc(sizeof(struct cg_dash_t)); + dash->offset = offset; + dash->data = malloc((size_t)ndash * sizeof(double)); + dash->size = ndash; + memcpy(dash->data, dashes, (size_t)ndash * sizeof(double)); + return dash; + } + return NULL; +} + +static struct cg_dash_t * cg_dash_clone(struct cg_dash_t * dash) +{ + if(dash) + return cg_dash_create(dash->data, dash->size, dash->offset); + return NULL; +} + +static void cg_dash_destroy(struct cg_dash_t * dash) +{ + if(dash) + { + free(dash->data); + free(dash); + } +} + +static inline struct cg_path_t * cg_dash_path(struct cg_dash_t * dash, struct cg_path_t * path) +{ + if((dash->data == NULL) || (dash->size <= 0)) + return cg_path_clone(path); + + int toggle = 1; + int offset = 0; + double phase = dash->offset; + while(phase >= dash->data[offset]) + { + toggle = !toggle; + phase -= dash->data[offset]; + offset += 1; + if(offset == dash->size) + offset = 0; + } + + struct cg_path_t * flat = cg_path_clone_flat(path); + struct cg_path_t * result = cg_path_create(); + cg_array_ensure(result->elements, flat->elements.size); + cg_array_ensure(result->points, flat->points.size); + + enum cg_path_element_t * elements = flat->elements.data; + enum cg_path_element_t * end = elements + flat->elements.size; + struct cg_point_t * points = flat->points.data; + while(elements < end) + { + int itoggle = toggle; + int ioffset = offset; + double iphase = phase; + double x0 = points->x; + double y0 = points->y; + if(itoggle) + cg_path_move_to(result, x0, y0); + ++elements; + ++points; + while((elements < end) && (*elements == CG_PATH_ELEMENT_LINE_TO)) + { + double dx = points->x - x0; + double dy = points->y - y0; + double dist0 = sqrt(dx * dx + dy * dy); + double dist1 = 0; + while(dist0 - dist1 > dash->data[ioffset] - iphase) + { + dist1 += dash->data[ioffset] - iphase; + double a = dist1 / dist0; + double x = x0 + a * dx; + double y = y0 + a * dy; + if(itoggle) + cg_path_line_to(result, x, y); + else + cg_path_move_to(result, x, y); + itoggle = !itoggle; + iphase = 0; + ioffset += 1; + if(ioffset == dash->size) + ioffset = 0; + } + iphase += dist0 - dist1; + x0 = points->x; + y0 = points->y; + if(itoggle) + cg_path_line_to(result, x0, y0); + ++elements; + ++points; + } + } + cg_path_destroy(flat); + return result; +} + +#define ALIGN_SIZE(size) (((size) + 7ul) & ~7ul) +static void ft_outline_init(XCG_FT_Outline * outline, struct cg_ctx_t * ctx, int points, int contours) +{ + size_t size_a = ALIGN_SIZE((points + contours) * sizeof(XCG_FT_Vector)); + size_t size_b = ALIGN_SIZE((points + contours) * sizeof(char)); + size_t size_c = ALIGN_SIZE(contours * sizeof(int)); + size_t size_d = ALIGN_SIZE(contours * sizeof(char)); + size_t size_n = size_a + size_b + size_c + size_d; + if(size_n > ctx->outline_size) + { + ctx->outline_data = realloc(ctx->outline_data, size_n); + ctx->outline_size = size_n; + } + + XCG_FT_Byte * data = ctx->outline_data; + outline->points = (XCG_FT_Vector *)(data); + outline->tags = outline->contours_flag = NULL; + outline->contours = NULL; + if(data) + { + outline->tags = (char *)(data + size_a); + outline->contours = (int *)(data + size_a + size_b); + outline->contours_flag = (char*)(data + size_a + size_b + size_c); + } + outline->n_points = 0; + outline->n_contours = 0; + outline->flags = 0x0; +} + +#define FT_COORD(x) (XCG_FT_Pos)((x) * 64) +static void ft_outline_move_to(XCG_FT_Outline * ft, double x, double y) +{ + ft->points[ft->n_points].x = FT_COORD(x); + ft->points[ft->n_points].y = FT_COORD(y); + ft->tags[ft->n_points] = XCG_FT_CURVE_TAG_ON; + if(ft->n_points) + { + ft->contours[ft->n_contours] = ft->n_points - 1; + ft->n_contours++; + } + ft->contours_flag[ft->n_contours] = 1; + ft->n_points++; +} + +static void ft_outline_line_to(XCG_FT_Outline * ft, double x, double y) +{ + ft->points[ft->n_points].x = FT_COORD(x); + ft->points[ft->n_points].y = FT_COORD(y); + ft->tags[ft->n_points] = XCG_FT_CURVE_TAG_ON; + ft->n_points++; +} + +static void ft_outline_curve_to(XCG_FT_Outline * ft, double x1, double y1, double x2, double y2, double x3, double y3) +{ + ft->points[ft->n_points].x = FT_COORD(x1); + ft->points[ft->n_points].y = FT_COORD(y1); + ft->tags[ft->n_points] = XCG_FT_CURVE_TAG_CUBIC; + ft->n_points++; + + ft->points[ft->n_points].x = FT_COORD(x2); + ft->points[ft->n_points].y = FT_COORD(y2); + ft->tags[ft->n_points] = XCG_FT_CURVE_TAG_CUBIC; + ft->n_points++; + + ft->points[ft->n_points].x = FT_COORD(x3); + ft->points[ft->n_points].y = FT_COORD(y3); + ft->tags[ft->n_points] = XCG_FT_CURVE_TAG_ON; + ft->n_points++; +} + +static void ft_outline_close(XCG_FT_Outline * ft) +{ + ft->contours_flag[ft->n_contours] = 0; + int index = ft->n_contours ? ft->contours[ft->n_contours - 1] + 1 : 0; + if(index == ft->n_points) + return; + + ft->points[ft->n_points].x = ft->points[index].x; + ft->points[ft->n_points].y = ft->points[index].y; + ft->tags[ft->n_points] = XCG_FT_CURVE_TAG_ON; + ft->n_points++; +} + +static void ft_outline_end(XCG_FT_Outline * ft) +{ + if(ft->n_points) + { + ft->contours[ft->n_contours] = ft->n_points - 1; + ft->n_contours++; + } +} + +static void ft_outline_convert(XCG_FT_Outline * outline, struct cg_ctx_t * ctx, struct cg_path_t * path, struct cg_matrix_t * matrix) +{ + ft_outline_init(outline, ctx, path->points.size, path->contours); + enum cg_path_element_t * elements = path->elements.data; + struct cg_point_t * points = path->points.data; + struct cg_point_t p[3]; + for(int i = 0; i < path->elements.size; i++) + { + switch(elements[i]) + { + case CG_PATH_ELEMENT_MOVE_TO: + cg_matrix_map_point(matrix, &points[0], &p[0]); + ft_outline_move_to(outline, p[0].x, p[0].y); + points += 1; + break; + case CG_PATH_ELEMENT_LINE_TO: + cg_matrix_map_point(matrix, &points[0], &p[0]); + ft_outline_line_to(outline, p[0].x, p[0].y); + points += 1; + break; + case CG_PATH_ELEMENT_CURVE_TO: + cg_matrix_map_point(matrix, &points[0], &p[0]); + cg_matrix_map_point(matrix, &points[1], &p[1]); + cg_matrix_map_point(matrix, &points[2], &p[2]); + ft_outline_curve_to(outline, p[0].x, p[0].y, p[1].x, p[1].y, p[2].x, p[2].y); + points += 3; + break; + case CG_PATH_ELEMENT_CLOSE: + ft_outline_close(outline); + points += 1; + break; + } + } + ft_outline_end(outline); +} + +static void ft_outline_convert_dash(XCG_FT_Outline * outline, struct cg_ctx_t * ctx, struct cg_path_t * path, struct cg_matrix_t * matrix, struct cg_dash_t * dash) +{ + struct cg_path_t * dashed = cg_dash_path(dash, path); + ft_outline_convert(outline, ctx, dashed, matrix); + cg_path_destroy(dashed); +} + +static void generation_callback(int count, const XCG_FT_Span * spans, void * user) +{ + struct cg_rle_t * rle = user; + cg_array_ensure(rle->spans, count); + struct cg_span_t * data = rle->spans.data + rle->spans.size; + memcpy(data, spans, (size_t)count * sizeof(struct cg_span_t)); + rle->spans.size += count; +} + +static struct cg_rle_t * cg_rle_create(void) +{ + struct cg_rle_t * rle = malloc(sizeof(struct cg_rle_t)); + cg_array_init(rle->spans); + rle->x = 0; + rle->y = 0; + rle->w = 0; + rle->h = 0; + return rle; +} + +static void cg_rle_destroy(struct cg_rle_t * rle) +{ + if(rle) + { + free(rle->spans.data); + free(rle); + } +} + +static void cg_rle_rasterize(struct cg_ctx_t * ctx, struct cg_rle_t * rle, struct cg_path_t * path, struct cg_matrix_t * m, struct cg_rect_t * clip, struct cg_stroke_data_t * stroke, enum cg_fill_rule_t winding) +{ + XCG_FT_Raster_Params params; + params.flags = XCG_FT_RASTER_FLAG_DIRECT | XCG_FT_RASTER_FLAG_AA; + params.gray_spans = generation_callback; + params.user = rle; + if(clip) + { + params.flags |= XCG_FT_RASTER_FLAG_CLIP; + params.clip_box.xMin = (XCG_FT_Pos)(clip->x); + params.clip_box.yMin = (XCG_FT_Pos)(clip->y); + params.clip_box.xMax = (XCG_FT_Pos)(clip->x + clip->w); + params.clip_box.yMax = (XCG_FT_Pos)(clip->y + clip->h); + } + if(stroke) + { + XCG_FT_Outline outline; + if(stroke->dash == NULL) + ft_outline_convert(&outline, ctx, path, m); + else + ft_outline_convert_dash(&outline, ctx, path, m, stroke->dash); + XCG_FT_Stroker_LineCap ftCap; + XCG_FT_Stroker_LineJoin ftJoin; + XCG_FT_Fixed ftWidth; + XCG_FT_Fixed ftMiterLimit; + + struct cg_point_t p1 = { 0, 0 }; + struct cg_point_t p2 = { 1.41421356237309504880, 1.41421356237309504880 }; + struct cg_point_t p3; + + cg_matrix_map_point(m, &p1, &p1); + cg_matrix_map_point(m, &p2, &p2); + + p3.x = p2.x - p1.x; + p3.y = p2.y - p1.y; + + double scale = sqrt(p3.x * p3.x + p3.y * p3.y) / 2.0; + + ftWidth = (XCG_FT_Fixed)(stroke->width * scale * 0.5 * (1 << 6)); + ftMiterLimit = (XCG_FT_Fixed)(stroke->miterlimit * (1 << 16)); + + switch(stroke->cap) + { + case CG_LINE_CAP_SQUARE: + ftCap = XCG_FT_STROKER_LINECAP_SQUARE; + break; + case CG_LINE_CAP_ROUND: + ftCap = XCG_FT_STROKER_LINECAP_ROUND; + break; + default: + ftCap = XCG_FT_STROKER_LINECAP_BUTT; + break; + } + switch(stroke->join) + { + case CG_LINE_JOIN_BEVEL: + ftJoin = XCG_FT_STROKER_LINEJOIN_BEVEL; + break; + case CG_LINE_JOIN_ROUND: + ftJoin = XCG_FT_STROKER_LINEJOIN_ROUND; + break; + default: + ftJoin = XCG_FT_STROKER_LINEJOIN_MITER_FIXED; + break; + } + XCG_FT_Stroker stroker; + XCG_FT_Stroker_New(&stroker); + XCG_FT_Stroker_Set(stroker, ftWidth, ftCap, ftJoin, ftMiterLimit); + XCG_FT_Stroker_ParseOutline(stroker, &outline); + + XCG_FT_UInt points; + XCG_FT_UInt contours; + XCG_FT_Stroker_GetCounts(stroker, &points, &contours); + + ft_outline_init(&outline, ctx, points, contours); + XCG_FT_Stroker_Export(stroker, &outline); + XCG_FT_Stroker_Done(stroker); + + outline.flags = XCG_FT_OUTLINE_NONE; + params.source = &outline; + XCG_FT_Raster_Render(¶ms); + } + else + { + XCG_FT_Outline outline; + ft_outline_convert(&outline, ctx, path, m); + switch(winding) + { + case CG_FILL_RULE_EVEN_ODD: + outline.flags = XCG_FT_OUTLINE_EVEN_ODD_FILL; + break; + default: + outline.flags = XCG_FT_OUTLINE_NONE; + break; + } + + params.source = &outline; + XCG_FT_Raster_Render(¶ms); + } + + if(rle->spans.size == 0) + { + rle->x = 0; + rle->y = 0; + rle->w = 0; + rle->h = 0; + return; + } + + struct cg_span_t *spans = rle->spans.data; + int x1 = INT_MAX; + int y1 = spans[0].y; + int x2 = 0; + int y2 = spans[rle->spans.size - 1].y; + for(int i = 0; i < rle->spans.size; i++) + { + if(spans[i].x < x1) + x1 = spans[i].x; + if(spans[i].x + spans[i].len > x2) + x2 = spans[i].x + spans[i].len; + } + + rle->x = x1; + rle->y = y1; + rle->w = x2 - x1; + rle->h = y2 - y1 + 1; +} + +static struct cg_rle_t * cg_rle_intersection(struct cg_rle_t * a, struct cg_rle_t * b) +{ + struct cg_rle_t * result = malloc(sizeof(struct cg_rle_t)); + cg_array_init(result->spans); + cg_array_ensure(result->spans, CG_MAX(a->spans.size, b->spans.size)); + + struct cg_span_t * a_spans = a->spans.data; + struct cg_span_t * a_end = a_spans + a->spans.size; + struct cg_span_t * b_spans = b->spans.data; + struct cg_span_t * b_end = b_spans + b->spans.size; + while((a_spans < a_end) && (b_spans < b_end)) + { + if(b_spans->y > a_spans->y) + { + ++a_spans; + continue; + } + if(a_spans->y != b_spans->y) + { + ++b_spans; + continue; + } + int ax1 = a_spans->x; + int ax2 = ax1 + a_spans->len; + int bx1 = b_spans->x; + int bx2 = bx1 + b_spans->len; + if(bx1 < ax1 && bx2 < ax1) + { + ++b_spans; + continue; + } + else if(ax1 < bx1 && ax2 < bx1) + { + ++a_spans; + continue; + } + int x = CG_MAX(ax1, bx1); + int len = CG_MIN(ax2, bx2) - x; + if(len) + { + struct cg_span_t * span = result->spans.data + result->spans.size; + span->x = x; + span->len = len; + span->y = a_spans->y; + span->coverage = CG_DIV255(a_spans->coverage * b_spans->coverage); + result->spans.size += 1; + } + if(ax2 < bx2) + { + ++a_spans; + } + else + { + ++b_spans; + } + } + if(result->spans.size == 0) + { + result->x = 0; + result->y = 0; + result->w = 0; + result->h = 0; + return result; + } + struct cg_span_t * spans = result->spans.data; + int x1 = INT_MAX; + int y1 = spans[0].y; + int x2 = 0; + int y2 = spans[result->spans.size - 1].y; + for(int i = 0; i < result->spans.size; i++) + { + if(spans[i].x < x1) + x1 = spans[i].x; + if(spans[i].x + spans[i].len > x2) + x2 = spans[i].x + spans[i].len; + } + result->x = x1; + result->y = y1; + result->w = x2 - x1; + result->h = y2 - y1 + 1; + + return result; +} + +static void cg_rle_clip_path(struct cg_rle_t * rle, struct cg_rle_t * clip) +{ + if(rle && clip) + { + struct cg_rle_t * result = cg_rle_intersection(rle, clip); + cg_array_ensure(rle->spans, result->spans.size); + memcpy(rle->spans.data, result->spans.data, (size_t)result->spans.size * sizeof(struct cg_span_t)); + rle->spans.size = result->spans.size; + rle->x = result->x; + rle->y = result->y; + rle->w = result->w; + rle->h = result->h; + cg_rle_destroy(result); + } +} + +static struct cg_rle_t * cg_rle_clone(struct cg_rle_t * rle) +{ + if(rle) + { + struct cg_rle_t * result = malloc(sizeof(struct cg_rle_t)); + cg_array_init(result->spans); + cg_array_ensure(result->spans, rle->spans.size); + memcpy(result->spans.data, rle->spans.data, (size_t)rle->spans.size * sizeof(struct cg_span_t)); + result->spans.size = rle->spans.size; + result->x = rle->x; + result->y = rle->y; + result->w = rle->w; + result->h = rle->h; + return result; + } + return NULL; +} + +static inline void cg_rle_clear(struct cg_rle_t * rle) +{ + rle->spans.size = 0; + rle->x = 0; + rle->y = 0; + rle->w = 0; + rle->h = 0; +} + +static void cg_gradient_init_linear(struct cg_gradient_t * gradient, double x1, double y1, double x2, double y2) +{ + gradient->type = CG_GRADIENT_TYPE_LINEAR; + gradient->spread = CG_SPREAD_METHOD_PAD; + gradient->opacity = 1.0; + gradient->stops.size = 0; + cg_matrix_init_identity(&gradient->matrix); + cg_gradient_set_values_linear(gradient, x1, y1, x2, y2); +} + +static void cg_gradient_init_radial(struct cg_gradient_t * gradient, double cx, double cy, double cr, double fx, double fy, double fr) +{ + gradient->type = CG_GRADIENT_TYPE_RADIAL; + gradient->spread = CG_SPREAD_METHOD_PAD; + gradient->opacity = 1.0; + gradient->stops.size = 0; + cg_matrix_init_identity(&gradient->matrix); + cg_gradient_set_values_radial(gradient, cx, cy, cr, fx, fy, fr); +} + +void cg_gradient_set_values_linear(struct cg_gradient_t * gradient, double x1, double y1, double x2, double y2) +{ + gradient->values[0] = x1; + gradient->values[1] = y1; + gradient->values[2] = x2; + gradient->values[3] = y2; +} + +void cg_gradient_set_values_radial(struct cg_gradient_t * gradient, double cx, double cy, double cr, double fx, double fy, double fr) +{ + gradient->values[0] = cx; + gradient->values[1] = cy; + gradient->values[2] = cr; + gradient->values[3] = fx; + gradient->values[4] = fy; + gradient->values[5] = fr; +} + +void cg_gradient_set_spread(struct cg_gradient_t * gradient, enum cg_spread_method_t spread) +{ + gradient->spread = spread; +} + +void cg_gradient_set_matrix(struct cg_gradient_t * gradient, struct cg_matrix_t * m) +{ + memcpy(&gradient->matrix, m, sizeof(struct cg_matrix_t)); +} + +void cg_gradient_set_opacity(struct cg_gradient_t * gradient, double opacity) +{ + gradient->opacity = CG_CLAMP(opacity, 0.0, 1.0); +} + +void cg_gradient_add_stop_rgb(struct cg_gradient_t * gradient, double offset, double r, double g, double b) +{ + cg_gradient_add_stop_rgba(gradient, offset, r, g, b, 1.0); +} + +void cg_gradient_add_stop_rgba(struct cg_gradient_t * gradient, double offset, double r, double g, double b, double a) +{ + if(offset < 0.0) + offset = 0.0; + if(offset > 1.0) + offset = 1.0; + cg_array_ensure(gradient->stops, 1); + struct cg_gradient_stop_t * stops = gradient->stops.data; + int nstops = gradient->stops.size; + int i; + for(i = 0; i < nstops; i++) + { + if(offset < stops[i].offset) + { + memmove(&stops[i + 1], &stops[i], (size_t)(nstops - i) * sizeof(struct cg_gradient_stop_t)); + break; + } + } + struct cg_gradient_stop_t * stop = &stops[i]; + stop->offset = offset; + cg_color_init_rgba(&stop->color, r, g, b, a); + gradient->stops.size += 1; +} + +void cg_gradient_add_stop_color(struct cg_gradient_t * gradient, double offset, struct cg_color_t * color) +{ + cg_gradient_add_stop_rgba(gradient, offset, color->r, color->g, color->b, color->a); +} + +void cg_gradient_add_stop(struct cg_gradient_t * gradient, struct cg_gradient_stop_t * stop) +{ + cg_gradient_add_stop_rgba(gradient, stop->offset, stop->color.r, stop->color.g, stop->color.b, stop->color.a); +} + +void cg_gradient_clear_stops(struct cg_gradient_t * gradient) +{ + gradient->stops.size = 0; +} + +static void cg_gradient_copy(struct cg_gradient_t * gradient, struct cg_gradient_t * source) +{ + gradient->type = source->type; + gradient->spread = source->spread; + gradient->matrix = source->matrix; + gradient->opacity = source->opacity; + cg_array_ensure(gradient->stops, source->stops.size); + memcpy(gradient->values, source->values, sizeof(source->values)); + memcpy(gradient->stops.data, source->stops.data, source->stops.size * sizeof(struct cg_gradient_stop_t)); +} + +static void cg_gradient_destroy(struct cg_gradient_t * gradient) +{ + if(gradient->stops.data) + free(gradient->stops.data); +} + +static void cg_texture_init(struct cg_texture_t * texture, struct cg_surface_t * surface, enum cg_texture_type_t type) +{ + surface = cg_surface_reference(surface); + cg_surface_destroy(texture->surface); + texture->type = type; + texture->surface = surface; + texture->opacity = 1.0; + cg_matrix_init_identity(&texture->matrix); +} + +void cg_texture_set_type(struct cg_texture_t * texture, enum cg_texture_type_t type) +{ + texture->type = type; +} + +void cg_texture_set_matrix(struct cg_texture_t * texture, struct cg_matrix_t * m) +{ + memcpy(&texture->matrix, m, sizeof(struct cg_matrix_t)); +} + +void cg_texture_set_surface(struct cg_texture_t * texture, struct cg_surface_t * surface) +{ + surface = cg_surface_reference(surface); + cg_surface_destroy(texture->surface); + texture->surface = surface; +} + +void cg_texture_set_opacity(struct cg_texture_t * texture, double opacity) +{ + texture->opacity = CG_CLAMP(opacity, 0.0, 1.0); +} + +static void cg_texture_copy(struct cg_texture_t * texture, struct cg_texture_t * source) +{ + struct cg_surface_t * surface = cg_surface_reference(source->surface); + cg_surface_destroy(texture->surface); + texture->type = source->type; + texture->surface = surface; + texture->opacity = source->opacity; + texture->matrix = source->matrix; +} + +static void cg_texture_destroy(struct cg_texture_t * texture) +{ + cg_surface_destroy(texture->surface); +} + +static void cg_paint_init(struct cg_paint_t * paint) +{ + paint->type = CG_PAINT_TYPE_COLOR; + paint->texture.surface = NULL; + cg_array_init(paint->gradient.stops); + cg_color_init_rgba(&paint->color, 0, 0, 0, 1.0); +} + +static void cg_paint_destroy(struct cg_paint_t * paint) +{ + cg_texture_destroy(&paint->texture); + cg_gradient_destroy(&paint->gradient); +} + +static void cg_paint_copy(struct cg_paint_t * paint, struct cg_paint_t * source) +{ + paint->type = source->type; + switch(source->type) + { + case CG_PAINT_TYPE_COLOR: + paint->color = source->color; + break; + case CG_PAINT_TYPE_GRADIENT: + cg_gradient_copy(&paint->gradient, &source->gradient); + break; + case CG_PAINT_TYPE_TEXTURE: + cg_texture_copy(&paint->texture, &source->texture); + break; + default: + break; + } +} + +struct cg_gradient_data_t { + enum cg_spread_method_t spread; + struct cg_matrix_t matrix; + uint32_t colortable[1024]; + union { + struct { + double x1, y1; + double x2, y2; + } linear; + struct { + double cx, cy, cr; + double fx, fy, fr; + } radial; + }; +}; + +struct cg_texture_data_t { + struct cg_matrix_t matrix; + int width; + int height; + int stride; + int alpha; + void * pixels; +}; + +struct cg_linear_gradient_values_t { + double dx; + double dy; + double l; + double off; +}; + +struct cg_radial_gradient_values_t { + double dx; + double dy; + double dr; + double sqrfr; + double a; + double inv2a; + int extended; +}; + +static inline uint32_t premultiply_color(struct cg_color_t * color, double opacity) +{ + uint32_t alpha = (uint8_t)(color->a * opacity * 255); + uint32_t pr = (uint8_t)(color->r * alpha); + uint32_t pg = (uint8_t)(color->g * alpha); + uint32_t pb = (uint8_t)(color->b * alpha); + return (alpha << 24) | (pr << 16) | (pg << 8) | (pb); +} + +static inline uint32_t combine_opacity(struct cg_color_t * color, double opacity) +{ + uint32_t a = (uint8_t)(color->a * opacity * 255); + uint32_t r = (uint8_t)(color->r * 255); + uint32_t g = (uint8_t)(color->g * 255); + uint32_t b = (uint8_t)(color->b * 255); + return (a << 24) | (r << 16) | (g << 8) | (b); +} + +static inline uint32_t premultiply_pixel(uint32_t color) +{ + uint32_t a = CG_ALPHA(color); + uint32_t r = (color >> 16) & 0xff; + uint32_t g = (color >> 8) & 0xff; + uint32_t b = (color >> 0) & 0xff; + uint32_t pr = (r * a) / 255; + uint32_t pg = (g * a) / 255; + uint32_t pb = (b * a) / 255; + return (a << 24) | (pr << 16) | (pg << 8) | (pb); +} + +static inline uint32_t interpolate_pixel(uint32_t x, uint32_t a, uint32_t y, uint32_t b) +{ + uint32_t t = (x & 0xff00ff) * a + (y & 0xff00ff) * b; + t = (t + ((t >> 8) & 0xff00ff) + 0x800080) >> 8; + t &= 0xff00ff; + x = ((x >> 8) & 0xff00ff) * a + ((y >> 8) & 0xff00ff) * b; + x = (x + ((x >> 8) & 0xff00ff) + 0x800080); + x &= 0xff00ff00; + x |= t; + return x; +} + +static void __cg_memfill32(uint32_t * dst, uint32_t val, int len) +{ + for(int i = 0; i < len; i++) + dst[i] = val; +} +extern __typeof(__cg_memfill32) cg_memfill32 __attribute__((weak, alias("__cg_memfill32"))); + +static inline int gradient_clamp(struct cg_gradient_data_t * gradient, int ipos) +{ + switch(gradient->spread) + { + case CG_SPREAD_METHOD_PAD: + if(ipos < 0) + ipos = 0; + else if(ipos >= 1024) + ipos = 1024 - 1; + break; + case CG_SPREAD_METHOD_REFLECT: + ipos = ipos % 2048; + ipos = ipos < 0 ? 2048 + ipos : ipos; + ipos = ipos >= 1024 ? 2048 - 1 - ipos : ipos; + break; + case CG_SPREAD_METHOD_REPEAT: + ipos = ipos % 1024; + ipos = ipos < 0 ? 1024 + ipos : ipos; + break; + default: + break; + } + return ipos; +} + +#define FIXPT_BITS (8) +#define FIXPT_SIZE (1 << FIXPT_BITS) + +static inline uint32_t gradient_pixel_fixed(struct cg_gradient_data_t * gradient, int fixed_pos) +{ + int ipos = (fixed_pos + (FIXPT_SIZE / 2)) >> FIXPT_BITS; + return gradient->colortable[gradient_clamp(gradient, ipos)]; +} + +static inline uint32_t gradient_pixel(struct cg_gradient_data_t * gradient, double pos) +{ + int ipos = (int)(pos * (1024 - 1) + 0.5); + return gradient->colortable[gradient_clamp(gradient, ipos)]; +} + +static inline void fetch_linear_gradient(uint32_t * buffer, struct cg_linear_gradient_values_t * v, struct cg_gradient_data_t * gradient, int y, int x, int length) +{ + double t, inc; + double rx = 0, ry = 0; + + if(v->l == 0.0) + { + t = inc = 0; + } + else + { + rx = gradient->matrix.c * (y + 0.5) + gradient->matrix.a * (x + 0.5) + gradient->matrix.tx; + ry = gradient->matrix.d * (y + 0.5) + gradient->matrix.b * (x + 0.5) + gradient->matrix.ty; + t = v->dx * rx + v->dy * ry + v->off; + inc = v->dx * gradient->matrix.a + v->dy * gradient->matrix.b; + t *= (1024 - 1); + inc *= (1024 - 1); + } + uint32_t * end = buffer + length; + if((inc > -1e-5) && (inc < 1e-5)) + { + cg_memfill32(buffer, gradient_pixel_fixed(gradient, (int)(t * FIXPT_SIZE)), length); + } + else + { + if(t + inc * length < (double)(INT_MAX >> (FIXPT_BITS + 1)) && t + inc * length > (double)(INT_MIN >> (FIXPT_BITS + 1))) + { + int t_fixed = (int)(t * FIXPT_SIZE); + int inc_fixed = (int)(inc * FIXPT_SIZE); + while(buffer < end) + { + *buffer = gradient_pixel_fixed(gradient, t_fixed); + t_fixed += inc_fixed; + ++buffer; + } + } + else + { + while(buffer < end) + { + *buffer = gradient_pixel(gradient, t / 1024); + t += inc; + ++buffer; + } + } + } +} + +static inline void fetch_radial_gradient(uint32_t * buffer, struct cg_radial_gradient_values_t * v, struct cg_gradient_data_t * gradient, int y, int x, int length) +{ + if(v->a == 0.0) + { + cg_memfill32(buffer, 0, length); + return; + } + + double rx = gradient->matrix.c * (y + 0.5) + gradient->matrix.tx + gradient->matrix.a * (x + 0.5); + double ry = gradient->matrix.d * (y + 0.5) + gradient->matrix.ty + gradient->matrix.b * (x + 0.5); + rx -= gradient->radial.fx; + ry -= gradient->radial.fy; + + double inv_a = 1.0 / (2.0 * v->a); + double delta_rx = gradient->matrix.a; + double delta_ry = gradient->matrix.b; + + double b = 2 * (v->dr * gradient->radial.fr + rx * v->dx + ry * v->dy); + double delta_b = 2 * (delta_rx * v->dx + delta_ry * v->dy); + double b_delta_b = 2 * b * delta_b; + double delta_b_delta_b = 2 * delta_b * delta_b; + + double bb = b * b; + double delta_bb = delta_b * delta_b; + + b *= inv_a; + delta_b *= inv_a; + + double rxrxryry = rx * rx + ry * ry; + double delta_rxrxryry = delta_rx * delta_rx + delta_ry * delta_ry; + double rx_plus_ry = 2 * (rx * delta_rx + ry * delta_ry); + double delta_rx_plus_ry = 2 * delta_rxrxryry; + + inv_a *= inv_a; + + double det = (bb - 4 * v->a * (v->sqrfr - rxrxryry)) * inv_a; + double delta_det = (b_delta_b + delta_bb + 4 * v->a * (rx_plus_ry + delta_rxrxryry)) * inv_a; + double delta_delta_det = (delta_b_delta_b + 4 * v->a * delta_rx_plus_ry) * inv_a; + + uint32_t * end = buffer + length; + if(v->extended) + { + while(buffer < end) + { + uint32_t result = 0; + det = fabs(det) < DBL_EPSILON ? 0.0 : det; + if(det >= 0) + { + double w = sqrt(det) - b; + if(gradient->radial.fr + v->dr * w >= 0) + result = gradient_pixel(gradient, w); + } + *buffer = result; + det += delta_det; + delta_det += delta_delta_det; + b += delta_b; + ++buffer; + } + } + else + { + while(buffer < end) + { + det = fabs(det) < DBL_EPSILON ? 0.0 : det; + uint32_t result = 0; + if(det >= 0) + result = gradient_pixel(gradient, sqrt(det) - b); + *buffer++ = result; + det += delta_det; + delta_det += delta_delta_det; + b += delta_b; + } + } +} + +static void __cg_comp_solid_source(uint32_t * dst, int len, uint32_t color, uint32_t alpha) +{ + if(alpha == 255) + { + cg_memfill32(dst, color, len); + } + else + { + uint32_t ialpha = 255 - alpha; + color = CG_BYTE_MUL(color, alpha); + for(int i = 0; i < len; i++) + dst[i] = color + CG_BYTE_MUL(dst[i], ialpha); + } +} +extern __typeof(__cg_comp_solid_source) cg_comp_solid_source __attribute__((weak, alias("__cg_comp_solid_source"))); + +static void __cg_comp_solid_source_over(uint32_t * dst, int len, uint32_t color, uint32_t alpha) +{ + if((alpha & CG_ALPHA(color)) == 255) + { + cg_memfill32(dst, color, len); + } + else + { + if(alpha != 255) + color = CG_BYTE_MUL(color, alpha); + uint32_t ialpha = 255 - CG_ALPHA(color); + for(int i = 0; i < len; i++) + dst[i] = color + CG_BYTE_MUL(dst[i], ialpha); + } +} +extern __typeof(__cg_comp_solid_source_over) cg_comp_solid_source_over __attribute__((weak, alias("__cg_comp_solid_source_over"))); + +static void __cg_comp_solid_destination_in(uint32_t * dst, int len, uint32_t color, uint32_t alpha) +{ + uint32_t a = CG_ALPHA(color); + if(alpha != 255) + a = CG_BYTE_MUL(a, alpha) + 255 - alpha; + for(int i = 0; i < len; i++) + dst[i] = CG_BYTE_MUL(dst[i], a); +} +extern __typeof(__cg_comp_solid_destination_in) cg_comp_solid_destination_in __attribute__((weak, alias("__cg_comp_solid_destination_in"))); + +static void __cg_comp_solid_destination_out(uint32_t * dst, int len, uint32_t color, uint32_t alpha) +{ + uint32_t a = CG_ALPHA(~color); + if(alpha != 255) + a = CG_BYTE_MUL(a, alpha) + 255 - alpha; + for(int i = 0; i < len; i++) + dst[i] = CG_BYTE_MUL(dst[i], a); +} +extern __typeof(__cg_comp_solid_destination_out) cg_comp_solid_destination_out __attribute__((weak, alias("__cg_comp_solid_destination_out"))); + +static void __cg_comp_source(uint32_t * dst, int len, uint32_t * src, uint32_t alpha) +{ + if(alpha == 255) + { + memcpy(dst, src, (size_t)(len) * sizeof(uint32_t)); + } + else + { + uint32_t ialpha = 255 - alpha; + for(int i = 0; i < len; i++) + dst[i] = interpolate_pixel(src[i], alpha, dst[i], ialpha); + } +} +extern __typeof(__cg_comp_source) cg_comp_source __attribute__((weak, alias("__cg_comp_source"))); + +static void __cg_comp_source_over(uint32_t * dst, int len, uint32_t * src, uint32_t alpha) +{ + uint32_t s, sia; + if(alpha == 255) + { + for(int i = 0; i < len; i++) + { + s = src[i]; + if(s >= 0xff000000) + dst[i] = s; + else if(s != 0) + { + sia = CG_ALPHA(~s); + dst[i] = s + CG_BYTE_MUL(dst[i], sia); + } + } + } + else + { + for(int i = 0; i < len; i++) + { + s = CG_BYTE_MUL(src[i], alpha); + sia = CG_ALPHA(~s); + dst[i] = s + CG_BYTE_MUL(dst[i], sia); + } + } +} +extern __typeof(__cg_comp_source_over) cg_comp_source_over __attribute__((weak, alias("__cg_comp_source_over"))); + +static void __cg_comp_destination_in(uint32_t * dst, int len, uint32_t * src, uint32_t alpha) +{ + if(alpha == 255) + { + for(int i = 0; i < len; i++) + dst[i] = CG_BYTE_MUL(dst[i], CG_ALPHA(src[i])); + } + else + { + uint32_t cia = 255 - alpha; + uint32_t a; + for(int i = 0; i < len; i++) + { + a = CG_BYTE_MUL(CG_ALPHA(src[i]), alpha) + cia; + dst[i] = CG_BYTE_MUL(dst[i], a); + } + } +} +extern __typeof(__cg_comp_destination_in) cg_comp_destination_in __attribute__((weak, alias("__cg_comp_destination_in"))); + +static void __cg_comp_destination_out(uint32_t * dst, int len, uint32_t * src, uint32_t alpha) +{ + if(alpha == 255) + { + for(int i = 0; i < len; i++) + dst[i] = CG_BYTE_MUL(dst[i], CG_ALPHA(~src[i])); + } + else + { + uint32_t cia = 255 - alpha; + uint32_t sia; + for(int i = 0; i < len; i++) + { + sia = CG_BYTE_MUL(CG_ALPHA(~src[i]), alpha) + cia; + dst[i] = CG_BYTE_MUL(dst[i], sia); + } + } +} +extern __typeof(__cg_comp_destination_out) cg_comp_destination_out __attribute__((weak, alias("__cg_comp_destination_out"))); + +typedef void (*cg_comp_solid_function_t)(uint32_t * dst, int len, uint32_t color, uint32_t alpha); +static const cg_comp_solid_function_t cg_comp_solid_map[] = { + cg_comp_solid_source, + cg_comp_solid_source_over, + cg_comp_solid_destination_in, + cg_comp_solid_destination_out, +}; + +typedef void (*cg_comp_function_t)(uint32_t * dst, int len, uint32_t * src, uint32_t alpha); +static const cg_comp_function_t cg_comp_map[] = { + cg_comp_source, + cg_comp_source_over, + cg_comp_destination_in, + cg_comp_destination_out, +}; + +static inline void blend_solid(struct cg_surface_t * surface, enum cg_operator_t op, struct cg_rle_t * rle, uint32_t solid) +{ + cg_comp_solid_function_t func = cg_comp_solid_map[op]; + int count = rle->spans.size; + struct cg_span_t * spans = rle->spans.data; + while(count--) + { + uint32_t * target = (uint32_t *)(surface->pixels + spans->y * surface->stride) + spans->x; + func(target, spans->len, solid, spans->coverage); + ++spans; + } +} + +static inline void blend_linear_gradient(struct cg_surface_t * surface, enum cg_operator_t op, struct cg_rle_t * rle, struct cg_gradient_data_t * gradient) +{ + cg_comp_function_t func = cg_comp_map[op]; + unsigned int buffer[1024]; + + struct cg_linear_gradient_values_t v; + v.dx = gradient->linear.x2 - gradient->linear.x1; + v.dy = gradient->linear.y2 - gradient->linear.y1; + v.l = v.dx * v.dx + v.dy * v.dy; + v.off = 0.0; + if(v.l != 0.0) + { + v.dx /= v.l; + v.dy /= v.l; + v.off = -v.dx * gradient->linear.x1 - v.dy * gradient->linear.y1; + } + + int count = rle->spans.size; + struct cg_span_t * spans = rle->spans.data; + while(count--) + { + int length = spans->len; + int x = spans->x; + while(length) + { + int l = CG_MIN(length, 1024); + fetch_linear_gradient(buffer, &v, gradient, spans->y, x, l); + uint32_t * target = (uint32_t *)(surface->pixels + spans->y * surface->stride) + x; + func(target, l, buffer, spans->coverage); + x += l; + length -= l; + } + ++spans; + } +} + +static inline void blend_radial_gradient(struct cg_surface_t * surface, enum cg_operator_t op, struct cg_rle_t * rle, struct cg_gradient_data_t * gradient) +{ + cg_comp_function_t func = cg_comp_map[op]; + unsigned int buffer[1024]; + + struct cg_radial_gradient_values_t v; + v.dx = gradient->radial.cx - gradient->radial.fx; + v.dy = gradient->radial.cy - gradient->radial.fy; + v.dr = gradient->radial.cr - gradient->radial.fr; + v.sqrfr = gradient->radial.fr * gradient->radial.fr; + v.a = v.dr * v.dr - v.dx * v.dx - v.dy * v.dy; + v.inv2a = 1.0 / (2.0 * v.a); + v.extended = gradient->radial.fr != 0.0 || v.a <= 0.0; + + int count = rle->spans.size; + struct cg_span_t * spans = rle->spans.data; + while(count--) + { + int length = spans->len; + int x = spans->x; + while(length) + { + int l = CG_MIN(length, 1024); + fetch_radial_gradient(buffer, &v, gradient, spans->y, x, l); + uint32_t * target = (uint32_t *)(surface->pixels + spans->y * surface->stride) + x; + func(target, l, buffer, spans->coverage); + x += l; + length -= l; + } + ++spans; + } +} + +#define FIXED_SCALE (1 << 16) +static inline void blend_untransformed_argb(struct cg_surface_t * surface, enum cg_operator_t op, struct cg_rle_t * rle, struct cg_texture_data_t * texture) +{ + cg_comp_function_t func = cg_comp_map[op]; + + int image_width = texture->width; + int image_height = texture->height; + int xoff = (int)(texture->matrix.tx); + int yoff = (int)(texture->matrix.ty); + int count = rle->spans.size; + struct cg_span_t * spans = rle->spans.data; + while(count--) + { + int x = spans->x; + int length = spans->len; + int sx = xoff + x; + int sy = yoff + spans->y; + if(sy >= 0 && sy < image_height && sx < image_width) + { + if(sx < 0) + { + x -= sx; + length += sx; + sx = 0; + } + if(sx + length > image_width) + length = image_width - sx; + if(length > 0) + { + int coverage = (spans->coverage * texture->alpha) >> 8; + uint32_t * src = (uint32_t *)(texture->pixels + sy * texture->stride) + sx; + uint32_t * dst = (uint32_t *)(surface->pixels + spans->y * surface->stride) + x; + func(dst, length, src, coverage); + } + } + ++spans; + } +} + +static inline void blend_transformed_argb(struct cg_surface_t * surface, enum cg_operator_t op, struct cg_rle_t * rle, struct cg_texture_data_t * texture) +{ + cg_comp_function_t func = cg_comp_map[op]; + uint32_t buffer[1024]; + + int image_width = texture->width; + int image_height = texture->height; + int fdx = (int)(texture->matrix.a * FIXED_SCALE); + int fdy = (int)(texture->matrix.b * FIXED_SCALE); + int count = rle->spans.size; + struct cg_span_t * spans = rle->spans.data; + while(count--) + { + uint32_t * target = (uint32_t *)(surface->pixels + spans->y * surface->stride) + spans->x; + double cx = spans->x + 0.5; + double cy = spans->y + 0.5; + int x = (int)((texture->matrix.c * cy + texture->matrix.a * cx + texture->matrix.tx) * FIXED_SCALE); + int y = (int)((texture->matrix.d * cy + texture->matrix.b * cx + texture->matrix.ty) * FIXED_SCALE); + int length = spans->len; + int coverage = (spans->coverage * texture->alpha) >> 8; + while(length) + { + int l = CG_MIN(length, 1024); + uint32_t * end = buffer + l; + uint32_t * b = buffer; + int start = 0; + int clen = 0; + while(b < end) + { + int px = x >> 16; + int py = y >> 16; + if(((unsigned int)px < (unsigned int)image_width) && ((unsigned int)py < (unsigned int)image_height)) + { + *b = ((uint32_t *)(texture->pixels + py * texture->stride))[px]; + clen++; + } + x += fdx; + y += fdy; + ++b; + if(clen == 0) + start++; + } + func(target + start, clen, buffer + start, coverage); + target += l; + length -= l; + } + ++spans; + } +} + +static inline void blend_untransformed_tiled_argb(struct cg_surface_t * surface, enum cg_operator_t op, struct cg_rle_t * rle, struct cg_texture_data_t * texture) +{ + cg_comp_function_t func = cg_comp_map[op]; + + int image_width = texture->width; + int image_height = texture->height; + int xoff = (int)(texture->matrix.tx) % image_width; + int yoff = (int)(texture->matrix.ty) % image_height; + if(xoff < 0) + xoff += image_width; + if(yoff < 0) + yoff += image_height; + int count = rle->spans.size; + struct cg_span_t * spans = rle->spans.data; + while(count--) + { + int x = spans->x; + int length = spans->len; + int sx = (xoff + spans->x) % image_width; + int sy = (spans->y + yoff) % image_height; + if(sx < 0) + sx += image_width; + if(sy < 0) + sy += image_height; + int coverage = (spans->coverage * texture->alpha) >> 8; + while(length) + { + int l = CG_MIN(image_width - sx, length); + if(1024 < l) + l = 1024; + uint32_t * src = (uint32_t *)(texture->pixels + sy * texture->stride) + sx; + uint32_t * dst = (uint32_t *)(surface->pixels + spans->y * surface->stride) + x; + func(dst, l, src, coverage); + x += l; + length -= l; + sx = 0; + } + ++spans; + } +} + +static inline void blend_transformed_tiled_argb(struct cg_surface_t * surface, enum cg_operator_t op, struct cg_rle_t * rle, struct cg_texture_data_t * texture) +{ + cg_comp_function_t func = cg_comp_map[op]; + uint32_t buffer[1024]; + + int image_width = texture->width; + int image_height = texture->height; + int scanline_offset = texture->stride / 4; + int fdx = (int)(texture->matrix.a * FIXED_SCALE); + int fdy = (int)(texture->matrix.b * FIXED_SCALE); + int count = rle->spans.size; + struct cg_span_t * spans = rle->spans.data; + while(count--) + { + uint32_t * target = (uint32_t *)(surface->pixels + spans->y * surface->stride) + spans->x; + uint32_t * image_bits = (uint32_t *)texture->pixels; + double cx = spans->x + 0.5; + double cy = spans->y + 0.5; + int x = (int)((texture->matrix.c * cy + texture->matrix.a * cx + texture->matrix.tx) * FIXED_SCALE); + int y = (int)((texture->matrix.d * cy + texture->matrix.b * cx + texture->matrix.ty) * FIXED_SCALE); + int coverage = (spans->coverage * texture->alpha) >> 8; + int length = spans->len; + while(length) + { + int l = CG_MIN(length, 1024); + uint32_t * end = buffer + l; + uint32_t * b = buffer; + int px16 = x % (image_width << 16); + int py16 = y % (image_height << 16); + int px_delta = fdx % (image_width << 16); + int py_delta = fdy % (image_height << 16); + while(b < end) + { + if(px16 < 0) + px16 += image_width << 16; + if(py16 < 0) + py16 += image_height << 16; + int px = px16 >> 16; + int py = py16 >> 16; + int y_offset = py * scanline_offset; + + *b = image_bits[y_offset + px]; + x += fdx; + y += fdy; + px16 += px_delta; + if(px16 >= image_width << 16) + px16 -= image_width << 16; + py16 += py_delta; + if(py16 >= image_height << 16) + py16 -= image_height << 16; + ++b; + } + func(target, l, buffer, coverage); + target += l; + length -= l; + } + ++spans; + } +} + +static inline void cg_blend_color(struct cg_ctx_t * ctx, struct cg_rle_t * rle, struct cg_color_t * color) +{ + if(color) + { + struct cg_state_t * state = ctx->state; + uint32_t solid = premultiply_color(color, state->opacity); + if((CG_ALPHA(solid) == 255) && (state->op == CG_OPERATOR_SRC_OVER)) + blend_solid(ctx->surface, CG_OPERATOR_SRC, rle, solid); + else + blend_solid(ctx->surface, state->op, rle, solid); + } +} + +static inline void cg_blend_gradient(struct cg_ctx_t * ctx, struct cg_rle_t * rle, struct cg_gradient_t * gradient) +{ + if(gradient && (gradient->stops.size > 0)) + { + struct cg_state_t * state = ctx->state; + struct cg_gradient_data_t data; + int i, pos = 0, nstop = gradient->stops.size; + struct cg_gradient_stop_t *curr, *next, *start, *last; + uint32_t curr_color, next_color, last_color; + uint32_t dist, idist; + double delta, t, incr, fpos; + double opacity = state->opacity * gradient->opacity; + + start = gradient->stops.data; + curr = start; + curr_color = combine_opacity(&curr->color, opacity); + + data.colortable[pos] = premultiply_pixel(curr_color); + ++pos; + incr = 1.0 / 1024; + fpos = 1.5 * incr; + + while(fpos <= curr->offset) + { + data.colortable[pos] = data.colortable[pos - 1]; + ++pos; + fpos += incr; + } + for(i = 0; i < nstop - 1; i++) + { + curr = (start + i); + next = (start + i + 1); + delta = 1.0 / (next->offset - curr->offset); + next_color = combine_opacity(&next->color, opacity); + while(fpos < next->offset && pos < 1024) + { + t = (fpos - curr->offset) * delta; + dist = (uint32_t)(255 * t); + idist = 255 - dist; + data.colortable[pos] = premultiply_pixel(interpolate_pixel(curr_color, idist, next_color, dist)); + ++pos; + fpos += incr; + } + curr_color = next_color; + } + + last = start + nstop - 1; + last_color = premultiply_color(&last->color, opacity); + for(; pos < 1024; ++pos) + data.colortable[pos] = last_color; + + data.spread = gradient->spread; + data.matrix = gradient->matrix; + cg_matrix_multiply(&data.matrix, &data.matrix, &state->matrix); + cg_matrix_invert(&data.matrix); + + if(gradient->type == CG_GRADIENT_TYPE_LINEAR) + { + data.linear.x1 = gradient->values[0]; + data.linear.y1 = gradient->values[1]; + data.linear.x2 = gradient->values[2]; + data.linear.y2 = gradient->values[3]; + blend_linear_gradient(ctx->surface, state->op, rle, &data); + } + else + { + data.radial.cx = gradient->values[0]; + data.radial.cy = gradient->values[1]; + data.radial.cr = gradient->values[2]; + data.radial.fx = gradient->values[3]; + data.radial.fy = gradient->values[4]; + data.radial.fr = gradient->values[5]; + blend_radial_gradient(ctx->surface, state->op, rle, &data); + } + } +} + +static inline void cg_blend_texture(struct cg_ctx_t * ctx, struct cg_rle_t * rle, struct cg_texture_t * texture) +{ + if(texture) + { + struct cg_state_t * state = ctx->state; + struct cg_texture_data_t data; + data.width = texture->surface->width; + data.height = texture->surface->height; + data.stride = texture->surface->stride; + data.alpha = (int)(state->opacity * texture->opacity * 256.0); + data.pixels = texture->surface->pixels; + data.matrix = texture->matrix; + cg_matrix_multiply(&data.matrix, &data.matrix, &state->matrix); + cg_matrix_invert(&data.matrix); + struct cg_matrix_t * m = &data.matrix; + if((m->a == 1.0) && (m->b == 0.0) && (m->c == 0.0) && (m->d == 1.0)) + { + if(texture->type == CG_TEXTURE_TYPE_PLAIN) + blend_untransformed_argb(ctx->surface, state->op, rle, &data); + else + blend_untransformed_tiled_argb(ctx->surface, state->op, rle, &data); + } + else + { + if(texture->type == CG_TEXTURE_TYPE_PLAIN) + blend_transformed_argb(ctx->surface, state->op, rle, &data); + else + blend_transformed_tiled_argb(ctx->surface, state->op, rle, &data); + } + } +} + +static void cg_blend(struct cg_ctx_t * ctx, struct cg_rle_t * rle) +{ + if(rle && (rle->spans.size > 0)) + { + struct cg_paint_t * source = &ctx->state->paint; + switch(source->type) + { + case CG_PAINT_TYPE_COLOR: + cg_blend_color(ctx, rle, &source->color); + break; + case CG_PAINT_TYPE_GRADIENT: + cg_blend_gradient(ctx, rle, &source->gradient); + break; + case CG_PAINT_TYPE_TEXTURE: + cg_blend_texture(ctx, rle, &source->texture); + default: + break; + } + } +} + +static struct cg_state_t * cg_state_create(void) +{ + struct cg_state_t * state = malloc(sizeof(struct cg_state_t)); + state->clippath = NULL; + cg_paint_init(&state->paint); + cg_matrix_init_identity(&state->matrix); + state->winding = CG_FILL_RULE_NON_ZERO; + state->stroke.width = 1.0; + state->stroke.miterlimit = 10.0; + state->stroke.cap = CG_LINE_CAP_BUTT; + state->stroke.join = CG_LINE_JOIN_MITER; + state->stroke.dash = NULL; + state->op = CG_OPERATOR_SRC_OVER; + state->opacity = 1.0; + state->next = NULL; + return state; +} + +static struct cg_state_t * cg_state_clone(struct cg_state_t * state) +{ + struct cg_state_t * newstate = cg_state_create(); + newstate->clippath = cg_rle_clone(state->clippath); + cg_paint_copy(&newstate->paint, &state->paint); + newstate->matrix = state->matrix; + newstate->winding = state->winding; + newstate->stroke.width = state->stroke.width; + newstate->stroke.miterlimit = state->stroke.miterlimit; + newstate->stroke.cap = state->stroke.cap; + newstate->stroke.join = state->stroke.join; + newstate->stroke.dash = cg_dash_clone(state->stroke.dash); + newstate->op = state->op; + newstate->opacity = state->opacity; + newstate->next = NULL; + return newstate; +} + +static void cg_state_destroy(struct cg_state_t * state) +{ + cg_rle_destroy(state->clippath); + cg_paint_destroy(&state->paint); + cg_dash_destroy(state->stroke.dash); + free(state); +} + +struct cg_ctx_t * cg_create(struct cg_surface_t * surface) +{ + struct cg_ctx_t * ctx = malloc(sizeof(struct cg_ctx_t)); + ctx->surface = cg_surface_reference(surface); + ctx->state = cg_state_create(); + ctx->path = cg_path_create(); + ctx->rle = cg_rle_create(); + ctx->clippath = NULL; + ctx->clip.x = 0.0; + ctx->clip.y = 0.0; + ctx->clip.w = surface->width; + ctx->clip.h = surface->height; + ctx->outline_data = NULL; + ctx->outline_size = 0; + return ctx; +} + +void cg_destroy(struct cg_ctx_t * ctx) +{ + if(ctx) + { + while(ctx->state) + { + struct cg_state_t * state = ctx->state; + ctx->state = state->next; + cg_state_destroy(state); + } + cg_surface_destroy(ctx->surface); + cg_path_destroy(ctx->path); + cg_rle_destroy(ctx->rle); + cg_rle_destroy(ctx->clippath); + if(ctx->outline_data) + free(ctx->outline_data); + free(ctx); + } +} + +void cg_save(struct cg_ctx_t * ctx) +{ + struct cg_state_t * state = cg_state_clone(ctx->state); + state->next = ctx->state; + ctx->state = state; +} + +void cg_restore(struct cg_ctx_t * ctx) +{ + struct cg_state_t * state = ctx->state; + ctx->state = state->next; + cg_state_destroy(state); +} + +struct cg_color_t * cg_set_source_rgb(struct cg_ctx_t * ctx, double r, double g, double b) +{ + return cg_set_source_rgba(ctx, r, g, b, 1.0); +} + +struct cg_color_t * cg_set_source_rgba(struct cg_ctx_t * ctx, double r, double g, double b, double a) +{ + struct cg_paint_t * paint = &ctx->state->paint; + paint->type = CG_PAINT_TYPE_COLOR; + cg_color_init_rgba(&paint->color, r, g, b, a); + return &paint->color; +} + +struct cg_color_t * cg_set_source_color(struct cg_ctx_t * ctx, struct cg_color_t * color) +{ + return cg_set_source_rgba(ctx, color->r, color->g, color->b, color->a); +} + +struct cg_gradient_t * cg_set_source_linear_gradient(struct cg_ctx_t * ctx, double x1, double y1, double x2, double y2) +{ + struct cg_paint_t * paint = &ctx->state->paint; + paint->type = CG_PAINT_TYPE_GRADIENT; + cg_gradient_init_linear(&paint->gradient, x1, y1, x2, y2); + return &paint->gradient; +} + +struct cg_gradient_t * cg_set_source_radial_gradient(struct cg_ctx_t * ctx, double cx, double cy, double cr, double fx, double fy, double fr) +{ + struct cg_paint_t * paint = &ctx->state->paint; + paint->type = CG_PAINT_TYPE_GRADIENT; + cg_gradient_init_radial(&paint->gradient, cx, cy, cr, fx, fy, fr); + return &paint->gradient; +} + +static inline struct cg_texture_t * cg_set_texture(struct cg_ctx_t *ctx, struct cg_surface_t * surface, enum cg_texture_type_t type) +{ + struct cg_paint_t * paint = &ctx->state->paint; + paint->type = CG_PAINT_TYPE_TEXTURE; + cg_texture_init(&paint->texture, surface, type); + return &paint->texture; +} + +struct cg_texture_t * cg_set_source_surface(struct cg_ctx_t * ctx, struct cg_surface_t * surface, double x, double y) +{ + struct cg_texture_t * texture = cg_set_texture(ctx, surface, CG_TEXTURE_TYPE_PLAIN); + cg_matrix_init_translate(&texture->matrix, x, y); + return texture; +} + +void cg_set_operator(struct cg_ctx_t * ctx, enum cg_operator_t op) +{ + ctx->state->op = op; +} + +void cg_set_opacity(struct cg_ctx_t * ctx, double opacity) +{ + ctx->state->opacity = CG_CLAMP(opacity, 0.0, 1.0); +} + +void cg_set_fill_rule(struct cg_ctx_t * ctx, enum cg_fill_rule_t winding) +{ + ctx->state->winding = winding; +} + +void cg_set_line_width(struct cg_ctx_t * ctx, double width) +{ + ctx->state->stroke.width = width; +} + +void cg_set_line_cap(struct cg_ctx_t * ctx, enum cg_line_cap_t cap) +{ + ctx->state->stroke.cap = cap; +} + +void cg_set_line_join(struct cg_ctx_t * ctx, enum cg_line_join_t join) +{ + ctx->state->stroke.join = join; +} + +void cg_set_miter_limit(struct cg_ctx_t * ctx, double limit) +{ + ctx->state->stroke.miterlimit = limit; +} + +void cg_set_dash(struct cg_ctx_t * ctx, double * dashes, int ndash, double offset) +{ + cg_dash_destroy(ctx->state->stroke.dash); + ctx->state->stroke.dash = cg_dash_create(dashes, ndash, offset); +} + +void cg_translate(struct cg_ctx_t * ctx, double tx, double ty) +{ + cg_matrix_translate(&ctx->state->matrix, tx, ty); +} + +void cg_scale(struct cg_ctx_t * ctx, double sx, double sy) +{ + cg_matrix_scale(&ctx->state->matrix, sx, sy); +} + +void cg_rotate(struct cg_ctx_t * ctx, double r) +{ + cg_matrix_rotate(&ctx->state->matrix, r); +} + +void cg_transform(struct cg_ctx_t * ctx, struct cg_matrix_t * m) +{ + cg_matrix_multiply(&ctx->state->matrix, m, &ctx->state->matrix); +} + +void cg_set_matrix(struct cg_ctx_t * ctx, struct cg_matrix_t * m) +{ + memcpy(&ctx->state->matrix, m, sizeof(struct cg_matrix_t)); +} + +void cg_identity_matrix(struct cg_ctx_t * ctx) +{ + cg_matrix_init_identity(&ctx->state->matrix); +} + +void cg_move_to(struct cg_ctx_t * ctx, double x, double y) +{ + cg_path_move_to(ctx->path, x, y); +} + +void cg_line_to(struct cg_ctx_t * ctx, double x, double y) +{ + cg_path_line_to(ctx->path, x, y); +} + +void cg_curve_to(struct cg_ctx_t * ctx, double x1, double y1, double x2, double y2, double x3, double y3) +{ + cg_path_curve_to(ctx->path, x1, y1, x2, y2, x3, y3); +} + +void cg_quad_to(struct cg_ctx_t * ctx, double x1, double y1, double x2, double y2) +{ + cg_path_quad_to(ctx->path, x1, y1, x2, y2); +} + +void cg_rel_move_to(struct cg_ctx_t * ctx, double dx, double dy) +{ + cg_path_rel_move_to(ctx->path, dx, dy); +} + +void cg_rel_line_to(struct cg_ctx_t * ctx, double dx, double dy) +{ + cg_path_rel_line_to(ctx->path, dx, dy); +} + +void cg_rel_curve_to(struct cg_ctx_t * ctx, double dx1, double dy1, double dx2, double dy2, double dx3, double dy3) +{ + cg_path_rel_curve_to(ctx->path, dx1, dy1, dx2, dy2, dx3, dy3); +} + +void cg_rel_quad_to(struct cg_ctx_t * ctx, double dx1, double dy1, double dx2, double dy2) +{ + cg_path_rel_quad_to(ctx->path, dx1, dy1, dx2, dy2); +} + +void cg_rectangle(struct cg_ctx_t * ctx, double x, double y, double w, double h) +{ + cg_path_add_rectangle(ctx->path, x, y, w, h); +} + +void cg_round_rectangle(struct cg_ctx_t * ctx, double x, double y, double w, double h, double rx, double ry) +{ + cg_path_add_round_rectangle(ctx->path, x, y, w, h, rx, ry); +} + +void cg_ellipse(struct cg_ctx_t * ctx, double cx, double cy, double rx, double ry) +{ + cg_path_add_ellipse(ctx->path, cx, cy, rx, ry); +} + +void cg_circle(struct cg_ctx_t * ctx, double cx, double cy, double r) +{ + cg_path_add_ellipse(ctx->path, cx, cy, r, r); +} + +void cg_arc(struct cg_ctx_t * ctx, double cx, double cy, double r, double a0, double a1) +{ + cg_path_add_arc(ctx->path, cx, cy, r, a0, a1, 0); +} + +void cg_arc_negative(struct cg_ctx_t * ctx, double cx, double cy, double r, double a0, double a1) +{ + cg_path_add_arc(ctx->path, cx, cy, r, a0, a1, 1); +} + +void cg_new_path(struct cg_ctx_t * ctx) +{ + cg_path_clear(ctx->path); +} + +void cg_close_path(struct cg_ctx_t * ctx) +{ + cg_path_close(ctx->path); +} + +void cg_reset_clip(struct cg_ctx_t * ctx) +{ + cg_rle_destroy(ctx->state->clippath); + ctx->state->clippath = NULL; +} + +void cg_clip(struct cg_ctx_t * ctx) +{ + cg_clip_preserve(ctx); + cg_new_path(ctx); +} + +void cg_clip_preserve(struct cg_ctx_t * ctx) +{ + struct cg_state_t * state = ctx->state; + if(state->clippath) + { + cg_rle_clear(ctx->rle); + cg_rle_rasterize(ctx, ctx->rle, ctx->path, &state->matrix, &ctx->clip, NULL, state->winding); + cg_rle_clip_path(state->clippath, ctx->rle); + } + else + { + state->clippath = cg_rle_create(); + cg_rle_rasterize(ctx, state->clippath, ctx->path, &state->matrix, &ctx->clip, NULL, state->winding); + } +} + +void cg_fill(struct cg_ctx_t * ctx) +{ + cg_fill_preserve(ctx); + cg_new_path(ctx); +} + +void cg_fill_preserve(struct cg_ctx_t * ctx) +{ + struct cg_state_t * state = ctx->state; + cg_rle_clear(ctx->rle); + cg_rle_rasterize(ctx, ctx->rle, ctx->path, &state->matrix, &ctx->clip, NULL, state->winding); + cg_rle_clip_path(ctx->rle, state->clippath); + cg_blend(ctx, ctx->rle); +} + +void cg_stroke(struct cg_ctx_t * ctx) +{ + cg_stroke_preserve(ctx); + cg_new_path(ctx); +} + +void cg_stroke_preserve(struct cg_ctx_t * ctx) +{ + struct cg_state_t * state = ctx->state; + cg_rle_clear(ctx->rle); + cg_rle_rasterize(ctx, ctx->rle, ctx->path, &state->matrix, &ctx->clip, &state->stroke, CG_FILL_RULE_NON_ZERO); + cg_rle_clip_path(ctx->rle, state->clippath); + cg_blend(ctx, ctx->rle); +} + +void cg_paint(struct cg_ctx_t * ctx) +{ + struct cg_state_t * state = ctx->state; + if((state->clippath == NULL) && (ctx->clippath == NULL)) + { + struct cg_path_t * path = cg_path_create(); + cg_path_add_rectangle(path, ctx->clip.x, ctx->clip.y, ctx->clip.w, ctx->clip.h); + struct cg_matrix_t m; + cg_matrix_init_identity(&m); + ctx->clippath = cg_rle_create(); + cg_rle_rasterize(ctx, ctx->clippath, path, &m, &ctx->clip, NULL, CG_FILL_RULE_NON_ZERO); + cg_path_destroy(path); + } + struct cg_rle_t * rle = state->clippath ? state->clippath : ctx->clippath; + cg_blend(ctx, rle); +} diff --git a/libmui/mui/cg.h b/libmui/mui/cg.h new file mode 100644 index 0000000..ca7b3d0 --- /dev/null +++ b/libmui/mui/cg.h @@ -0,0 +1,314 @@ +#ifndef __CG_H__ +#define __CG_H__ + +#ifdef __cplusplus +extern "C" { +#endif + +#include "xft.h" + +struct cg_point_t { + double x; + double y; +}; + +struct cg_rect_t { + double x; + double y; + double w; + double h; +}; + +struct cg_matrix_t { + double a; double b; + double c; double d; + double tx; double ty; +}; + +struct cg_color_t { + double r; + double g; + double b; + double a; +}; + +struct cg_gradient_stop_t { + double offset; + struct cg_color_t color; +}; + +enum cg_path_element_t { + CG_PATH_ELEMENT_MOVE_TO = 0, + CG_PATH_ELEMENT_LINE_TO = 1, + CG_PATH_ELEMENT_CURVE_TO = 2, + CG_PATH_ELEMENT_CLOSE = 3, +}; + +enum cg_spread_method_t { + CG_SPREAD_METHOD_PAD = 0, + CG_SPREAD_METHOD_REFLECT = 1, + CG_SPREAD_METHOD_REPEAT = 2, +}; + +enum cg_gradient_type_t { + CG_GRADIENT_TYPE_LINEAR = 0, + CG_GRADIENT_TYPE_RADIAL = 1, +}; + +enum cg_texture_type_t { + CG_TEXTURE_TYPE_PLAIN = 0, + CG_TEXTURE_TYPE_TILED = 1, +}; + +enum cg_line_cap_t { + CG_LINE_CAP_BUTT = 0, + CG_LINE_CAP_ROUND = 1, + CG_LINE_CAP_SQUARE = 2, +}; + +enum cg_line_join_t { + CG_LINE_JOIN_MITER = 0, + CG_LINE_JOIN_ROUND = 1, + CG_LINE_JOIN_BEVEL = 2, +}; + +enum cg_fill_rule_t { + CG_FILL_RULE_NON_ZERO = 0, + CG_FILL_RULE_EVEN_ODD = 1, +}; + +enum cg_paint_type_t { + CG_PAINT_TYPE_COLOR = 0, + CG_PAINT_TYPE_GRADIENT = 1, + CG_PAINT_TYPE_TEXTURE = 2, +}; + +enum cg_operator_t { + CG_OPERATOR_SRC = 0, /* r = s * ca + d * cia */ + CG_OPERATOR_SRC_OVER = 1, /* r = (s + d * sia) * ca + d * cia */ + CG_OPERATOR_DST_IN = 2, /* r = d * sa * ca + d * cia */ + CG_OPERATOR_DST_OUT = 3, /* r = d * sia * ca + d * cia */ +}; + +struct cg_surface_t { + int ref; + int width; + int height; + int stride; + int owndata; + void * pixels; +}; + +struct cg_path_t { + int contours; + struct cg_point_t start; + struct { + enum cg_path_element_t * data; + int size; + int capacity; + } elements; + struct { + struct cg_point_t * data; + int size; + int capacity; + } points; +}; + +struct cg_gradient_t { + enum cg_gradient_type_t type; + enum cg_spread_method_t spread; + struct cg_matrix_t matrix; + double values[6]; + double opacity; + struct { + struct cg_gradient_stop_t * data; + int size; + int capacity; + } stops; +}; + +struct cg_texture_t { + enum cg_texture_type_t type; + struct cg_surface_t * surface; + struct cg_matrix_t matrix; + double opacity; +}; + +struct cg_paint_t { + enum cg_paint_type_t type; + struct cg_color_t color; + struct cg_gradient_t gradient; + struct cg_texture_t texture; +}; + +struct cg_span_t { + int x; + int len; + int y; + unsigned char coverage; +}; + +struct cg_rle_t { + struct { + struct cg_span_t * data; + int size; + int capacity; + } spans; + int x; + int y; + int w; + int h; +}; + +struct cg_dash_t { + double offset; + double * data; + int size; +}; + +struct cg_stroke_data_t { + double width; + double miterlimit; + enum cg_line_cap_t cap; + enum cg_line_join_t join; + struct cg_dash_t * dash; +}; + +struct cg_state_t { + struct cg_rle_t * clippath; + struct cg_paint_t paint; + struct cg_matrix_t matrix; + enum cg_fill_rule_t winding; + struct cg_stroke_data_t stroke; + enum cg_operator_t op; + double opacity; + struct cg_state_t * next; +}; + +struct cg_ctx_t { + struct cg_surface_t * surface; + struct cg_state_t * state; + struct cg_path_t * path; + struct cg_rle_t * rle; + struct cg_rle_t * clippath; + struct cg_rect_t clip; + void * outline_data; + size_t outline_size; +}; + +#ifndef CG_MIN +#define CG_MIN(a, b) ({typeof(a) _amin = (a); typeof(b) _bmin = (b); (void)(&_amin == &_bmin); _amin < _bmin ? _amin : _bmin;}) +#endif +#ifndef CG_MAX +#define CG_MAX(a, b) ({typeof(a) _amax = (a); typeof(b) _bmax = (b); (void)(&_amax == &_bmax); _amax > _bmax ? _amax : _bmax;}) +#endif +#ifndef CG_CLAMP +#define CG_CLAMP(v, a, b) CG_MIN(CG_MAX(a, v), b) +#endif +#ifndef CG_ALPHA +#define CG_ALPHA(c) ((c) >> 24) +#endif +#ifndef CG_DIV255 +#define CG_DIV255(x) (((x) + ((x) >> 8) + 0x80) >> 8) +#endif +#ifndef CG_BYTE_MUL +#define CG_BYTE_MUL(x, a) ((((((x) >> 8) & 0x00ff00ff) * (a)) & 0xff00ff00) + (((((x) & 0x00ff00ff) * (a)) >> 8) & 0x00ff00ff)) +#endif + +void cg_memfill32(uint32_t * dst, uint32_t val, int len); +void cg_comp_solid_source(uint32_t * dst, int len, uint32_t color, uint32_t alpha); +void cg_comp_solid_source_over(uint32_t * dst, int len, uint32_t color, uint32_t alpha); +void cg_comp_solid_destination_in(uint32_t * dst, int len, uint32_t color, uint32_t alpha); +void cg_comp_solid_destination_out(uint32_t * dst, int len, uint32_t color, uint32_t alpha); +void cg_comp_source(uint32_t * dst, int len, uint32_t * src, uint32_t alpha); +void cg_comp_source_over(uint32_t * dst, int len, uint32_t * src, uint32_t alpha); +void cg_comp_destination_in(uint32_t * dst, int len, uint32_t * src, uint32_t alpha); +void cg_comp_destination_out(uint32_t * dst, int len, uint32_t * src, uint32_t alpha); + +void cg_matrix_init(struct cg_matrix_t * m, double a, double b, double c, double d, double tx, double ty); +void cg_matrix_init_identity(struct cg_matrix_t * m); +void cg_matrix_init_translate(struct cg_matrix_t * m, double tx, double ty); +void cg_matrix_init_scale(struct cg_matrix_t * m, double sx, double sy); +void cg_matrix_init_rotate(struct cg_matrix_t * m, double r); +void cg_matrix_translate(struct cg_matrix_t * m, double tx, double ty); +void cg_matrix_scale(struct cg_matrix_t * m, double sx, double sy); +void cg_matrix_rotate(struct cg_matrix_t * m, double r); +void cg_matrix_multiply(struct cg_matrix_t * m, struct cg_matrix_t * m1, struct cg_matrix_t * m2); +void cg_matrix_invert(struct cg_matrix_t * m); +void cg_matrix_map_point(struct cg_matrix_t * m, struct cg_point_t * p1, struct cg_point_t * p2); + +struct cg_surface_t * cg_surface_create(int width, int height); +struct cg_surface_t * cg_surface_create_for_data(int width, int height, void * pixels); +void cg_surface_destroy(struct cg_surface_t * surface); +struct cg_surface_t * cg_surface_reference(struct cg_surface_t * surface); + +void cg_gradient_set_values_linear(struct cg_gradient_t * gradient, double x1, double y1, double x2, double y2); +void cg_gradient_set_values_radial(struct cg_gradient_t * gradient, double cx, double cy, double cr, double fx, double fy, double fr); +void cg_gradient_set_spread(struct cg_gradient_t * gradient, enum cg_spread_method_t spread); +void cg_gradient_set_matrix(struct cg_gradient_t * gradient, struct cg_matrix_t * m); +void cg_gradient_set_opacity(struct cg_gradient_t * gradient, double opacity); +void cg_gradient_add_stop_rgb(struct cg_gradient_t * gradient, double offset, double r, double g, double b); +void cg_gradient_add_stop_rgba(struct cg_gradient_t * gradient, double offset, double r, double g, double b, double a); +void cg_gradient_add_stop_color(struct cg_gradient_t * gradient, double offset, struct cg_color_t * color); +void cg_gradient_add_stop(struct cg_gradient_t * gradient, struct cg_gradient_stop_t * stop); +void cg_gradient_clear_stops(struct cg_gradient_t * gradient); + +void cg_texture_set_type(struct cg_texture_t * texture, enum cg_texture_type_t type); +void cg_texture_set_matrix(struct cg_texture_t * texture, struct cg_matrix_t * m); +void cg_texture_set_surface(struct cg_texture_t * texture, struct cg_surface_t * surface); +void cg_texture_set_opacity(struct cg_texture_t * texture, double opacity); + +struct cg_ctx_t * cg_create(struct cg_surface_t * surface); +void cg_destroy(struct cg_ctx_t * ctx); +void cg_save(struct cg_ctx_t * ctx); +void cg_restore(struct cg_ctx_t * ctx); +struct cg_color_t * cg_set_source_rgb(struct cg_ctx_t * ctx, double r, double g, double b); +struct cg_color_t * cg_set_source_rgba(struct cg_ctx_t * ctx, double r, double g, double b, double a); +struct cg_color_t * cg_set_source_color(struct cg_ctx_t * ctx, struct cg_color_t * color); +struct cg_gradient_t * cg_set_source_linear_gradient(struct cg_ctx_t * ctx, double x1, double y1, double x2, double y2); +struct cg_gradient_t * cg_set_source_radial_gradient(struct cg_ctx_t * ctx, double cx, double cy, double cr, double fx, double fy, double fr); +struct cg_texture_t * cg_set_source_surface(struct cg_ctx_t * ctx, struct cg_surface_t * surface, double x, double y); +void cg_set_operator(struct cg_ctx_t * ctx, enum cg_operator_t op); +void cg_set_opacity(struct cg_ctx_t * ctx, double opacity); +void cg_set_fill_rule(struct cg_ctx_t * ctx, enum cg_fill_rule_t winding); +void cg_set_line_width(struct cg_ctx_t * ctx, double width); +void cg_set_line_cap(struct cg_ctx_t * ctx, enum cg_line_cap_t cap); +void cg_set_line_join(struct cg_ctx_t * ctx, enum cg_line_join_t join); +void cg_set_miter_limit(struct cg_ctx_t * ctx, double limit); +void cg_set_dash(struct cg_ctx_t * ctx, double * dashes, int ndash, double offset); +void cg_translate(struct cg_ctx_t * ctx, double tx, double ty); +void cg_scale(struct cg_ctx_t * ctx, double sx, double sy); +void cg_rotate(struct cg_ctx_t * ctx, double r); +void cg_transform(struct cg_ctx_t * ctx, struct cg_matrix_t * m); +void cg_set_matrix(struct cg_ctx_t * ctx, struct cg_matrix_t * m); +void cg_identity_matrix(struct cg_ctx_t * ctx); +void cg_move_to(struct cg_ctx_t * ctx, double x, double y); +void cg_line_to(struct cg_ctx_t * ctx, double x, double y); +void cg_curve_to(struct cg_ctx_t * ctx, double x1, double y1, double x2, double y2, double x3, double y3); +void cg_quad_to(struct cg_ctx_t * ctx, double x1, double y1, double x2, double y2); +void cg_rel_move_to(struct cg_ctx_t * ctx, double dx, double dy); +void cg_rel_line_to(struct cg_ctx_t * ctx, double dx, double dy); +void cg_rel_curve_to(struct cg_ctx_t * ctx, double dx1, double dy1, double dx2, double dy2, double dx3, double dy3); +void cg_rel_quad_to(struct cg_ctx_t * ctx, double dx1, double dy1, double dx2, double dy2); +void cg_rectangle(struct cg_ctx_t * ctx, double x, double y, double w, double h); +void cg_round_rectangle(struct cg_ctx_t * ctx, double x, double y, double w, double h, double rx, double ry); +void cg_ellipse(struct cg_ctx_t * ctx, double cx, double cy, double rx, double ry); +void cg_circle(struct cg_ctx_t * ctx, double cx, double cy, double r); +void cg_arc(struct cg_ctx_t * ctx, double cx, double cy, double r, double a0, double a1); +void cg_arc_negative(struct cg_ctx_t * ctx, double cx, double cy, double r, double a0, double a1); +void cg_new_path(struct cg_ctx_t * ctx); +void cg_close_path(struct cg_ctx_t * ctx); +void cg_reset_clip(struct cg_ctx_t * ctx); +void cg_clip(struct cg_ctx_t * ctx); +void cg_clip_preserve(struct cg_ctx_t * ctx); +void cg_fill(struct cg_ctx_t * ctx); +void cg_fill_preserve(struct cg_ctx_t * ctx); +void cg_stroke(struct cg_ctx_t * ctx); +void cg_stroke_preserve(struct cg_ctx_t * ctx); +void cg_paint(struct cg_ctx_t * ctx); + +#ifdef __cplusplus +} +#endif + +#endif /* __CG_H__ */ diff --git a/libmui/mui/mui.c b/libmui/mui/mui.c new file mode 100644 index 0000000..c0655ee --- /dev/null +++ b/libmui/mui/mui.c @@ -0,0 +1,285 @@ +/* + * mui.c + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ + +#include +#include +#include +#include +#include + +#include "mui_priv.h" + +void +mui_init( + mui_t *ui) +{ + //memset(ui, 0, sizeof(*ui)); + ui->clear_color = MUI_COLOR(0xccccccff); + TAILQ_INIT(&ui->windows); + TAILQ_INIT(&ui->zombies); + TAILQ_INIT(&ui->fonts); + mui_font_init(ui); + pixman_region32_init(&ui->redraw); + c2_rect_t whole = C2_RECT(0, 0, ui->screen_size.x, ui->screen_size.y); + pixman_region32_reset(&ui->inval, (pixman_box32_t*)&whole); +} + +void +mui_dispose( + mui_t *ui) +{ + pixman_region32_fini(&ui->inval); + mui_font_dispose(ui); + mui_window_t *w; + while ((w = TAILQ_FIRST(&ui->windows))) { + mui_window_dispose(w); + } +} + +void +mui_draw( + mui_t *ui, + mui_drawable_t *dr, + uint16_t all) +{ + if (!(all || pixman_region32_not_empty(&ui->inval))) + return; + if (all) { + // printf("%s: all\n", __func__); + c2_rect_t whole = C2_RECT(0, 0, dr->pix.size.x, dr->pix.size.y); + pixman_region32_reset(&ui->inval, (pixman_box32_t*)&whole); + } + mui_drawable_set_clip(dr, NULL); + + /* + * Windows are drawn top to bottom, their area/rectangle is added to the + * done region, the done region (any windows that are overlaping others) + * is substracted to any other windows update region before drawing... + * once all windows are done, the 'done' region (sum of all the windows), + * is substracted from the 'desk' area and erased. + */ + pixman_region32_t done = {}; + + mui_window_t * win; + TAILQ_FOREACH_REVERSE(win, &ui->windows, windows, self) { + pixman_region32_intersect_rect(&win->inval, &win->inval, + win->frame.l, win->frame.t, + c2_rect_width(&win->frame), c2_rect_height(&win->frame)); + + mui_drawable_set_clip(dr, NULL); + if (!all) + mui_drawable_clip_push_region(dr, &win->inval); + else + mui_drawable_clip_push(dr, &win->frame); + pixman_region32_clear(&win->inval); + + mui_drawable_clip_substract_region(dr, &done); + mui_window_draw(win, dr); + // printf(" %s : %s\n", win->title, c2_rect_as_str(&win->frame)); + pixman_region32_union_rect(&done, &done, + win->frame.l, win->frame.t, + c2_rect_width(&win->frame), c2_rect_height(&win->frame)); + } + + mui_drawable_set_clip(dr, NULL); + pixman_region32_t sect = {}; + c2_rect_t desk = C2_RECT(0, 0, dr->pix.size.x, dr->pix.size.y); + pixman_region32_inverse(§, &done, (pixman_box32_t*)&desk); + + mui_drawable_clip_push_region(dr, §); + + pixman_image_fill_boxes( + ui->clear_color.value ? PIXMAN_OP_SRC : PIXMAN_OP_CLEAR, + mui_drawable_get_pixman(dr), + &PIXMAN_COLOR(ui->clear_color), 1, (pixman_box32_t*)&desk); + pixman_region32_fini(§); + pixman_region32_fini(&done); + + pixman_region32_union(&ui->redraw, &ui->redraw, &ui->inval); + pixman_region32_clear(&ui->inval); + if (ui->draw_debug) { + // save a png of the current screen + ui->draw_debug = 0; + printf("%s: saving debug.png\n", __func__); + // mui_drawable_save_to_png(dr, "debug.png"); + } +} + +bool +mui_handle_event( + mui_t *ui, + mui_event_t *ev) +{ + bool res = false; + if (!ev->when) + ev->when = mui_get_time(); + ui->action_active++; + switch (ev->type) { + case MUI_EVENT_KEYUP: + case MUI_EVENT_KEYDOWN: { + if (ev->modifiers & MUI_MODIFIER_EVENT_TRACE) + printf("%s modifiers %04x key %x\n", __func__, + ev->modifiers, ev->key.key); + mui_window_t *w, *safe; + TAILQ_FOREACH_REVERSE_SAFE(w, &ui->windows, windows, self, safe) { + if ((res = mui_window_handle_keyboard(w, ev))) { + if (ev->modifiers & MUI_MODIFIER_EVENT_TRACE) + printf(" window:%s handled it\n", + w->title); + break; + } + } + if (ev->modifiers & MUI_MODIFIER_EVENT_TRACE) + if (!res) + printf(" no window handled it\n"); + } break; + case MUI_EVENT_BUTTONUP: + case MUI_EVENT_BUTTONDOWN: + case MUI_EVENT_WHEEL: + case MUI_EVENT_DRAG: { + if (ev->type == MUI_EVENT_BUTTONDOWN && ev->mouse.button > 1) { + printf("%s: button %d not handled\n", __func__, + ev->mouse.button); + ui->draw_debug++; + c2_rect_t whole = C2_RECT(0, 0, ui->screen_size.x, ui->screen_size.y); + pixman_region32_reset(&ui->inval, (pixman_box32_t*)&whole); + } + if (ev->modifiers & MUI_MODIFIER_EVENT_TRACE) + printf("%s %d mouse %d %3dx%3d capture:%s\n", __func__, + ev->type, ev->mouse.button, + ev->mouse.where.x, ev->mouse.where.y, + ui->event_capture ? + ui->event_capture->title : "(none)"); + if (ui->event_capture) { + res = mui_window_handle_mouse(ui->event_capture, ev); + break; + } else { + mui_window_t *w, *safe; + TAILQ_FOREACH_REVERSE_SAFE(w, &ui->windows, windows, self, safe) { + if ((res = mui_window_handle_mouse(w, ev))) { + if (ev->modifiers & MUI_MODIFIER_EVENT_TRACE) + printf(" window:%s handled it\n", + w->title); + break; + } + } + } + if (ev->modifiers & MUI_MODIFIER_EVENT_TRACE) + if (!res) + printf(" no window handled it\n"); + } break; + } + ui->action_active--; + return res; +} + +static uint16_t +_mui_simplify_mods( + uint16_t mods) +{ + uint16_t res = 0; + if (mods & MUI_MODIFIER_SHIFT) + res |= MUI_MODIFIER_RSHIFT; + if (mods & MUI_MODIFIER_CTRL) + res |= MUI_MODIFIER_RCTRL; + if (mods & MUI_MODIFIER_ALT) + res |= MUI_MODIFIER_RALT; + if (mods & MUI_MODIFIER_SUPER) + res |= MUI_MODIFIER_RSUPER; + return res; +} + +bool +mui_event_match_key( + mui_event_t *ev, + mui_key_equ_t key_equ) +{ + if (ev->type != MUI_EVENT_KEYUP && ev->type != MUI_EVENT_KEYDOWN) + return false; + if (toupper(ev->key.key) != toupper(key_equ.key)) + return false; + if (_mui_simplify_mods(ev->modifiers) != _mui_simplify_mods(key_equ.mod)) + return false; + return true; +} + +uint8_t +mui_timer_register( + mui_t *ui, + mui_timer_p cb, + void *param, + uint32_t delay) +{ + if (ui->timer.map == (uint64_t)-1) { + fprintf(stderr, "%s ran out of timers\n", __func__); + return -1; + } + int ti = ffsl(~ui->timer.map) - 1; + ui->timer.map |= 1 << ti; + ui->timer.timers[ti].cb = cb; + ui->timer.timers[ti].param = param; + ui->timer.timers[ti].when = mui_get_time() + delay; + return 0; +} + +void +mui_timers_run( + mui_t *ui ) +{ + uint64_t now = mui_get_time(); + uint64_t map = ui->timer.map; + while (map) { + int ti = ffsl(map) - 1; + map &= ~(1 << ti); + if (ui->timer.timers[ti].when > now) + continue; + mui_time_t r = ui->timer.timers[ti].cb( + ui, now, + ui->timer.timers[ti].param); + if (r == 0) + ui->timer.map &= ~(1 << ti); + else + ui->timer.timers[ti].when += r; + } +} + +void +_mui_window_free( + mui_window_t *win); + +void +mui_garbage_collect( + mui_t * ui) +{ + mui_window_t *win, *safe; + TAILQ_FOREACH_SAFE(win, &ui->zombies, self, safe) { + TAILQ_REMOVE(&ui->zombies, win, self); + _mui_window_free(win); + } +} + +void +mui_run( + mui_t *ui) +{ + mui_timers_run(ui); + mui_garbage_collect(ui); +} + +bool +mui_has_active_windows( + mui_t *ui) +{ + mui_window_t *win; + TAILQ_FOREACH(win, &ui->windows, self) { + if (mui_menubar_window(win) || win->flags.hidden) + continue; + return true; + } + return false; +} diff --git a/libmui/mui/mui.h b/libmui/mui/mui.h new file mode 100644 index 0000000..6aa38d5 --- /dev/null +++ b/libmui/mui/mui.h @@ -0,0 +1,984 @@ +/* + * mui.h + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ + +/* + * This is the main include file for the libmui UI library, it should be + * the only one you need to include. + */ + +#pragma once + +#include +#include +#include +#include "c2_arrays.h" +#include "bsd_queue.h" +#include "stb_ttc.h" + +/* Four Character Constants are used everywhere. Wish this had become a standard, + * as it is so handy -- but nope, thus the macro. Annoyingly, the little- + * endianess of them makes it a pain to do a printf() with them, this is why + * the values are reversed here. + */ +#ifndef FCC +#define FCC(_a,_b,_c,_d) (((_d)<<24)|((_c)<<16)|((_b)<<8)|(_a)) +#endif + +enum mui_event_e { + MUI_EVENT_KEYUP = 0, + MUI_EVENT_KEYDOWN, + MUI_EVENT_BUTTONUP, + MUI_EVENT_BUTTONDOWN, + MUI_EVENT_WHEEL, + MUI_EVENT_DRAG, + // the following ones aren't supported yet + MUI_EVENT_MOUSEENTER, + MUI_EVENT_MOUSELEAVE, + MUI_EVENT_RESIZE, + MUI_EVENT_CLOSE, + MUI_EVENT_COUNT, +}; + +enum mui_key_e { + MUI_KEY_ESCAPE = 0x1b, + MUI_KEY_LEFT = 0x80, + MUI_KEY_UP, + MUI_KEY_RIGHT, + MUI_KEY_DOWN, + MUI_KEY_INSERT, + MUI_KEY_HOME, + MUI_KEY_END, + MUI_KEY_PAGEUP, + MUI_KEY_PAGEDOWN, + MUI_KEY_MODIFIERS = 0x90, + MUI_KEY_LSHIFT = MUI_KEY_MODIFIERS, + MUI_KEY_RSHIFT, + MUI_KEY_LCTRL, + MUI_KEY_RCTRL, + MUI_KEY_LALT, + MUI_KEY_RALT, + MUI_KEY_RSUPER, + MUI_KEY_LSUPER, + MUI_KEY_MODIFIERS_LAST, + MUI_KEY_F1 = 0x100, + MUI_KEY_F2, + MUI_KEY_F3, + MUI_KEY_F4, + MUI_KEY_F5, + MUI_KEY_F6, + MUI_KEY_F7, + MUI_KEY_F8, + MUI_KEY_F9, + MUI_KEY_F10, + MUI_KEY_F11, + MUI_KEY_F12, +}; + +enum mui_modifier_e { + MUI_MODIFIER_LSHIFT = (1 << (MUI_KEY_LSHIFT - MUI_KEY_MODIFIERS)), + MUI_MODIFIER_RSHIFT = (1 << (MUI_KEY_RSHIFT - MUI_KEY_MODIFIERS)), + MUI_MODIFIER_LCTRL = (1 << (MUI_KEY_LCTRL - MUI_KEY_MODIFIERS)), + MUI_MODIFIER_RCTRL = (1 << (MUI_KEY_RCTRL - MUI_KEY_MODIFIERS)), + MUI_MODIFIER_LALT = (1 << (MUI_KEY_LALT - MUI_KEY_MODIFIERS)), + MUI_MODIFIER_RALT = (1 << (MUI_KEY_RALT - MUI_KEY_MODIFIERS)), + MUI_MODIFIER_RSUPER = (1 << (MUI_KEY_RSUPER - MUI_KEY_MODIFIERS)), + MUI_MODIFIER_LSUPER = (1 << (MUI_KEY_LSUPER - MUI_KEY_MODIFIERS)), + + // special flag, trace events handling for this event + MUI_MODIFIER_EVENT_TRACE= (1 << 15), + MUI_MODIFIER_SHIFT = (MUI_MODIFIER_LSHIFT | MUI_MODIFIER_RSHIFT), + MUI_MODIFIER_CTRL = (MUI_MODIFIER_LCTRL | MUI_MODIFIER_RCTRL), + MUI_MODIFIER_ALT = (MUI_MODIFIER_LALT | MUI_MODIFIER_RALT), + MUI_MODIFIER_SUPER = (MUI_MODIFIER_LSUPER | MUI_MODIFIER_RSUPER), +}; + +/* + * The following constants are in UTF8 format, and relate to glyphs in + * the TTF fonts + */ +/* These are from the icon font */ +#define MUI_ICON_FOLDER "" +#define MUI_ICON_FOLDER_OPEN "" +#define MUI_ICON_ROOT "" +#define MUI_ICON_FILE "" +#define MUI_ICON_POPUP_ARROWS "" +#define MUI_ICON_HOME "" +#define MUI_ICON_SBAR_UP "" +#define MUI_ICON_SBAR_DOWN "" + +/* These are specific the Charcoal System font */ +#define MUI_GLYPH_APPLE "" // solid apple +#define MUI_GLYPH_OAPPLE "" // open apple +#define MUI_GLYPH_COMMAND "" +#define MUI_GLYPH_OPTION "" +#define MUI_GLYPH_CONTROL "" +#define MUI_GLYPH_SHIFT "" +#define MUI_GLYPH_TICK "" // tickmark for menus +#define MUI_GLYPH_SUBMENU "" // custom, for the hierarchical menus +#define MUI_GLYPH_IIE "" // custom, IIe glyph +/* These are also from Charcoal System font (added to the original) */ +#define MUI_GLYPH_F1 "" +#define MUI_GLYPH_F2 "" +#define MUI_GLYPH_F3 "" +#define MUI_GLYPH_F4 "" +#define MUI_GLYPH_F5 "" +#define MUI_GLYPH_F6 "" +#define MUI_GLYPH_F7 "" +#define MUI_GLYPH_F8 "" +#define MUI_GLYPH_F9 "" +#define MUI_GLYPH_F10 "" +#define MUI_GLYPH_F11 "" +#define MUI_GLYPH_F12 "" + +typedef uint64_t mui_time_t; + +/* + * Even description. pretty standard stuff here -- the 'when' field is + * only used really to detect double clicks so far. + * + * Even handlers should return true if the event was handled, (in which case + * even processing stops for that event) or false to continue passing the even + * down the chain. + * + * Events are passed to the top window first, and then down the chain of + * windows, until one of them returns true. + * Implicitely, it means the menubar gets to see the events first, even clicks, + * even if the click wasn't in the menubar. This is also true of key events of + * course, which allows the menu to detect key combos, first. + */ +typedef struct mui_event_t { + uint8_t type; + mui_time_t when; + uint32_t modifiers; + union { + struct key { + uint32_t key; + bool up; + } key; + struct { + uint32_t button; + c2_pt_t where; + } mouse; + struct { + int32_t delta; + c2_pt_t where; + } wheel; + }; +} mui_event_t; + +/* + * Key equivalent, used to match key events to menu items + * Might be extended to controls, right now only the 'key' is checked, + * mostly for Return and ESC. + */ +typedef union mui_key_equ_t { + struct { + uint16_t mod; + uint16_t key; + }; + uint32_t value; +} mui_key_equ_t; + +#define MUI_KEY_EQU(_mask, _key) \ + (mui_key_equ_t){ .mod = (_mask), .key = (_key) } + +struct mui_t; + +typedef struct mui_listbox_elem_t { + uint32_t disabled : 1; + char icon[8]; + void * elem; // char * or... ? +} mui_listbox_elem_t; + +DECLARE_C_ARRAY(mui_listbox_elem_t, mui_listbox_elems, 2); +IMPLEMENT_C_ARRAY(mui_listbox_elems); + +struct mui_control_t; +struct mui_window_t; +struct mui_listbox_elem_t; + +/* + * Window DEFinition -- Handle all related to a window, from drawing to + * even handling. + */ +enum { + MUI_WDEF_INIT = 0, + MUI_WDEF_DISPOSE, + MUI_WDEF_DRAW, + MUI_WDEF_EVENT, +}; +typedef bool (*mui_wdef_p)( + struct mui_window_t * win, + uint8_t what, + void * param); +enum { + MUI_CDEF_INIT = 0, + MUI_CDEF_DISPOSE, + MUI_CDEF_DRAW, + MUI_CDEF_EVENT, + MUI_CDEF_SET_STATE, + MUI_CDEF_SET_VALUE, + MUI_CDEF_SET_TITLE, + MUI_CDEF_SELECT, +}; +typedef bool (*mui_cdef_p)( + struct mui_control_t * c, + uint8_t what, + void * param); +typedef void (*mui_ldef_p)( + struct mui_control_t * c, + uint32_t elem_index, + struct mui_listbox_elem_t * elem); + +/* + * Timer callback definition. Behaves in a pretty standard way; the timer + * returns 0 to be cancelled (for one shot timers for example) or return + * the delay to the next call. + */ +typedef mui_time_t (*mui_timer_p)( + struct mui_t * mui, + mui_time_t now, + void * param); +/* + * Actions are the provided way to add custom response to events for the + * application; action handlers are called for a variety of things, from clicks + * in controls, to menu selections, to window close etc. + * + * The 'what' parameter is a 4 character code, that can be used to identify + * the action, and the 'param' is a pointer to a structure that depends on + * the 'what' action (hopefully documented with that action constant) + * + * the 'cb_param' is specific to this action function pointer and is passed as + * is to the callback. + */ +typedef int (*mui_window_action_p)( + struct mui_window_t * win, + void * cb_param, + uint32_t what, + void * param); +typedef int (*mui_control_action_p)( + struct mui_control_t *c, + void * cb_param, + uint32_t what, + void * param); +/* + * This is a standardized way of installing 'action' handlers onto windows + * and controls. The 'current' field is used to prevent re-entrance. This structure + * is opaque and is not accessible by the application, typically. + */ +typedef struct mui_action_t { + STAILQ_ENTRY(mui_action_t) self; + uint32_t current; // prevents re-entrance + union { + mui_window_action_p window_cb; + mui_control_action_p control_cb; + }; + void * cb_param; +} mui_action_t; + +typedef STAILQ_HEAD(, mui_action_t) mui_action_queue_t; + +struct cg_surface_t; +struct cg_ctx_t; + +/* + * Describes a pixmap. Currently only used for the screen destination pixels. + * And really, only bpp:43 for ARGB is supported. + */ +typedef struct mui_pixmap_t { + uint8_t * pixels; + uint32_t bpp : 8; + c2_pt_t size; + uint32_t row_bytes; +} mui_pixmap_t; + +typedef pixman_region32_t mui_region_t; + +DECLARE_C_ARRAY(mui_region_t, mui_clip_stack, 2); + +/* + * The Drawable is a drawing context -- currently there's only one for the + * whole screen, but technically we could have several. The important feature + * of this is that it keeps a context for the pixman library destination + * image, AND also the context for the 'cg' vectorial library. + * Furthermore it keeps track of a stack of clipping rectangles, and is able + * to 'sync' the current clipping area for either (or both) cg and libpixman. + */ +typedef struct mui_drawable_t { + mui_pixmap_t pix; // *has* to be first in struct + void * _pix_hash; // used to detect if pix has changed + struct cg_surface_t * cg_surface; + struct cg_ctx_t * cg; + union pixman_image * pixman; // (try) not to use these directly + unsigned int pixman_clip_dirty: 1, + cg_clip_dirty : 1; + mui_clip_stack_t clip; +} mui_drawable_t; + +/* + * Drawable related + */ +void +mui_drawable_dispose( + mui_drawable_t * dr); +// get/allocate a pixman structure for this drawable +union pixman_image * +mui_drawable_get_pixman( + mui_drawable_t * dr); +// get/allocate a cg drawing context for this +struct cg_ctx_t * +mui_drawable_get_cg( + mui_drawable_t * dr); +// return 0 (no intersect), 1: fully contained and 2: partial contains +int +mui_drawable_clip_intersects( + mui_drawable_t * dr, + c2_rect_p r ); +void +mui_drawable_set_clip( + mui_drawable_t * dr, + c2_rect_array_p clip ); +int +mui_drawable_clip_push( + mui_drawable_t * dr, + c2_rect_p r ); +int +mui_drawable_clip_push_region( + mui_drawable_t * dr, + pixman_region32_t * rgn ); +int +mui_drawable_clip_substract_region( + mui_drawable_t * dr, + pixman_region32_t * rgn ); +void +mui_drawable_clip_pop( + mui_drawable_t * dr ); +pixman_region32_t * +mui_drawable_clip_get( + mui_drawable_t * dr); + + +/* + * Your typical ARGB color. Note that the components are NOT + * alpha-premultiplied at this stage. + * This struct should be able to be passed as a value, not a pointer + */ +typedef union mui_color_t { + struct { + uint8_t a,r,g,b; + } __attribute__((packed)); + uint32_t value; + uint8_t v[4]; +} mui_color_t; + +typedef struct mui_control_color_t { + mui_color_t fill, frame, text; +} mui_control_color_t; + +#define MUI_COLOR(_v) ((mui_color_t){ .value = (_v)}) + +#define CG_COLOR(_c) (struct cg_color_t){ \ + .a = (_c).a / 255.0, .r = (_c).r / 255.0, \ + .g = (_c).g / 255.0, .b = (_c).b / 255.0 } +/* + * Pixman use premultiplied alpha values + */ +#define PIXMAN_COLOR(_c) (pixman_color_t){ \ + .alpha = (_c).a * 257, .red = (_c).r * (_c).a, \ + .green = (_c).g * (_c).a, .blue = (_c).b * (_c).a } + + +typedef struct mui_font_t { + mui_drawable_t font; // points to ttc pixels! + char * name; // not filename, internal name, aka 'main' + unsigned int size; // in pixels + TAILQ_ENTRY(mui_font_t) self; + struct stb_ttc_info ttc; +} mui_font_t; + +/* + * Font related + */ +void +mui_font_init( + struct mui_t * ui); +void +mui_font_dispose( + struct mui_t * ui); + +mui_font_t * +mui_font_find( + struct mui_t * ui, + const char * name); +void +mui_font_text_draw( + mui_font_t * font, + mui_drawable_t *dr, + c2_pt_t where, + const char * text, + unsigned int text_len, + mui_color_t color); +int +mui_font_text_measure( + mui_font_t * font, + const char * text, + struct stb_ttc_measure *m ); + +enum mui_text_align_e { + MUI_TEXT_ALIGN_LEFT = 0, + MUI_TEXT_ALIGN_CENTER = (1 << 0), + MUI_TEXT_ALIGN_RIGHT = (1 << 1), + MUI_TEXT_ALIGN_TOP = 0, + MUI_TEXT_ALIGN_MIDDLE = (MUI_TEXT_ALIGN_CENTER << 2), + MUI_TEXT_ALIGN_BOTTOM = (MUI_TEXT_ALIGN_RIGHT << 2), +}; + +void +mui_font_textbox( + mui_font_t * font, + mui_drawable_t *dr, + c2_rect_t bbox, + const char * text, + unsigned int text_len, + mui_color_t color, + uint16_t flags ); +DECLARE_C_ARRAY(stb_ttc_g*, mui_glyph_array, 8, int x, y, w; ); +DECLARE_C_ARRAY(mui_glyph_array_t, mui_glyph_line_array, 8); + +/* + * Measure a text string, return the number of lines, and each glyphs + * position already aligned to the MUI_TEXT_ALIGN_* flags. + */ +void +mui_font_measure( + mui_font_t * font, + c2_rect_t bbox, + const char * text, + unsigned int text_len, + mui_glyph_line_array_t *lines, + uint16_t flags); +// clear all the lines, and glyph lists. Use it after mui_font_measure +void +mui_font_measure_clear( + mui_glyph_line_array_t *lines); + + +enum mui_window_layer_e { + MUI_WINDOW_LAYER_NORMAL = 0, + MUI_WINDOW_LAYER_MODAL = 3, + MUI_WINDOW_LAYER_ALERT = 5, + MUI_WINDOW_LAYER_TOP = 15, + // Menubar and Menus (popups) are also windows + MUI_WINDOW_MENUBAR_LAYER = MUI_WINDOW_LAYER_TOP - 1, + MUI_WINDOW_MENU_LAYER, +}; + +enum mui_window_action_e { + MUI_WINDOW_ACTION_NONE = 0, + MUI_WINDOW_ACTION_CLOSE = FCC('w','c','l','s'), +}; + +typedef struct mui_window_t { + TAILQ_ENTRY(mui_window_t) self; + struct mui_t * ui; + mui_wdef_p wdef; + uint32_t uid; // optional, pseudo unique id + struct { + unsigned long hidden: 1, + zombie: 1, // is in pre-delete ui->zombies + layer : 4, + hit_part : 8; + } flags; + c2_pt_t click_loc; + struct mui_drawable_t * dr; + // both these rectangles are in screen coordinates, even tho + // 'contents' is fully included in 'frame' + c2_rect_t frame, content; + char * title; + mui_action_queue_t actions; + TAILQ_HEAD(controls, mui_control_t) controls; + // anything deleted during an action goes in zombies + TAILQ_HEAD(zombies, mui_control_t) zombies; + struct mui_control_t * control_clicked; + mui_region_t inval; +} mui_window_t; + +/* + * Window related + */ +mui_window_t * +mui_window_create( + struct mui_t * ui, + c2_rect_t frame, + mui_wdef_p wdef, + uint8_t layer, + const char * title, + uint32_t instance_size); +// Dispose a window and it's content (controls). +/* + * Note: if an action is in progress the window is not freed immediately + * but added to the zombie list, and freed when the action is done. + * This is to prevent re-entrance problems. This allows window actions to + * delete their own window without crashing. + */ +void +mui_window_dispose( + mui_window_t * win); +// Invalidate 'r' in window coordinates, or the whole window if 'r' is NULL +void +mui_window_inval( + mui_window_t * win, + c2_rect_t * r); +// return true if the window is the frontmost window (in that window's layer) +bool +mui_window_isfront( + mui_window_t * win); +// return the top (non menubar/menu) window +mui_window_t * +mui_window_front( + struct mui_t *ui); + +// move win to the front (of its layer), return true if it was moved +bool +mui_window_select( + mui_window_t * win); +// call the window action callback, if any +void +mui_window_action( + mui_window_t * c, + uint32_t what, + void * param ); +// add an action callback for this window +void +mui_window_set_action( + mui_window_t * c, + mui_window_action_p cb, + void * param ); +// return the window whose UID is 'uid', or NULL if not found +mui_window_t * +mui_window_get_by_id( + struct mui_t * ui, + uint32_t uid ); +// set the window UID +void +mui_window_set_id( + mui_window_t * win, + uint32_t uid); + +struct mui_menu_items_t; + +/* + * This is a menu item descriptor (also used for the titles, bar a few bits). + * This does not a *control* in the window, instead this is used to describe + * the menus and menu item controls that are created when the menu becomes visible. + */ +typedef struct mui_menu_item_t { + uint32_t disabled : 1, hilited : 1; + uint32_t index: 9; + uint32_t uid; + char * title; + char mark[8]; // UTF8 -- Charcoal + char icon[8]; // UTF8 -- Wider, icon font + char kcombo[16]; // UTF8 -- display only + mui_key_equ_t key_equ; // keystroke to select this item + struct mui_menu_item_t * submenu; + c2_pt_t location; // calculated by menu creation code +} mui_menu_item_t; + +/* + * The menu item array is atypical as the items ('e' field) are not allocated + * by the array, but by the menu creation code. This is because the menu + * reuses the pointer to the items that is passed when the menu is added to + * the menubar. + * the 'read only' field is used to prevent the array from trying to free the + * items when being disposed. + */ +DECLARE_C_ARRAY(mui_menu_item_t, mui_menu_items, 2, + bool read_only; ); +IMPLEMENT_C_ARRAY(mui_menu_items); + +enum { + // parameter is a mui_menu_item_t* for the first item of the menu, + // this is exactly the parameter passed to add_simple() + // you can use this to disable/enable menu items etc + MUI_MENUBAR_ACTION_PREPARE = FCC('m','b','p','r'), + // parameter 'target' is a mui_menuitem_t* + MUI_MENUBAR_ACTION_SELECT = FCC('m','b','a','r'), +}; +/* + * Menu related. + * Menubar, and menus/popups are windows as well, in a layer above the + * normal ones. + */ +mui_window_t * +mui_menubar_new( + struct mui_t * ui ); +// return the previously created menubar (or NULL) +mui_window_t * +mui_menubar_get( + struct mui_t * ui ); + +/* + * Add a menu to the menubar. 'items' is an array of mui_menu_item_t + * terminated by an element with a NULL title. + * Note: The array is NOT const, it will be tweaked for storing items position, + * it can also be tweaked to set/reset the disabled state, check marks etc + */ +struct mui_control_t * +mui_menubar_add_simple( + mui_window_t * win, + const char * title, + uint32_t menu_uid, + mui_menu_item_t * items ); +/* Turn off any highlighted menu titles */ +mui_window_t * +mui_menubar_highlight( + mui_window_t * win, + bool ignored ); + +enum mui_control_type_e { + MUI_CONTROL_NONE = 0, + MUI_CONTROL_BUTTON, + MUI_CONTROL_GROUP, + MUI_CONTROL_SEPARATOR, + MUI_CONTROL_TEXTBOX, + MUI_CONTROL_GROUPBOX, + MUI_CONTROL_LISTBOX, + MUI_CONTROL_SCROLLBAR, + MUI_CONTROL_MENUTITLE, + MUI_CONTROL_MENUITEM, + MUI_CONTROL_SUBMENUITEM, + MUI_CONTROL_POPUP, +}; + +enum { + MUI_BUTTON_STYLE_NORMAL = 0, + MUI_BUTTON_STYLE_DEFAULT = 1, + MUI_BUTTON_STYLE_RADIO, + MUI_BUTTON_STYLE_CHECKBOX, +}; + +enum { + MUI_CONTROL_STATE_NORMAL = 0, + MUI_CONTROL_STATE_HOVER, + MUI_CONTROL_STATE_CLICKED, + MUI_CONTROL_STATE_DISABLED, + MUI_CONTROL_STATE_COUNT +}; + +enum { + MUI_CONTROL_ACTION_NONE = 0, + MUI_CONTROL_ACTION_VALUE_CHANGED = FCC('c','v','a','l'), + MUI_CONTROL_ACTION_CLICKED = FCC('c','l','k','d'), + MUI_CONTROL_ACTION_SELECT = FCC('c','s','e','l'), + MUI_CONTROL_ACTION_DOUBLECLICK = FCC('c','d','c','l'), +}; + +/* + * Control record... this are the 'common' fields, most of the controls + * have their own 'extended' record using their own fields. + */ +typedef struct mui_control_t { + TAILQ_ENTRY(mui_control_t) self; + struct mui_window_t * win; + mui_cdef_p cdef; + uint32_t state; + uint32_t type; + uint32_t style; + struct { + unsigned int hidden : 1, + zombie : 1, + hit_part : 8; + } flags; + uint32_t value; + uint32_t uid; + uint32_t uid_mask; // for radio buttons + c2_rect_t frame; + mui_key_equ_t key_equ; // keystroke to select this control + char * title; + mui_action_queue_t actions; +} mui_control_t; + +/* + * Control related + */ +mui_control_t * +mui_control_new( + mui_window_t * win, + uint8_t type, + mui_cdef_p cdef, + c2_rect_t frame, + const char * title, + uint32_t uid, + uint32_t instance_size ); +void +mui_control_dispose( + mui_control_t * c ); +uint32_t +mui_control_get_type( + mui_control_t * c ); +uint32_t +mui_control_get_uid( + mui_control_t * c ); +mui_control_t * +mui_control_locate( + mui_window_t * win, + c2_pt_t pt ); +mui_control_t * +mui_control_get_by_id( + mui_window_t * win, + uint32_t uid ); +void +mui_control_inval( + mui_control_t * c ); +void +mui_control_action( + mui_control_t * c, + uint32_t what, + void * param ); +void +mui_control_set_action( + mui_control_t * c, + mui_control_action_p cb, + void * param ); +void +mui_control_set_state( + mui_control_t * c, + uint32_t state ); +uint32_t +mui_control_get_state( + mui_control_t * c ); + +int32_t +mui_control_get_value( + mui_control_t * c); +int32_t +mui_control_set_value( + mui_control_t * c, + int32_t selected); +const char * +mui_control_get_title( + mui_control_t * c ); +void +mui_control_set_title( + mui_control_t * c, + const char * text ); + +mui_control_t * +mui_button_new( + mui_window_t * win, + c2_rect_t frame, + uint8_t style, + const char * title, + uint32_t uid ); +/* + * Create a static text box. Font is optional, flags correponds to the MUI_TEXT_ALIGN_* + * PLUS the extrast listed below. + */ +enum mui_textbox_e { + MUI_CONTROL_TEXTBOX_FRAME = (1 << 8), +}; +mui_control_t * +mui_textbox_new( + mui_window_t * win, + c2_rect_t frame, + const char * text, + const char * font, + uint16_t flags ); +mui_control_t * +mui_groupbox_new( + mui_window_t * win, + c2_rect_t frame, + const char * title, + uint16_t flags ); + +mui_control_t * +mui_scrollbar_new( + mui_window_t * win, + c2_rect_t frame, + uint32_t uid ); +uint32_t +mui_scrollbar_get_max( + mui_control_t * c); +void +mui_scrollbar_set_max( + mui_control_t * c, + uint32_t max); +void +mui_scrollbar_set_page( + mui_control_t * c, + uint32_t page); + +mui_control_t * +mui_listbox_new( + mui_window_t * win, + c2_rect_t frame, + uint32_t uid ); +void +mui_listbox_prepare( + mui_control_t * c); +mui_listbox_elems_t * +mui_listbox_get_elems( + mui_control_t * c); +mui_control_t * +mui_separator_new( + mui_window_t * win, + c2_rect_t frame); +mui_control_t * +mui_popupmenu_new( + mui_window_t * win, + c2_rect_t frame, + const char * title, + uint32_t uid ); +mui_menu_items_t * +mui_popupmenu_get_items( + mui_control_t * c); +void +mui_popupmenu_prepare( + mui_control_t * c); + +/* + * Standard file dialog + */ +enum mui_std_action_e { + MUI_STDF_ACTION_NONE = 0, + // parameter 'target' is a char * with full pathname of selected file + MUI_STDF_ACTION_SELECT = FCC('s','t','d','s'), + MUI_STDF_ACTION_CANCEL = FCC('s','t','d','c'), +}; + +mui_window_t * +mui_stdfile_get( + struct mui_t * ui, + c2_pt_t where, + const char * prompt, + const char * regexp, + const char * start_path ); +// return the curently selected pathname -- caller must free() it +char * +mui_stdfile_get_selected_path( + mui_window_t * w ); + +/* + * Alert dialog + */ +enum { + MUI_ALERT_FLAG_OK = (1 << 0), + MUI_ALERT_FLAG_CANCEL = (1 << 1), + + MUI_ALERT_ICON_INFO = (1 << 8), + + MUI_ALERT_INFO = (MUI_ALERT_FLAG_OK | MUI_ALERT_ICON_INFO), + MUI_ALERT_WARN = (MUI_ALERT_FLAG_OK | MUI_ALERT_FLAG_CANCEL), +}; + +enum { + MUI_ALERT_BUTTON_OK = FCC('o','k',' ',' '), + MUI_ALERT_BUTTON_CANCEL = FCC('c','a','n','c'), +}; + +mui_window_t * +mui_alert( + struct mui_t * ui, + c2_pt_t where, // (0,0) will center it + const char * title, + const char * message, + uint16_t flags ); + +enum { + MUI_TIME_RES = 1, + MUI_TIME_SECOND = 1000000, + MUI_TIME_MS = (MUI_TIME_SECOND/1000), +}; +mui_time_t +mui_get_time(); + +typedef struct mui_timer_group_t { + uint64_t map; + struct { + mui_time_t when; + mui_timer_p cb; + void * param; + } timers[64]; +} mui_timer_group_t; + +/* + * Register 'cb' to be called after 'delay'. Returns a timer id (0 to 63) + * or 0xff if no timer is available. The timer function cb can return 0 for a one + * shot timer, or another delay that will be added to the current stamp for a further + * call of the timer. + * 'param' will be also passed to the timer callback. + */ +uint8_t +mui_timer_register( + struct mui_t * ui, + mui_timer_p cb, + void * param, + uint32_t delay); + +typedef struct mui_t { + c2_pt_t screen_size; + mui_color_t clear_color; + uint16_t modifier_keys; + int draw_debug; + // this is the sum of all the window's dirty regions, inc moved windows etc + mui_region_t inval; + // once the pixels have been refreshed, 'inval' is copied to 'redraw' + // to push the pixels to the screen. + mui_region_t redraw; + + TAILQ_HEAD(, mui_font_t) fonts; + TAILQ_HEAD(windows, mui_window_t) windows; + mui_window_t * menubar; + TAILQ_HEAD(, mui_window_t) zombies; + // this is used to track any active action callbacks to + // prevent recursion problem and track any 'delete' happening + // during an action callback + uint32_t action_active; + mui_window_t * event_capture; + mui_timer_group_t timer; + char * pref_directory; /* optional */ +} mui_t; + +void +mui_init( + mui_t * ui); +void +mui_dispose( + mui_t * ui); +void +mui_draw( + mui_t * ui, + mui_drawable_t *dr, + uint16_t all); +void +mui_run( + mui_t * ui); +// return true if the event was handled by the ui +bool +mui_handle_event( + mui_t * ui, + mui_event_t * ev); +// return true if event 'ev' is a key combo matching key_equ +bool +mui_event_match_key( + mui_event_t * ev, + mui_key_equ_t key_equ); +/* Return true if the ui has any active windows, ie, not hidden, zombie; + * This does not include the menubar, but it does include any menus or popups + * + * This is used to decide wether to hide the mouse cursor or not + */ +bool +mui_has_active_windows( + mui_t * ui); + +/* Return a hash value for string inString */ +uint32_t +mui_hash( + const char * inString ); diff --git a/libmui/mui/mui_alert.c b/libmui/mui/mui_alert.c new file mode 100644 index 0000000..0647bff --- /dev/null +++ b/libmui/mui/mui_alert.c @@ -0,0 +1,90 @@ +/* + * mui_alert.c + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ + +#include +#include +#include +#include + +#include "mui.h" + + +typedef struct mui_alert_t { + mui_window_t win; + mui_control_t * ok, *cancel; +} mui_alert_t; + +static int +_mui_alert_button_cb( + mui_control_t * c, + void * cb_param, + uint32_t what, + void * param) +{ +// mui_alert_t * alert = (mui_alert_t *)c->win; + + // notify the window handler of the control action + mui_window_action(c->win, what, c); + mui_window_dispose(c->win); + return 0; +} + +mui_window_t * +mui_alert( + struct mui_t * ui, + c2_pt_t where, + const char * title, + const char * message, + uint16_t flags ) +{ + c2_rect_t cf = C2_RECT_WH(0, 0, 540, 200); + + if (where.x && where.y) + c2_rect_offset(&cf, where.x, where.y); + else + c2_rect_offset(&cf, + (ui->screen_size.x / 2) - (c2_rect_width(&cf) / 2), + (ui->screen_size.y * 0.3) - (c2_rect_height(&cf) / 2)); + mui_window_t *w = mui_window_create(ui, cf, + NULL, MUI_WINDOW_LAYER_ALERT, + title, sizeof(mui_alert_t)); + mui_alert_t * alert = (mui_alert_t *)w; + mui_control_t * c = NULL; + + cf = C2_RECT_WH(0, 0, 120, 40); + c2_rect_left_of(&cf, c2_rect_width(&w->content), 20); + c2_rect_top_of(&cf, c2_rect_height(&w->content), 20); + if (flags & MUI_ALERT_FLAG_OK) { + alert->ok = c = mui_button_new(w, cf, MUI_BUTTON_STYLE_DEFAULT, + "OK", MUI_ALERT_BUTTON_OK); + alert->ok->key_equ = MUI_KEY_EQU(0, 13); // return + c2_rect_left_of(&cf, cf.l, 30); + } + if (flags & MUI_ALERT_FLAG_CANCEL) { + alert->cancel = c = mui_button_new(w, cf, MUI_BUTTON_STYLE_NORMAL, + "Cancel", MUI_ALERT_BUTTON_CANCEL); + alert->cancel->key_equ = MUI_KEY_EQU(0, 27); // ESC + } + cf = C2_RECT_WH(0, 10, 540-140, 70); + c2_rect_left_of(&cf, c2_rect_width(&w->content), 20); + c = mui_textbox_new(w, cf, message, NULL, 0); + cf = C2_RECT_WH(10, 10, 80, 75); + c = mui_textbox_new(w, cf, + "", "icon_large", + MUI_TEXT_ALIGN_CENTER | MUI_TEXT_ALIGN_MIDDLE); + + c = NULL; + TAILQ_FOREACH(c, &w->controls, self) { + if (mui_control_get_uid(c) == 0) + continue; + mui_control_set_action(c, _mui_alert_button_cb, alert); + } + + return w; +} + diff --git a/libmui/mui/mui_cdef_boxes.c b/libmui/mui/mui_cdef_boxes.c new file mode 100644 index 0000000..bb3e723 --- /dev/null +++ b/libmui/mui/mui_cdef_boxes.c @@ -0,0 +1,188 @@ +/* + * mui_cdef_boxes.c + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ + +#include +#include +#include +#include +#include + +#include "mui.h" +#include "cg.h" + +typedef struct mui_textbox_control_t { + mui_control_t control; + mui_font_t * font; + uint16_t flags; +} mui_textbox_control_t; + +extern const mui_control_color_t mui_control_color[MUI_CONTROL_STATE_COUNT]; + +static void +mui_textbox_draw( + mui_window_t * win, + mui_control_t * c, + mui_drawable_t *dr ) +{ + c2_rect_t f = c->frame; + c2_rect_offset(&f, win->content.l, win->content.t); + + mui_textbox_control_t *tb = (mui_textbox_control_t *)c; + + c2_rect_t text_frame = f; + mui_drawable_clip_push(dr, &f); + mui_font_textbox(tb->font, dr, + text_frame, c->title, strlen(c->title), + mui_control_color[c->state].text, + tb->flags); + if (tb->flags & MUI_CONTROL_TEXTBOX_FRAME) { + struct cg_ctx_t * cg = mui_drawable_get_cg(dr); + cg_set_line_width(cg, 1); + cg_set_source_color(cg, &CG_COLOR(mui_control_color[c->state].frame)); + cg_rectangle(cg, f.l, f.t, + c2_rect_width(&f), c2_rect_height(&f)); + cg_stroke(cg); + } + mui_drawable_clip_pop(dr); +} + +static void +mui_groupbox_draw( + mui_window_t * win, + mui_control_t * c, + mui_drawable_t *dr ) +{ + c2_rect_t f = c->frame; + c2_rect_offset(&f, win->content.l, win->content.t); + + mui_textbox_control_t *tb = (mui_textbox_control_t *)c; + + c2_rect_t text_frame = f; + c2_rect_t box_frame = f; + + mui_font_t * main = mui_font_find(win->ui, "main"); + stb_ttc_measure m = {}; + mui_font_text_measure(main, c->title, &m); + text_frame.l += main->size * 0.3; + text_frame.b = text_frame.t + main->size; + text_frame.r = text_frame.l + m.x1 + m.x0; + box_frame.t += m.ascent * 0.85; + + mui_color_t contentFill = MUI_COLOR(0xf0f0f0ff); + mui_color_t decoColor = MUI_COLOR(0x666666ff); + + struct cg_ctx_t * cg = mui_drawable_get_cg(dr); + cg_set_line_width(cg, 1); + cg_set_source_color(cg, &CG_COLOR(decoColor)); + cg_rectangle(cg, box_frame.l, box_frame.t, + c2_rect_width(&box_frame), c2_rect_height(&box_frame)); + cg_stroke(cg); + // now erase text background + cg_set_source_color(cg, &CG_COLOR(contentFill)); + cg_rectangle(cg, text_frame.l, text_frame.t, + c2_rect_width(&text_frame), c2_rect_height(&text_frame)); + cg_fill(cg); + +// mui_drawable_clip_push(dr, &f); + mui_font_textbox(main, dr, + text_frame, c->title, strlen(c->title), + mui_control_color[c->state].text, + tb->flags); +// mui_drawable_clip_pop(dr); +} + +static void +mui_separator_draw( + mui_window_t * win, + mui_control_t * c, + mui_drawable_t *dr ) +{ + c2_rect_t f = c->frame; + c2_rect_offset(&f, win->content.l, win->content.t); + +// mui_drawable_clip_push(dr, &f); + struct cg_ctx_t * cg = mui_drawable_get_cg(dr); + cg_set_line_width(cg, 1); + + mui_color_t decoColor = MUI_COLOR(0x666666ff); + cg_set_source_color(cg, &CG_COLOR(decoColor)); + cg_move_to(cg, f.l + 0, f.t); + cg_line_to(cg, f.r - 0, f.t); + cg_stroke(cg); +// mui_drawable_clip_pop(dr); +} + +static bool +mui_cdef_boxes( + struct mui_control_t * c, + uint8_t what, + void * param) +{ +// mui_textbox_control_t *tb = (mui_textbox_control_t *)c; + switch (what) { + case MUI_CDEF_DRAW: { + mui_drawable_t * dr = param; + switch (c->type) { + case MUI_CONTROL_SEPARATOR: + mui_separator_draw(c->win, c, dr); + break; + case MUI_CONTROL_GROUPBOX: + mui_groupbox_draw(c->win, c, dr); + break; + case MUI_CONTROL_TEXTBOX: + mui_textbox_draw(c->win, c, dr); + break; + default: + break; + } + } break; + } + return false; +} + +mui_control_t * +mui_textbox_new( + mui_window_t * win, + c2_rect_t frame, + const char * text, + const char * font, + uint16_t flags ) +{ + mui_control_t * c = mui_control_new( + win, MUI_CONTROL_TEXTBOX, mui_cdef_boxes, + frame, text, 0, sizeof(mui_textbox_control_t)); + mui_textbox_control_t *tb = (mui_textbox_control_t *)c; + tb->font = mui_font_find(win->ui, font ? font : "main"); + tb->flags = flags; + return c; +} + +mui_control_t * +mui_separator_new( + mui_window_t * win, + c2_rect_t frame) +{ + return mui_control_new( + win, MUI_CONTROL_SEPARATOR, mui_cdef_boxes, + frame, NULL, 0, sizeof(mui_textbox_control_t)); +} + +mui_control_t * +mui_groupbox_new( + mui_window_t * win, + c2_rect_t frame, + const char * title, + uint16_t flags ) +{ + mui_control_t * c = mui_control_new( + win, MUI_CONTROL_GROUPBOX, mui_cdef_boxes, + frame, title, 0, sizeof(mui_textbox_control_t)); + mui_textbox_control_t *tb = (mui_textbox_control_t *)c; + tb->flags = flags; + return c; +} diff --git a/libmui/mui/mui_cdef_buttons.c b/libmui/mui/mui_cdef_buttons.c new file mode 100644 index 0000000..183a840 --- /dev/null +++ b/libmui/mui/mui_cdef_buttons.c @@ -0,0 +1,267 @@ +/* + * mui_cdef_buttons.c + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ + +#include +#include +#include +#include +#include + +#include "mui.h" +#include "cg.h" + + +extern const mui_control_color_t mui_control_color[MUI_CONTROL_STATE_COUNT]; + +#define BUTTON_INSET 4 +void +mui_button_draw( + mui_window_t * win, + mui_control_t * c, + mui_drawable_t *dr ) +{ + c2_rect_t f = c->frame; + c2_rect_offset(&f, win->content.l, win->content.t); + + struct cg_ctx_t * cg = mui_drawable_get_cg(dr); + + cg_set_source_color(cg, &CG_COLOR(mui_control_color[c->state].frame)); + if (c->style == MUI_BUTTON_STYLE_DEFAULT) { + cg_set_line_width(cg, 3); + cg_round_rectangle(cg, f.l, f.t, + c2_rect_width(&f), c2_rect_height(&f), + 10, 10); + cg_stroke(cg); + c2_rect_inset(&f, BUTTON_INSET, BUTTON_INSET); + } + mui_font_t * main = TAILQ_FIRST(&win->ui->fonts); + stb_ttc_measure m = {}; + mui_font_text_measure(main, c->title, &m); + + int title_width = m.x1 - m.x0; + c2_rect_t title = f; + title.r = title.l + title_width + 1; + title.b = title.t + m.ascent - m.descent; + c2_rect_offset(&title, -m.x0 + + (int)((c2_rect_width(&f) / 2) - (c2_rect_width(&title)) / 2), + (c2_rect_height(&f) / 2) - (c2_rect_height(&title) / 2)); + mui_drawable_clip_push(dr, &f); + cg = mui_drawable_get_cg(dr); + c2_rect_t inner = f; + c2_rect_inset(&inner, 1, 1); + cg_set_line_width(cg, 2); + cg_round_rectangle(cg, inner.l, inner.t, + c2_rect_width(&inner), c2_rect_height(&inner), 6, 6); + cg_set_source_color(cg, &CG_COLOR(mui_control_color[c->state].fill)); + cg_fill_preserve(cg); +// cg_rectangle(cg, title.l, title.t, +// c2_rect_width(&title), c2_rect_height(&title)); + cg_set_source_color(cg, &CG_COLOR(mui_control_color[c->state].frame)); + cg_stroke(cg); + mui_font_text_draw(main, dr, + C2_PT(title.l, title.t), c->title, strlen(c->title), + mui_control_color[c->state].text); + mui_drawable_clip_pop(dr); +} + +void +mui_check_rad_draw( + mui_window_t * win, + mui_control_t * c, + mui_drawable_t *dr ) +{ + mui_font_t * main = mui_font_find(win->ui, "main"); + c2_rect_t f = c->frame; + c2_rect_offset(&f, win->content.l, win->content.t); + + c2_rect_t box = f; + box.r = box.l + (main->size * 0.95); + box.b = box.t + (main->size * 0.95); + c2_rect_offset(&box, 0, (c2_rect_height(&f) / 2) - (c2_rect_height(&box) / 2)); + c2_rect_t title = f; + title.l = box.r + 8; + + struct cg_ctx_t * cg = mui_drawable_get_cg(dr); + + mui_drawable_clip_push(dr, &f); + + // draw the box/circle + if (c->style == MUI_BUTTON_STYLE_RADIO) { + cg_circle(cg, box.l + (c2_rect_width(&box) / 2), + box.t + (c2_rect_height(&box) / 2), + c2_rect_width(&box) / 2); + cg_set_source_color(cg, &CG_COLOR(mui_control_color[c->state].fill)); + cg_fill_preserve(cg); + cg_set_line_width(cg, 2); + cg_set_source_color(cg, &CG_COLOR(mui_control_color[c->state].frame)); + cg_stroke(cg); + if (c->value) { // fill the inside circle + c2_rect_inset(&box, 5, 5); + cg_circle(cg, box.l + (c2_rect_width(&box) / 2), + box.t + (c2_rect_height(&box) / 2), + c2_rect_width(&box) / 2); + cg_fill(cg); + } + } else { + cg_rectangle(cg, box.l, box.t, + c2_rect_width(&box), c2_rect_height(&box)); + cg_set_source_color(cg, &CG_COLOR(mui_control_color[c->state].fill)); + cg_fill_preserve(cg); + cg_set_line_width(cg, 2); + cg_set_source_color(cg, &CG_COLOR(mui_control_color[c->state].frame)); + cg_stroke(cg); + // now the cross + if (c->value) { + cg_set_line_width(cg, 2); + cg_move_to(cg, box.l, box.t); + cg_line_to(cg, box.r, box.b); + cg_move_to(cg, box.r, box.t); + cg_line_to(cg, box.l, box.b); + cg_stroke(cg); + } + } + mui_font_textbox(main, dr, + title, c->title, 0, + mui_control_color[0].text, + MUI_TEXT_ALIGN_MIDDLE); + mui_drawable_clip_pop(dr); +} + + +static bool +mui_button_mouse( + struct mui_control_t * c, + mui_event_t * ev) +{ + if (c->state == MUI_CONTROL_STATE_DISABLED) + return false; + + c2_rect_t f = c->frame; + c2_rect_offset(&f, c->win->content.l, c->win->content.t); + + switch (ev->type) { + case MUI_EVENT_BUTTONDOWN: { + if (c2_rect_contains_pt(&f, &ev->mouse.where)) + mui_control_set_state(c, MUI_CONTROL_STATE_CLICKED); + } break; + case MUI_EVENT_BUTTONUP: { + if (c->state != MUI_CONTROL_STATE_CLICKED) + break; + mui_control_set_state(c, MUI_CONTROL_STATE_NORMAL); + switch (c->style) { + case MUI_BUTTON_STYLE_NORMAL: + case MUI_BUTTON_STYLE_DEFAULT: + break; + /* Look for all other matching radio buttons, turn + * their values off, before setting this one to on */ + case MUI_BUTTON_STYLE_RADIO: { + if (c->uid_mask) { + mui_control_t * c2 = NULL; + TAILQ_FOREACH(c2, &c->win->controls, self) { + if (c2->type == MUI_CONTROL_BUTTON && + c2->style == MUI_BUTTON_STYLE_RADIO && + (c2->uid & c->uid_mask) == (c->uid & c->uid_mask) && + c2 != c) { + // printf("OFF %4.4s\n", (char*)&c2->uid); + mui_control_set_value(c2, false); + } + } + } + // printf("ON %4.4s\n", (char*)&c->uid); + mui_control_set_value(c, true); + } break; + case MUI_BUTTON_STYLE_CHECKBOX: + mui_control_set_value(c, + !mui_control_get_value(c)); + break; + } + mui_control_action(c, MUI_CONTROL_ACTION_SELECT, NULL); + } break; + case MUI_EVENT_DRAG: { + if (c2_rect_contains_pt(&f, &ev->mouse.where)) + mui_control_set_state(c, MUI_CONTROL_STATE_CLICKED); + else + mui_control_set_state(c, MUI_CONTROL_STATE_NORMAL); + } break; + } + return true; +} + +static bool +mui_cdef_button( + struct mui_control_t * c, + uint8_t what, + void * param) +{ + switch (what) { + case MUI_CDEF_INIT: + break; + case MUI_CDEF_DISPOSE: + break; + case MUI_CDEF_DRAW: { + mui_drawable_t * dr = param; + switch (c->style) { + case MUI_BUTTON_STYLE_NORMAL: + case MUI_BUTTON_STYLE_DEFAULT: + mui_button_draw(c->win, c, dr); + break; + case MUI_BUTTON_STYLE_CHECKBOX: + case MUI_BUTTON_STYLE_RADIO: + mui_check_rad_draw(c->win, c, dr); + break; + default: + return false; + } + } break; + case MUI_CDEF_EVENT: { + mui_event_t *ev = param; + // printf("%s event %d where %dx%d\n", __func__, ev->type, + // ev->mouse.where.x, + // ev->mouse.where.y); + switch (ev->type) { + case MUI_EVENT_BUTTONUP: + case MUI_EVENT_DRAG: + case MUI_EVENT_BUTTONDOWN: { + return mui_button_mouse(c, ev); + } break; + } + } break; + case MUI_CDEF_SELECT: { + if (c->style == MUI_BUTTON_STYLE_CHECKBOX) { + mui_control_set_value(c, !c->value); + } + } break; + } + return false; +} + +mui_control_t * +mui_button_new( + mui_window_t * win, + c2_rect_t frame, + uint8_t style, + const char * title, + uint32_t uid ) +{ + switch (style) { + case MUI_BUTTON_STYLE_NORMAL: + break; + case MUI_BUTTON_STYLE_DEFAULT: + c2_rect_inset(&frame, -BUTTON_INSET, -BUTTON_INSET); + break; + case MUI_BUTTON_STYLE_CHECKBOX: + break; + case MUI_BUTTON_STYLE_RADIO: + break; + } + mui_control_t * c = mui_control_new( + win, MUI_CONTROL_BUTTON, mui_cdef_button, + frame, title, uid, 0); + c->style = style; + return c; +} diff --git a/libmui/mui/mui_cdef_listbox.c b/libmui/mui/mui_cdef_listbox.c new file mode 100644 index 0000000..564f427 --- /dev/null +++ b/libmui/mui/mui_cdef_listbox.c @@ -0,0 +1,307 @@ +/* + * mui_cdef_listbox.c + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ + +#include +#include +#include +#include +#include + +#include "mui.h" +#include "cg.h" + +typedef struct mui_listbox_control_t { + mui_control_t control; + struct mui_control_t * scrollbar; + int32_t scroll; + uint8_t elem_height; + mui_listbox_elems_t elems; + mui_ldef_p ldef; + // to handle double-click + mui_time_t last_click; +} mui_listbox_control_t; + +extern const mui_control_color_t mui_control_color[MUI_CONTROL_STATE_COUNT]; + +static void +mui_listbox_draw( + mui_window_t * win, + mui_control_t * c, + mui_drawable_t *dr ) +{ + c2_rect_t f = c->frame; + c2_rect_offset(&f, win->content.l, win->content.t); + + struct cg_ctx_t * cg = mui_drawable_get_cg(dr); + cg_set_line_width(cg, 1); + cg_set_source_color(cg, &CG_COLOR(mui_control_color[c->state].frame)); + cg_rectangle(cg, f.l, f.t, + c2_rect_width(&f), c2_rect_height(&f)); + cg_stroke(cg); + { + c2_rect_t clip = f; + c2_rect_inset(&clip, 1, 1); + mui_drawable_clip_push(dr, &clip); + } + mui_listbox_control_t *lb = (mui_listbox_control_t *)c; + uint32_t top_element = lb->scroll / lb->elem_height; + uint32_t bottom_element = top_element + 1 + + (c2_rect_height(&f) / lb->elem_height); +// printf("%s draw from %d to %d\n", __func__, top_element, bottom_element); + + mui_font_t * icons = mui_font_find(win->ui, "icon_small"); + mui_font_t * main = mui_font_find(win->ui, "main"); + mui_color_t highlight = MUI_COLOR(0xd6fcc0ff); + + for (unsigned int ii = top_element; + ii < lb->elems.count && ii < bottom_element; ii++) { + c2_rect_t ef = f; + ef.b = ef.t + lb->elem_height; + c2_rect_offset(&ef, 0, ii * lb->elem_height - lb->scroll); + if (ii == c->value) { + struct cg_ctx_t * cg = mui_drawable_get_cg(dr); + cg_set_line_width(cg, 1); + cg_set_source_color(cg, &CG_COLOR(highlight)); + cg_rectangle(cg, ef.l, ef.t, + c2_rect_width(&ef), c2_rect_height(&ef)); + cg_fill(cg); + } + ef.l += 8; + mui_listbox_elem_t *e = &lb->elems.e[ii]; + if (lb->elems.e[ii].icon[0]) + mui_font_text_draw(icons, dr, ef.tl, lb->elems.e[ii].icon, 0, + mui_control_color[e->disabled ? + MUI_CONTROL_STATE_DISABLED : 0].text); + ef.l += 26; + mui_font_text_draw(main, dr, ef.tl, e->elem, 0, + mui_control_color[e->disabled ? + MUI_CONTROL_STATE_DISABLED : 0].text); + } + mui_drawable_clip_pop(dr); +} + +static bool +mui_listbox_key( + mui_control_t * c, + mui_event_t * ev) +{ + mui_listbox_control_t *lb = (mui_listbox_control_t *)c; + printf("%s key: %d\n", __func__, ev->key.key); + c2_rect_t f = c->frame; + c2_rect_offset(&f, -f.l, -f.t); + uint32_t page_size = (c2_rect_height(&f) / lb->elem_height)-1; + + int delta = 0; + if (ev->modifiers & (MUI_MODIFIER_SUPER | MUI_MODIFIER_CTRL)) + return false; + switch (ev->key.key) { + case MUI_KEY_UP: delta = -1; break; + case MUI_KEY_DOWN: delta = 1; break; + case MUI_KEY_PAGEUP: delta = -page_size; break; + case MUI_KEY_PAGEDOWN: delta = page_size; break; + } + if (!delta) + return false; + int nsel = c->value + delta; + if (nsel < 0) + nsel = 0; + if (nsel >= (int)lb->elems.count) + nsel = lb->elems.count - 1; + if (nsel != (int)c->value) { + c->value = nsel; + c2_rect_t e = c->frame; + e.b = e.t + lb->elem_height; + + c2_rect_offset(&e, -e.l, + -e.t + (c->value * lb->elem_height)); + c2_rect_t w = f; + c2_rect_offset(&w, 0, lb->scroll); + printf(" e:%s f:%s\n", c2_rect_as_str(&e), c2_rect_as_str(&w)); + if (e.b > w.b) { + lb->scroll = (e.b - c2_rect_height(&c->frame)); + printf(" over %d\n", lb->scroll); + } + if (e.t < w.t) + lb->scroll = e.t; + printf(" scroll:%d\n", lb->scroll); + mui_control_set_value(lb->scrollbar, lb->scroll); + mui_control_inval(c); +// mui_control_inval(lb->scrollbar); + mui_control_action(c, MUI_CONTROL_ACTION_VALUE_CHANGED, + &lb->elems.e[nsel]); + return true; + } + return false; +} + +static bool +mui_cdef_event( + struct mui_control_t * c, + mui_event_t *ev) +{ + mui_listbox_control_t *lb = (mui_listbox_control_t *)c; + switch (ev->type) { + case MUI_EVENT_BUTTONDOWN: { + c2_rect_t f = c->frame; + c2_rect_offset(&f, c->win->content.l, c->win->content.t); + // uint32_t page_size = (c2_rect_height(&f) / lb->elem_height)-1; + int nsel = lb->scroll + (ev->mouse.where.y - f.t); + nsel /= lb->elem_height; + if (nsel < 0) + nsel = 0; + if (nsel >= (int)lb->elems.count) + nsel = lb->elems.count - 1; + if (nsel != (int)c->value) { + mui_control_set_value(c, nsel); + mui_control_action(c, + MUI_CONTROL_ACTION_VALUE_CHANGED, + &lb->elems.e[nsel]); + } + { + mui_time_t now = mui_get_time(); + if ((now - lb->last_click) < + (MUI_TIME_MS * 300)) { + lb->last_click = 0; + if (lb->elems.e[nsel].disabled) + return true; + mui_control_action(c, + MUI_CONTROL_ACTION_SELECT, + &lb->elems.e[nsel]); + } else + lb->last_click = now; + } + return true; + } break; + case MUI_EVENT_KEYUP: { // ignore keydowns + if (ev->key.key == 13) { + if (!lb->elems.e[c->value].disabled) { + mui_control_action(c, + MUI_CONTROL_ACTION_SELECT, + &lb->elems.e[c->value]); + } + return true; + } + if (mui_listbox_key(c, ev)) + return true; + } break; + case MUI_EVENT_WHEEL: { + // printf("%s wheel delta %d\n", __func__, ev->wheel.delta); + lb->scroll += ev->wheel.delta * 20; + if (lb->scroll < 0) + lb->scroll = 0; + if (lb->scroll > + (int32_t)((lb->elems.count * lb->elem_height) - + c2_rect_height(&c->frame))) + lb->scroll = (lb->elems.count * lb->elem_height) - + c2_rect_height(&c->frame); + mui_control_set_value(lb->scrollbar, lb->scroll); + mui_control_inval(c); + return true; + } break; + } + return false; +} + +bool +mui_cdef_listbox( + struct mui_control_t * c, + uint8_t what, + void * param) +{ + switch (what) { + case MUI_CDEF_INIT: + break; + case MUI_CDEF_DISPOSE: + break; + case MUI_CDEF_DRAW: { + mui_drawable_t * dr = param; + mui_listbox_draw(c->win, c, dr); + } break; + case MUI_CDEF_EVENT: { + mui_event_t *ev = param; + return mui_cdef_event(c, ev); + } break; + } + return false; +} + +static int +mui_listbox_sbar_action( + mui_control_t * c, + void * cb_param, + uint32_t what, + void * param) +{ + mui_listbox_control_t *lb = (mui_listbox_control_t *)cb_param; + lb->scroll = mui_control_get_value(lb->scrollbar); +// printf("%s scroll %d\n", __func__, lb->scroll); + mui_control_inval(&lb->control); + return 0; +} + +mui_listbox_elems_t * +mui_listbox_get_elems( + mui_control_t * c) +{ + mui_listbox_control_t *lb = (mui_listbox_control_t *)c; + return &lb->elems; +} + + +mui_control_t * +mui_listbox_new( + mui_window_t * win, + c2_rect_t frame, + uint32_t uid ) +{ + c2_rect_t lbf = frame; + c2_rect_t sb = frame; + mui_font_t * main = mui_font_find(win->ui, "main"); + lbf.r -= main->size; + sb.l = sb.r - main->size; + mui_control_t *c = mui_control_new( + win, MUI_CONTROL_LISTBOX, mui_cdef_listbox, + lbf, NULL, uid, sizeof(mui_listbox_control_t)); + mui_listbox_control_t *lb = (mui_listbox_control_t *)c; + lb->scrollbar = mui_scrollbar_new(win, sb, 0); + mui_control_set_action(lb->scrollbar, mui_listbox_sbar_action, c); + lb->elem_height = main->size + 2; + + return c; +} + +void +mui_listbox_prepare( + mui_control_t * c) +{ + mui_listbox_control_t *lb = (mui_listbox_control_t *)c; + c2_rect_t content = C2_RECT_WH(0, 0, + c2_rect_width(&c->frame), c2_rect_height(&c->frame)); + content.b = lb->elems.count * lb->elem_height; + + c2_rect_offset(&content, 0, lb->scroll); + if (content.b < c2_rect_height(&c->frame)) { + c2_rect_offset(&content, 0, c2_rect_height(&c->frame) - content.b); + } + if (content.t > 0) { + c2_rect_offset(&content, 0, -content.t); + } + lb->scroll = content.t; + if (c2_rect_height(&content) > c2_rect_height(&c->frame)) { + mui_scrollbar_set_max(lb->scrollbar, + c2_rect_height(&content)); + mui_control_set_value(lb->scrollbar, -lb->scroll); + mui_scrollbar_set_page(lb->scrollbar, c2_rect_height(&c->frame)); + } else { + mui_scrollbar_set_max(lb->scrollbar, 0); + mui_control_set_value(lb->scrollbar, 0); + mui_scrollbar_set_page(lb->scrollbar, 0); + } + mui_control_inval(lb->scrollbar); + mui_control_inval(c); +} diff --git a/libmui/mui/mui_cdef_scrollbar.c b/libmui/mui/mui_cdef_scrollbar.c new file mode 100644 index 0000000..10d412a --- /dev/null +++ b/libmui/mui/mui_cdef_scrollbar.c @@ -0,0 +1,366 @@ +/* + * mui_cdef_scrollbar.c + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ + +#include +#include +#include +#include +#include + +#include "mui.h" +#include "cg.h" + + +extern const mui_control_color_t mui_control_color[MUI_CONTROL_STATE_COUNT]; + +enum mui_sb_part_e { + MUI_SB_PART_FRAME = 0, + MUI_SB_PART_UP, + MUI_SB_PART_DOWN, + MUI_SB_PART_PAGEUP, + MUI_SB_PART_PAGEDOWN, + MUI_SB_PART_THUMB, + MUI_SB_PART_COUNT, +}; + +typedef struct mui_scrollbar_control_t { + mui_control_t control; + uint32_t visible; + uint32_t max; + c2_pt_t drag_offset; + uint32_t saved_value; // to handle 'snapback' +} mui_scrollbar_control_t; + +static void +mui_scrollbar_make_rects( + mui_control_t * c, + c2_rect_t * parts) +{ + c2_rect_t f = c->frame; + c2_rect_offset(&f, c->win->content.l, c->win->content.t); + parts[MUI_SB_PART_FRAME] = f; + + c2_rect_t part = f; + part.b = part.t + c2_rect_width(&part); + parts[MUI_SB_PART_UP] = part; + part = f; + part.t = part.b - c2_rect_width(&part); + parts[MUI_SB_PART_DOWN] = part; + + mui_scrollbar_control_t *sb = (mui_scrollbar_control_t *)c; + if (sb->max <= sb->visible) { + c2_rect_t z = {}; + parts[MUI_SB_PART_THUMB] = z; + parts[MUI_SB_PART_PAGEUP] = z; + parts[MUI_SB_PART_PAGEDOWN] = z; + return; + } + part = f; + part.t = parts[MUI_SB_PART_UP].b + 1; + part.b = parts[MUI_SB_PART_DOWN].t - 1; + + float visible = sb->visible / (float)sb->max; + float thumb_size = visible * sb->visible; + if (thumb_size < 20) + thumb_size = 20; + float thumb_pos = c->value / ((float)sb->max- sb->visible); + float thumb_offset = 0.5 + thumb_pos * (c2_rect_height(&part) - thumb_size); +// printf("%s visible:%.2f ts: %.2f thumb_pos:%.2f thumb_offset:%.2f\n", +// __func__, visible, thumb_size, thumb_pos, thumb_offset); + + part.b = part.t + thumb_size; + c2_rect_offset(&part, 0, thumb_offset); + if (part.b > parts[MUI_SB_PART_DOWN].t) { + c2_rect_offset(&part, 0, parts[MUI_SB_PART_DOWN].t - part.b); + } + parts[MUI_SB_PART_THUMB] = part; + part = f; + part.t = parts[MUI_SB_PART_UP].b + 1; + part.b = parts[MUI_SB_PART_THUMB].t - 1; + parts[MUI_SB_PART_PAGEUP] = part; + part = f; + part.t = parts[MUI_SB_PART_THUMB].b + 1; + part.b = parts[MUI_SB_PART_DOWN].t - 1; + parts[MUI_SB_PART_PAGEDOWN] = part; +} + +static void +mui_scrollbar_draw( + mui_window_t * win, + mui_control_t * c, + mui_drawable_t *dr ) +{ + c2_rect_t f = c->frame; + c2_rect_offset(&f, win->content.l, win->content.t); + + struct cg_ctx_t * cg = mui_drawable_get_cg(dr); + cg_set_line_width(cg, 1); + + cg_rectangle(cg, f.l, f.t, + c2_rect_width(&f), c2_rect_height(&f)); + cg_set_source_color(cg, &CG_COLOR(mui_control_color[c->state].fill)); + cg_fill_preserve(cg); + cg_set_source_color(cg, &CG_COLOR(mui_control_color[c->state].frame)); + cg_stroke(cg); + + mui_font_t * icons = mui_font_find(win->ui, "icon_small"); + + c2_rect_t parts[MUI_SB_PART_COUNT]; + mui_scrollbar_make_rects(c, parts); + + mui_color_t contentFill = MUI_COLOR(0xa0a0a0ff); + mui_color_t decoColor = MUI_COLOR(0x666666ff); + + c2_rect_t pf; + pf = parts[MUI_SB_PART_UP]; + cg_rectangle(cg, pf.l, pf.t, + c2_rect_width(&pf), c2_rect_height(&pf)); + cg_set_source_color(cg, + c->flags.hit_part == MUI_SB_PART_UP ? + &CG_COLOR(decoColor) : + &CG_COLOR(mui_control_color[c->state].fill)); + cg_fill_preserve(cg); + cg_set_source_color(cg, &CG_COLOR(mui_control_color[c->state].frame)); + cg_stroke(cg); + + stb_ttc_measure m = {}; + mui_font_text_measure(icons, "", &m); + pf.l = pf.l + (c2_rect_width(&pf) - m.x1 - m.x0) / 2; + mui_font_text_draw(icons, dr, pf.tl, "", 0, + mui_control_color[c->state].text); + + pf = parts[MUI_SB_PART_DOWN]; + cg_rectangle(cg, pf.l, pf.t, + c2_rect_width(&pf), c2_rect_height(&pf)); + cg_set_source_color(cg, + c->flags.hit_part == MUI_SB_PART_DOWN ? + &CG_COLOR(decoColor) : + &CG_COLOR(mui_control_color[c->state].fill)); + cg_fill_preserve(cg); + cg_set_source_color(cg, &CG_COLOR(mui_control_color[c->state].frame)); + cg_stroke(cg); + + mui_font_text_measure(icons, "", &m); + pf.l = pf.l + (c2_rect_width(&pf) - m.x1 - m.x0) / 2; + mui_font_text_draw(icons, dr, pf.tl, "", 0, + mui_control_color[c->state].text); + + pf = parts[MUI_SB_PART_PAGEUP]; + if (c2_rect_height(&pf) > 0) { + cg_rectangle(cg, pf.l, pf.t, + c2_rect_width(&pf), c2_rect_height(&pf)); + cg_set_source_color(cg, + c->flags.hit_part == MUI_SB_PART_PAGEUP ? + &CG_COLOR(decoColor) : + &CG_COLOR(contentFill)); + cg_fill(cg); + } + pf = parts[MUI_SB_PART_PAGEDOWN]; + if (c2_rect_height(&pf) > 0) { + cg_rectangle(cg, pf.l, pf.t, + c2_rect_width(&pf), c2_rect_height(&pf)); + cg_set_source_color(cg, + c->flags.hit_part == MUI_SB_PART_PAGEDOWN ? + &CG_COLOR(decoColor) : + &CG_COLOR(contentFill)); + cg_fill(cg); + } + pf = parts[MUI_SB_PART_THUMB]; + if (c2_rect_height(&pf) > 0) { + cg_rectangle(cg, pf.l, pf.t, + c2_rect_width(&pf), c2_rect_height(&pf)); + cg_set_source_color(cg, + c->flags.hit_part == MUI_SB_PART_THUMB ? + &CG_COLOR(decoColor) : + &CG_COLOR(mui_control_color[c->state].fill)); + cg_fill_preserve(cg); + cg_set_source_color(cg, + &CG_COLOR(mui_control_color[c->state].frame)); + cg_stroke(cg); + } +} + +static void +_mui_scrollbar_scroll( + mui_control_t * c, + int32_t delta ) +{ + mui_scrollbar_control_t *sb = (mui_scrollbar_control_t *)c; + int32_t v = (int32_t)c->value + delta; + if (v < 0) + v = 0; + if (v > ((int32_t)sb->max - (int32_t)sb->visible)) + v = sb->max - sb->visible; + c->value = v; + mui_control_inval(c); + mui_control_action(c, MUI_CONTROL_ACTION_VALUE_CHANGED, NULL); +} + +static bool +mui_scrollbar_mouse( + struct mui_control_t * c, + mui_event_t * ev) +{ + mui_scrollbar_control_t *sb = (mui_scrollbar_control_t *)c; + c2_rect_t parts[MUI_SB_PART_COUNT]; + mui_scrollbar_make_rects(c, parts); + + switch (ev->type) { + case MUI_EVENT_BUTTONDOWN: { + for (int i = 1; i < MUI_SB_PART_COUNT; ++i) { + if (c2_rect_contains_pt(&parts[i], &ev->mouse.where)) { + c->flags.hit_part = i; + sb->drag_offset.x = + ev->mouse.where.x - parts[i].l; + sb->drag_offset.y = + ev->mouse.where.y - parts[i].t; + sb->saved_value = c->value; + break; + } + } + int part = c->flags.hit_part % MUI_SB_PART_COUNT; + switch (part) { + case MUI_SB_PART_DOWN: + case MUI_SB_PART_UP: { + _mui_scrollbar_scroll(c, + part == MUI_SB_PART_UP ? -30 : 30); + } break; + case MUI_SB_PART_PAGEUP: + case MUI_SB_PART_PAGEDOWN: { + int32_t offset = sb->visible; + _mui_scrollbar_scroll(c, + part == MUI_SB_PART_PAGEUP ? + -offset : offset); + } break; + case MUI_SB_PART_THUMB: + mui_control_inval(c); + break; + } + // printf("%s hit part %d\n", __func__, c->flags.hit_part); + } break; + case MUI_EVENT_DRAG: { + if (!c->flags.hit_part) + break; + int part = c->flags.hit_part % MUI_SB_PART_COUNT; + c2_rect_t test_rect = parts[part]; + if (part == MUI_SB_PART_THUMB) + c2_rect_inset(&test_rect, -60, -60); + if (c2_rect_contains_pt(&test_rect, &ev->mouse.where)) { + c->flags.hit_part = part; + switch (part) { + case MUI_SB_PART_THUMB: { + c2_rect_t nt = parts[part]; + c2_rect_offset(&nt, 0, + -(nt.t + parts[MUI_SB_PART_UP].b) + + ev->mouse.where.y - sb->drag_offset.y); + // printf("%s thumb %s\n", __func__, c2_rect_as_str(&nt)); + if (nt.t < 0) + c2_rect_offset(&nt, 0, -nt.t); + if (nt.b > parts[MUI_SB_PART_DOWN].t) + c2_rect_offset(&nt, 0, parts[MUI_SB_PART_DOWN].t - nt.b); + + int max_pixels = parts[MUI_SB_PART_DOWN].t - + parts[MUI_SB_PART_UP].b - + c2_rect_height(&nt); + uint32_t nv = nt.t * (sb->max - sb->visible) / max_pixels; + if (nv > (sb->max - sb->visible)) + nv = sb->max - sb->visible; + c->value = nv; + // printf("v is %d vs %d max %d = %d new val %d\n", + // nt.t, max_pixels, sb->max, nv, + // mui_control_get_value(c)); + mui_control_inval(c); + mui_control_action(c, MUI_CONTROL_ACTION_VALUE_CHANGED, NULL); + } break; + } + } else { + c->flags.hit_part = part + MUI_SB_PART_COUNT; + if (part == MUI_SB_PART_THUMB) { + c->value = sb->saved_value; + mui_control_inval(c); + mui_control_action(c, MUI_CONTROL_ACTION_VALUE_CHANGED, NULL); + } + } + } break; + case MUI_EVENT_BUTTONUP: { + if (!c->flags.hit_part) + break; + mui_control_inval(c); + c->flags.hit_part = 0; + } break; + } + return true; +} + +static bool +mui_cdef_scrollbar( + struct mui_control_t * c, + uint8_t what, + void * param) +{ + switch (what) { + case MUI_CDEF_INIT: + break; + case MUI_CDEF_DISPOSE: + break; + case MUI_CDEF_DRAW: { + mui_drawable_t * dr = param; + mui_scrollbar_draw(c->win, c, dr); + } break; + case MUI_CDEF_EVENT: { + // printf("%s event\n", __func__); + mui_event_t *ev = param; + switch (ev->type) { + case MUI_EVENT_BUTTONUP: + case MUI_EVENT_DRAG: + case MUI_EVENT_BUTTONDOWN: { + return mui_scrollbar_mouse(c, ev); + } break; + } + } break; + } + return false; +} + +uint32_t +mui_scrollbar_get_max( + mui_control_t * c) +{ + mui_scrollbar_control_t *sb = (mui_scrollbar_control_t *)c; + return sb->max; +} +void +mui_scrollbar_set_max( + mui_control_t * c, + uint32_t max) +{ + mui_scrollbar_control_t *sb = (mui_scrollbar_control_t *)c; + sb->max = max; + mui_control_inval(c); +} +void +mui_scrollbar_set_page( + mui_control_t * c, + uint32_t page) +{ + mui_scrollbar_control_t *sb = (mui_scrollbar_control_t *)c; + sb->visible = page; + mui_control_inval(c); +} + +mui_control_t * +mui_scrollbar_new( + mui_window_t * win, + c2_rect_t frame, + uint32_t uid ) +{ + return mui_control_new( + win, MUI_CONTROL_SCROLLBAR, mui_cdef_scrollbar, + frame, NULL, uid, + sizeof(mui_scrollbar_control_t)); +} diff --git a/libmui/mui/mui_controls.c b/libmui/mui/mui_controls.c new file mode 100644 index 0000000..e7ec48d --- /dev/null +++ b/libmui/mui/mui_controls.c @@ -0,0 +1,318 @@ +/* + * mui_controls.c + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ + +#include +#include +#include +#include +#include + +#include "mui.h" + + +const mui_control_color_t mui_control_color[MUI_CONTROL_STATE_COUNT] = { + [MUI_CONTROL_STATE_NORMAL] = { + .fill = MUI_COLOR(0xeeeeeeff), + .frame = MUI_COLOR(0x000000ff), + .text = MUI_COLOR(0x000000ff), + }, + [MUI_CONTROL_STATE_HOVER] = { + .fill = MUI_COLOR(0xaaaaaaff), + .frame = MUI_COLOR(0x000000ff), + .text = MUI_COLOR(0x0000ffff), + }, + [MUI_CONTROL_STATE_CLICKED] = { + .fill = MUI_COLOR(0x777777ff), + .frame = MUI_COLOR(0x000000ff), + .text = MUI_COLOR(0xffffffff), + }, + [MUI_CONTROL_STATE_DISABLED] = { + .fill = MUI_COLOR(0xeeeeeeff), + .frame = MUI_COLOR(0x666666ff), + .text = MUI_COLOR(0xccccccff), + }, +}; + +void +mui_control_draw( + mui_window_t * win, + mui_control_t * c, + mui_drawable_t *dr ) +{ + if (!c) + return; + if (c->cdef) + c->cdef(c, MUI_CDEF_DRAW, dr); +} + +mui_control_t * +mui_control_new( + mui_window_t * win, + uint8_t type, + mui_cdef_p cdef, + c2_rect_t frame, + const char * title, + uint32_t uid, + uint32_t instance_size ) +{ + if (!win) + return NULL; + mui_control_t *c = calloc(1, instance_size >= sizeof(*c) ? + instance_size : sizeof(*c)); + c->type = type; + c->cdef = cdef; + c->frame = frame; + c->title = title ? strdup(title) : NULL; + c->win = win; + c->uid = uid; + STAILQ_INIT(&c->actions); + TAILQ_INSERT_TAIL(&win->controls, c, self); + if (c->cdef) + c->cdef(c, MUI_CDEF_INIT, NULL); + return c; +} + +void +_mui_control_free( + mui_control_t * c ) +{ + if (!c) + return; + if (c->title) + free(c->title); + c->title = NULL; + if (c->cdef) + c->cdef(c, MUI_CDEF_DISPOSE, NULL); + free(c); +} + +void +mui_control_dispose( + mui_control_t * c ) +{ + if (!c) + return; + if (c->flags.zombie) { + printf("%s: DOUBLE delete %s\n", __func__, c->title); + return; + } + TAILQ_REMOVE(&c->win->controls, c, self); + if (c->win->flags.zombie || c->win->ui->action_active) { + c->flags.zombie = true; + TAILQ_INSERT_TAIL(&c->win->zombies, c, self); + } else + _mui_control_free(c); +} + +uint32_t +mui_control_get_type( + mui_control_t * c ) +{ + if (!c) + return 0; + return c->type; +} +uint32_t +mui_control_get_uid( + mui_control_t * c ) +{ + if (!c) + return 0; + return c->uid; +} + +mui_control_t * +mui_control_locate( + mui_window_t * win, + c2_pt_t pt ) +{ + if (!win) + return NULL; + mui_control_t * c; + TAILQ_FOREACH(c, &win->controls, self) { + c2_rect_t f = c->frame; + c2_rect_offset(&f, win->content.l, win->content.t); + if (c2_rect_contains_pt(&f, &pt)) + return c; + } + return NULL; +} + +static mui_time_t +_mui_control_highlight_timer_cb( + struct mui_t * mui, + mui_time_t now, + void * param) +{ + mui_control_t * c = param; + + printf("%s: %s\n", __func__, c->title); + mui_control_set_state(c, MUI_CONTROL_STATE_NORMAL); + if (c->cdef) + c->cdef(c, MUI_CDEF_SELECT, NULL); + mui_control_action(c, MUI_CONTROL_ACTION_SELECT, NULL); + + return 0; +} + +int32_t +mui_control_get_value( + mui_control_t * c) +{ + if (!c) + return 0; + return c->value; +} + +int32_t +mui_control_set_value( + mui_control_t * c, + int32_t value) +{ + if (!c) + return 0; + if (c->cdef && c->cdef(c, MUI_CDEF_SET_VALUE, &value)) + return c->value; + if (value != (int)c->value) + mui_control_inval(c); + c->value = value; + return c->value; +} + +bool +mui_control_event( + mui_control_t * c, + mui_event_t * ev ) +{ + if (!c) + return false; + bool res = false; + + res = c->cdef && c->cdef(c, MUI_CDEF_EVENT, ev); + if (res || !c->key_equ.key) + return res; + switch (ev->type) { + case MUI_EVENT_KEYDOWN: + if (c->state != MUI_CONTROL_STATE_DISABLED && + mui_event_match_key(ev, c->key_equ)) { + mui_control_set_state(c, MUI_CONTROL_STATE_CLICKED); + mui_timer_register( + c->win->ui, _mui_control_highlight_timer_cb, + c, MUI_TIME_SECOND / 10); + res = true; + } + break; + } + return res; +} + +void +mui_control_inval( + mui_control_t * c ) +{ + if (!c) + return; + mui_window_inval(c->win, &c->frame); +} + +void +mui_control_set_state( + mui_control_t * c, + uint32_t state ) +{ + if (!c) + return; + if (c->cdef && c->cdef(c, MUI_CDEF_SET_STATE, &state)) + return; + if (c->state == state) + return; + c->state = state; + mui_control_inval(c); +} + +uint32_t +mui_control_get_state( + mui_control_t * c ) +{ + if (!c) + return 0; + return c->state; +} + +const char * +mui_control_get_title( + mui_control_t * c ) +{ + if (!c) + return NULL; + return c->title; +} + +void +mui_control_set_title( + mui_control_t * c, + const char * text ) +{ + if (!c) + return; + if (c->cdef && c->cdef(c, MUI_CDEF_SET_TITLE, (void*)text)) + return; + if (text && c->title && !strcmp(text, c->title)) + return; + if (c->title) + free(c->title); + c->title = text ? strdup(text) : NULL; + mui_control_inval(c); +} + +void +mui_control_action( + mui_control_t * c, + uint32_t what, + void * param ) +{ + if (!c) + return; + c->win->ui->action_active++; + mui_action_t *a; + STAILQ_FOREACH(a, &c->actions, self) { + if (!a->control_cb) + continue; + a->control_cb(c, a->cb_param, what, param); + } + c->win->ui->action_active--; +} + +void +mui_control_set_action( + mui_control_t * c, + mui_control_action_p cb, + void * param ) +{ + if (!c) + return; + mui_action_t *a = calloc(1, sizeof(*a)); + a->control_cb = cb; + a->cb_param = param; + STAILQ_INSERT_TAIL(&c->actions, a, self); +} + +mui_control_t * +mui_control_get_by_id( + mui_window_t * win, + uint32_t uid ) +{ + if (!win) + return NULL; + mui_control_t *c; + TAILQ_FOREACH(c, &win->controls, self) { + if (c->uid == uid) + return c; + } + return NULL; +} diff --git a/libmui/mui/mui_drawable.c b/libmui/mui/mui_drawable.c new file mode 100644 index 0000000..c1bbe47 --- /dev/null +++ b/libmui/mui/mui_drawable.c @@ -0,0 +1,280 @@ +/* + * mui_drawable.c + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ + +#include +#include +#include +#include + +#include +#include "mui.h" +#include "cg.h" + + +IMPLEMENT_C_ARRAY(mui_clip_stack); + +void +mui_drawable_clear( + mui_drawable_t * dr) +{ + if (!dr) + return; + if (dr->cg) + cg_destroy(dr->cg); + dr->cg = NULL; + if (dr->cg_surface) + cg_surface_destroy(dr->cg_surface); + dr->cg_surface = NULL; + if (dr->pixman) + pixman_image_unref(dr->pixman); + dr->pixman = NULL; + for (int i = 0; i < (int)dr->clip.count; i++) + pixman_region32_fini(&dr->clip.e[i]); + mui_clip_stack_clear(&dr->clip); + dr->_pix_hash = NULL; +} + +void +mui_drawable_dispose( + mui_drawable_t * dr) +{ + if (!dr) + return; + mui_drawable_clear(dr); + mui_clip_stack_free(&dr->clip); +// free(dr); +} + +static struct cg_ctx_t * +_cg_updated_clip( + mui_drawable_t * dr) +{ + if (!dr->cg_clip_dirty) + return dr->cg; + dr->cg_clip_dirty = 0; + cg_reset_clip(dr->cg); + if (!dr->clip.count) + return dr->cg; + int cnt = 0; + pixman_box32_t *r = pixman_region32_rectangles( + &dr->clip.e[dr->clip.count-1], &cnt); + for (int i = 0; i < cnt; i++) { + cg_rectangle(dr->cg, r[i].x1, r[i].y1, + r[i].x2 - r[i].x1, r[i].y2 - r[i].y1); + } + cg_clip(dr->cg); + return dr->cg; +} + +struct cg_ctx_t * +mui_drawable_get_cg( + mui_drawable_t * dr) +{ + if (!dr) + return NULL; + + if (dr->cg) { + // in case the pix structure had changed... + dr->cg_surface->stride = dr->pix.row_bytes; + dr->cg_surface->pixels = dr->pix.pixels; + dr->cg_surface->width = dr->pix.size.x; + dr->cg_surface->height = dr->pix.size.y; + return _cg_updated_clip(dr); + } + dr->cg_surface = cg_surface_create_for_data( + dr->pix.size.x, dr->pix.size.y, + dr->pix.pixels); + dr->cg_surface->stride = dr->pix.row_bytes; + dr->cg = cg_create(dr->cg_surface); + return _cg_updated_clip(dr); +} + +/* + * Update clip to the 'latest' one from the stack of clips. + */ +static union pixman_image * +_pixman_updated_clip( + mui_drawable_t * dr) +{ + if (!dr->pixman_clip_dirty) + return dr->pixman; + dr->pixman_clip_dirty = 0; + if (dr->clip.count) + pixman_image_set_clip_region32( + dr->pixman, &dr->clip.e[dr->clip.count-1]); + else + pixman_image_set_clip_region32( + dr->pixman, NULL); + return dr->pixman; +} + +union pixman_image * +mui_pixmap_make_pixman( + mui_pixmap_t * pix) +{ + pixman_format_code_t k = PIXMAN_a8r8g8b8; + switch (pix->bpp) { + case 8: k = PIXMAN_a8; break; + case 16: k = PIXMAN_r5g6b5; break; + case 24: k = PIXMAN_r8g8b8; break; + case 32: k = PIXMAN_a8r8g8b8; break; + } + union pixman_image * res = pixman_image_create_bits_no_clear( + k, pix->size.x, pix->size.y, + (uint32_t*)pix->pixels, pix->row_bytes); + return res; +} + +union pixman_image * +mui_drawable_get_pixman( + mui_drawable_t * dr) +{ + if (!dr) + return NULL; + void * _hash = dr->pix.pixels + dr->pix.size.y * dr->pix.row_bytes; + if (dr->_pix_hash != _hash) { + dr->_pix_hash = _hash; + dr->_pix_hash = dr->pix.pixels + dr->pix.size.y * dr->pix.row_bytes; + if (dr->pixman) + pixman_image_unref(dr->pixman); + dr->pixman = NULL; + } +// return _pixman_updated_clip(dr); + if (dr->pixman) + return _pixman_updated_clip(dr); + dr->pixman = mui_pixmap_make_pixman(&dr->pix); + dr->pixman_clip_dirty = 1; + return _pixman_updated_clip(dr); +} + +pixman_region32_t * +mui_drawable_clip_get( + mui_drawable_t * dr) +{ + if (!dr || !dr->clip.count) + return NULL; + return &dr->clip.e[dr->clip.count-1]; +} + +void +mui_drawable_set_clip( + mui_drawable_t * dr, + c2_rect_array_p clip ) +{ + if (!dr) + return; + for (int i = 0; i < (int)dr->clip.count; i++) + pixman_region32_fini(&dr->clip.e[i]); + mui_clip_stack_clear(&dr->clip); + if (clip && clip->count) { + pixman_region32_t r = {}; + + pixman_region32_init_rects(&r, + (pixman_box32_t*)clip->e, clip->count); + // intersect the whole lot with the whole pixmap boundary + pixman_region32_t rg = {}; + pixman_region32_intersect_rect(&rg, &r, + 0, 0, dr->pix.size.x, dr->pix.size.y); + pixman_region32_fini(&r); + + mui_clip_stack_add(&dr->clip, rg); + } + dr->pixman_clip_dirty = 1; + dr->cg_clip_dirty = 1; +} + +int +mui_drawable_clip_intersects( + mui_drawable_t * dr, + c2_rect_p r ) +{ + if (!dr || !r) + return 0; + if (!dr->clip.count) + return 1; + // conveniently, c2_rect_t is the same as pixman_box_32_t + pixman_region_overlap_t o = pixman_region32_contains_rectangle( + &dr->clip.e[dr->clip.count-1], (pixman_box32_t*)r); + + return o == PIXMAN_REGION_OUT ? 0 : + o == PIXMAN_REGION_IN ? 1 : -1; +} + +int +mui_drawable_clip_push( + mui_drawable_t * dr, + c2_rect_p r ) +{ + if (!dr || !r) + return 0; + pixman_region32_t rg = {}; + if (dr->clip.count == 0) { + pixman_region32_init_rect(&rg, + r->l, r->t, c2_rect_width(r), c2_rect_height(r)); + } else { + pixman_region32_intersect_rect(&rg, &dr->clip.e[dr->clip.count-1], + r->l, r->t, c2_rect_width(r), c2_rect_height(r)); + } + mui_clip_stack_add(&dr->clip, rg); + dr->pixman_clip_dirty = 1; + dr->cg_clip_dirty = 1; + + return dr->clip.count; +} + +int +mui_drawable_clip_push_region( + mui_drawable_t * dr, + pixman_region32_t * rgn ) +{ + if (!dr || !rgn) + return 0; + + pixman_region32_t rg = {}; + if (dr->clip.count == 0) { + pixman_region32_copy(&rg, rgn); + } else { + pixman_region32_intersect(&rg, &dr->clip.e[dr->clip.count-1], rgn); + } + mui_clip_stack_add(&dr->clip, rg); + dr->pixman_clip_dirty = 1; + dr->cg_clip_dirty = 1; + + return dr->clip.count; +} + +int +mui_drawable_clip_substract_region( + mui_drawable_t * dr, + pixman_region32_t * rgn ) +{ + if (!dr || !rgn) + return 0; + + pixman_region32_t rg = {}; + if (dr->clip.count != 0) { + pixman_region32_subtract(&rg, &dr->clip.e[dr->clip.count-1], rgn); + } + mui_clip_stack_add(&dr->clip, rg); + dr->pixman_clip_dirty = 1; + dr->cg_clip_dirty = 1; + + return dr->clip.count; +} + +void +mui_drawable_clip_pop( + mui_drawable_t * dr ) +{ + if (!dr || !dr->clip.count) + return; + pixman_region32_fini(&dr->clip.e[dr->clip.count-1]); + dr->clip.count--; + dr->pixman_clip_dirty = 1; + dr->cg_clip_dirty = 1; +} diff --git a/libmui/mui/mui_font.c b/libmui/mui/mui_font.c new file mode 100644 index 0000000..606881d --- /dev/null +++ b/libmui/mui/mui_font.c @@ -0,0 +1,351 @@ +/* + * mui_font.c + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ + +#include +#include +#include +#include +#include + +#define STB_TRUETYPE_IMPLEMENTATION +#define STB_TTC_IMPLEMENTATION +#include "stb_ttc.h" + + +#define INCBIN_STYLE INCBIN_STYLE_SNAKE +#define INCBIN_PREFIX mui_ +#include "incbin.h" + +INCBIN(main_font, "fonts/Charcoal_mui.ttf"); +INCBIN(icon_font, "fonts/typicon.ttf"); +INCBIN(dingbat_font, "fonts/Dingbat.ttf"); + +#include "mui.h" + +mui_font_t * +mui_font_find( + mui_t *ui, + const char *name) +{ + mui_font_t *f; + TAILQ_FOREACH(f, &ui->fonts, self) { + if (!strcmp(f->name, name)) + return f; + } + return NULL; +} + +static void +_mui_font_pixman_prep( + mui_font_t *f) +{ + f->font.pix.bpp = 8; + f->font.pix.size.x = f->ttc.p_stride; + f->font.pix.size.y = f->ttc.p_height; + f->font.pix.row_bytes = f->ttc.p_stride; + f->font.pix.pixels = f->ttc.pixels; +} + +mui_font_t * +mui_font_from_mem( + mui_t *ui, + const char *name, + unsigned int size, + const void *font_data, + unsigned int font_size ) +{ + mui_font_t *f = calloc(1, sizeof(*f)); + f->name = strdup(name); + f->size = size; + stb_ttc_LoadFont(&f->ttc, font_data, font_size); + TAILQ_INSERT_TAIL(&ui->fonts, f, self); + printf("%s: Loaded font %s:%d\n", __func__, name, size); + + return f; +} + +void +mui_font_init( + mui_t *ui) +{ + printf("%s: Loading fonts\n", __func__); + mui_font_from_mem(ui, "main", 28, + mui_main_font_data, mui_main_font_size); + mui_font_from_mem(ui, "icon_large", 96, + mui_icon_font_data, mui_icon_font_size); + mui_font_from_mem(ui, "icon_small", 30, + mui_icon_font_data, mui_icon_font_size); +} + +void +mui_font_dispose( + mui_t *ui) +{ + mui_font_t *f; + while ((f = TAILQ_FIRST(&ui->fonts))) { + TAILQ_REMOVE(&ui->fonts, f, self); + stb_ttc_Free(&f->ttc); + free(f->name); + free(f); + } +} + +int +mui_font_text_measure( + mui_font_t *font, + const char *text, + stb_ttc_measure *m ) +{ + struct stb_ttc_info * ttc = &font->ttc; + float scale = stbtt_ScaleForPixelHeight(&ttc->font, font->size); + int w = stb_ttc_MeasureText(ttc, scale, text, m); + return w; +} + + +void +mui_font_text_draw( + mui_font_t *font, + mui_drawable_t *dr, + c2_pt_t where, + const char *text, + unsigned int text_len, + mui_color_t color) +{ + struct stb_ttc_info * ttc = &font->ttc; + unsigned int state = 0; + float scale = stbtt_ScaleForPixelHeight(&ttc->font, font->size); + double xpos = 0; + unsigned int last = 0; + unsigned int cp = 0; + + if (!text_len) + text_len = strlen(text); + mui_drawable_t * src = &font->font; + mui_drawable_t * dst = dr; + + pixman_color_t pc = PIXMAN_COLOR(color); + pixman_image_t * fill = pixman_image_create_solid_fill(&pc); + + where.y += font->ttc.ascent * scale; + for (unsigned int ch = 0; text[ch] && ch < text_len; ch++) { + if (stb_ttc__UTF8_Decode(&state, &cp, text[ch]) != UTF8_ACCEPT) + continue; + if (last) { + int kern = scale * stb_ttc__CodepointsGetKerning(ttc, last, cp); + xpos += kern; + } + last = cp; + int gl = stb_ttc__CodepointGetGlyph(ttc, cp); + if (gl == -1) + continue; + stb_ttc_g *gc = stb_ttc__ScaledGlyphGetCache(ttc, gl, scale); + if (!gc) + continue; + if (gc->p_y == (unsigned short) -1) + stb_ttc__ScaledGlyphRenderToCache(ttc, gc); +// int pxpos = gc->x0 + ((xpos + gc->lsb) * scale); + int pxpos = where.x + gc->x0 + ((xpos + 0) * scale); + // if (gc->lsb) + // printf("glyph %3d : %04x:%c lsb %d\n", ch, cp, cp < 32 ? '.' : cp, gc->lsb); + + int ph = gc->y1 - gc->y0; + int pw = gc->x1 - gc->x0; + _mui_font_pixman_prep(font); + pixman_image_composite32( + PIXMAN_OP_OVER, + fill, + mui_drawable_get_pixman(src), + mui_drawable_get_pixman(dst), + 0, 0, gc->p_x, gc->p_y, + pxpos, where.y + gc->y0, pw, ph); + xpos += gc->advance; + } + pixman_image_unref(fill); +} + +IMPLEMENT_C_ARRAY(mui_glyph_array); +IMPLEMENT_C_ARRAY(mui_glyph_line_array); + +void +mui_font_measure( + mui_font_t *font, + c2_rect_t bbox, + const char *text, + unsigned int text_len, + mui_glyph_line_array_t *lines, + uint16_t flags) +{ + struct stb_ttc_info * ttc = &font->ttc; + unsigned int state = 0; + float scale = stbtt_ScaleForPixelHeight(&ttc->font, font->size); + unsigned int last = 0; + unsigned int cp = 0; + + if (!text_len) + text_len = strlen(text); + + c2_pt_t where = {}; + unsigned int ch = 0; + int wrap_chi = 0; + int wrap_w = 0; + int wrap_count = 0; + do { + where.y += font->ttc.ascent * scale; + const mui_glyph_array_t zero = {}; + mui_glyph_line_array_push(lines, zero); + mui_glyph_array_t * line = &lines->e[lines->count - 1]; + line->x = 0; + line->y = where.y; + line->w = 0; + wrap_chi = ch; + wrap_w = 0; + wrap_count = 0; + for (;text[ch]; ch++) { + if (stb_ttc__UTF8_Decode(&state, &cp, text[ch]) != UTF8_ACCEPT) + continue; + if (last) { + int kern = scale * stb_ttc__CodepointsGetKerning(ttc, last, cp); + line->w += kern; + } + last = cp; +// printf("glyph %3d : %04x:%c\n", ch, cp, cp < 32 ? '.' : cp); + if (cp == '\n') { + ch++; + break; + } + if (isspace(cp) || ispunct(cp)) { + wrap_chi = ch; + wrap_w = line->w; + wrap_count = line->count; + } + int gl = stb_ttc__CodepointGetGlyph(ttc, cp); + if (gl == -1) + continue; + stb_ttc_g *gc = stb_ttc__ScaledGlyphGetCache(ttc, gl, scale); + if (!gc) + continue; + if (gc->p_y == (unsigned short) -1) + stb_ttc__ScaledGlyphRenderToCache(ttc, gc); + if (((line->w + gc->advance) * scale) > c2_rect_width(&bbox)) { + if (wrap_count) { + ch = wrap_chi + 1; + line->count = wrap_count; + line->w = wrap_w; + } + break; + } + line->w += gc->advance; + mui_glyph_array_push(line, gc); + }; + } while (text[ch] && ch < text_len); + int bh = 0; + for (int i = 0; i < (int)lines->count; i++) { + mui_glyph_array_t * line = &lines->e[i]; + bh = line->y - (font->ttc.descent * scale); + line->w *= scale; +// printf(" line %d y %3d size %d width %d\n", i, +// line->y, line->count, line->w); + } +// printf("box height is %d/%d\n", bh, c2_rect_height(&bbox)); + int ydiff = 0; + if (flags & MUI_TEXT_ALIGN_MIDDLE) { + ydiff = (c2_rect_height(&bbox) - bh) / 2; + } else if (flags & MUI_TEXT_ALIGN_BOTTOM) { + ydiff = c2_rect_height(&bbox) - bh; + } + for (int i = 0; i < (int)lines->count; i++) { + mui_glyph_array_t * line = &lines->e[i]; + line->y += ydiff; + if (flags & MUI_TEXT_ALIGN_RIGHT) { + line->x = c2_rect_width(&bbox) - line->w; + } else if (flags & MUI_TEXT_ALIGN_CENTER) { + line->x = (c2_rect_width(&bbox) - line->w) / 2; + } + } +} + +void +mui_font_measure_clear( + mui_glyph_line_array_t *lines) +{ + if (!lines) + return; + for (int i = 0; i < (int)lines->count; i++) { + mui_glyph_array_t * line = &lines->e[i]; + mui_glyph_array_free(line); + } + mui_glyph_line_array_free(lines); +} + + +void +mui_font_measure_draw( + mui_font_t *font, + mui_drawable_t *dr, + c2_rect_t bbox, + mui_glyph_line_array_t *lines, + mui_color_t color, + uint16_t flags) +{ + pixman_color_t pc = PIXMAN_COLOR(color); + pixman_image_t * fill = pixman_image_create_solid_fill(&pc); + struct stb_ttc_info * ttc = &font->ttc; + float scale = stbtt_ScaleForPixelHeight(&ttc->font, font->size); + + mui_drawable_t * src = &font->font; + mui_drawable_t * dst = dr; + + // all glyphs we need were loaded, update the pixman texture + _mui_font_pixman_prep(font); + + for (int li = 0; li < (int)lines->count; li++) { + mui_glyph_array_t * line = &lines->e[li]; + int xpos = 0;//where.x / scale; + for (int ci = 0; ci < (int)line->count; ci++) { + stb_ttc_g *gc = line->e[ci]; +// int pxpos = gc->x0 + ((xpos + gc->lsb) * scale); + int pxpos = gc->x0 + ((xpos + 0) * scale); + + int ph = gc->y1 - gc->y0; + int pw = gc->x1 - gc->x0; + pixman_image_composite32( + PIXMAN_OP_OVER, + fill, + mui_drawable_get_pixman(src), + mui_drawable_get_pixman(dst), + 0, 0, gc->p_x, gc->p_y, + bbox.l + line->x + pxpos, + bbox.t + line->y + gc->y0, pw, ph); + xpos += gc->advance; + } + } + pixman_image_unref(fill); +} + +void +mui_font_textbox( + mui_font_t *font, + mui_drawable_t *dr, + c2_rect_t bbox, + const char *text, + unsigned int text_len, + mui_color_t color, + uint16_t flags) +{ + mui_glyph_line_array_t lines = {}; + + if (!text_len) + text_len = strlen(text); + + mui_font_measure(font, bbox, text, text_len, &lines, flags); + + mui_font_measure_draw(font, dr, bbox, &lines, color, flags); + + mui_font_measure_clear(&lines); +} diff --git a/libmui/mui/mui_menus.c b/libmui/mui/mui_menus.c new file mode 100644 index 0000000..6e20639 --- /dev/null +++ b/libmui/mui/mui_menus.c @@ -0,0 +1,916 @@ +/* + * mui_menus.c + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ + +#include +#include +#include +#include +#include + +#include "mui.h" +#include "mui_priv.h" +#include "cg.h" + +#ifndef D +#define D(_x) //_x +#endif + +/* These are *window action* -- parameter 'target' is a mui_menu_t* */ +enum mui_menu_action_e { + MUI_MENU_ACTION_NONE = 0, + MUI_MENU_ACTION_OPEN = FCC('m','e','n','o'), + MUI_MENU_ACTION_CLOSE = FCC('m','e','n','c'), + MUI_MENU_ACTION_SELECT = FCC('m','e','n','s'), + // BEFORE a menu/submenu get opened/created + MUI_MENU_ACTION_PREPARE = FCC('m','e','p','r'), +}; + +struct mui_menu_control_t; +struct mui_menubar_t; + +typedef struct mui_menu_t { + mui_window_t win; + unsigned int click_inside : 1, + drag_ev : 1, + closing: 1, // prevent double-delete + timer_call_count : 2; // used by mui_menu_close_timer_cb + mui_control_t * highlighted; + mui_time_t sub_open_stamp; + // currently open menu, if any + struct mui_menu_control_t * sub; + struct mui_menubar_t * menubar; +} mui_menu_t; + +typedef struct mui_menubar_t { + mui_window_t win; + unsigned int click_inside : 1, + drag_ev : 1, + was_highlighted : 1, + timer_call_count : 2; // used by mui_menu_close_timer_cb + + // currently open menu title + struct mui_menu_control_t * selected_title; + // keep track of the menus, and their own submenus as they are being opened + // this is to keep track of the 'hierarchy' of menus, so that we can close + // them all when the user clicks outside of them, or release the mouse. + mui_menu_t * open[8]; + int open_count; + bool delayed_closing; +} mui_menubar_t; + +/* These are *control action* -- parameter is a menu title or popup, + * parameter is a mui_menu_control_t* + */ +enum mui_menu_control_action_e { + MUI_MENU_CONTROL_ACTION_NONE = 0, + // called *before* the window is opened, you can change the items state/list + MUI_MENU_CONTROL_ACTION_PREPARE = FCC('m','c','p','r'), + // when an item is selected. + MUI_MENU_CONTROL_ACTION_SELECT = FCC('m','c','s','l'), +}; + + +static mui_window_t * +_mui_menu_create( + mui_t * ui, + mui_menubar_t * mbar, + c2_pt_t origin, + mui_menu_item_t* items ); +static void +mui_menu_close( + mui_window_t * win ); +static bool +mui_wdef_menu( + struct mui_window_t * win, + uint8_t what, + void * param); +static bool +mui_wdef_menubar( + struct mui_window_t * win, + uint8_t what, + void * param); +static bool +mui_cdef_popup( + struct mui_control_t * c, + uint8_t what, + void * param); + +#define MUI_MENU_CLOSE_BLINK_DELAY (MUI_TIME_SECOND / 20) + + +// The menu bar title is a control, reset it's state to normal, and close +// any open menus (and their submenus, if any) +static bool +_mui_menubar_close_menu( + mui_menubar_t *mbar ) +{ + if (mbar->delayed_closing) + return false; + mbar->click_inside = false; + mui_control_set_state((mui_control_t*)mbar->selected_title, 0); + if (mbar->selected_title) + mbar->selected_title->menu_window = NULL; + if (!mbar->open_count) + return false; + mbar->selected_title = NULL; + for (int i = 0; i < mbar->open_count; i++) { + mui_menu_close(&mbar->open[i]->win); + mbar->open[i] = NULL; + } + mbar->open_count = 0; + return true; +} + +// close the submenu from a hierarchical menu item +static bool +_mui_menu_close_submenu( + mui_menu_t * menu ) +{ + mui_menu_control_t * sub = menu->sub; + if (!sub) + return false; + menu->sub = NULL; + mui_control_set_state((mui_control_t*)sub, 0); + if (sub->menu_window) { + mui_menu_close(sub->menu_window); + } + sub->menu_window = NULL; + + return true; +} + +/* + * This does the blinking of the selected item, and closes the menu + */ +static mui_time_t +mui_menu_close_timer_cb( + struct mui_t * mui, + mui_time_t now, + void * param) +{ + mui_menu_t * menu = param; + if (!menu->highlighted) { + printf("%s: no selected item, closing\n", __func__); + mui_window_dispose(&menu->win); + return 0; + } + menu->timer_call_count++; + mui_control_set_state(menu->highlighted, + menu->highlighted->state == MUI_CONTROL_STATE_CLICKED ? + MUI_CONTROL_STATE_NORMAL : MUI_CONTROL_STATE_CLICKED); + if (menu->timer_call_count == 3) { + // we are done! + mui_menuitem_control_t * item = (mui_menuitem_control_t*)menu->highlighted; + mui_window_action(&menu->win, MUI_MENU_ACTION_SELECT, &item->item); + // mui_menu_close_all(&menu->win); + if (menu->menubar) { + mui_menubar_t * mbar = (mui_menubar_t*)menu->menubar; + mui_window_action(&mbar->win, + MUI_MENUBAR_ACTION_SELECT, &item->item); + mbar->delayed_closing = false; + _mui_menubar_close_menu((mui_menubar_t*)menu->menubar); + } else + mui_menu_close(&menu->win); + return 0; + } + return MUI_MENU_CLOSE_BLINK_DELAY; +} +/* + * This restore the menubar to normal after a keystroke/menu equivalent was hit + */ +static mui_time_t +mui_menubar_unhighlight_cb( + struct mui_t * mui, + mui_time_t now, + void * param) +{ + mui_menubar_t * mbar = param; + mui_menubar_highlight(&mbar->win, false); + return 0; +} + +static int +mui_menu_action_cb( + mui_window_t * win, + void * cb_param, + uint32_t what, + void * param) // menu item here +{ + mui_menubar_t * mbar = cb_param; + D(printf("%s %s %4.4s\n", __func__, mbar->win.title, (char*)&what);) + switch (what) { + /* + * If a submenu is created, add ourselves as a callback to it as well, + * so we get notified when it is closed or an item is selected + */ + case MUI_MENU_ACTION_OPEN: { + mui_menu_t * submenu = param; + mui_window_set_action(&submenu->win, mui_menu_action_cb, mbar); + } break; + case MUI_MENU_ACTION_SELECT: { + mui_menu_item_t * item = param; + D(printf(" %s calling menubar action: %s\n", __func__, item->title);) + // this is typically application code! + mui_window_action(&mbar->win, MUI_MENUBAR_ACTION_SELECT, item); + } break; + case MUI_WINDOW_ACTION_CLOSE: { + D(mui_menu_t * menu = (mui_menu_t*)win;) + D(printf("%s closing %s\n", __func__, menu->win.title);) +// mui_menu_close_all(&mbar->win); + } break; + } + return 0; +} + +static int +mui_submenu_action_cb( + mui_window_t * win, + void * cb_param, + uint32_t what, + void * param) // menu item here +{ + mui_menu_t * menu = cb_param; + D(printf("%s %s %4.4s\n", __func__, menu->win.title, (char*)&what);) + switch (what) { + case MUI_MENU_ACTION_SELECT: { + mui_menu_item_t * item = param; + D(printf(" %s calling menubar action: %s\n", __func__, item->title);) + // this is typically application code! + mui_window_action(&menu->win, MUI_MENUBAR_ACTION_SELECT, item); + } break; + } + return 0; +} + + +static bool +mui_menubar_handle_mouse( + mui_menubar_t * mbar, + mui_event_t * ev ) +{ + mui_window_t * win = &mbar->win; + + bool inside = c2_rect_contains_pt(&win->frame, &ev->mouse.where); + mui_control_t * c = inside ? mui_control_locate(win, ev->mouse.where) : NULL; + switch (ev->type) { + case MUI_EVENT_BUTTONUP: { + D(printf("%s up drag %d click in:%d high:%d was:%d\n", __func__, + mbar->drag_ev, mbar->click_inside, + mbar->selected_title ? 1 : 0, mbar->was_highlighted);) + if (mbar->drag_ev == 0 && mbar->click_inside) { + if (mbar->was_highlighted) { + return _mui_menubar_close_menu(mbar); + } else + mbar->click_inside = false; + return true; + } else if (mbar->drag_ev == 0 && !mbar->click_inside) { + return false; + } + return _mui_menubar_close_menu(mbar); + } break; + case MUI_EVENT_DRAG: + if (!mbar->click_inside) + break; + mbar->drag_ev = 1; + // fallthrough + case MUI_EVENT_BUTTONDOWN: { + // printf("%s button %d\n", __func__, ev->mouse.button); + if (ev->mouse.button > 1) + break; + // save click, for detecting click+drag vs sticky menus + if (ev->type == MUI_EVENT_BUTTONDOWN) { + D(printf("%s click inside %d\n", __func__, inside);) + mbar->drag_ev = 0; + mbar->click_inside = inside; + mbar->was_highlighted = mbar->selected_title != NULL; + } + if (c && mui_control_get_state(c) != MUI_CONTROL_STATE_DISABLED) { + if (mbar->selected_title && + c != (mui_control_t*)mbar->selected_title) + _mui_menubar_close_menu(mbar); + mbar->click_inside = true; + mui_control_set_state(c, MUI_CONTROL_STATE_CLICKED); + mui_menu_control_t *title = (mui_menu_control_t*)c; + mbar->selected_title = title; + if (mui_control_get_type(c) == MUI_CONTROL_MENUTITLE) { + if (title->menu_window == NULL) { + title->menu_window = _mui_menu_create( + win->ui, mbar, C2_PT(c->frame.l, c->frame.b), + title->menu.e); + } + } + return true; + } + } break; + } + return false; +} + +/* + * Look into a menu item list for a key_equ match -- also handle + * recursive submenus, return true the match itself, if found + */ +static mui_menu_item_t * +mui_menu_handle_keydown( + mui_menu_item_t * items, + mui_event_t *ev ) +{ +// D(printf("%s\n", __func__);) + for (int ii = 0; items[ii].title; ii++) { + mui_menu_item_t * item = &items[ii]; + if (item->submenu) { + mui_menu_item_t * sub = mui_menu_handle_keydown(item->submenu, ev); + if (sub) + return sub; + continue; + } + if (!item->key_equ.value || item->disabled) + continue; + if (mui_event_match_key(ev, item->key_equ)) { + // printf("%s KEY MATCH %s\n", __func__, item->title); + return item; + } + } + return NULL; +} + +static bool +mui_menubar_handle_keydown( + mui_menubar_t * mbar, + mui_event_t *ev ) +{ + // iterate all the menus, check any with a key_equ that is + // non-zero, if would, highlight that menu (temporarily) + // and call the action function witht that menu item. score! + mui_window_t * win = &mbar->win; + for (mui_control_t * c = TAILQ_FIRST(&win->controls); + c; c = TAILQ_NEXT(c, self)) { + if (c->type != MUI_CONTROL_MENUTITLE) + continue; + mui_menu_control_t * title = (mui_menu_control_t*)c; + // printf("%s title %s\n", __func__, c->title); + mui_menu_item_t * item = mui_menu_handle_keydown( + title->menu.e, ev); + if (item) { + // printf("%s KEY MATCH %s\n", __func__, item->title); + mui_control_set_state(c, MUI_CONTROL_STATE_CLICKED); + mui_window_action(&mbar->win, + MUI_MENUBAR_ACTION_SELECT, item); + mui_timer_register(win->ui, mui_menubar_unhighlight_cb, mbar, + MUI_MENU_CLOSE_BLINK_DELAY); + return true; + } + } + return false; +} + +static bool +mui_wdef_menubar( + struct mui_window_t * win, + uint8_t what, + void * param) +{ + mui_menubar_t * mbar = (mui_menubar_t*)win; + switch (what) { + case MUI_WDEF_DRAW: { + mui_drawable_t * dr = param; + mui_wdef_menubar_draw(win, dr); + } break; + case MUI_WDEF_EVENT: { + mui_event_t *ev = param; + switch (ev->type) { + case MUI_EVENT_BUTTONUP: + case MUI_EVENT_DRAG: + case MUI_EVENT_BUTTONDOWN: { + if (mui_menubar_handle_mouse(mbar, ev)) { + // printf("%s handled\n", __func__); + return true; + } + } break; + case MUI_EVENT_KEYDOWN: { + // printf("%s keydown %c\n", __func__, ev->key.key); + if (mui_menubar_handle_keydown(mbar, ev)) + return true; + } break; + } + } break; + } + return false; +} + +static bool +mui_menu_handle_mouse( + mui_menu_t * menu, + mui_event_t * ev ) +{ + mui_window_t * win = &menu->win; + +// bool inside = c2_rect_contains_pt(&win->frame, &ev->mouse.where); + bool is_front = mui_window_isfront(win); + mui_control_t * c = mui_control_locate(win, ev->mouse.where); + switch (ev->type) { + case MUI_EVENT_BUTTONUP: { + D(printf("%s (%s) up drag %d\n", __func__, win->title, menu->drag_ev);) + if (menu->drag_ev == 0) { + // return true; + } + mui_menubar_t * mbar = (mui_menubar_t*)menu->menubar; + if (menu->highlighted && + menu->highlighted->type != MUI_CONTROL_SUBMENUITEM) { + /* + * This tells the normal closing code that we are + * taking care of the closing with the timer, and not *now* + */ + if (mbar) + mbar->delayed_closing = true; + menu->timer_call_count = 0; + mui_timer_register(win->ui, mui_menu_close_timer_cb, menu, + MUI_MENU_CLOSE_BLINK_DELAY); + } else { + menu->highlighted = NULL; + if (mbar) + _mui_menubar_close_menu(mbar); + else + mui_menu_close(win); + } + } break; + case MUI_EVENT_DRAG: + /* + * if we've just opened a submenu, make it 'sticky' for a while + * so that the user can drag the mouse to it without it closing + */ + if (!is_front && + (mui_get_time() - menu->sub_open_stamp) < (MUI_TIME_SECOND / 2)) + return false; + menu->drag_ev = 1; + // fallthrough + case MUI_EVENT_BUTTONDOWN: { + // save click, for detecting click+drag vs sticky menus + if (ev->type == MUI_EVENT_BUTTONDOWN) { + menu->drag_ev = 0; + } + // printf("%s in:%d c:%s\n", __func__, inside, c ? c->title : ""); + if (c && mui_control_get_state(c) != MUI_CONTROL_STATE_DISABLED) { + if (menu->sub && c != (mui_control_t*)menu->sub) + _mui_menu_close_submenu(menu); + if (menu->highlighted && c != menu->highlighted) + mui_control_set_state(menu->highlighted, 0); + mui_control_set_state(c, MUI_CONTROL_STATE_CLICKED); + menu->highlighted = c; + if (c->type == MUI_CONTROL_SUBMENUITEM) { + mui_menu_control_t *title = (mui_menu_control_t*)c; + if (title->menu_window == NULL) { + c2_pt_t where = C2_PT(c->frame.r, c->frame.t); + c2_pt_offset(&where, win->content.l, win->content.t); + title->menu_window = _mui_menu_create( + win->ui, + (mui_menubar_t*)menu->menubar, where, + title->menu.e); + menu->sub = title; + menu->sub_open_stamp = mui_get_time(); + mui_window_action(&menu->win, MUI_MENU_ACTION_OPEN, + title->menu_window); + mui_window_set_action(title->menu_window, + mui_submenu_action_cb, menu); + } + } + } else { + if (!menu->sub) { + if (menu->highlighted) + mui_control_set_state(menu->highlighted, 0); + menu->highlighted = NULL; + } + } + } break; + } + return false; +} + +static bool +mui_wdef_menu( + struct mui_window_t * win, + uint8_t what, + void * param) +{ + mui_menu_t * menu = (mui_menu_t*)win; + switch (what) { + case MUI_WDEF_DISPOSE: { + _mui_menu_close_submenu(menu); + } break; + case MUI_WDEF_DRAW: { + mui_drawable_t * dr = param; + // shared drawing code for the menubar and menu + mui_wdef_menubar_draw(win, dr); + } break; + case MUI_WDEF_EVENT: { + mui_event_t *ev = param; + switch (ev->type) { + case MUI_EVENT_BUTTONUP: + case MUI_EVENT_DRAG: + case MUI_EVENT_BUTTONDOWN: { + if (mui_menu_handle_mouse(menu, ev)) + return true; + } break; + } + } break; + } + return false; +} + + +mui_window_t * +mui_menubar_new( + mui_t * ui ) +{ + mui_font_t * main = mui_font_find(ui, "main"); + c2_rect_t mbf = C2_RECT_WH(0, 0, ui->screen_size.x, main->size + 4); + +// printf("%s frame %s\n", __func__, c2_rect_as_str(&mbf)); + mui_menubar_t * mbar = (mui_menubar_t*)mui_window_create( + ui, mbf, + mui_wdef_menubar, MUI_WINDOW_MENUBAR_LAYER, + "Menubar", sizeof(*mbar)); + ui->menubar = &mbar->win; + return &mbar->win; +} + +mui_window_t * +mui_menubar_get( + mui_t * ui ) +{ + return ui ? ui->menubar : NULL; +} + +bool +mui_menubar_window( + mui_window_t * win) +{ + return win && win->wdef == mui_wdef_menubar; +} + +mui_control_t * +mui_menubar_add_simple( + mui_window_t *win, + const char * title, + uint32_t menu_uid, + mui_menu_item_t * items ) +{ + mui_font_t * main = mui_font_find(win->ui, "main"); + stb_ttc_measure m = {}; + mui_font_text_measure(main, title, &m); + + int title_width = m.x1 - m.x0 + (main->size / 2); + c2_rect_t title_rect = { .t = 2 }; + + mui_control_t * last = TAILQ_LAST(&win->controls, controls); + if (last) { + c2_rect_offset(&title_rect, last->frame.r, 0); + } else + title_rect.l = 4; + title_rect.r = title_rect.l + title_width + 6; + title_rect.b = win->content.b + 2;// title_rect.t + m.ascent - m.descent; + + mui_control_t * c = mui_control_new( + win, MUI_CONTROL_MENUTITLE, mui_cdef_popup, + title_rect, title, menu_uid, + sizeof(mui_menu_control_t)); + //printf("%s title %s rect %s\n", __func__, title, c2_rect_as_str(&title_rect)); + + mui_menu_control_t *menu = (mui_menu_control_t*)c; + menu->menubar = win; + /* + * We do not clone the items, so they must be static + * from somewhere -- we do not free them either. + */ + int sub_count = 0; + for (int ii = 0; items[ii].title; ii++) + sub_count++; + menu->menu.count = sub_count; + menu->menu.e = items; + menu->menu.read_only = 1; + + return c; +} + +mui_window_t * +mui_menubar_highlight( + mui_window_t * win, + bool ignored ) +{ +// mui_menubar_t * mbar = (mui_menubar_t*)win; + mui_control_t * c = NULL; + TAILQ_FOREACH(c, &win->controls, self) { + if (c->type == MUI_CONTROL_MENUTITLE || + mui_control_get_state(c)) { + mui_control_set_state(c, 0); + } + } + return win; +} + + +static c2_rect_t +mui_menu_get_enclosing_rect( + mui_t * ui, + mui_menu_item_t * items) +{ + c2_rect_t frame = {}; + + mui_font_t * main = mui_font_find(ui, "main"); + stb_ttc_measure m = {}; + for (int i = 0; items[i].title; i++) { + items[i].location = frame.br; + if (items[i].title && items[i].title[0] != '-') { + mui_font_text_measure(main, items[i].title, &m); + int title_width = main->size + m.x1 - m.x0 ; + + if (items[i].kcombo[0]) { + mui_font_text_measure(main, items[i].kcombo, &m); + title_width += m.x1 - m.x0 + main->size; + } else + title_width += main->size; + + if (title_width > frame.r) + frame.r = title_width; + frame.b += main->size + 2; + } else { + frame.b += main->size / 4; + } + } + return frame; +} + +/* + * Given a list of items, create the actual windows & controls to show + * it onscreen + */ +static mui_window_t * +_mui_menu_create( + mui_t * ui, + mui_menubar_t * mbar, + c2_pt_t origin, + mui_menu_item_t * items ) +{ + if (mbar) + mui_window_action(&mbar->win, + MUI_MENUBAR_ACTION_PREPARE, + items); + + c2_rect_t frame = mui_menu_get_enclosing_rect(ui, items); + c2_rect_t on_screen = frame; + c2_rect_offset(&on_screen, origin.x, origin.y); + + /* Make sure the whole popup is on screen */ + /* TODO: handle very large popups, that'll be considerably more + * complicated to do, with the arrows, scrolling and stuff */ + c2_rect_t screen = C2_RECT_WH(0, 0, ui->screen_size.x, ui->screen_size.y); + if (on_screen.r > screen.r) + c2_rect_offset(&on_screen, screen.r - on_screen.r, 0); + if (on_screen.b > screen.b) + c2_rect_offset(&on_screen, 0, screen.b - on_screen.b); + if (on_screen.t < screen.t) + c2_rect_offset(&on_screen, 0, screen.t - on_screen.t); + if (on_screen.l < screen.l) + c2_rect_offset(&on_screen, screen.l - on_screen.l, 0); + + mui_menu_t * menu = (mui_menu_t*)mui_window_create( + ui, on_screen, + mui_wdef_menu, MUI_WINDOW_MENU_LAYER, + items[0].title, sizeof(*menu)); + if (mbar) { + mbar->open[mbar->open_count++] = menu; + menu->menubar = mbar; + } + /* Walk all the items in out static structure, and create the controls + * for each of them with their own corresponding item */ + for (int i = 0; items[i].title; i++) { + mui_menu_item_t * item = &items[i]; + item->index = i; + c2_rect_t title_rect = frame; + title_rect.t = item->location.y - 1; + if (items[i+1].title) + title_rect.b = items[i+1].location.y; + else + title_rect.b = frame.b; + + mui_control_t * c = NULL; + if (item->submenu) { + // printf("%s submenu %s\n", __func__, item->title); + c = mui_control_new( + &menu->win, + MUI_CONTROL_SUBMENUITEM, + mui_cdef_popup, + title_rect, item->title, item->uid, + sizeof(mui_menu_control_t)); + mui_menu_control_t *sub = (mui_menu_control_t*)c; + /* Note: menuitems aren't copied, we use the same static structure */ + sub->menu.count = 0; + for (int i = 0; item->submenu[i].title; i++) + sub->menu.count++; + sub->menu.e = item->submenu; + sub->menu.read_only = 1; + } else { + c = mui_control_new( + &menu->win, + MUI_CONTROL_MENUITEM, + mui_cdef_popup, + title_rect, item->title, item->uid, + sizeof(mui_menuitem_control_t)); + } + if (item->disabled) + mui_control_set_state(c, MUI_CONTROL_STATE_DISABLED); + // both item and submenu share a menuitem control + mui_menuitem_control_t *mic = (mui_menuitem_control_t*)c; + mic->item = *item; + } + return &menu->win; +} + +static void +mui_menu_close( + mui_window_t * win ) +{ + mui_menu_t * menu = (mui_menu_t*)win; + mui_menubar_t * mbar = (mui_menubar_t*)menu->menubar; // can be NULL + + menu->highlighted = NULL; + if (mbar && mbar->open_count) { + mbar->open[mbar->open_count-1] = NULL; + mbar->open_count--; + } + mui_window_dispose(win); +} + + +static int +mui_popupmenu_action_cb( + mui_window_t * win, + void * cb_param, + uint32_t what, + void * param) // popup control here +{ + mui_menu_control_t * pop = cb_param; + printf("%s %4.4s\n", __func__, (char*)&what); + switch (what) { + case MUI_MENU_ACTION_SELECT: { + mui_menu_item_t * item = param; + pop->item.control.value = item->index; + mui_control_inval(&pop->item.control); + mui_control_action(&pop->item.control, + MUI_CONTROL_ACTION_VALUE_CHANGED, NULL); + } break; + } + return 0; +} + +static bool +mui_popupmenu_handle_mouse( + mui_menu_control_t * pop, + mui_event_t * ev ) +{ + mui_control_t * c = &pop->item.control; + switch (ev->type) { + case MUI_EVENT_BUTTONUP: { + printf("%s up has popup %d\n", __func__, pop->menu_window != NULL); + if (pop->menu_window) { +// mui_menu_close(pop->menu_window); + pop->menu_window = NULL; + } + mui_control_set_state(c, 0); + } break; + case MUI_EVENT_BUTTONDOWN: { + mui_control_set_state(c, + ev->type != MUI_EVENT_BUTTONUP ? + MUI_CONTROL_STATE_CLICKED : 0); + if (!pop->menu_window) { + c2_pt_t loc = pop->menu_frame.tl; + c2_pt_offset(&loc, c->win->content.l, c->win->content.t); + c2_pt_offset(&loc, 0, -pop->menu.e[c->value].location.y); + pop->menu_window = _mui_menu_create( + c->win->ui, NULL, loc, + pop->menu.e); + mui_window_set_action(pop->menu_window, + mui_popupmenu_action_cb, pop); + // pass the mousedown to the new popup + // mui_window_handle_mouse(pop->menu_window, ev); + } + mui_control_inval(c); + } break; + } + return true; +} + +static bool +mui_cdef_popup( + struct mui_control_t * c, + uint8_t what, + void * param) +{ + switch (what) { + case MUI_CDEF_INIT: { + } break; + case MUI_CDEF_DISPOSE: + switch (c->type) { + case MUI_CONTROL_POPUP: + case MUI_CONTROL_MENUTITLE: { + mui_menu_control_t *pop = (mui_menu_control_t*)c; + if (pop->menu_window) { + mui_menu_close(pop->menu_window); + pop->menu_window = NULL; + } + mui_menu_items_clear(&pop->menu); + if (!pop->menu.read_only) + mui_menu_items_free(&pop->menu); + } break; + } + break; + case MUI_CDEF_DRAW: { + mui_drawable_t * dr = param; + switch (c->type) { + case MUI_CONTROL_POPUP: + mui_popuptitle_draw(c->win, c, dr); + break; + case MUI_CONTROL_MENUTITLE: + mui_menutitle_draw(c->win, c, dr); + break; + case MUI_CONTROL_MENUITEM: + case MUI_CONTROL_SUBMENUITEM: + mui_menuitem_draw(c->win, c, dr); + break; + } + } break; + case MUI_CDEF_EVENT: { + mui_event_t *ev = param; + // printf("%s event %d\n", __func__, ev->type); + switch (ev->type) { + case MUI_EVENT_BUTTONUP: + case MUI_EVENT_DRAG: + case MUI_EVENT_BUTTONDOWN: { + switch (c->type) { + case MUI_CONTROL_POPUP: { + mui_menu_control_t *pop = (mui_menu_control_t*)c; + return mui_popupmenu_handle_mouse(pop, ev); + } break; + case MUI_CONTROL_MENUTITLE: + // mui_menutitle_draw(c->win, c, dr); + break; + } + } break; + } + } break; + } + return false; +} + +mui_control_t * +mui_popupmenu_new( + mui_window_t * win, + c2_rect_t frame, + const char * title, + uint32_t uid ) +{ + return mui_control_new( + win, MUI_CONTROL_POPUP, mui_cdef_popup, + frame, title, uid, sizeof(mui_menu_control_t)); +} + +mui_menu_items_t * +mui_popupmenu_get_items( + mui_control_t * c) +{ + if (!c) + return NULL; + if (c->type != MUI_CONTROL_POPUP && c->type != MUI_CONTROL_MENUTITLE) { + printf("%s: not a popup or menutitle\n", __func__); + return NULL; + } + mui_menu_control_t *pop = (mui_menu_control_t*)c; + return &pop->menu; +} + +void +mui_popupmenu_prepare( + mui_control_t * c) +{ + mui_menu_control_t *pop = (mui_menu_control_t*)c; + if (pop->menu_window) { + mui_window_dispose(pop->menu_window); + pop->menu_window = NULL; + } + c2_rect_t frame = mui_menu_get_enclosing_rect(c->win->ui, pop->menu.e); + pop->menu_frame = frame; + c2_rect_offset(&frame, c->frame.l, c->frame.t); + frame.r += 32; // add the popup symbol width + if (c2_rect_width(&frame) < c2_rect_width(&c->frame)) { + c2_rect_offset(&frame, (c2_rect_width(&c->frame) / 2) - + (c2_rect_width(&frame) / 2), 0); + } + pop->menu_frame = frame; + c->value = 0; + mui_control_inval(c); +} diff --git a/libmui/mui/mui_menus_draw.c b/libmui/mui/mui_menus_draw.c new file mode 100644 index 0000000..8d7e1ad --- /dev/null +++ b/libmui/mui/mui_menus_draw.c @@ -0,0 +1,240 @@ +/* + * mui_menus_draw.c + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ + +#include +#include +#include +#include +#include + +#include "mui.h" +#include "mui_priv.h" +#include "cg.h" + + +/* + * Menubar/menus frames is easy as pie -- just a framed rectangle + */ +void +mui_wdef_menubar_draw( + struct mui_window_t * win, + mui_drawable_t * dr) +{ + c2_rect_t content = win->frame; + win->content = content; + + struct cg_ctx_t * cg = mui_drawable_get_cg(dr); + + mui_color_t frameColor = MUI_COLOR(0x000000ff); + mui_color_t contentFill = MUI_COLOR(0xf0f0f0ff); + cg_set_line_width(cg, 1); + cg_rectangle(cg, win->frame.l + 0.5f, win->frame.t + 0.5f, + c2_rect_width(&win->frame) - 1, c2_rect_height(&win->frame) - 1); + cg_set_source_color(cg, &CG_COLOR(contentFill)); + cg_fill_preserve(cg); + cg_set_source_color(cg, &CG_COLOR(frameColor)); + cg_stroke(cg); +} + + +extern const mui_control_color_t mui_control_color[MUI_CONTROL_STATE_COUNT]; + +void +mui_menutitle_draw( + mui_window_t * win, + mui_control_t * c, + mui_drawable_t *dr ) +{ + c2_rect_t f = c->frame; + c2_rect_offset(&f, win->content.l, win->content.t); + + struct cg_ctx_t * cg = mui_drawable_get_cg(dr); + + mui_font_t * main = mui_font_find(win->ui, "main"); + stb_ttc_measure m = {}; + mui_font_text_measure(main, c->title, &m); + + int title_width = m.x1 - m.x0; + c2_rect_t title = f; + title.r = title.l + title_width + 1; + title.b = title.t + m.ascent - m.descent; + c2_rect_offset(&title, //-m.x0 + + (int)((c2_rect_width(&f) / 2) - (c2_rect_width(&title)) / 2), + (c2_rect_height(&f) / 2) - (c2_rect_height(&title) / 2)); + mui_drawable_clip_push(dr, &f); + uint32_t state = mui_control_get_state(c); + if (state) { + cg_set_source_color(cg, &CG_COLOR(mui_control_color[state].fill)); + cg_rectangle(cg, f.l, f.t, c2_rect_width(&f), c2_rect_height(&f)); + cg_fill(cg); + } + mui_font_text_draw(main, dr, + C2_PT(title.l, title.t), c->title, strlen(c->title), + mui_control_color[state].text); + mui_drawable_clip_pop(dr); +} + + +static void +mui_menuitem_get_locations( + mui_t * ui, + c2_rect_t * frame, + mui_menu_item_t * item, + c2_pt_t out[3] ) +{ + mui_font_t * main = mui_font_find(ui, "main"); + + stb_ttc_measure m = {}; + mui_font_text_measure(main, item->title, &m); + + c2_rect_t title = *frame; + title.b = title.t + m.ascent - m.descent; + c2_rect_offset(&title, 0, + (c2_rect_height(frame) / 2) - (c2_rect_height(&title) / 2)); + + if (item->icon[0]) { + title.l += 6; + mui_font_t * icons = mui_font_find(ui, "icon_small"); + + mui_font_text_measure(icons, item->icon, &m); + c2_pt_t loc = title.tl; + loc.x = loc.x + (icons->size / 2) - ((m.x1 - m.x0) / 2); + out[0] = loc; + title.l += 6; + } else if (item->mark[0]) { + mui_font_text_measure(main, item->mark, &m); + c2_pt_t loc = title.tl; + loc.x = loc.x + (main->size / 2) - ((m.x1 - m.x0) / 2); + out[0] = loc; + } + title.l += main->size; + out[1] = title.tl; + + if (item->kcombo[0]) { + mui_font_text_measure(main, item->kcombo, &m); + + c2_pt_t loc = C2_PT(title.r - m.x1 - m.x0 - (main->size/3), title.t); + out[2] = loc; + } +} + + +extern const mui_control_color_t mui_control_color[MUI_CONTROL_STATE_COUNT]; + +void +mui_menuitem_draw( + mui_window_t * win, + mui_control_t * c, + mui_drawable_t *dr ) +{ + c2_rect_t f = c->frame; + c2_rect_offset(&f, win->content.l, win->content.t); + + struct cg_ctx_t * cg = mui_drawable_get_cg(dr); + mui_drawable_clip_push(dr, &f); + + mui_font_t * main = TAILQ_FIRST(&win->ui->fonts); + if (c->title && c->title[0] != '-') { + c2_pt_t loc[3]; + mui_menuitem_control_t *mic = (mui_menuitem_control_t*)c; + mui_menuitem_get_locations(win->ui, &f, &mic->item, loc); + + uint32_t state = mui_control_get_state(c); + if (state && state != MUI_CONTROL_STATE_DISABLED) { + c2_rect_t b = f; + c2_rect_inset(&b, 1, 0); + cg_set_source_color(cg, &CG_COLOR(mui_control_color[state].fill)); + cg_rectangle(cg, b.l, b.t, c2_rect_width(&b), c2_rect_height(&b)); + cg_fill(cg); + } + if (mic->item.icon[0]) { + mui_font_t * icons = mui_font_find(win->ui, "icon_small"); + mui_font_text_draw(icons, dr, + loc[0], mic->item.icon, 0, + mui_control_color[state].text); + } else if (mic->item.mark[0]) { + mui_font_text_draw(main, dr, + loc[0], mic->item.mark, 0, + mui_control_color[state].text); + } + mui_font_text_draw(main, dr, + loc[1], mic->item.title, 0, + mui_control_color[state].text); + + if (mic->item.kcombo[0]) { + mui_font_text_draw(main, dr, + loc[2], mic->item.kcombo, 0, + mui_control_color[state].text); + } + } else { + cg_move_to(cg, f.l, f.t + c2_rect_height(&f) / 2); + cg_line_to(cg, f.r, f.t + c2_rect_height(&f) / 2); + mui_color_t decoColor = MUI_COLOR(0x666666ff); + + cg_set_source_color(cg, &CG_COLOR(decoColor)); + cg_stroke(cg); + } + mui_drawable_clip_pop(dr); +} + + +void +mui_popuptitle_draw( + mui_window_t * win, + mui_control_t * c, + mui_drawable_t *dr ) +{ + mui_menu_control_t *pop = (mui_menu_control_t*)c; + c2_rect_t f = c->frame; + if (c2_rect_width(&pop->menu_frame) && + c2_rect_width(&pop->menu_frame) < c2_rect_width(&f)) { + f = pop->menu_frame; + f.b = c->frame.b; + } + c2_rect_offset(&f, win->content.l, win->content.t); + + mui_font_t * main = TAILQ_FIRST(&win->ui->fonts); + mui_font_t * icons = mui_font_find(win->ui, "icon_small"); + uint32_t state = mui_control_get_state(c); + + mui_drawable_clip_push(dr, &f); + struct cg_ctx_t * cg = mui_drawable_get_cg(dr); + c2_rect_t inner = f; + c2_rect_inset(&inner, 1, 1); + cg_set_line_width(cg, 2); + cg_round_rectangle(cg, inner.l, inner.t, + c2_rect_width(&inner), c2_rect_height(&inner), 3, 3); + cg_set_source_color(cg, &CG_COLOR(mui_control_color[state].fill)); + cg_fill_preserve(cg); + cg_set_source_color(cg, &CG_COLOR(mui_control_color[state].frame)); + cg_stroke(cg); + cg_move_to(cg, inner.r - 32, inner.t + 2); + cg_line_to(cg, inner.r - 32, inner.b - 2); + mui_color_t decoColor = MUI_COLOR(0x666666ff); + cg_set_source_color(cg, &CG_COLOR(decoColor)); + cg_stroke(cg); + + if (pop->menu.count) { + mui_menu_item_t item = pop->menu.e[c->value]; + c2_pt_t loc[3]; + c2_rect_offset(&f, 0, -1); + mui_menuitem_get_locations(win->ui, &f, &item, loc); + + if (item.icon[0]) + mui_font_text_draw(icons, dr, + loc[0], item.icon, 0, + mui_control_color[state].text); + mui_font_text_draw(main, dr, + loc[1], item.title, 0, + mui_control_color[state].text); + } + mui_font_text_draw(icons, dr, + C2_PT(inner.r - 32 + 8, inner.t + 2), "", 0, + mui_control_color[state].text); + mui_drawable_clip_pop(dr); +} diff --git a/libmui/mui/mui_priv.h b/libmui/mui/mui_priv.h new file mode 100644 index 0000000..b321817 --- /dev/null +++ b/libmui/mui/mui_priv.h @@ -0,0 +1,87 @@ +/* + * mui_priv.h + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ + +/* + * This are functions that are used internally by the MUI library itself, + * but are not part of the public API. + */ + +#pragma once + +#include "mui.h" + +// private, used my mui_draw() -- do not use outside of this context +void +mui_window_draw( + mui_window_t *win, + mui_drawable_t *dr); +bool +mui_window_handle_mouse( + mui_window_t *win, + mui_event_t *event); +bool +mui_window_handle_keyboard( + mui_window_t *win, + mui_event_t *event); + + +void +mui_control_draw( + mui_window_t * win, + mui_control_t * c, + mui_drawable_t *dr ); + +bool +mui_control_event( + mui_control_t * c, + mui_event_t * ev ); + + +/* This is common to: + * - Menu titles in the menubar + * - Menu items in a popup menu + * - Menu items pointing to a sub-menu + */ +typedef struct mui_menuitem_control_t { + mui_control_t control; + mui_menu_item_t item; +} mui_menuitem_control_t; + +/* this is for menu title, popup menu labels, and hierarchical menu items */ +typedef struct mui_menu_control_t { + mui_menuitem_control_t item; + mui_menu_items_t menu; + c2_rect_t menu_frame; + mui_window_t * menubar; + mui_window_t * menu_window; // when open +} mui_menu_control_t; + +void +mui_wdef_menubar_draw( + struct mui_window_t * win, + mui_drawable_t * dr); +void +mui_popuptitle_draw( + mui_window_t * win, + mui_control_t * c, + mui_drawable_t *dr ); +void +mui_menuitem_draw( + mui_window_t * win, + mui_control_t * c, + mui_drawable_t *dr ); +void +mui_menutitle_draw( + mui_window_t * win, + mui_control_t * c, + mui_drawable_t *dr ); + +// return true if window win is the menubar +bool +mui_menubar_window( + mui_window_t * win); diff --git a/libmui/mui/mui_stdfile.c b/libmui/mui/mui_stdfile.c new file mode 100644 index 0000000..9aafd6c --- /dev/null +++ b/libmui/mui/mui_stdfile.c @@ -0,0 +1,409 @@ +/* + * mui_stdfile.c + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ + +#define _GNU_SOURCE +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "mui.h" +#include "c2_geometry.h" + +#include +#include + +DECLARE_C_ARRAY(char*, string_array, 2); +IMPLEMENT_C_ARRAY(string_array); + +typedef struct mui_stdfile_t { + mui_window_t win; + mui_control_t * ok, *cancel, *home, *root; + mui_control_t * listbox, *popup; + char * pref_file; // pathname we put last path used + char * re_pattern; + char * current_path; + char * selected_path; + regex_t re; + string_array_t pop_path; +} mui_stdfile_t; + +enum { + MUI_STD_FILE_PART_FRAME = 0, + MUI_STD_FILE_PART_OK, + MUI_STD_FILE_PART_CANCEL, + MUI_STD_FILE_PART_HOME, + MUI_STD_FILE_PART_ROOT, + MUI_STD_FILE_PART_LISTBOX, + MUI_STD_FILE_PART_POPUP, + MUI_STD_FILE_PART_COUNT, +}; + +static int +_mui_stdfile_sort_cb( + const void * a, + const void * b) +{ + const mui_listbox_elem_t * ea = a; + const mui_listbox_elem_t * eb = b; + #if 0 + if (ea->icon == MUI_ICON_FOLDER && eb->icon != MUI_ICON_FOLDER) + return -1; + if (ea->icon != MUI_ICON_FOLDER && eb->icon == MUI_ICON_FOLDER) + return 1; + #endif + return strcmp(ea->elem, eb->elem); +} + +static int +_mui_stdfile_populate( + mui_stdfile_t * std, + const char * path) +{ + if (std->current_path && !strcmp(std->current_path, path)) + return 0; + + printf("%s %s\n", __func__, path); + errno = 0; + DIR * dir = opendir(path); + if (!dir) { + // show an alert of some sort + char * msg = NULL; + asprintf(&msg, "%s\n%s", path, + strerror(errno)); + mui_alert(std->win.ui, C2_PT(0,0), + "Could not open directory", + msg, + MUI_ALERT_FLAG_OK); + return -1; + } + if (std->current_path) + free(std->current_path); + std->current_path = strdup(path); + path = NULL; // this COULD be in the list we are now deleting! + for (int i = 0; i < (int)std->pop_path.count; i++) + free(std->pop_path.e[i]); + std->pop_path.count = 0; + + mui_control_t *pop = std->popup; + mui_menu_items_t * items = mui_popupmenu_get_items(pop); + mui_menu_items_clear(items); + char * p = strdup(std->current_path); + char * d; + const char *home = getenv("HOME"); + int item_id = 1000; + while ((d = basename(p)) != NULL) { + mui_menu_item_t i = { + .title = strdup(d), + .uid = item_id++, + }; + if (!strcmp(d, "/")) + strcpy(i.icon, MUI_ICON_ROOT); + else if (home && !strcmp(p, home)) + strcpy(i.icon, MUI_ICON_HOME); + else + strcpy(i.icon, MUI_ICON_FOLDER_OPEN); + mui_menu_items_push(items, i); + // printf(" %s - %s\n", p, d); + string_array_push(&std->pop_path, strdup(p)); + if (!strcmp(d, "/")) + break; + *d = 0; + } + free(p); + mui_menu_item_t z = {}; + mui_menu_items_push(items, z); + mui_popupmenu_prepare(pop); + + mui_control_t * lb = std->listbox; + mui_listbox_elems_t * elems = mui_listbox_get_elems(lb); + mui_listbox_elems_clear(elems); + struct dirent * ent; + while ((ent = readdir(dir))) { + if (ent->d_name[0] == '.') + continue; + struct stat st; + char * full_path = NULL; + asprintf(&full_path, "%s/%s", std->current_path, ent->d_name); + stat(full_path, &st); + free(full_path); + mui_listbox_elem_t e = {}; + // usr the regex to filter file names + if (std->re_pattern) { + if (!S_ISDIR(st.st_mode) && regexec(&std->re, ent->d_name, 0, NULL, 0)) + e.disabled = 1; + } + e.elem = strdup(ent->d_name); + if (S_ISDIR(st.st_mode)) + strcpy(e.icon, MUI_ICON_FOLDER); + else + strcpy(e.icon, MUI_ICON_FILE); + mui_listbox_elems_push(elems, e); + } + qsort(elems->e, elems->count, + sizeof(mui_listbox_elem_t), _mui_stdfile_sort_cb); + mui_control_set_value(lb, 0); + mui_listbox_prepare(lb); + closedir(dir); + return 0; +} + +static int +_mui_stdfile_window_action( + mui_window_t * win, + void * cb_param, + uint32_t what, + void * param) +{ + mui_stdfile_t * std = (mui_stdfile_t*)win; + + switch (what) { + case MUI_WINDOW_ACTION_CLOSE: { + // dispose of anything we had allocated + printf("%s close\n", __func__); + if (std->pref_file) + free(std->pref_file); + if (std->re_pattern) + free(std->re_pattern); + if (std->current_path) + free(std->current_path); + if (std->selected_path) + free(std->selected_path); + regfree(&std->re); + for (int i = 0; i < (int)std->pop_path.count; i++) + free(std->pop_path.e[i]); + std->pop_path.count = 0; + + } break; + } + return 0; +} + +static int +_mui_stdfile_control_action( + mui_control_t * c, + void * cb_param, + uint32_t what, + void * param) +{ + mui_stdfile_t * std = cb_param; + switch (c->uid) { + case MUI_STD_FILE_PART_OK: { + mui_listbox_elem_t * e = mui_listbox_get_elems(std->listbox)->e; + int idx = mui_control_get_value(std->listbox); + if (idx < 0 || idx >= (int)mui_listbox_get_elems(std->listbox)->count) + return 0; + mui_listbox_elem_t * elem = &e[idx]; + if (elem->disabled) + break; + // save pref file + if (std->pref_file) { + FILE * f = fopen(std->pref_file, "w"); + if (f) { + fprintf(f, "%s\n", std->current_path); + fclose(f); + } + } + _mui_stdfile_control_action(std->listbox, std, + MUI_CONTROL_ACTION_SELECT, elem); + } break; + case MUI_STD_FILE_PART_CANCEL: + mui_window_action(&std->win, MUI_STDF_ACTION_CANCEL, NULL); + break; + case MUI_STD_FILE_PART_HOME: + // printf("%s Home\n", __func__); + _mui_stdfile_populate(std, getenv("HOME")); + break; + case MUI_STD_FILE_PART_ROOT: + // printf("%s Root\n", __func__); + _mui_stdfile_populate(std, "/"); + break; + case MUI_STD_FILE_PART_LISTBOX: { + // printf("%s Listbox\n", __func__); + if (what == MUI_CONTROL_ACTION_SELECT || + what == MUI_CONTROL_ACTION_DOUBLECLICK) { + mui_listbox_elem_t * e = param; + if (e->disabled) + break; + char * full_path = NULL; + asprintf(&full_path, "%s/%s", + std->current_path, (char*)e->elem); + char *dbl; + while ((dbl = strstr(full_path, "//")) != NULL) { + memmove(dbl, dbl + 1, strlen(dbl)); // include zero + } + struct stat st; + stat(full_path, &st); + if (S_ISDIR(st.st_mode)) { + _mui_stdfile_populate(std, full_path); + } else { + printf("Selected: %s\n", full_path); + mui_window_action(&std->win, MUI_STDF_ACTION_SELECT, NULL); + } + free(full_path); + } + } break; + case MUI_STD_FILE_PART_POPUP: + // printf("%s POPUP\n", __func__); + if (what == MUI_CONTROL_ACTION_VALUE_CHANGED) { + int idx = mui_control_get_value(c); + printf("Selected: %s\n", std->pop_path.e[idx]); + _mui_stdfile_populate(std, std->pop_path.e[idx]); + } + break; + } + return 0; +} + +mui_window_t * +mui_stdfile_get( + struct mui_t * ui, + c2_pt_t where, + const char * prompt, + const char * regexp, + const char * start_path ) +{ + c2_rect_t wpos = C2_RECT_WH(where.x, where.y, 700, 400); + if (where.x == 0 && where.y == 0) + c2_rect_offset(&wpos, + (ui->screen_size.x / 2) - (c2_rect_width(&wpos) / 2), + (ui->screen_size.y * 0.4) - (c2_rect_height(&wpos) / 2)); + mui_window_t *w = mui_window_create(ui, + wpos, + NULL, MUI_WINDOW_LAYER_MODAL, + prompt, sizeof(mui_stdfile_t)); + mui_window_set_action(w, _mui_stdfile_window_action, NULL); + mui_stdfile_t *std = (mui_stdfile_t *)w; + if (regexp) { + std->re_pattern = strdup(regexp); + int re = regcomp(&std->re, std->re_pattern, REG_EXTENDED); + if (re) { + char * msg = NULL; + asprintf(&msg, "%s\n%s", std->re_pattern, + strerror(errno)); + mui_alert(std->win.ui, C2_PT(0,0), + "Could not compile regexp", + msg, + MUI_ALERT_FLAG_OK); + free(std->re_pattern); + std->re_pattern = NULL; + } + } + mui_control_t * c = NULL; + c2_rect_t cf; + cf = C2_RECT_WH(0, 0, 120, 40); + c2_rect_left_of(&cf, c2_rect_width(&w->content), 20); + c2_rect_top_of(&cf, c2_rect_height(&w->content), 20); + std->cancel = c = mui_button_new(w, + cf, MUI_BUTTON_STYLE_NORMAL, + "Cancel", MUI_STD_FILE_PART_CANCEL); + c2_rect_top_of(&cf, cf.t, 20); + std->ok = c = mui_button_new(w, + cf, MUI_BUTTON_STYLE_DEFAULT, + "Select", MUI_STD_FILE_PART_OK); + + std->ok->key_equ = MUI_KEY_EQU(0, 13); // return + std->cancel->key_equ = MUI_KEY_EQU(0, 27); // ESC + + c2_rect_t t = cf; + t.b = t.t + 1; + c2_rect_top_of(&t, cf.t, 25); + c = mui_separator_new(w, t); + + c2_rect_top_of(&cf, cf.t, 40); + std->home = c = mui_button_new(w, + cf, MUI_BUTTON_STYLE_NORMAL, + "Home", MUI_STD_FILE_PART_HOME); + c->key_equ = MUI_KEY_EQU(MUI_MODIFIER_ALT, 'h'); + + c2_rect_top_of(&cf, cf.t, 20); + std->root = c = mui_button_new(w, + cf, MUI_BUTTON_STYLE_NORMAL, + "Root", MUI_STD_FILE_PART_ROOT); + c->key_equ = MUI_KEY_EQU(MUI_MODIFIER_ALT, '/'); + + cf = C2_RECT_WH(15, 45, 700-185, 300); + std->listbox = c = mui_listbox_new(w, cf, + MUI_STD_FILE_PART_LISTBOX); + + cf = C2_RECT_WH(15, 0, 700-185, 34); + c2_rect_top_of(&cf, std->listbox->frame.t, 6); + std->popup = c = mui_popupmenu_new(w, cf, + "Popup", MUI_STD_FILE_PART_POPUP); +// printf("Popup: %p\n", c); + c = NULL; + TAILQ_FOREACH(c, &w->controls, self) { + if (mui_control_get_uid(c) == 0) + continue; + mui_control_set_action(c, _mui_stdfile_control_action, std); + } + int dopop = 1; // populate to start_path by default + if (ui->pref_directory) { + uint32_t hash = std->re_pattern ? mui_hash(std->re_pattern) : 0; + asprintf(&std->pref_file, "%s/std_path_%04x", ui->pref_directory, hash); + printf("%s pref file: %s\n", __func__, std->pref_file); + /* read last used pathname */ + FILE * f = fopen(std->pref_file, "r"); + if (f) { + char * path = NULL; + size_t len = 0; + getline(&path, &len, f); + fclose(f); + if (path) { + char *nl = strrchr(path, '\n'); + if (nl) + *nl = 0; + if (_mui_stdfile_populate(std, path) == 0) { + printf("%s last path[%s]: %s\n", __func__, + std->re_pattern, path); + dopop = 0; + } + free(path); + } + } + } + if (dopop) + _mui_stdfile_populate(std, start_path); + + return w; +} + +char * +mui_stdfile_get_path( + mui_window_t * w ) +{ + mui_stdfile_t * std = (mui_stdfile_t *)w; + return std->current_path; +} + +char * +mui_stdfile_get_selected_path( + mui_window_t * w ) +{ + mui_stdfile_t * std = (mui_stdfile_t *)w; + + mui_listbox_elem_t * e = mui_listbox_get_elems(std->listbox)->e; + int idx = mui_control_get_value(std->listbox); + if (idx < 0 || idx >= (int)mui_listbox_get_elems(std->listbox)->count) + return NULL; + mui_listbox_elem_t * elem = &e[idx]; + char * full_path = NULL; + asprintf(&full_path, "%s/%s", + std->current_path, (char*)elem->elem); + char *dbl; + while ((dbl = strstr(full_path, "//")) != NULL) { + memmove(dbl, dbl + 1, strlen(dbl)); // include zero + } + return full_path; +} diff --git a/libmui/mui/mui_utils.c b/libmui/mui/mui_utils.c new file mode 100644 index 0000000..e5b86f7 --- /dev/null +++ b/libmui/mui/mui_utils.c @@ -0,0 +1,40 @@ +/* + * mui_utils.c + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ + +#include +#include "mui.h" + +mui_time_t +mui_get_time() +{ + struct timespec tim; + clock_gettime(CLOCK_MONOTONIC_RAW, &tim); + uint64_t time = ((uint64_t)tim.tv_sec) * (1000000 / MUI_TIME_RES) + + tim.tv_nsec / (1000 * MUI_TIME_RES); + return time; +} + + +uint32_t +mui_hash( + const char * inString ) +{ + if (!inString) + return 0; + /* Described http://papa.bretmulvey.com/post/124027987928/hash-functions */ + const uint32_t p = 16777619; + uint32_t hash = 0x811c9dc5; + while (*inString) + hash = (hash ^ (*inString++)) * p; + hash += hash << 13; + hash ^= hash >> 7; + hash += hash << 3; + hash ^= hash >> 17; + hash += hash << 5; + return hash; +} \ No newline at end of file diff --git a/libmui/mui/mui_window.c b/libmui/mui/mui_window.c new file mode 100644 index 0000000..bfcf94d --- /dev/null +++ b/libmui/mui/mui_window.c @@ -0,0 +1,503 @@ +/* + * mui_window.c + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ + +#include +#include +#include +#include +#include + +#include "mui_priv.h" +#include "cg.h" + +enum mui_window_part_e { + MUI_WINDOW_PART_NONE = 0, + MUI_WINDOW_PART_CONTENT, + MUI_WINDOW_PART_TITLE, + MUI_WINDOW_PART_FRAME, + MUI_WINDOW_PART_COUNT, +}; + +static void +mui_window_update_rects( + mui_window_t *win, + mui_font_t * main ) +{ + int title_height = main->size; + c2_rect_t content = win->frame; + c2_rect_inset(&content, 4, 4); + content.t += title_height - 1; + win->content = content; +} + +void +mui_titled_window_draw( + struct mui_t *ui, + mui_window_t *win, + mui_drawable_t *dr) +{ + mui_font_t * main = mui_font_find(ui, "main"); + if (!main) + return; + mui_window_update_rects(win, main); + int title_height = main->size; + + struct cg_ctx_t * cg = mui_drawable_get_cg(dr); + mui_color_t frameFill = MUI_COLOR(0xbbbbbbff); + mui_color_t contentFill = MUI_COLOR(0xf0f0f0ff); + mui_color_t frameColor = MUI_COLOR(0x000000ff); + mui_color_t decoColor = MUI_COLOR(0x999999ff); + mui_color_t titleColor = MUI_COLOR(0x000000aa); + mui_color_t dimTitleColor = MUI_COLOR(0x00000055); + + cg_set_line_width(cg, 1); + cg_rectangle(cg, win->frame.l + 0.5f, win->frame.t + 0.5f, + c2_rect_width(&win->frame) - 1, c2_rect_height(&win->frame) - 1); + cg_rectangle(cg, win->content.l + 0.5f, win->content.t + 0.5f, + c2_rect_width(&win->content) - 1, c2_rect_height(&win->content) - 1); + cg_set_source_color(cg, &CG_COLOR(frameFill)); + cg_fill_preserve(cg); + cg_set_source_color(cg, &CG_COLOR(frameColor)); + cg_stroke(cg); + + bool isFront = mui_window_front(ui) == win; + if (isFront) { + const int lrMargin = 6; + const int steps = 6; + cg_set_line_width(cg, 2); + for (int i = 1; i < (title_height + 4) / steps; i++) { + cg_move_to(cg, win->frame.l + lrMargin, win->frame.t + i * steps); + cg_line_to(cg, win->frame.r - lrMargin, win->frame.t + i * steps); + } + cg_set_source_color(cg, &CG_COLOR(decoColor)); + cg_stroke(cg); + } + if (win->title) { + stb_ttc_measure m = {}; + mui_font_text_measure(main, win->title, &m); + + int title_width = m.x1 - m.x0; + c2_rect_t title = win->frame; + c2_rect_offset(&title, 0, 1); + title.b = title.t + title_height; + title.l += (c2_rect_width(&win->frame) / 2) - (title_width / 2); + title.r = title.l + title_width; + if (isFront) { + c2_rect_t titleBack = title; + c2_rect_inset(&titleBack, -6, 0); + cg_round_rectangle(cg, titleBack.l, titleBack.t, + c2_rect_width(&titleBack), c2_rect_height(&titleBack), 12, 12); + cg_set_source_color(cg, &CG_COLOR(frameFill)); + cg_fill(cg); + } + mui_font_text_draw(main, dr, + C2_PT(-m.x0 + 1 + title.l, title.t + 0), + win->title, strlen(win->title), + isFront ? titleColor : dimTitleColor); + } + cg_set_source_color(cg, &CG_COLOR(contentFill)); + cg_rectangle(cg, win->content.l + 0.5f, win->content.t + 0.5f, + c2_rect_width(&win->content) - 1, c2_rect_height(&win->content) - 1); + cg_fill(cg); +} + +mui_window_t * +mui_window_create( + struct mui_t *ui, + c2_rect_t frame, + mui_wdef_p wdef, + uint8_t layer, + const char *title, + uint32_t instance_size) +{ + mui_window_t * w = calloc(1, + instance_size >= sizeof(*w) ? + instance_size : sizeof(*w)); + w->ui = ui; + w->frame = frame; + w->title = title ? strdup(title) : NULL; + w->wdef = wdef; + w->flags.layer = layer; + TAILQ_INIT(&w->controls); + TAILQ_INIT(&w->zombies); + STAILQ_INIT(&w->actions); + pixman_region32_init(&w->inval); + + TAILQ_INSERT_HEAD(&ui->windows, w, self); + mui_window_select(w); // place it in it's own layer + mui_font_t * main = mui_font_find(ui, "main"); + mui_window_update_rects(w, main); + mui_window_inval(w, NULL); // just to mark the UI dirty + + return w; +} + +void +_mui_control_free( + mui_control_t * c ); +void +_mui_window_free( + mui_window_t *win) +{ + if (!win) + return; + pixman_region32_fini(&win->inval); + mui_control_t * c; + while ((c = TAILQ_FIRST(&win->controls))) { + mui_control_dispose(c); + } + while ((c = TAILQ_FIRST(&win->zombies))) { + TAILQ_REMOVE(&win->zombies, c, self); + _mui_control_free(c); + } + if (win->title) + free(win->title); + free(win); +} + +void +mui_window_dispose( + mui_window_t *win) +{ + if (!win) + return; + if (win->flags.zombie) { + printf("%s: DOUBLE delete %s\n", __func__, win->title); + return; + } + bool was_front = mui_window_isfront(win); + mui_window_action(win, MUI_WINDOW_ACTION_CLOSE, NULL); + mui_window_inval(win, NULL); // just to mark the UI dirty + if (win->wdef) + win->wdef(win, MUI_WDEF_DISPOSE, NULL); + struct mui_t *ui = win->ui; + TAILQ_REMOVE(&ui->windows, win, self); + if (ui->event_capture == win) + ui->event_capture = NULL; + if (ui->action_active) { + // printf("%s %s is now zombie\n", __func__, win->title); + win->flags.zombie = true; + TAILQ_INSERT_TAIL(&ui->zombies, win, self); + } else + _mui_window_free(win); + if (was_front) { + mui_window_t * front = mui_window_front(ui); + if (front) + mui_window_inval(front, NULL); + } +} + +void +mui_window_draw( + mui_window_t *win, + mui_drawable_t *dr) +{ + mui_drawable_clip_push(dr, &win->frame); + if (win->wdef) + win->wdef(win, MUI_WDEF_DRAW, dr); + else + mui_titled_window_draw(win->ui, win, dr); + struct cg_ctx_t * cg = mui_drawable_get_cg(dr); + cg_save(cg); +// cg_translate(cg, content.l, content.t); + mui_control_t * c, *safe; + TAILQ_FOREACH_SAFE(c, &win->controls, self, safe) { + mui_control_draw(win, c, dr); + } + cg_restore(cg); + + mui_drawable_clip_pop(dr); +} + +bool +mui_window_handle_keyboard( + mui_window_t *win, + mui_event_t *event) +{ + if (!mui_window_isfront(win)) + return false; + if (win->wdef && win->wdef(win, MUI_WDEF_EVENT, event)) { +// printf("%s %s handled it\n", __func__, win->title); + return true; + } +// printf("%s %s checkint controls\n", __func__, win->title); + mui_control_t * c, *safe; + TAILQ_FOREACH_SAFE(c, &win->controls, self, safe) { + if (mui_control_event(c, event)) { +// printf("%s control %s handled it\n", __func__, c->title); + return true; + } + } + return false; +} + +bool +mui_window_handle_mouse( + mui_window_t *win, + mui_event_t *event) +{ + if (win->wdef && win->wdef(win, MUI_WDEF_EVENT, event)) + return true; + switch (event->type) { + case MUI_EVENT_WHEEL: { + if (!c2_rect_contains_pt(&win->frame, &event->wheel.where)) + return false; + mui_control_t * c = mui_control_locate(win, event->wheel.where); + if (!c) + return false; + if (c->cdef && c->cdef(c, MUI_CDEF_EVENT, event)) { + return true; + } + } break; + case MUI_EVENT_BUTTONDOWN: { + if (!c2_rect_contains_pt(&win->frame, &event->mouse.where)) + return false; + mui_control_t * c = mui_control_locate(win, event->mouse.where); + /* if modifiers like control is down, don't select */ + if (!(event->modifiers & MUI_MODIFIER_CTRL)) + mui_window_select(win); + if (mui_window_front(win->ui) != win) + c = NULL; + if (!c) { + /* find where we clicked in the window */ + win->ui->event_capture = win; + win->click_loc = event->mouse.where; + c2_pt_offset(&win->click_loc, -win->frame.l, -win->frame.t); + win->flags.hit_part = MUI_WINDOW_PART_FRAME; + if (event->mouse.where.y < win->content.t) + win->flags.hit_part = MUI_WINDOW_PART_TITLE; + else if (c2_rect_contains_pt(&win->content, &event->mouse.where)) + win->flags.hit_part = MUI_WINDOW_PART_CONTENT; + } else + win->flags.hit_part = MUI_WINDOW_PART_CONTENT; + if (c) { + if (c->cdef && c->cdef(c, MUI_CDEF_EVENT, event)) { + // c->state = MUI_CONTROL_STATE_CLICKED; + win->control_clicked = c; + } + } + return true; + } break; + case MUI_EVENT_DRAG: + if (win->flags.hit_part == MUI_WINDOW_PART_TITLE) { + c2_rect_t frame = win->frame; + c2_rect_offset(&frame, + -win->frame.l + event->mouse.where.x - win->click_loc.x, + -win->frame.t + event->mouse.where.y - win->click_loc.y); + // todo, get that visibel rectangle from somewhere else + c2_rect_t screen = { .br = win->ui->screen_size }; + screen.t += 35; + c2_rect_t title_bar = frame; + title_bar.b = title_bar.t + 35; // TODO fix that + if (c2_rect_intersect_rect(&title_bar, &screen)) { + c2_rect_t o; + c2_rect_clip_rect(&title_bar, &screen, &o); + if (c2_rect_width(&o) > 10 && c2_rect_height(&o) > 10) { + mui_window_inval(win, NULL); // old frame + win->frame = frame; + mui_window_inval(win, NULL); // new frame + } + } + // mui_window_inval(win, NULL); + return true; + } + if (win->control_clicked) { + mui_control_t * c = win->control_clicked; + if (c->cdef && c->cdef(c, MUI_CDEF_EVENT, event)) { + return true; + } else + win->control_clicked = NULL; + } + return win->flags.hit_part != MUI_WINDOW_PART_NONE; + break; + case MUI_EVENT_BUTTONUP: { + int part = win->flags.hit_part; + win->flags.hit_part = MUI_WINDOW_PART_NONE; + win->ui->event_capture = NULL; + if (win->control_clicked) { + mui_control_t * c = win->control_clicked; + win->control_clicked = NULL; + if (c->cdef && c->cdef(c, MUI_CDEF_EVENT, event)) + return true; + } + return part != MUI_WINDOW_PART_NONE; + } break; + case MUI_EVENT_MOUSEENTER: + case MUI_EVENT_MOUSELEAVE: + break; + } +// printf("MOUSE %s button %d\n", __func__, event->mouse.button); +// printf("MOUSE %s %s\n", __func__, c->title); + return false; +} + +void +mui_window_inval( + mui_window_t *win, + c2_rect_t * r) +{ + if (!win) + return; + c2_rect_t frame = win->frame; + c2_rect_t forward = {}; + + if (!r) { + // printf("%s %s inval %s (whole)\n", __func__, win->title, c2_rect_as_str(&frame)); + pixman_region32_reset(&win->inval, (pixman_box32_t*)&frame); + forward = frame; + + mui_window_t * w; + TAILQ_FOREACH(w, &win->ui->windows, self) { + if (w == win || !c2_rect_intersect_rect(&w->frame, &forward)) + continue; + pixman_region32_union_rect(&w->inval, &w->inval, + forward.l, forward.t, + c2_rect_width(&forward), c2_rect_height(&forward)); + } + } else { + c2_rect_t local = *r; + c2_rect_offset(&local, win->content.l, win->content.t); + forward = local; + + pixman_region32_union_rect(&win->inval, &win->inval, + forward.l, forward.t, + c2_rect_width(&forward), c2_rect_height(&forward)); + } + if (c2_rect_isempty(&forward)) + return; + pixman_region32_union_rect(&win->ui->inval, &win->ui->inval, + forward.l, forward.t, + c2_rect_width(&forward), c2_rect_height(&forward)); +} + +mui_window_t * +mui_window_front( + struct mui_t *ui) +{ + if (!ui) + return NULL; + mui_window_t * w; + TAILQ_FOREACH_REVERSE(w, &ui->windows, windows, self) { + if (w->flags.hidden) + continue; + if (w->flags.layer < MUI_WINDOW_MENUBAR_LAYER) + return w; + } + return NULL; +} + +bool +mui_window_isfront( + mui_window_t *win) +{ + if (!win) + return NULL; + mui_window_t * next = TAILQ_NEXT(win, self); + while (next && next->flags.hidden) + next = TAILQ_NEXT(next, self); + if (!next) + return true; + if (next->flags.layer > win->flags.layer) + return true; + return false; +} + +bool +mui_window_select( + mui_window_t *win) +{ + bool res = false; + if (!win) + return false; + mui_window_t *last = NULL; + if (mui_window_isfront(win)) + goto done; + res = true; + mui_window_inval(win, NULL); + TAILQ_REMOVE(&win->ui->windows, win, self); + mui_window_t *w, *save; + TAILQ_FOREACH_SAFE(w, &win->ui->windows, self, save) { + if (w->flags.layer > win->flags.layer) { + TAILQ_INSERT_BEFORE(w, win, self); + goto done; + } + last = w; + } + TAILQ_INSERT_TAIL(&win->ui->windows, win, self); +done: + if (last) // we are deselecting this one, so redraw it + mui_window_inval(last, NULL); +#if 0 + printf("%s %s res:%d stack is now:\n", __func__, win->title, res); + TAILQ_FOREACH(w, &win->ui->windows, self) { + printf(" L:%2d T:%s\n", w->flags.layer, w->title); + } +#endif + return res; +} + +void +mui_window_action( + mui_window_t * win, + uint32_t what, + void * param ) +{ + if (!win) + return; + win->ui->action_active++; + mui_action_t *a; + STAILQ_FOREACH(a, &win->actions, self) { + if (!a->window_cb) + continue; + a->window_cb(win, a->cb_param, what, param); + } + win->ui->action_active--; +} + +void +mui_window_set_action( + mui_window_t * win, + mui_window_action_p cb, + void * param ) +{ + if (!win || !cb) + return; + + mui_action_t *a; + STAILQ_FOREACH(a, &win->actions, self) { + if (a->window_cb == cb && a->cb_param == param) + return; + } + a = calloc(1, sizeof(*a)); + a->window_cb = cb; + a->cb_param = param; + STAILQ_INSERT_TAIL(&win->actions, a, self); +} + +mui_window_t * +mui_window_get_by_id( + struct mui_t *ui, + uint32_t uid ) +{ + mui_window_t *w; + TAILQ_FOREACH(w, &ui->windows, self) { + if (w->uid == uid) + return w; + } + return NULL; +} + +void +mui_window_set_id( + mui_window_t *win, + uint32_t uid) +{ + if (!win) + return; + win->uid = uid; +} diff --git a/libmui/mui/stb_truetype.h b/libmui/mui/stb_truetype.h new file mode 100644 index 0000000..bbf2284 --- /dev/null +++ b/libmui/mui/stb_truetype.h @@ -0,0 +1,5077 @@ +// stb_truetype.h - v1.26 - public domain +// authored from 2009-2021 by Sean Barrett / RAD Game Tools +// +// ======================================================================= +// +// NO SECURITY GUARANTEE -- DO NOT USE THIS ON UNTRUSTED FONT FILES +// +// This library does no range checking of the offsets found in the file, +// meaning an attacker can use it to read arbitrary memory. +// +// ======================================================================= +// +// This library processes TrueType files: +// parse files +// extract glyph metrics +// extract glyph shapes +// render glyphs to one-channel bitmaps with antialiasing (box filter) +// render glyphs to one-channel SDF bitmaps (signed-distance field/function) +// +// Todo: +// non-MS cmaps +// crashproof on bad data +// hinting? (no longer patented) +// cleartype-style AA? +// optimize: use simple memory allocator for intermediates +// optimize: build edge-list directly from curves +// optimize: rasterize directly from curves? +// +// ADDITIONAL CONTRIBUTORS +// +// Mikko Mononen: compound shape support, more cmap formats +// Tor Andersson: kerning, subpixel rendering +// Dougall Johnson: OpenType / Type 2 font handling +// Daniel Ribeiro Maciel: basic GPOS-based kerning +// +// Misc other: +// Ryan Gordon +// Simon Glass +// github:IntellectualKitty +// Imanol Celaya +// Daniel Ribeiro Maciel +// +// Bug/warning reports/fixes: +// "Zer" on mollyrocket Fabian "ryg" Giesen github:NiLuJe +// Cass Everitt Martins Mozeiko github:aloucks +// stoiko (Haemimont Games) Cap Petschulat github:oyvindjam +// Brian Hook Omar Cornut github:vassvik +// Walter van Niftrik Ryan Griege +// David Gow Peter LaValle +// David Given Sergey Popov +// Ivan-Assen Ivanov Giumo X. Clanjor +// Anthony Pesch Higor Euripedes +// Johan Duparc Thomas Fields +// Hou Qiming Derek Vinyard +// Rob Loach Cort Stratton +// Kenney Phillis Jr. Brian Costabile +// Ken Voskuil (kaesve) +// +// VERSION HISTORY +// +// 1.26 (2021-08-28) fix broken rasterizer +// 1.25 (2021-07-11) many fixes +// 1.24 (2020-02-05) fix warning +// 1.23 (2020-02-02) query SVG data for glyphs; query whole kerning table (but only kern not GPOS) +// 1.22 (2019-08-11) minimize missing-glyph duplication; fix kerning if both 'GPOS' and 'kern' are defined +// 1.21 (2019-02-25) fix warning +// 1.20 (2019-02-07) PackFontRange skips missing codepoints; GetScaleFontVMetrics() +// 1.19 (2018-02-11) GPOS kerning, STBTT_fmod +// 1.18 (2018-01-29) add missing function +// 1.17 (2017-07-23) make more arguments const; doc fix +// 1.16 (2017-07-12) SDF support +// 1.15 (2017-03-03) make more arguments const +// 1.14 (2017-01-16) num-fonts-in-TTC function +// 1.13 (2017-01-02) support OpenType fonts, certain Apple fonts +// 1.12 (2016-10-25) suppress warnings about casting away const with -Wcast-qual +// 1.11 (2016-04-02) fix unused-variable warning +// 1.10 (2016-04-02) user-defined fabs(); rare memory leak; remove duplicate typedef +// 1.09 (2016-01-16) warning fix; avoid crash on outofmem; use allocation userdata properly +// 1.08 (2015-09-13) document stbtt_Rasterize(); fixes for vertical & horizontal edges +// 1.07 (2015-08-01) allow PackFontRanges to accept arrays of sparse codepoints; +// variant PackFontRanges to pack and render in separate phases; +// fix stbtt_GetFontOFfsetForIndex (never worked for non-0 input?); +// fixed an assert() bug in the new rasterizer +// replace assert() with STBTT_assert() in new rasterizer +// +// Full history can be found at the end of this file. +// +// LICENSE +// +// See end of file for license information. +// +// USAGE +// +// Include this file in whatever places need to refer to it. In ONE C/C++ +// file, write: +// #define STB_TRUETYPE_IMPLEMENTATION +// before the #include of this file. This expands out the actual +// implementation into that C/C++ file. +// +// To make the implementation private to the file that generates the implementation, +// #define STBTT_STATIC +// +// Simple 3D API (don't ship this, but it's fine for tools and quick start) +// stbtt_BakeFontBitmap() -- bake a font to a bitmap for use as texture +// stbtt_GetBakedQuad() -- compute quad to draw for a given char +// +// Improved 3D API (more shippable): +// #include "stb_rect_pack.h" -- optional, but you really want it +// stbtt_PackBegin() +// stbtt_PackSetOversampling() -- for improved quality on small fonts +// stbtt_PackFontRanges() -- pack and renders +// stbtt_PackEnd() +// stbtt_GetPackedQuad() +// +// "Load" a font file from a memory buffer (you have to keep the buffer loaded) +// stbtt_InitFont() +// stbtt_GetFontOffsetForIndex() -- indexing for TTC font collections +// stbtt_GetNumberOfFonts() -- number of fonts for TTC font collections +// +// Render a unicode codepoint to a bitmap +// stbtt_GetCodepointBitmap() -- allocates and returns a bitmap +// stbtt_MakeCodepointBitmap() -- renders into bitmap you provide +// stbtt_GetCodepointBitmapBox() -- how big the bitmap must be +// +// Character advance/positioning +// stbtt_GetCodepointHMetrics() +// stbtt_GetFontVMetrics() +// stbtt_GetFontVMetricsOS2() +// stbtt_GetCodepointKernAdvance() +// +// Starting with version 1.06, the rasterizer was replaced with a new, +// faster and generally-more-precise rasterizer. The new rasterizer more +// accurately measures pixel coverage for anti-aliasing, except in the case +// where multiple shapes overlap, in which case it overestimates the AA pixel +// coverage. Thus, anti-aliasing of intersecting shapes may look wrong. If +// this turns out to be a problem, you can re-enable the old rasterizer with +// #define STBTT_RASTERIZER_VERSION 1 +// which will incur about a 15% speed hit. +// +// ADDITIONAL DOCUMENTATION +// +// Immediately after this block comment are a series of sample programs. +// +// After the sample programs is the "header file" section. This section +// includes documentation for each API function. +// +// Some important concepts to understand to use this library: +// +// Codepoint +// Characters are defined by unicode codepoints, e.g. 65 is +// uppercase A, 231 is lowercase c with a cedilla, 0x7e30 is +// the hiragana for "ma". +// +// Glyph +// A visual character shape (every codepoint is rendered as +// some glyph) +// +// Glyph index +// A font-specific integer ID representing a glyph +// +// Baseline +// Glyph shapes are defined relative to a baseline, which is the +// bottom of uppercase characters. Characters extend both above +// and below the baseline. +// +// Current Point +// As you draw text to the screen, you keep track of a "current point" +// which is the origin of each character. The current point's vertical +// position is the baseline. Even "baked fonts" use this model. +// +// Vertical Font Metrics +// The vertical qualities of the font, used to vertically position +// and space the characters. See docs for stbtt_GetFontVMetrics. +// +// Font Size in Pixels or Points +// The preferred interface for specifying font sizes in stb_truetype +// is to specify how tall the font's vertical extent should be in pixels. +// If that sounds good enough, skip the next paragraph. +// +// Most font APIs instead use "points", which are a common typographic +// measurement for describing font size, defined as 72 points per inch. +// stb_truetype provides a point API for compatibility. However, true +// "per inch" conventions don't make much sense on computer displays +// since different monitors have different number of pixels per +// inch. For example, Windows traditionally uses a convention that +// there are 96 pixels per inch, thus making 'inch' measurements have +// nothing to do with inches, and thus effectively defining a point to +// be 1.333 pixels. Additionally, the TrueType font data provides +// an explicit scale factor to scale a given font's glyphs to points, +// but the author has observed that this scale factor is often wrong +// for non-commercial fonts, thus making fonts scaled in points +// according to the TrueType spec incoherently sized in practice. +// +// DETAILED USAGE: +// +// Scale: +// Select how high you want the font to be, in points or pixels. +// Call ScaleForPixelHeight or ScaleForMappingEmToPixels to compute +// a scale factor SF that will be used by all other functions. +// +// Baseline: +// You need to select a y-coordinate that is the baseline of where +// your text will appear. Call GetFontBoundingBox to get the baseline-relative +// bounding box for all characters. SF*-y0 will be the distance in pixels +// that the worst-case character could extend above the baseline, so if +// you want the top edge of characters to appear at the top of the +// screen where y=0, then you would set the baseline to SF*-y0. +// +// Current point: +// Set the current point where the first character will appear. The +// first character could extend left of the current point; this is font +// dependent. You can either choose a current point that is the leftmost +// point and hope, or add some padding, or check the bounding box or +// left-side-bearing of the first character to be displayed and set +// the current point based on that. +// +// Displaying a character: +// Compute the bounding box of the character. It will contain signed values +// relative to . I.e. if it returns x0,y0,x1,y1, +// then the character should be displayed in the rectangle from +// to = 32 && *text < 128) { + stbtt_aligned_quad q; + stbtt_GetBakedQuad(cdata, 512,512, *text-32, &x,&y,&q,1);//1=opengl & d3d10+,0=d3d9 + glTexCoord2f(q.s0,q.t0); glVertex2f(q.x0,q.y0); + glTexCoord2f(q.s1,q.t0); glVertex2f(q.x1,q.y0); + glTexCoord2f(q.s1,q.t1); glVertex2f(q.x1,q.y1); + glTexCoord2f(q.s0,q.t1); glVertex2f(q.x0,q.y1); + } + ++text; + } + glEnd(); +} +#endif +// +// +////////////////////////////////////////////////////////////////////////////// +// +// Complete program (this compiles): get a single bitmap, print as ASCII art +// +#if 0 +#include +#define STB_TRUETYPE_IMPLEMENTATION // force following include to generate implementation +#include "stb_truetype.h" + +char ttf_buffer[1<<25]; + +int main(int argc, char **argv) +{ + stbtt_fontinfo font; + unsigned char *bitmap; + int w,h,i,j,c = (argc > 1 ? atoi(argv[1]) : 'a'), s = (argc > 2 ? atoi(argv[2]) : 20); + + fread(ttf_buffer, 1, 1<<25, fopen(argc > 3 ? argv[3] : "c:/windows/fonts/arialbd.ttf", "rb")); + + stbtt_InitFont(&font, ttf_buffer, stbtt_GetFontOffsetForIndex(ttf_buffer,0)); + bitmap = stbtt_GetCodepointBitmap(&font, 0,stbtt_ScaleForPixelHeight(&font, s), c, &w, &h, 0,0); + + for (j=0; j < h; ++j) { + for (i=0; i < w; ++i) + putchar(" .:ioVM@"[bitmap[j*w+i]>>5]); + putchar('\n'); + } + return 0; +} +#endif +// +// Output: +// +// .ii. +// @@@@@@. +// V@Mio@@o +// :i. V@V +// :oM@@M +// :@@@MM@M +// @@o o@M +// :@@. M@M +// @@@o@@@@ +// :M@@V:@@. +// +////////////////////////////////////////////////////////////////////////////// +// +// Complete program: print "Hello World!" banner, with bugs +// +#if 0 +char buffer[24<<20]; +unsigned char screen[20][79]; + +int main(int arg, char **argv) +{ + stbtt_fontinfo font; + int i,j,ascent,baseline,ch=0; + float scale, xpos=2; // leave a little padding in case the character extends left + char *text = "Heljo World!"; // intentionally misspelled to show 'lj' brokenness + + fread(buffer, 1, 1000000, fopen("c:/windows/fonts/arialbd.ttf", "rb")); + stbtt_InitFont(&font, buffer, 0); + + scale = stbtt_ScaleForPixelHeight(&font, 15); + stbtt_GetFontVMetrics(&font, &ascent,0,0); + baseline = (int) (ascent*scale); + + while (text[ch]) { + int advance,lsb,x0,y0,x1,y1; + float x_shift = xpos - (float) floor(xpos); + stbtt_GetCodepointHMetrics(&font, text[ch], &advance, &lsb); + stbtt_GetCodepointBitmapBoxSubpixel(&font, text[ch], scale,scale,x_shift,0, &x0,&y0,&x1,&y1); + stbtt_MakeCodepointBitmapSubpixel(&font, &screen[baseline + y0][(int) xpos + x0], x1-x0,y1-y0, 79, scale,scale,x_shift,0, text[ch]); + // note that this stomps the old data, so where character boxes overlap (e.g. 'lj') it's wrong + // because this API is really for baking character bitmaps into textures. if you want to render + // a sequence of characters, you really need to render each bitmap to a temp buffer, then + // "alpha blend" that into the working buffer + xpos += (advance * scale); + if (text[ch+1]) + xpos += scale*stbtt_GetCodepointKernAdvance(&font, text[ch],text[ch+1]); + ++ch; + } + + for (j=0; j < 20; ++j) { + for (i=0; i < 78; ++i) + putchar(" .:ioVM@"[screen[j][i]>>5]); + putchar('\n'); + } + + return 0; +} +#endif + + +////////////////////////////////////////////////////////////////////////////// +////////////////////////////////////////////////////////////////////////////// +//// +//// INTEGRATION WITH YOUR CODEBASE +//// +//// The following sections allow you to supply alternate definitions +//// of C library functions used by stb_truetype, e.g. if you don't +//// link with the C runtime library. + +#ifdef STB_TRUETYPE_IMPLEMENTATION + // #define your own (u)stbtt_int8/16/32 before including to override this + #ifndef stbtt_uint8 + typedef unsigned char stbtt_uint8; + typedef signed char stbtt_int8; + typedef unsigned short stbtt_uint16; + typedef signed short stbtt_int16; + typedef unsigned int stbtt_uint32; + typedef signed int stbtt_int32; + #endif + + typedef char stbtt__check_size32[sizeof(stbtt_int32)==4 ? 1 : -1]; + typedef char stbtt__check_size16[sizeof(stbtt_int16)==2 ? 1 : -1]; + + // e.g. #define your own STBTT_ifloor/STBTT_iceil() to avoid math.h + #ifndef STBTT_ifloor + #include + #define STBTT_ifloor(x) ((int) floor(x)) + #define STBTT_iceil(x) ((int) ceil(x)) + #endif + + #ifndef STBTT_sqrt + #include + #define STBTT_sqrt(x) sqrt(x) + #define STBTT_pow(x,y) pow(x,y) + #endif + + #ifndef STBTT_fmod + #include + #define STBTT_fmod(x,y) fmod(x,y) + #endif + + #ifndef STBTT_cos + #include + #define STBTT_cos(x) cos(x) + #define STBTT_acos(x) acos(x) + #endif + + #ifndef STBTT_fabs + #include + #define STBTT_fabs(x) fabs(x) + #endif + + // #define your own functions "STBTT_malloc" / "STBTT_free" to avoid malloc.h + #ifndef STBTT_malloc + #include + #define STBTT_malloc(x,u) ((void)(u),malloc(x)) + #define STBTT_free(x,u) ((void)(u),free(x)) + #endif + + #ifndef STBTT_assert + #include + #define STBTT_assert(x) assert(x) + #endif + + #ifndef STBTT_strlen + #include + #define STBTT_strlen(x) strlen(x) + #endif + + #ifndef STBTT_memcpy + #include + #define STBTT_memcpy memcpy + #define STBTT_memset memset + #endif +#endif + +/////////////////////////////////////////////////////////////////////////////// +/////////////////////////////////////////////////////////////////////////////// +//// +//// INTERFACE +//// +//// + +#ifndef __STB_INCLUDE_STB_TRUETYPE_H__ +#define __STB_INCLUDE_STB_TRUETYPE_H__ + +#ifdef STBTT_STATIC +#define STBTT_DEF static +#else +#define STBTT_DEF extern +#endif + +#ifdef __cplusplus +extern "C" { +#endif + +// private structure +typedef struct +{ + unsigned char *data; + int cursor; + int size; +} stbtt__buf; + +////////////////////////////////////////////////////////////////////////////// +// +// TEXTURE BAKING API +// +// If you use this API, you only have to call two functions ever. +// + +typedef struct +{ + unsigned short x0,y0,x1,y1; // coordinates of bbox in bitmap + float xoff,yoff,xadvance; +} stbtt_bakedchar; + +STBTT_DEF int stbtt_BakeFontBitmap(const unsigned char *data, int offset, // font location (use offset=0 for plain .ttf) + float pixel_height, // height of font in pixels + unsigned char *pixels, int pw, int ph, // bitmap to be filled in + int first_char, int num_chars, // characters to bake + stbtt_bakedchar *chardata); // you allocate this, it's num_chars long +// if return is positive, the first unused row of the bitmap +// if return is negative, returns the negative of the number of characters that fit +// if return is 0, no characters fit and no rows were used +// This uses a very crappy packing. + +typedef struct +{ + float x0,y0,s0,t0; // top-left + float x1,y1,s1,t1; // bottom-right +} stbtt_aligned_quad; + +STBTT_DEF void stbtt_GetBakedQuad(const stbtt_bakedchar *chardata, int pw, int ph, // same data as above + int char_index, // character to display + float *xpos, float *ypos, // pointers to current position in screen pixel space + stbtt_aligned_quad *q, // output: quad to draw + int opengl_fillrule); // true if opengl fill rule; false if DX9 or earlier +// Call GetBakedQuad with char_index = 'character - first_char', and it +// creates the quad you need to draw and advances the current position. +// +// The coordinate system used assumes y increases downwards. +// +// Characters will extend both above and below the current position; +// see discussion of "BASELINE" above. +// +// It's inefficient; you might want to c&p it and optimize it. + +STBTT_DEF void stbtt_GetScaledFontVMetrics(const unsigned char *fontdata, int index, float size, float *ascent, float *descent, float *lineGap); +// Query the font vertical metrics without having to create a font first. + + +////////////////////////////////////////////////////////////////////////////// +// +// NEW TEXTURE BAKING API +// +// This provides options for packing multiple fonts into one atlas, not +// perfectly but better than nothing. + +typedef struct +{ + unsigned short x0,y0,x1,y1; // coordinates of bbox in bitmap + float xoff,yoff,xadvance; + float xoff2,yoff2; +} stbtt_packedchar; + +typedef struct stbtt_pack_context stbtt_pack_context; +typedef struct stbtt_fontinfo stbtt_fontinfo; +#ifndef STB_RECT_PACK_VERSION +typedef struct stbrp_rect stbrp_rect; +#endif + +STBTT_DEF int stbtt_PackBegin(stbtt_pack_context *spc, unsigned char *pixels, int width, int height, int stride_in_bytes, int padding, void *alloc_context); +// Initializes a packing context stored in the passed-in stbtt_pack_context. +// Future calls using this context will pack characters into the bitmap passed +// in here: a 1-channel bitmap that is width * height. stride_in_bytes is +// the distance from one row to the next (or 0 to mean they are packed tightly +// together). "padding" is the amount of padding to leave between each +// character (normally you want '1' for bitmaps you'll use as textures with +// bilinear filtering). +// +// Returns 0 on failure, 1 on success. + +STBTT_DEF void stbtt_PackEnd (stbtt_pack_context *spc); +// Cleans up the packing context and frees all memory. + +#define STBTT_POINT_SIZE(x) (-(x)) + +STBTT_DEF int stbtt_PackFontRange(stbtt_pack_context *spc, const unsigned char *fontdata, int font_index, float font_size, + int first_unicode_char_in_range, int num_chars_in_range, stbtt_packedchar *chardata_for_range); +// Creates character bitmaps from the font_index'th font found in fontdata (use +// font_index=0 if you don't know what that is). It creates num_chars_in_range +// bitmaps for characters with unicode values starting at first_unicode_char_in_range +// and increasing. Data for how to render them is stored in chardata_for_range; +// pass these to stbtt_GetPackedQuad to get back renderable quads. +// +// font_size is the full height of the character from ascender to descender, +// as computed by stbtt_ScaleForPixelHeight. To use a point size as computed +// by stbtt_ScaleForMappingEmToPixels, wrap the point size in STBTT_POINT_SIZE() +// and pass that result as 'font_size': +// ..., 20 , ... // font max minus min y is 20 pixels tall +// ..., STBTT_POINT_SIZE(20), ... // 'M' is 20 pixels tall + +typedef struct +{ + float font_size; + int first_unicode_codepoint_in_range; // if non-zero, then the chars are continuous, and this is the first codepoint + int *array_of_unicode_codepoints; // if non-zero, then this is an array of unicode codepoints + int num_chars; + stbtt_packedchar *chardata_for_range; // output + unsigned char h_oversample, v_oversample; // don't set these, they're used internally +} stbtt_pack_range; + +STBTT_DEF int stbtt_PackFontRanges(stbtt_pack_context *spc, const unsigned char *fontdata, int font_index, stbtt_pack_range *ranges, int num_ranges); +// Creates character bitmaps from multiple ranges of characters stored in +// ranges. This will usually create a better-packed bitmap than multiple +// calls to stbtt_PackFontRange. Note that you can call this multiple +// times within a single PackBegin/PackEnd. + +STBTT_DEF void stbtt_PackSetOversampling(stbtt_pack_context *spc, unsigned int h_oversample, unsigned int v_oversample); +// Oversampling a font increases the quality by allowing higher-quality subpixel +// positioning, and is especially valuable at smaller text sizes. +// +// This function sets the amount of oversampling for all following calls to +// stbtt_PackFontRange(s) or stbtt_PackFontRangesGatherRects for a given +// pack context. The default (no oversampling) is achieved by h_oversample=1 +// and v_oversample=1. The total number of pixels required is +// h_oversample*v_oversample larger than the default; for example, 2x2 +// oversampling requires 4x the storage of 1x1. For best results, render +// oversampled textures with bilinear filtering. Look at the readme in +// stb/tests/oversample for information about oversampled fonts +// +// To use with PackFontRangesGather etc., you must set it before calls +// call to PackFontRangesGatherRects. + +STBTT_DEF void stbtt_PackSetSkipMissingCodepoints(stbtt_pack_context *spc, int skip); +// If skip != 0, this tells stb_truetype to skip any codepoints for which +// there is no corresponding glyph. If skip=0, which is the default, then +// codepoints without a glyph recived the font's "missing character" glyph, +// typically an empty box by convention. + +STBTT_DEF void stbtt_GetPackedQuad(const stbtt_packedchar *chardata, int pw, int ph, // same data as above + int char_index, // character to display + float *xpos, float *ypos, // pointers to current position in screen pixel space + stbtt_aligned_quad *q, // output: quad to draw + int align_to_integer); + +STBTT_DEF int stbtt_PackFontRangesGatherRects(stbtt_pack_context *spc, const stbtt_fontinfo *info, stbtt_pack_range *ranges, int num_ranges, stbrp_rect *rects); +STBTT_DEF void stbtt_PackFontRangesPackRects(stbtt_pack_context *spc, stbrp_rect *rects, int num_rects); +STBTT_DEF int stbtt_PackFontRangesRenderIntoRects(stbtt_pack_context *spc, const stbtt_fontinfo *info, stbtt_pack_range *ranges, int num_ranges, stbrp_rect *rects); +// Calling these functions in sequence is roughly equivalent to calling +// stbtt_PackFontRanges(). If you more control over the packing of multiple +// fonts, or if you want to pack custom data into a font texture, take a look +// at the source to of stbtt_PackFontRanges() and create a custom version +// using these functions, e.g. call GatherRects multiple times, +// building up a single array of rects, then call PackRects once, +// then call RenderIntoRects repeatedly. This may result in a +// better packing than calling PackFontRanges multiple times +// (or it may not). + +// this is an opaque structure that you shouldn't mess with which holds +// all the context needed from PackBegin to PackEnd. +struct stbtt_pack_context { + void *user_allocator_context; + void *pack_info; + int width; + int height; + int stride_in_bytes; + int padding; + int skip_missing; + unsigned int h_oversample, v_oversample; + unsigned char *pixels; + void *nodes; +}; + +////////////////////////////////////////////////////////////////////////////// +// +// FONT LOADING +// +// + +STBTT_DEF int stbtt_GetNumberOfFonts(const unsigned char *data); +// This function will determine the number of fonts in a font file. TrueType +// collection (.ttc) files may contain multiple fonts, while TrueType font +// (.ttf) files only contain one font. The number of fonts can be used for +// indexing with the previous function where the index is between zero and one +// less than the total fonts. If an error occurs, -1 is returned. + +STBTT_DEF int stbtt_GetFontOffsetForIndex(const unsigned char *data, int index); +// Each .ttf/.ttc file may have more than one font. Each font has a sequential +// index number starting from 0. Call this function to get the font offset for +// a given index; it returns -1 if the index is out of range. A regular .ttf +// file will only define one font and it always be at offset 0, so it will +// return '0' for index 0, and -1 for all other indices. + +// The following structure is defined publicly so you can declare one on +// the stack or as a global or etc, but you should treat it as opaque. +struct stbtt_fontinfo +{ + void * userdata; + unsigned char * data; // pointer to .ttf file + int fontstart; // offset of start of font + + int numGlyphs; // number of glyphs, needed for range checking + + int loca,head,glyf,hhea,hmtx,kern,gpos,svg; // table locations as offset from start of .ttf + int index_map; // a cmap mapping for our chosen character encoding + int indexToLocFormat; // format needed to map from glyph index to glyph + + stbtt__buf cff; // cff font data + stbtt__buf charstrings; // the charstring index + stbtt__buf gsubrs; // global charstring subroutines index + stbtt__buf subrs; // private charstring subroutines index + stbtt__buf fontdicts; // array of font dicts + stbtt__buf fdselect; // map from glyph to fontdict +}; + +STBTT_DEF int stbtt_InitFont(stbtt_fontinfo *info, const unsigned char *data, int offset); +// Given an offset into the file that defines a font, this function builds +// the necessary cached info for the rest of the system. You must allocate +// the stbtt_fontinfo yourself, and stbtt_InitFont will fill it out. You don't +// need to do anything special to free it, because the contents are pure +// value data with no additional data structures. Returns 0 on failure. + + +////////////////////////////////////////////////////////////////////////////// +// +// CHARACTER TO GLYPH-INDEX CONVERSIOn + +STBTT_DEF int stbtt_FindGlyphIndex(const stbtt_fontinfo *info, int unicode_codepoint); +// If you're going to perform multiple operations on the same character +// and you want a speed-up, call this function with the character you're +// going to process, then use glyph-based functions instead of the +// codepoint-based functions. +// Returns 0 if the character codepoint is not defined in the font. + + +////////////////////////////////////////////////////////////////////////////// +// +// CHARACTER PROPERTIES +// + +STBTT_DEF float stbtt_ScaleForPixelHeight(const stbtt_fontinfo *info, float pixels); +// computes a scale factor to produce a font whose "height" is 'pixels' tall. +// Height is measured as the distance from the highest ascender to the lowest +// descender; in other words, it's equivalent to calling stbtt_GetFontVMetrics +// and computing: +// scale = pixels / (ascent - descent) +// so if you prefer to measure height by the ascent only, use a similar calculation. + +STBTT_DEF float stbtt_ScaleForMappingEmToPixels(const stbtt_fontinfo *info, float pixels); +// computes a scale factor to produce a font whose EM size is mapped to +// 'pixels' tall. This is probably what traditional APIs compute, but +// I'm not positive. + +STBTT_DEF void stbtt_GetFontVMetrics(const stbtt_fontinfo *info, int *ascent, int *descent, int *lineGap); +// ascent is the coordinate above the baseline the font extends; descent +// is the coordinate below the baseline the font extends (i.e. it is typically negative) +// lineGap is the spacing between one row's descent and the next row's ascent... +// so you should advance the vertical position by "*ascent - *descent + *lineGap" +// these are expressed in unscaled coordinates, so you must multiply by +// the scale factor for a given size + +STBTT_DEF int stbtt_GetFontVMetricsOS2(const stbtt_fontinfo *info, int *typoAscent, int *typoDescent, int *typoLineGap); +// analogous to GetFontVMetrics, but returns the "typographic" values from the OS/2 +// table (specific to MS/Windows TTF files). +// +// Returns 1 on success (table present), 0 on failure. + +STBTT_DEF void stbtt_GetFontBoundingBox(const stbtt_fontinfo *info, int *x0, int *y0, int *x1, int *y1); +// the bounding box around all possible characters + +STBTT_DEF void stbtt_GetCodepointHMetrics(const stbtt_fontinfo *info, int codepoint, int *advanceWidth, int *leftSideBearing); +// leftSideBearing is the offset from the current horizontal position to the left edge of the character +// advanceWidth is the offset from the current horizontal position to the next horizontal position +// these are expressed in unscaled coordinates + +STBTT_DEF int stbtt_GetCodepointKernAdvance(const stbtt_fontinfo *info, int ch1, int ch2); +// an additional amount to add to the 'advance' value between ch1 and ch2 + +STBTT_DEF int stbtt_GetCodepointBox(const stbtt_fontinfo *info, int codepoint, int *x0, int *y0, int *x1, int *y1); +// Gets the bounding box of the visible part of the glyph, in unscaled coordinates + +STBTT_DEF void stbtt_GetGlyphHMetrics(const stbtt_fontinfo *info, int glyph_index, int *advanceWidth, int *leftSideBearing); +STBTT_DEF int stbtt_GetGlyphKernAdvance(const stbtt_fontinfo *info, int glyph1, int glyph2); +STBTT_DEF int stbtt_GetGlyphBox(const stbtt_fontinfo *info, int glyph_index, int *x0, int *y0, int *x1, int *y1); +// as above, but takes one or more glyph indices for greater efficiency + +typedef struct stbtt_kerningentry +{ + int glyph1; // use stbtt_FindGlyphIndex + int glyph2; + int advance; +} stbtt_kerningentry; + +STBTT_DEF int stbtt_GetKerningTableLength(const stbtt_fontinfo *info); +STBTT_DEF int stbtt_GetKerningTable(const stbtt_fontinfo *info, stbtt_kerningentry* table, int table_length); +// Retrieves a complete list of all of the kerning pairs provided by the font +// stbtt_GetKerningTable never writes more than table_length entries and returns how many entries it did write. +// The table will be sorted by (a.glyph1 == b.glyph1)?(a.glyph2 < b.glyph2):(a.glyph1 < b.glyph1) + +////////////////////////////////////////////////////////////////////////////// +// +// GLYPH SHAPES (you probably don't need these, but they have to go before +// the bitmaps for C declaration-order reasons) +// + +#ifndef STBTT_vmove // you can predefine these to use different values (but why?) + enum { + STBTT_vmove=1, + STBTT_vline, + STBTT_vcurve, + STBTT_vcubic + }; +#endif + +#ifndef stbtt_vertex // you can predefine this to use different values + // (we share this with other code at RAD) + #define stbtt_vertex_type short // can't use stbtt_int16 because that's not visible in the header file + typedef struct + { + stbtt_vertex_type x,y,cx,cy,cx1,cy1; + unsigned char type,padding; + } stbtt_vertex; +#endif + +STBTT_DEF int stbtt_IsGlyphEmpty(const stbtt_fontinfo *info, int glyph_index); +// returns non-zero if nothing is drawn for this glyph + +STBTT_DEF int stbtt_GetCodepointShape(const stbtt_fontinfo *info, int unicode_codepoint, stbtt_vertex **vertices); +STBTT_DEF int stbtt_GetGlyphShape(const stbtt_fontinfo *info, int glyph_index, stbtt_vertex **vertices); +// returns # of vertices and fills *vertices with the pointer to them +// these are expressed in "unscaled" coordinates +// +// The shape is a series of contours. Each one starts with +// a STBTT_moveto, then consists of a series of mixed +// STBTT_lineto and STBTT_curveto segments. A lineto +// draws a line from previous endpoint to its x,y; a curveto +// draws a quadratic bezier from previous endpoint to +// its x,y, using cx,cy as the bezier control point. + +STBTT_DEF void stbtt_FreeShape(const stbtt_fontinfo *info, stbtt_vertex *vertices); +// frees the data allocated above + +STBTT_DEF unsigned char *stbtt_FindSVGDoc(const stbtt_fontinfo *info, int gl); +STBTT_DEF int stbtt_GetCodepointSVG(const stbtt_fontinfo *info, int unicode_codepoint, const char **svg); +STBTT_DEF int stbtt_GetGlyphSVG(const stbtt_fontinfo *info, int gl, const char **svg); +// fills svg with the character's SVG data. +// returns data size or 0 if SVG not found. + +////////////////////////////////////////////////////////////////////////////// +// +// BITMAP RENDERING +// + +STBTT_DEF void stbtt_FreeBitmap(unsigned char *bitmap, void *userdata); +// frees the bitmap allocated below + +STBTT_DEF unsigned char *stbtt_GetCodepointBitmap(const stbtt_fontinfo *info, float scale_x, float scale_y, int codepoint, int *width, int *height, int *xoff, int *yoff); +// allocates a large-enough single-channel 8bpp bitmap and renders the +// specified character/glyph at the specified scale into it, with +// antialiasing. 0 is no coverage (transparent), 255 is fully covered (opaque). +// *width & *height are filled out with the width & height of the bitmap, +// which is stored left-to-right, top-to-bottom. +// +// xoff/yoff are the offset it pixel space from the glyph origin to the top-left of the bitmap + +STBTT_DEF unsigned char *stbtt_GetCodepointBitmapSubpixel(const stbtt_fontinfo *info, float scale_x, float scale_y, float shift_x, float shift_y, int codepoint, int *width, int *height, int *xoff, int *yoff); +// the same as stbtt_GetCodepoitnBitmap, but you can specify a subpixel +// shift for the character + +STBTT_DEF void stbtt_MakeCodepointBitmap(const stbtt_fontinfo *info, unsigned char *output, int out_w, int out_h, int out_stride, float scale_x, float scale_y, int codepoint); +// the same as stbtt_GetCodepointBitmap, but you pass in storage for the bitmap +// in the form of 'output', with row spacing of 'out_stride' bytes. the bitmap +// is clipped to out_w/out_h bytes. Call stbtt_GetCodepointBitmapBox to get the +// width and height and positioning info for it first. + +STBTT_DEF void stbtt_MakeCodepointBitmapSubpixel(const stbtt_fontinfo *info, unsigned char *output, int out_w, int out_h, int out_stride, float scale_x, float scale_y, float shift_x, float shift_y, int codepoint); +// same as stbtt_MakeCodepointBitmap, but you can specify a subpixel +// shift for the character + +STBTT_DEF void stbtt_MakeCodepointBitmapSubpixelPrefilter(const stbtt_fontinfo *info, unsigned char *output, int out_w, int out_h, int out_stride, float scale_x, float scale_y, float shift_x, float shift_y, int oversample_x, int oversample_y, float *sub_x, float *sub_y, int codepoint); +// same as stbtt_MakeCodepointBitmapSubpixel, but prefiltering +// is performed (see stbtt_PackSetOversampling) + +STBTT_DEF void stbtt_GetCodepointBitmapBox(const stbtt_fontinfo *font, int codepoint, float scale_x, float scale_y, int *ix0, int *iy0, int *ix1, int *iy1); +// get the bbox of the bitmap centered around the glyph origin; so the +// bitmap width is ix1-ix0, height is iy1-iy0, and location to place +// the bitmap top left is (leftSideBearing*scale,iy0). +// (Note that the bitmap uses y-increases-down, but the shape uses +// y-increases-up, so CodepointBitmapBox and CodepointBox are inverted.) + +STBTT_DEF void stbtt_GetCodepointBitmapBoxSubpixel(const stbtt_fontinfo *font, int codepoint, float scale_x, float scale_y, float shift_x, float shift_y, int *ix0, int *iy0, int *ix1, int *iy1); +// same as stbtt_GetCodepointBitmapBox, but you can specify a subpixel +// shift for the character + +// the following functions are equivalent to the above functions, but operate +// on glyph indices instead of Unicode codepoints (for efficiency) +STBTT_DEF unsigned char *stbtt_GetGlyphBitmap(const stbtt_fontinfo *info, float scale_x, float scale_y, int glyph, int *width, int *height, int *xoff, int *yoff); +STBTT_DEF unsigned char *stbtt_GetGlyphBitmapSubpixel(const stbtt_fontinfo *info, float scale_x, float scale_y, float shift_x, float shift_y, int glyph, int *width, int *height, int *xoff, int *yoff); +STBTT_DEF void stbtt_MakeGlyphBitmap(const stbtt_fontinfo *info, unsigned char *output, int out_w, int out_h, int out_stride, float scale_x, float scale_y, int glyph); +STBTT_DEF void stbtt_MakeGlyphBitmapSubpixel(const stbtt_fontinfo *info, unsigned char *output, int out_w, int out_h, int out_stride, float scale_x, float scale_y, float shift_x, float shift_y, int glyph); +STBTT_DEF void stbtt_MakeGlyphBitmapSubpixelPrefilter(const stbtt_fontinfo *info, unsigned char *output, int out_w, int out_h, int out_stride, float scale_x, float scale_y, float shift_x, float shift_y, int oversample_x, int oversample_y, float *sub_x, float *sub_y, int glyph); +STBTT_DEF void stbtt_GetGlyphBitmapBox(const stbtt_fontinfo *font, int glyph, float scale_x, float scale_y, int *ix0, int *iy0, int *ix1, int *iy1); +STBTT_DEF void stbtt_GetGlyphBitmapBoxSubpixel(const stbtt_fontinfo *font, int glyph, float scale_x, float scale_y,float shift_x, float shift_y, int *ix0, int *iy0, int *ix1, int *iy1); + + +// @TODO: don't expose this structure +typedef struct +{ + int w,h,stride; + unsigned char *pixels; +} stbtt__bitmap; + +// rasterize a shape with quadratic beziers into a bitmap +STBTT_DEF void stbtt_Rasterize(stbtt__bitmap *result, // 1-channel bitmap to draw into + float flatness_in_pixels, // allowable error of curve in pixels + stbtt_vertex *vertices, // array of vertices defining shape + int num_verts, // number of vertices in above array + float scale_x, float scale_y, // scale applied to input vertices + float shift_x, float shift_y, // translation applied to input vertices + int x_off, int y_off, // another translation applied to input + int invert, // if non-zero, vertically flip shape + void *userdata); // context for to STBTT_MALLOC + +////////////////////////////////////////////////////////////////////////////// +// +// Signed Distance Function (or Field) rendering + +STBTT_DEF void stbtt_FreeSDF(unsigned char *bitmap, void *userdata); +// frees the SDF bitmap allocated below + +STBTT_DEF unsigned char * stbtt_GetGlyphSDF(const stbtt_fontinfo *info, float scale, int glyph, int padding, unsigned char onedge_value, float pixel_dist_scale, int *width, int *height, int *xoff, int *yoff); +STBTT_DEF unsigned char * stbtt_GetCodepointSDF(const stbtt_fontinfo *info, float scale, int codepoint, int padding, unsigned char onedge_value, float pixel_dist_scale, int *width, int *height, int *xoff, int *yoff); +// These functions compute a discretized SDF field for a single character, suitable for storing +// in a single-channel texture, sampling with bilinear filtering, and testing against +// larger than some threshold to produce scalable fonts. +// info -- the font +// scale -- controls the size of the resulting SDF bitmap, same as it would be creating a regular bitmap +// glyph/codepoint -- the character to generate the SDF for +// padding -- extra "pixels" around the character which are filled with the distance to the character (not 0), +// which allows effects like bit outlines +// onedge_value -- value 0-255 to test the SDF against to reconstruct the character (i.e. the isocontour of the character) +// pixel_dist_scale -- what value the SDF should increase by when moving one SDF "pixel" away from the edge (on the 0..255 scale) +// if positive, > onedge_value is inside; if negative, < onedge_value is inside +// width,height -- output height & width of the SDF bitmap (including padding) +// xoff,yoff -- output origin of the character +// return value -- a 2D array of bytes 0..255, width*height in size +// +// pixel_dist_scale & onedge_value are a scale & bias that allows you to make +// optimal use of the limited 0..255 for your application, trading off precision +// and special effects. SDF values outside the range 0..255 are clamped to 0..255. +// +// Example: +// scale = stbtt_ScaleForPixelHeight(22) +// padding = 5 +// onedge_value = 180 +// pixel_dist_scale = 180/5.0 = 36.0 +// +// This will create an SDF bitmap in which the character is about 22 pixels +// high but the whole bitmap is about 22+5+5=32 pixels high. To produce a filled +// shape, sample the SDF at each pixel and fill the pixel if the SDF value +// is greater than or equal to 180/255. (You'll actually want to antialias, +// which is beyond the scope of this example.) Additionally, you can compute +// offset outlines (e.g. to stroke the character border inside & outside, +// or only outside). For example, to fill outside the character up to 3 SDF +// pixels, you would compare against (180-36.0*3)/255 = 72/255. The above +// choice of variables maps a range from 5 pixels outside the shape to +// 2 pixels inside the shape to 0..255; this is intended primarily for apply +// outside effects only (the interior range is needed to allow proper +// antialiasing of the font at *smaller* sizes) +// +// The function computes the SDF analytically at each SDF pixel, not by e.g. +// building a higher-res bitmap and approximating it. In theory the quality +// should be as high as possible for an SDF of this size & representation, but +// unclear if this is true in practice (perhaps building a higher-res bitmap +// and computing from that can allow drop-out prevention). +// +// The algorithm has not been optimized at all, so expect it to be slow +// if computing lots of characters or very large sizes. + + + +////////////////////////////////////////////////////////////////////////////// +// +// Finding the right font... +// +// You should really just solve this offline, keep your own tables +// of what font is what, and don't try to get it out of the .ttf file. +// That's because getting it out of the .ttf file is really hard, because +// the names in the file can appear in many possible encodings, in many +// possible languages, and e.g. if you need a case-insensitive comparison, +// the details of that depend on the encoding & language in a complex way +// (actually underspecified in truetype, but also gigantic). +// +// But you can use the provided functions in two possible ways: +// stbtt_FindMatchingFont() will use *case-sensitive* comparisons on +// unicode-encoded names to try to find the font you want; +// you can run this before calling stbtt_InitFont() +// +// stbtt_GetFontNameString() lets you get any of the various strings +// from the file yourself and do your own comparisons on them. +// You have to have called stbtt_InitFont() first. + + +STBTT_DEF int stbtt_FindMatchingFont(const unsigned char *fontdata, const char *name, int flags); +// returns the offset (not index) of the font that matches, or -1 if none +// if you use STBTT_MACSTYLE_DONTCARE, use a font name like "Arial Bold". +// if you use any other flag, use a font name like "Arial"; this checks +// the 'macStyle' header field; i don't know if fonts set this consistently +#define STBTT_MACSTYLE_DONTCARE 0 +#define STBTT_MACSTYLE_BOLD 1 +#define STBTT_MACSTYLE_ITALIC 2 +#define STBTT_MACSTYLE_UNDERSCORE 4 +#define STBTT_MACSTYLE_NONE 8 // <= not same as 0, this makes us check the bitfield is 0 + +STBTT_DEF int stbtt_CompareUTF8toUTF16_bigendian(const char *s1, int len1, const char *s2, int len2); +// returns 1/0 whether the first string interpreted as utf8 is identical to +// the second string interpreted as big-endian utf16... useful for strings from next func + +STBTT_DEF const char *stbtt_GetFontNameString(const stbtt_fontinfo *font, int *length, int platformID, int encodingID, int languageID, int nameID); +// returns the string (which may be big-endian double byte, e.g. for unicode) +// and puts the length in bytes in *length. +// +// some of the values for the IDs are below; for more see the truetype spec: +// http://developer.apple.com/textfonts/TTRefMan/RM06/Chap6name.html +// http://www.microsoft.com/typography/otspec/name.htm + +enum { // platformID + STBTT_PLATFORM_ID_UNICODE =0, + STBTT_PLATFORM_ID_MAC =1, + STBTT_PLATFORM_ID_ISO =2, + STBTT_PLATFORM_ID_MICROSOFT =3 +}; + +enum { // encodingID for STBTT_PLATFORM_ID_UNICODE + STBTT_UNICODE_EID_UNICODE_1_0 =0, + STBTT_UNICODE_EID_UNICODE_1_1 =1, + STBTT_UNICODE_EID_ISO_10646 =2, + STBTT_UNICODE_EID_UNICODE_2_0_BMP=3, + STBTT_UNICODE_EID_UNICODE_2_0_FULL=4 +}; + +enum { // encodingID for STBTT_PLATFORM_ID_MICROSOFT + STBTT_MS_EID_SYMBOL =0, + STBTT_MS_EID_UNICODE_BMP =1, + STBTT_MS_EID_SHIFTJIS =2, + STBTT_MS_EID_UNICODE_FULL =10 +}; + +enum { // encodingID for STBTT_PLATFORM_ID_MAC; same as Script Manager codes + STBTT_MAC_EID_ROMAN =0, STBTT_MAC_EID_ARABIC =4, + STBTT_MAC_EID_JAPANESE =1, STBTT_MAC_EID_HEBREW =5, + STBTT_MAC_EID_CHINESE_TRAD =2, STBTT_MAC_EID_GREEK =6, + STBTT_MAC_EID_KOREAN =3, STBTT_MAC_EID_RUSSIAN =7 +}; + +enum { // languageID for STBTT_PLATFORM_ID_MICROSOFT; same as LCID... + // problematic because there are e.g. 16 english LCIDs and 16 arabic LCIDs + STBTT_MS_LANG_ENGLISH =0x0409, STBTT_MS_LANG_ITALIAN =0x0410, + STBTT_MS_LANG_CHINESE =0x0804, STBTT_MS_LANG_JAPANESE =0x0411, + STBTT_MS_LANG_DUTCH =0x0413, STBTT_MS_LANG_KOREAN =0x0412, + STBTT_MS_LANG_FRENCH =0x040c, STBTT_MS_LANG_RUSSIAN =0x0419, + STBTT_MS_LANG_GERMAN =0x0407, STBTT_MS_LANG_SPANISH =0x0409, + STBTT_MS_LANG_HEBREW =0x040d, STBTT_MS_LANG_SWEDISH =0x041D +}; + +enum { // languageID for STBTT_PLATFORM_ID_MAC + STBTT_MAC_LANG_ENGLISH =0 , STBTT_MAC_LANG_JAPANESE =11, + STBTT_MAC_LANG_ARABIC =12, STBTT_MAC_LANG_KOREAN =23, + STBTT_MAC_LANG_DUTCH =4 , STBTT_MAC_LANG_RUSSIAN =32, + STBTT_MAC_LANG_FRENCH =1 , STBTT_MAC_LANG_SPANISH =6 , + STBTT_MAC_LANG_GERMAN =2 , STBTT_MAC_LANG_SWEDISH =5 , + STBTT_MAC_LANG_HEBREW =10, STBTT_MAC_LANG_CHINESE_SIMPLIFIED =33, + STBTT_MAC_LANG_ITALIAN =3 , STBTT_MAC_LANG_CHINESE_TRAD =19 +}; + +#ifdef __cplusplus +} +#endif + +#endif // __STB_INCLUDE_STB_TRUETYPE_H__ + +/////////////////////////////////////////////////////////////////////////////// +/////////////////////////////////////////////////////////////////////////////// +//// +//// IMPLEMENTATION +//// +//// + +#ifdef STB_TRUETYPE_IMPLEMENTATION + +#ifndef STBTT_MAX_OVERSAMPLE +#define STBTT_MAX_OVERSAMPLE 8 +#endif + +#if STBTT_MAX_OVERSAMPLE > 255 +#error "STBTT_MAX_OVERSAMPLE cannot be > 255" +#endif + +typedef int stbtt__test_oversample_pow2[(STBTT_MAX_OVERSAMPLE & (STBTT_MAX_OVERSAMPLE-1)) == 0 ? 1 : -1]; + +#ifndef STBTT_RASTERIZER_VERSION +#define STBTT_RASTERIZER_VERSION 2 +#endif + +#ifdef _MSC_VER +#define STBTT__NOTUSED(v) (void)(v) +#else +#define STBTT__NOTUSED(v) (void)sizeof(v) +#endif + +////////////////////////////////////////////////////////////////////////// +// +// stbtt__buf helpers to parse data from file +// + +static stbtt_uint8 stbtt__buf_get8(stbtt__buf *b) +{ + if (b->cursor >= b->size) + return 0; + return b->data[b->cursor++]; +} + +static stbtt_uint8 stbtt__buf_peek8(stbtt__buf *b) +{ + if (b->cursor >= b->size) + return 0; + return b->data[b->cursor]; +} + +static void stbtt__buf_seek(stbtt__buf *b, int o) +{ + STBTT_assert(!(o > b->size || o < 0)); + b->cursor = (o > b->size || o < 0) ? b->size : o; +} + +static void stbtt__buf_skip(stbtt__buf *b, int o) +{ + stbtt__buf_seek(b, b->cursor + o); +} + +static stbtt_uint32 stbtt__buf_get(stbtt__buf *b, int n) +{ + stbtt_uint32 v = 0; + int i; + STBTT_assert(n >= 1 && n <= 4); + for (i = 0; i < n; i++) + v = (v << 8) | stbtt__buf_get8(b); + return v; +} + +static stbtt__buf stbtt__new_buf(const void *p, size_t size) +{ + stbtt__buf r; + STBTT_assert(size < 0x40000000); + r.data = (stbtt_uint8*) p; + r.size = (int) size; + r.cursor = 0; + return r; +} + +#define stbtt__buf_get16(b) stbtt__buf_get((b), 2) +#define stbtt__buf_get32(b) stbtt__buf_get((b), 4) + +static stbtt__buf stbtt__buf_range(const stbtt__buf *b, int o, int s) +{ + stbtt__buf r = stbtt__new_buf(NULL, 0); + if (o < 0 || s < 0 || o > b->size || s > b->size - o) return r; + r.data = b->data + o; + r.size = s; + return r; +} + +static stbtt__buf stbtt__cff_get_index(stbtt__buf *b) +{ + int count, start, offsize; + start = b->cursor; + count = stbtt__buf_get16(b); + if (count) { + offsize = stbtt__buf_get8(b); + STBTT_assert(offsize >= 1 && offsize <= 4); + stbtt__buf_skip(b, offsize * count); + stbtt__buf_skip(b, stbtt__buf_get(b, offsize) - 1); + } + return stbtt__buf_range(b, start, b->cursor - start); +} + +static stbtt_uint32 stbtt__cff_int(stbtt__buf *b) +{ + int b0 = stbtt__buf_get8(b); + if (b0 >= 32 && b0 <= 246) return b0 - 139; + else if (b0 >= 247 && b0 <= 250) return (b0 - 247)*256 + stbtt__buf_get8(b) + 108; + else if (b0 >= 251 && b0 <= 254) return -(b0 - 251)*256 - stbtt__buf_get8(b) - 108; + else if (b0 == 28) return stbtt__buf_get16(b); + else if (b0 == 29) return stbtt__buf_get32(b); + STBTT_assert(0); + return 0; +} + +static void stbtt__cff_skip_operand(stbtt__buf *b) { + int v, b0 = stbtt__buf_peek8(b); + STBTT_assert(b0 >= 28); + if (b0 == 30) { + stbtt__buf_skip(b, 1); + while (b->cursor < b->size) { + v = stbtt__buf_get8(b); + if ((v & 0xF) == 0xF || (v >> 4) == 0xF) + break; + } + } else { + stbtt__cff_int(b); + } +} + +static stbtt__buf stbtt__dict_get(stbtt__buf *b, int key) +{ + stbtt__buf_seek(b, 0); + while (b->cursor < b->size) { + int start = b->cursor, end, op; + while (stbtt__buf_peek8(b) >= 28) + stbtt__cff_skip_operand(b); + end = b->cursor; + op = stbtt__buf_get8(b); + if (op == 12) op = stbtt__buf_get8(b) | 0x100; + if (op == key) return stbtt__buf_range(b, start, end-start); + } + return stbtt__buf_range(b, 0, 0); +} + +static void stbtt__dict_get_ints(stbtt__buf *b, int key, int outcount, stbtt_uint32 *out) +{ + int i; + stbtt__buf operands = stbtt__dict_get(b, key); + for (i = 0; i < outcount && operands.cursor < operands.size; i++) + out[i] = stbtt__cff_int(&operands); +} + +static int stbtt__cff_index_count(stbtt__buf *b) +{ + stbtt__buf_seek(b, 0); + return stbtt__buf_get16(b); +} + +static stbtt__buf stbtt__cff_index_get(stbtt__buf b, int i) +{ + int count, offsize, start, end; + stbtt__buf_seek(&b, 0); + count = stbtt__buf_get16(&b); + offsize = stbtt__buf_get8(&b); + STBTT_assert(i >= 0 && i < count); + STBTT_assert(offsize >= 1 && offsize <= 4); + stbtt__buf_skip(&b, i*offsize); + start = stbtt__buf_get(&b, offsize); + end = stbtt__buf_get(&b, offsize); + return stbtt__buf_range(&b, 2+(count+1)*offsize+start, end - start); +} + +////////////////////////////////////////////////////////////////////////// +// +// accessors to parse data from file +// + +// on platforms that don't allow misaligned reads, if we want to allow +// truetype fonts that aren't padded to alignment, define ALLOW_UNALIGNED_TRUETYPE + +#define ttBYTE(p) (* (stbtt_uint8 *) (p)) +#define ttCHAR(p) (* (stbtt_int8 *) (p)) +#define ttFixed(p) ttLONG(p) + +static stbtt_uint16 ttUSHORT(stbtt_uint8 *p) { return p[0]*256 + p[1]; } +static stbtt_int16 ttSHORT(stbtt_uint8 *p) { return p[0]*256 + p[1]; } +static stbtt_uint32 ttULONG(stbtt_uint8 *p) { return (p[0]<<24) + (p[1]<<16) + (p[2]<<8) + p[3]; } +static stbtt_int32 ttLONG(stbtt_uint8 *p) { return (p[0]<<24) + (p[1]<<16) + (p[2]<<8) + p[3]; } + +#define stbtt_tag4(p,c0,c1,c2,c3) ((p)[0] == (c0) && (p)[1] == (c1) && (p)[2] == (c2) && (p)[3] == (c3)) +#define stbtt_tag(p,str) stbtt_tag4(p,str[0],str[1],str[2],str[3]) + +static int stbtt__isfont(stbtt_uint8 *font) +{ + // check the version number + if (stbtt_tag4(font, '1',0,0,0)) return 1; // TrueType 1 + if (stbtt_tag(font, "typ1")) return 1; // TrueType with type 1 font -- we don't support this! + if (stbtt_tag(font, "OTTO")) return 1; // OpenType with CFF + if (stbtt_tag4(font, 0,1,0,0)) return 1; // OpenType 1.0 + if (stbtt_tag(font, "true")) return 1; // Apple specification for TrueType fonts + return 0; +} + +// @OPTIMIZE: binary search +static stbtt_uint32 stbtt__find_table(stbtt_uint8 *data, stbtt_uint32 fontstart, const char *tag) +{ + stbtt_int32 num_tables = ttUSHORT(data+fontstart+4); + stbtt_uint32 tabledir = fontstart + 12; + stbtt_int32 i; + for (i=0; i < num_tables; ++i) { + stbtt_uint32 loc = tabledir + 16*i; + if (stbtt_tag(data+loc+0, tag)) + return ttULONG(data+loc+8); + } + return 0; +} + +static int stbtt_GetFontOffsetForIndex_internal(unsigned char *font_collection, int index) +{ + // if it's just a font, there's only one valid index + if (stbtt__isfont(font_collection)) + return index == 0 ? 0 : -1; + + // check if it's a TTC + if (stbtt_tag(font_collection, "ttcf")) { + // version 1? + if (ttULONG(font_collection+4) == 0x00010000 || ttULONG(font_collection+4) == 0x00020000) { + stbtt_int32 n = ttLONG(font_collection+8); + if (index >= n) + return -1; + return ttULONG(font_collection+12+index*4); + } + } + return -1; +} + +static int stbtt_GetNumberOfFonts_internal(unsigned char *font_collection) +{ + // if it's just a font, there's only one valid font + if (stbtt__isfont(font_collection)) + return 1; + + // check if it's a TTC + if (stbtt_tag(font_collection, "ttcf")) { + // version 1? + if (ttULONG(font_collection+4) == 0x00010000 || ttULONG(font_collection+4) == 0x00020000) { + return ttLONG(font_collection+8); + } + } + return 0; +} + +static stbtt__buf stbtt__get_subrs(stbtt__buf cff, stbtt__buf fontdict) +{ + stbtt_uint32 subrsoff = 0, private_loc[2] = { 0, 0 }; + stbtt__buf pdict; + stbtt__dict_get_ints(&fontdict, 18, 2, private_loc); + if (!private_loc[1] || !private_loc[0]) return stbtt__new_buf(NULL, 0); + pdict = stbtt__buf_range(&cff, private_loc[1], private_loc[0]); + stbtt__dict_get_ints(&pdict, 19, 1, &subrsoff); + if (!subrsoff) return stbtt__new_buf(NULL, 0); + stbtt__buf_seek(&cff, private_loc[1]+subrsoff); + return stbtt__cff_get_index(&cff); +} + +// since most people won't use this, find this table the first time it's needed +static int stbtt__get_svg(stbtt_fontinfo *info) +{ + stbtt_uint32 t; + if (info->svg < 0) { + t = stbtt__find_table(info->data, info->fontstart, "SVG "); + if (t) { + stbtt_uint32 offset = ttULONG(info->data + t + 2); + info->svg = t + offset; + } else { + info->svg = 0; + } + } + return info->svg; +} + +static int stbtt_InitFont_internal(stbtt_fontinfo *info, unsigned char *data, int fontstart) +{ + stbtt_uint32 cmap, t; + stbtt_int32 i,numTables; + + info->data = data; + info->fontstart = fontstart; + info->cff = stbtt__new_buf(NULL, 0); + + cmap = stbtt__find_table(data, fontstart, "cmap"); // required + info->loca = stbtt__find_table(data, fontstart, "loca"); // required + info->head = stbtt__find_table(data, fontstart, "head"); // required + info->glyf = stbtt__find_table(data, fontstart, "glyf"); // required + info->hhea = stbtt__find_table(data, fontstart, "hhea"); // required + info->hmtx = stbtt__find_table(data, fontstart, "hmtx"); // required + info->kern = stbtt__find_table(data, fontstart, "kern"); // not required + info->gpos = stbtt__find_table(data, fontstart, "GPOS"); // not required + + if (!cmap || !info->head || !info->hhea || !info->hmtx) + return 0; + if (info->glyf) { + // required for truetype + if (!info->loca) return 0; + } else { + // initialization for CFF / Type2 fonts (OTF) + stbtt__buf b, topdict, topdictidx; + stbtt_uint32 cstype = 2, charstrings = 0, fdarrayoff = 0, fdselectoff = 0; + stbtt_uint32 cff; + + cff = stbtt__find_table(data, fontstart, "CFF "); + if (!cff) return 0; + + info->fontdicts = stbtt__new_buf(NULL, 0); + info->fdselect = stbtt__new_buf(NULL, 0); + + // @TODO this should use size from table (not 512MB) + info->cff = stbtt__new_buf(data+cff, 512*1024*1024); + b = info->cff; + + // read the header + stbtt__buf_skip(&b, 2); + stbtt__buf_seek(&b, stbtt__buf_get8(&b)); // hdrsize + + // @TODO the name INDEX could list multiple fonts, + // but we just use the first one. + stbtt__cff_get_index(&b); // name INDEX + topdictidx = stbtt__cff_get_index(&b); + topdict = stbtt__cff_index_get(topdictidx, 0); + stbtt__cff_get_index(&b); // string INDEX + info->gsubrs = stbtt__cff_get_index(&b); + + stbtt__dict_get_ints(&topdict, 17, 1, &charstrings); + stbtt__dict_get_ints(&topdict, 0x100 | 6, 1, &cstype); + stbtt__dict_get_ints(&topdict, 0x100 | 36, 1, &fdarrayoff); + stbtt__dict_get_ints(&topdict, 0x100 | 37, 1, &fdselectoff); + info->subrs = stbtt__get_subrs(b, topdict); + + // we only support Type 2 charstrings + if (cstype != 2) return 0; + if (charstrings == 0) return 0; + + if (fdarrayoff) { + // looks like a CID font + if (!fdselectoff) return 0; + stbtt__buf_seek(&b, fdarrayoff); + info->fontdicts = stbtt__cff_get_index(&b); + info->fdselect = stbtt__buf_range(&b, fdselectoff, b.size-fdselectoff); + } + + stbtt__buf_seek(&b, charstrings); + info->charstrings = stbtt__cff_get_index(&b); + } + + t = stbtt__find_table(data, fontstart, "maxp"); + if (t) + info->numGlyphs = ttUSHORT(data+t+4); + else + info->numGlyphs = 0xffff; + + info->svg = -1; + + // find a cmap encoding table we understand *now* to avoid searching + // later. (todo: could make this installable) + // the same regardless of glyph. + numTables = ttUSHORT(data + cmap + 2); + info->index_map = 0; + for (i=0; i < numTables; ++i) { + stbtt_uint32 encoding_record = cmap + 4 + 8 * i; + // find an encoding we understand: + switch(ttUSHORT(data+encoding_record)) { + case STBTT_PLATFORM_ID_MICROSOFT: + switch (ttUSHORT(data+encoding_record+2)) { + case STBTT_MS_EID_UNICODE_BMP: + case STBTT_MS_EID_UNICODE_FULL: + // MS/Unicode + info->index_map = cmap + ttULONG(data+encoding_record+4); + break; + } + break; + case STBTT_PLATFORM_ID_UNICODE: + // Mac/iOS has these + // all the encodingIDs are unicode, so we don't bother to check it + info->index_map = cmap + ttULONG(data+encoding_record+4); + break; + } + } + if (info->index_map == 0) + return 0; + + info->indexToLocFormat = ttUSHORT(data+info->head + 50); + return 1; +} + +STBTT_DEF int stbtt_FindGlyphIndex(const stbtt_fontinfo *info, int unicode_codepoint) +{ + stbtt_uint8 *data = info->data; + stbtt_uint32 index_map = info->index_map; + + stbtt_uint16 format = ttUSHORT(data + index_map + 0); + if (format == 0) { // apple byte encoding + stbtt_int32 bytes = ttUSHORT(data + index_map + 2); + if (unicode_codepoint < bytes-6) + return ttBYTE(data + index_map + 6 + unicode_codepoint); + return 0; + } else if (format == 6) { + stbtt_uint32 first = ttUSHORT(data + index_map + 6); + stbtt_uint32 count = ttUSHORT(data + index_map + 8); + if ((stbtt_uint32) unicode_codepoint >= first && (stbtt_uint32) unicode_codepoint < first+count) + return ttUSHORT(data + index_map + 10 + (unicode_codepoint - first)*2); + return 0; + } else if (format == 2) { + STBTT_assert(0); // @TODO: high-byte mapping for japanese/chinese/korean + return 0; + } else if (format == 4) { // standard mapping for windows fonts: binary search collection of ranges + stbtt_uint16 segcount = ttUSHORT(data+index_map+6) >> 1; + stbtt_uint16 searchRange = ttUSHORT(data+index_map+8) >> 1; + stbtt_uint16 entrySelector = ttUSHORT(data+index_map+10); + stbtt_uint16 rangeShift = ttUSHORT(data+index_map+12) >> 1; + + // do a binary search of the segments + stbtt_uint32 endCount = index_map + 14; + stbtt_uint32 search = endCount; + + if (unicode_codepoint > 0xffff) + return 0; + + // they lie from endCount .. endCount + segCount + // but searchRange is the nearest power of two, so... + if (unicode_codepoint >= ttUSHORT(data + search + rangeShift*2)) + search += rangeShift*2; + + // now decrement to bias correctly to find smallest + search -= 2; + while (entrySelector) { + stbtt_uint16 end; + searchRange >>= 1; + end = ttUSHORT(data + search + searchRange*2); + if (unicode_codepoint > end) + search += searchRange*2; + --entrySelector; + } + search += 2; + + { + stbtt_uint16 offset, start, last; + stbtt_uint16 item = (stbtt_uint16) ((search - endCount) >> 1); + + start = ttUSHORT(data + index_map + 14 + segcount*2 + 2 + 2*item); + last = ttUSHORT(data + endCount + 2*item); + if (unicode_codepoint < start || unicode_codepoint > last) + return 0; + + offset = ttUSHORT(data + index_map + 14 + segcount*6 + 2 + 2*item); + if (offset == 0) + return (stbtt_uint16) (unicode_codepoint + ttSHORT(data + index_map + 14 + segcount*4 + 2 + 2*item)); + + return ttUSHORT(data + offset + (unicode_codepoint-start)*2 + index_map + 14 + segcount*6 + 2 + 2*item); + } + } else if (format == 12 || format == 13) { + stbtt_uint32 ngroups = ttULONG(data+index_map+12); + stbtt_int32 low,high; + low = 0; high = (stbtt_int32)ngroups; + // Binary search the right group. + while (low < high) { + stbtt_int32 mid = low + ((high-low) >> 1); // rounds down, so low <= mid < high + stbtt_uint32 start_char = ttULONG(data+index_map+16+mid*12); + stbtt_uint32 end_char = ttULONG(data+index_map+16+mid*12+4); + if ((stbtt_uint32) unicode_codepoint < start_char) + high = mid; + else if ((stbtt_uint32) unicode_codepoint > end_char) + low = mid+1; + else { + stbtt_uint32 start_glyph = ttULONG(data+index_map+16+mid*12+8); + if (format == 12) + return start_glyph + unicode_codepoint-start_char; + else // format == 13 + return start_glyph; + } + } + return 0; // not found + } + // @TODO + STBTT_assert(0); + return 0; +} + +STBTT_DEF int stbtt_GetCodepointShape(const stbtt_fontinfo *info, int unicode_codepoint, stbtt_vertex **vertices) +{ + return stbtt_GetGlyphShape(info, stbtt_FindGlyphIndex(info, unicode_codepoint), vertices); +} + +static void stbtt_setvertex(stbtt_vertex *v, stbtt_uint8 type, stbtt_int32 x, stbtt_int32 y, stbtt_int32 cx, stbtt_int32 cy) +{ + v->type = type; + v->x = (stbtt_int16) x; + v->y = (stbtt_int16) y; + v->cx = (stbtt_int16) cx; + v->cy = (stbtt_int16) cy; +} + +static int stbtt__GetGlyfOffset(const stbtt_fontinfo *info, int glyph_index) +{ + int g1,g2; + + STBTT_assert(!info->cff.size); + + if (glyph_index >= info->numGlyphs) return -1; // glyph index out of range + if (info->indexToLocFormat >= 2) return -1; // unknown index->glyph map format + + if (info->indexToLocFormat == 0) { + g1 = info->glyf + ttUSHORT(info->data + info->loca + glyph_index * 2) * 2; + g2 = info->glyf + ttUSHORT(info->data + info->loca + glyph_index * 2 + 2) * 2; + } else { + g1 = info->glyf + ttULONG (info->data + info->loca + glyph_index * 4); + g2 = info->glyf + ttULONG (info->data + info->loca + glyph_index * 4 + 4); + } + + return g1==g2 ? -1 : g1; // if length is 0, return -1 +} + +static int stbtt__GetGlyphInfoT2(const stbtt_fontinfo *info, int glyph_index, int *x0, int *y0, int *x1, int *y1); + +STBTT_DEF int stbtt_GetGlyphBox(const stbtt_fontinfo *info, int glyph_index, int *x0, int *y0, int *x1, int *y1) +{ + if (info->cff.size) { + stbtt__GetGlyphInfoT2(info, glyph_index, x0, y0, x1, y1); + } else { + int g = stbtt__GetGlyfOffset(info, glyph_index); + if (g < 0) return 0; + + if (x0) *x0 = ttSHORT(info->data + g + 2); + if (y0) *y0 = ttSHORT(info->data + g + 4); + if (x1) *x1 = ttSHORT(info->data + g + 6); + if (y1) *y1 = ttSHORT(info->data + g + 8); + } + return 1; +} + +STBTT_DEF int stbtt_GetCodepointBox(const stbtt_fontinfo *info, int codepoint, int *x0, int *y0, int *x1, int *y1) +{ + return stbtt_GetGlyphBox(info, stbtt_FindGlyphIndex(info,codepoint), x0,y0,x1,y1); +} + +STBTT_DEF int stbtt_IsGlyphEmpty(const stbtt_fontinfo *info, int glyph_index) +{ + stbtt_int16 numberOfContours; + int g; + if (info->cff.size) + return stbtt__GetGlyphInfoT2(info, glyph_index, NULL, NULL, NULL, NULL) == 0; + g = stbtt__GetGlyfOffset(info, glyph_index); + if (g < 0) return 1; + numberOfContours = ttSHORT(info->data + g); + return numberOfContours == 0; +} + +static int stbtt__close_shape(stbtt_vertex *vertices, int num_vertices, int was_off, int start_off, + stbtt_int32 sx, stbtt_int32 sy, stbtt_int32 scx, stbtt_int32 scy, stbtt_int32 cx, stbtt_int32 cy) +{ + if (start_off) { + if (was_off) + stbtt_setvertex(&vertices[num_vertices++], STBTT_vcurve, (cx+scx)>>1, (cy+scy)>>1, cx,cy); + stbtt_setvertex(&vertices[num_vertices++], STBTT_vcurve, sx,sy,scx,scy); + } else { + if (was_off) + stbtt_setvertex(&vertices[num_vertices++], STBTT_vcurve,sx,sy,cx,cy); + else + stbtt_setvertex(&vertices[num_vertices++], STBTT_vline,sx,sy,0,0); + } + return num_vertices; +} + +static int stbtt__GetGlyphShapeTT(const stbtt_fontinfo *info, int glyph_index, stbtt_vertex **pvertices) +{ + stbtt_int16 numberOfContours; + stbtt_uint8 *endPtsOfContours; + stbtt_uint8 *data = info->data; + stbtt_vertex *vertices=0; + int num_vertices=0; + int g = stbtt__GetGlyfOffset(info, glyph_index); + + *pvertices = NULL; + + if (g < 0) return 0; + + numberOfContours = ttSHORT(data + g); + + if (numberOfContours > 0) { + stbtt_uint8 flags=0,flagcount; + stbtt_int32 ins, i,j=0,m,n, next_move, was_off=0, off, start_off=0; + stbtt_int32 x,y,cx,cy,sx,sy, scx,scy; + stbtt_uint8 *points; + endPtsOfContours = (data + g + 10); + ins = ttUSHORT(data + g + 10 + numberOfContours * 2); + points = data + g + 10 + numberOfContours * 2 + 2 + ins; + + n = 1+ttUSHORT(endPtsOfContours + numberOfContours*2-2); + + m = n + 2*numberOfContours; // a loose bound on how many vertices we might need + vertices = (stbtt_vertex *) STBTT_malloc(m * sizeof(vertices[0]), info->userdata); + if (vertices == 0) + return 0; + + next_move = 0; + flagcount=0; + + // in first pass, we load uninterpreted data into the allocated array + // above, shifted to the end of the array so we won't overwrite it when + // we create our final data starting from the front + + off = m - n; // starting offset for uninterpreted data, regardless of how m ends up being calculated + + // first load flags + + for (i=0; i < n; ++i) { + if (flagcount == 0) { + flags = *points++; + if (flags & 8) + flagcount = *points++; + } else + --flagcount; + vertices[off+i].type = flags; + } + + // now load x coordinates + x=0; + for (i=0; i < n; ++i) { + flags = vertices[off+i].type; + if (flags & 2) { + stbtt_int16 dx = *points++; + x += (flags & 16) ? dx : -dx; // ??? + } else { + if (!(flags & 16)) { + x = x + (stbtt_int16) (points[0]*256 + points[1]); + points += 2; + } + } + vertices[off+i].x = (stbtt_int16) x; + } + + // now load y coordinates + y=0; + for (i=0; i < n; ++i) { + flags = vertices[off+i].type; + if (flags & 4) { + stbtt_int16 dy = *points++; + y += (flags & 32) ? dy : -dy; // ??? + } else { + if (!(flags & 32)) { + y = y + (stbtt_int16) (points[0]*256 + points[1]); + points += 2; + } + } + vertices[off+i].y = (stbtt_int16) y; + } + + // now convert them to our format + num_vertices=0; + sx = sy = cx = cy = scx = scy = 0; + for (i=0; i < n; ++i) { + flags = vertices[off+i].type; + x = (stbtt_int16) vertices[off+i].x; + y = (stbtt_int16) vertices[off+i].y; + + if (next_move == i) { + if (i != 0) + num_vertices = stbtt__close_shape(vertices, num_vertices, was_off, start_off, sx,sy,scx,scy,cx,cy); + + // now start the new one + start_off = !(flags & 1); + if (start_off) { + // if we start off with an off-curve point, then when we need to find a point on the curve + // where we can start, and we need to save some state for when we wraparound. + scx = x; + scy = y; + if (!(vertices[off+i+1].type & 1)) { + // next point is also a curve point, so interpolate an on-point curve + sx = (x + (stbtt_int32) vertices[off+i+1].x) >> 1; + sy = (y + (stbtt_int32) vertices[off+i+1].y) >> 1; + } else { + // otherwise just use the next point as our start point + sx = (stbtt_int32) vertices[off+i+1].x; + sy = (stbtt_int32) vertices[off+i+1].y; + ++i; // we're using point i+1 as the starting point, so skip it + } + } else { + sx = x; + sy = y; + } + stbtt_setvertex(&vertices[num_vertices++], STBTT_vmove,sx,sy,0,0); + was_off = 0; + next_move = 1 + ttUSHORT(endPtsOfContours+j*2); + ++j; + } else { + if (!(flags & 1)) { // if it's a curve + if (was_off) // two off-curve control points in a row means interpolate an on-curve midpoint + stbtt_setvertex(&vertices[num_vertices++], STBTT_vcurve, (cx+x)>>1, (cy+y)>>1, cx, cy); + cx = x; + cy = y; + was_off = 1; + } else { + if (was_off) + stbtt_setvertex(&vertices[num_vertices++], STBTT_vcurve, x,y, cx, cy); + else + stbtt_setvertex(&vertices[num_vertices++], STBTT_vline, x,y,0,0); + was_off = 0; + } + } + } + num_vertices = stbtt__close_shape(vertices, num_vertices, was_off, start_off, sx,sy,scx,scy,cx,cy); + } else if (numberOfContours < 0) { + // Compound shapes. + int more = 1; + stbtt_uint8 *comp = data + g + 10; + num_vertices = 0; + vertices = 0; + while (more) { + stbtt_uint16 flags, gidx; + int comp_num_verts = 0, i; + stbtt_vertex *comp_verts = 0, *tmp = 0; + float mtx[6] = {1,0,0,1,0,0}, m, n; + + flags = ttSHORT(comp); comp+=2; + gidx = ttSHORT(comp); comp+=2; + + if (flags & 2) { // XY values + if (flags & 1) { // shorts + mtx[4] = ttSHORT(comp); comp+=2; + mtx[5] = ttSHORT(comp); comp+=2; + } else { + mtx[4] = ttCHAR(comp); comp+=1; + mtx[5] = ttCHAR(comp); comp+=1; + } + } + else { + // @TODO handle matching point + STBTT_assert(0); + } + if (flags & (1<<3)) { // WE_HAVE_A_SCALE + mtx[0] = mtx[3] = ttSHORT(comp)/16384.0f; comp+=2; + mtx[1] = mtx[2] = 0; + } else if (flags & (1<<6)) { // WE_HAVE_AN_X_AND_YSCALE + mtx[0] = ttSHORT(comp)/16384.0f; comp+=2; + mtx[1] = mtx[2] = 0; + mtx[3] = ttSHORT(comp)/16384.0f; comp+=2; + } else if (flags & (1<<7)) { // WE_HAVE_A_TWO_BY_TWO + mtx[0] = ttSHORT(comp)/16384.0f; comp+=2; + mtx[1] = ttSHORT(comp)/16384.0f; comp+=2; + mtx[2] = ttSHORT(comp)/16384.0f; comp+=2; + mtx[3] = ttSHORT(comp)/16384.0f; comp+=2; + } + + // Find transformation scales. + m = (float) STBTT_sqrt(mtx[0]*mtx[0] + mtx[1]*mtx[1]); + n = (float) STBTT_sqrt(mtx[2]*mtx[2] + mtx[3]*mtx[3]); + + // Get indexed glyph. + comp_num_verts = stbtt_GetGlyphShape(info, gidx, &comp_verts); + if (comp_num_verts > 0) { + // Transform vertices. + for (i = 0; i < comp_num_verts; ++i) { + stbtt_vertex* v = &comp_verts[i]; + stbtt_vertex_type x,y; + x=v->x; y=v->y; + v->x = (stbtt_vertex_type)(m * (mtx[0]*x + mtx[2]*y + mtx[4])); + v->y = (stbtt_vertex_type)(n * (mtx[1]*x + mtx[3]*y + mtx[5])); + x=v->cx; y=v->cy; + v->cx = (stbtt_vertex_type)(m * (mtx[0]*x + mtx[2]*y + mtx[4])); + v->cy = (stbtt_vertex_type)(n * (mtx[1]*x + mtx[3]*y + mtx[5])); + } + // Append vertices. + tmp = (stbtt_vertex*)STBTT_malloc((num_vertices+comp_num_verts)*sizeof(stbtt_vertex), info->userdata); + if (!tmp) { + if (vertices) STBTT_free(vertices, info->userdata); + if (comp_verts) STBTT_free(comp_verts, info->userdata); + return 0; + } + if (num_vertices > 0 && vertices) STBTT_memcpy(tmp, vertices, num_vertices*sizeof(stbtt_vertex)); + STBTT_memcpy(tmp+num_vertices, comp_verts, comp_num_verts*sizeof(stbtt_vertex)); + if (vertices) STBTT_free(vertices, info->userdata); + vertices = tmp; + STBTT_free(comp_verts, info->userdata); + num_vertices += comp_num_verts; + } + // More components ? + more = flags & (1<<5); + } + } else { + // numberOfCounters == 0, do nothing + } + + *pvertices = vertices; + return num_vertices; +} + +typedef struct +{ + int bounds; + int started; + float first_x, first_y; + float x, y; + stbtt_int32 min_x, max_x, min_y, max_y; + + stbtt_vertex *pvertices; + int num_vertices; +} stbtt__csctx; + +#define STBTT__CSCTX_INIT(bounds) {bounds,0, 0,0, 0,0, 0,0,0,0, NULL, 0} + +static void stbtt__track_vertex(stbtt__csctx *c, stbtt_int32 x, stbtt_int32 y) +{ + if (x > c->max_x || !c->started) c->max_x = x; + if (y > c->max_y || !c->started) c->max_y = y; + if (x < c->min_x || !c->started) c->min_x = x; + if (y < c->min_y || !c->started) c->min_y = y; + c->started = 1; +} + +static void stbtt__csctx_v(stbtt__csctx *c, stbtt_uint8 type, stbtt_int32 x, stbtt_int32 y, stbtt_int32 cx, stbtt_int32 cy, stbtt_int32 cx1, stbtt_int32 cy1) +{ + if (c->bounds) { + stbtt__track_vertex(c, x, y); + if (type == STBTT_vcubic) { + stbtt__track_vertex(c, cx, cy); + stbtt__track_vertex(c, cx1, cy1); + } + } else { + stbtt_setvertex(&c->pvertices[c->num_vertices], type, x, y, cx, cy); + c->pvertices[c->num_vertices].cx1 = (stbtt_int16) cx1; + c->pvertices[c->num_vertices].cy1 = (stbtt_int16) cy1; + } + c->num_vertices++; +} + +static void stbtt__csctx_close_shape(stbtt__csctx *ctx) +{ + if (ctx->first_x != ctx->x || ctx->first_y != ctx->y) + stbtt__csctx_v(ctx, STBTT_vline, (int)ctx->first_x, (int)ctx->first_y, 0, 0, 0, 0); +} + +static void stbtt__csctx_rmove_to(stbtt__csctx *ctx, float dx, float dy) +{ + stbtt__csctx_close_shape(ctx); + ctx->first_x = ctx->x = ctx->x + dx; + ctx->first_y = ctx->y = ctx->y + dy; + stbtt__csctx_v(ctx, STBTT_vmove, (int)ctx->x, (int)ctx->y, 0, 0, 0, 0); +} + +static void stbtt__csctx_rline_to(stbtt__csctx *ctx, float dx, float dy) +{ + ctx->x += dx; + ctx->y += dy; + stbtt__csctx_v(ctx, STBTT_vline, (int)ctx->x, (int)ctx->y, 0, 0, 0, 0); +} + +static void stbtt__csctx_rccurve_to(stbtt__csctx *ctx, float dx1, float dy1, float dx2, float dy2, float dx3, float dy3) +{ + float cx1 = ctx->x + dx1; + float cy1 = ctx->y + dy1; + float cx2 = cx1 + dx2; + float cy2 = cy1 + dy2; + ctx->x = cx2 + dx3; + ctx->y = cy2 + dy3; + stbtt__csctx_v(ctx, STBTT_vcubic, (int)ctx->x, (int)ctx->y, (int)cx1, (int)cy1, (int)cx2, (int)cy2); +} + +static stbtt__buf stbtt__get_subr(stbtt__buf idx, int n) +{ + int count = stbtt__cff_index_count(&idx); + int bias = 107; + if (count >= 33900) + bias = 32768; + else if (count >= 1240) + bias = 1131; + n += bias; + if (n < 0 || n >= count) + return stbtt__new_buf(NULL, 0); + return stbtt__cff_index_get(idx, n); +} + +static stbtt__buf stbtt__cid_get_glyph_subrs(const stbtt_fontinfo *info, int glyph_index) +{ + stbtt__buf fdselect = info->fdselect; + int nranges, start, end, v, fmt, fdselector = -1, i; + + stbtt__buf_seek(&fdselect, 0); + fmt = stbtt__buf_get8(&fdselect); + if (fmt == 0) { + // untested + stbtt__buf_skip(&fdselect, glyph_index); + fdselector = stbtt__buf_get8(&fdselect); + } else if (fmt == 3) { + nranges = stbtt__buf_get16(&fdselect); + start = stbtt__buf_get16(&fdselect); + for (i = 0; i < nranges; i++) { + v = stbtt__buf_get8(&fdselect); + end = stbtt__buf_get16(&fdselect); + if (glyph_index >= start && glyph_index < end) { + fdselector = v; + break; + } + start = end; + } + } + if (fdselector == -1) stbtt__new_buf(NULL, 0); + return stbtt__get_subrs(info->cff, stbtt__cff_index_get(info->fontdicts, fdselector)); +} + +static int stbtt__run_charstring(const stbtt_fontinfo *info, int glyph_index, stbtt__csctx *c) +{ + int in_header = 1, maskbits = 0, subr_stack_height = 0, sp = 0, v, i, b0; + int has_subrs = 0, clear_stack; + float s[48]; + stbtt__buf subr_stack[10], subrs = info->subrs, b; + float f; + +#define STBTT__CSERR(s) (0) + + // this currently ignores the initial width value, which isn't needed if we have hmtx + b = stbtt__cff_index_get(info->charstrings, glyph_index); + while (b.cursor < b.size) { + i = 0; + clear_stack = 1; + b0 = stbtt__buf_get8(&b); + switch (b0) { + // @TODO implement hinting + case 0x13: // hintmask + case 0x14: // cntrmask + if (in_header) + maskbits += (sp / 2); // implicit "vstem" + in_header = 0; + stbtt__buf_skip(&b, (maskbits + 7) / 8); + break; + + case 0x01: // hstem + case 0x03: // vstem + case 0x12: // hstemhm + case 0x17: // vstemhm + maskbits += (sp / 2); + break; + + case 0x15: // rmoveto + in_header = 0; + if (sp < 2) return STBTT__CSERR("rmoveto stack"); + stbtt__csctx_rmove_to(c, s[sp-2], s[sp-1]); + break; + case 0x04: // vmoveto + in_header = 0; + if (sp < 1) return STBTT__CSERR("vmoveto stack"); + stbtt__csctx_rmove_to(c, 0, s[sp-1]); + break; + case 0x16: // hmoveto + in_header = 0; + if (sp < 1) return STBTT__CSERR("hmoveto stack"); + stbtt__csctx_rmove_to(c, s[sp-1], 0); + break; + + case 0x05: // rlineto + if (sp < 2) return STBTT__CSERR("rlineto stack"); + for (; i + 1 < sp; i += 2) + stbtt__csctx_rline_to(c, s[i], s[i+1]); + break; + + // hlineto/vlineto and vhcurveto/hvcurveto alternate horizontal and vertical + // starting from a different place. + + case 0x07: // vlineto + if (sp < 1) return STBTT__CSERR("vlineto stack"); + goto vlineto; + case 0x06: // hlineto + if (sp < 1) return STBTT__CSERR("hlineto stack"); + for (;;) { + if (i >= sp) break; + stbtt__csctx_rline_to(c, s[i], 0); + i++; + vlineto: + if (i >= sp) break; + stbtt__csctx_rline_to(c, 0, s[i]); + i++; + } + break; + + case 0x1F: // hvcurveto + if (sp < 4) return STBTT__CSERR("hvcurveto stack"); + goto hvcurveto; + case 0x1E: // vhcurveto + if (sp < 4) return STBTT__CSERR("vhcurveto stack"); + for (;;) { + if (i + 3 >= sp) break; + stbtt__csctx_rccurve_to(c, 0, s[i], s[i+1], s[i+2], s[i+3], (sp - i == 5) ? s[i + 4] : 0.0f); + i += 4; + hvcurveto: + if (i + 3 >= sp) break; + stbtt__csctx_rccurve_to(c, s[i], 0, s[i+1], s[i+2], (sp - i == 5) ? s[i+4] : 0.0f, s[i+3]); + i += 4; + } + break; + + case 0x08: // rrcurveto + if (sp < 6) return STBTT__CSERR("rcurveline stack"); + for (; i + 5 < sp; i += 6) + stbtt__csctx_rccurve_to(c, s[i], s[i+1], s[i+2], s[i+3], s[i+4], s[i+5]); + break; + + case 0x18: // rcurveline + if (sp < 8) return STBTT__CSERR("rcurveline stack"); + for (; i + 5 < sp - 2; i += 6) + stbtt__csctx_rccurve_to(c, s[i], s[i+1], s[i+2], s[i+3], s[i+4], s[i+5]); + if (i + 1 >= sp) return STBTT__CSERR("rcurveline stack"); + stbtt__csctx_rline_to(c, s[i], s[i+1]); + break; + + case 0x19: // rlinecurve + if (sp < 8) return STBTT__CSERR("rlinecurve stack"); + for (; i + 1 < sp - 6; i += 2) + stbtt__csctx_rline_to(c, s[i], s[i+1]); + if (i + 5 >= sp) return STBTT__CSERR("rlinecurve stack"); + stbtt__csctx_rccurve_to(c, s[i], s[i+1], s[i+2], s[i+3], s[i+4], s[i+5]); + break; + + case 0x1A: // vvcurveto + case 0x1B: // hhcurveto + if (sp < 4) return STBTT__CSERR("(vv|hh)curveto stack"); + f = 0.0; + if (sp & 1) { f = s[i]; i++; } + for (; i + 3 < sp; i += 4) { + if (b0 == 0x1B) + stbtt__csctx_rccurve_to(c, s[i], f, s[i+1], s[i+2], s[i+3], 0.0); + else + stbtt__csctx_rccurve_to(c, f, s[i], s[i+1], s[i+2], 0.0, s[i+3]); + f = 0.0; + } + break; + + case 0x0A: // callsubr + if (!has_subrs) { + if (info->fdselect.size) + subrs = stbtt__cid_get_glyph_subrs(info, glyph_index); + has_subrs = 1; + } + // FALLTHROUGH + case 0x1D: // callgsubr + if (sp < 1) return STBTT__CSERR("call(g|)subr stack"); + v = (int) s[--sp]; + if (subr_stack_height >= 10) return STBTT__CSERR("recursion limit"); + subr_stack[subr_stack_height++] = b; + b = stbtt__get_subr(b0 == 0x0A ? subrs : info->gsubrs, v); + if (b.size == 0) return STBTT__CSERR("subr not found"); + b.cursor = 0; + clear_stack = 0; + break; + + case 0x0B: // return + if (subr_stack_height <= 0) return STBTT__CSERR("return outside subr"); + b = subr_stack[--subr_stack_height]; + clear_stack = 0; + break; + + case 0x0E: // endchar + stbtt__csctx_close_shape(c); + return 1; + + case 0x0C: { // two-byte escape + float dx1, dx2, dx3, dx4, dx5, dx6, dy1, dy2, dy3, dy4, dy5, dy6; + float dx, dy; + int b1 = stbtt__buf_get8(&b); + switch (b1) { + // @TODO These "flex" implementations ignore the flex-depth and resolution, + // and always draw beziers. + case 0x22: // hflex + if (sp < 7) return STBTT__CSERR("hflex stack"); + dx1 = s[0]; + dx2 = s[1]; + dy2 = s[2]; + dx3 = s[3]; + dx4 = s[4]; + dx5 = s[5]; + dx6 = s[6]; + stbtt__csctx_rccurve_to(c, dx1, 0, dx2, dy2, dx3, 0); + stbtt__csctx_rccurve_to(c, dx4, 0, dx5, -dy2, dx6, 0); + break; + + case 0x23: // flex + if (sp < 13) return STBTT__CSERR("flex stack"); + dx1 = s[0]; + dy1 = s[1]; + dx2 = s[2]; + dy2 = s[3]; + dx3 = s[4]; + dy3 = s[5]; + dx4 = s[6]; + dy4 = s[7]; + dx5 = s[8]; + dy5 = s[9]; + dx6 = s[10]; + dy6 = s[11]; + //fd is s[12] + stbtt__csctx_rccurve_to(c, dx1, dy1, dx2, dy2, dx3, dy3); + stbtt__csctx_rccurve_to(c, dx4, dy4, dx5, dy5, dx6, dy6); + break; + + case 0x24: // hflex1 + if (sp < 9) return STBTT__CSERR("hflex1 stack"); + dx1 = s[0]; + dy1 = s[1]; + dx2 = s[2]; + dy2 = s[3]; + dx3 = s[4]; + dx4 = s[5]; + dx5 = s[6]; + dy5 = s[7]; + dx6 = s[8]; + stbtt__csctx_rccurve_to(c, dx1, dy1, dx2, dy2, dx3, 0); + stbtt__csctx_rccurve_to(c, dx4, 0, dx5, dy5, dx6, -(dy1+dy2+dy5)); + break; + + case 0x25: // flex1 + if (sp < 11) return STBTT__CSERR("flex1 stack"); + dx1 = s[0]; + dy1 = s[1]; + dx2 = s[2]; + dy2 = s[3]; + dx3 = s[4]; + dy3 = s[5]; + dx4 = s[6]; + dy4 = s[7]; + dx5 = s[8]; + dy5 = s[9]; + dx6 = dy6 = s[10]; + dx = dx1+dx2+dx3+dx4+dx5; + dy = dy1+dy2+dy3+dy4+dy5; + if (STBTT_fabs(dx) > STBTT_fabs(dy)) + dy6 = -dy; + else + dx6 = -dx; + stbtt__csctx_rccurve_to(c, dx1, dy1, dx2, dy2, dx3, dy3); + stbtt__csctx_rccurve_to(c, dx4, dy4, dx5, dy5, dx6, dy6); + break; + + default: + return STBTT__CSERR("unimplemented"); + } + } break; + + default: + if (b0 != 255 && b0 != 28 && b0 < 32) + return STBTT__CSERR("reserved operator"); + + // push immediate + if (b0 == 255) { + f = (float)(stbtt_int32)stbtt__buf_get32(&b) / 0x10000; + } else { + stbtt__buf_skip(&b, -1); + f = (float)(stbtt_int16)stbtt__cff_int(&b); + } + if (sp >= 48) return STBTT__CSERR("push stack overflow"); + s[sp++] = f; + clear_stack = 0; + break; + } + if (clear_stack) sp = 0; + } + return STBTT__CSERR("no endchar"); + +#undef STBTT__CSERR +} + +static int stbtt__GetGlyphShapeT2(const stbtt_fontinfo *info, int glyph_index, stbtt_vertex **pvertices) +{ + // runs the charstring twice, once to count and once to output (to avoid realloc) + stbtt__csctx count_ctx = STBTT__CSCTX_INIT(1); + stbtt__csctx output_ctx = STBTT__CSCTX_INIT(0); + if (stbtt__run_charstring(info, glyph_index, &count_ctx)) { + *pvertices = (stbtt_vertex*)STBTT_malloc(count_ctx.num_vertices*sizeof(stbtt_vertex), info->userdata); + output_ctx.pvertices = *pvertices; + if (stbtt__run_charstring(info, glyph_index, &output_ctx)) { + STBTT_assert(output_ctx.num_vertices == count_ctx.num_vertices); + return output_ctx.num_vertices; + } + } + *pvertices = NULL; + return 0; +} + +static int stbtt__GetGlyphInfoT2(const stbtt_fontinfo *info, int glyph_index, int *x0, int *y0, int *x1, int *y1) +{ + stbtt__csctx c = STBTT__CSCTX_INIT(1); + int r = stbtt__run_charstring(info, glyph_index, &c); + if (x0) *x0 = r ? c.min_x : 0; + if (y0) *y0 = r ? c.min_y : 0; + if (x1) *x1 = r ? c.max_x : 0; + if (y1) *y1 = r ? c.max_y : 0; + return r ? c.num_vertices : 0; +} + +STBTT_DEF int stbtt_GetGlyphShape(const stbtt_fontinfo *info, int glyph_index, stbtt_vertex **pvertices) +{ + if (!info->cff.size) + return stbtt__GetGlyphShapeTT(info, glyph_index, pvertices); + else + return stbtt__GetGlyphShapeT2(info, glyph_index, pvertices); +} + +STBTT_DEF void stbtt_GetGlyphHMetrics(const stbtt_fontinfo *info, int glyph_index, int *advanceWidth, int *leftSideBearing) +{ + stbtt_uint16 numOfLongHorMetrics = ttUSHORT(info->data+info->hhea + 34); + if (glyph_index < numOfLongHorMetrics) { + if (advanceWidth) *advanceWidth = ttSHORT(info->data + info->hmtx + 4*glyph_index); + if (leftSideBearing) *leftSideBearing = ttSHORT(info->data + info->hmtx + 4*glyph_index + 2); + } else { + if (advanceWidth) *advanceWidth = ttSHORT(info->data + info->hmtx + 4*(numOfLongHorMetrics-1)); + if (leftSideBearing) *leftSideBearing = ttSHORT(info->data + info->hmtx + 4*numOfLongHorMetrics + 2*(glyph_index - numOfLongHorMetrics)); + } +} + +STBTT_DEF int stbtt_GetKerningTableLength(const stbtt_fontinfo *info) +{ + stbtt_uint8 *data = info->data + info->kern; + + // we only look at the first table. it must be 'horizontal' and format 0. + if (!info->kern) + return 0; + if (ttUSHORT(data+2) < 1) // number of tables, need at least 1 + return 0; + if (ttUSHORT(data+8) != 1) // horizontal flag must be set in format + return 0; + + return ttUSHORT(data+10); +} + +STBTT_DEF int stbtt_GetKerningTable(const stbtt_fontinfo *info, stbtt_kerningentry* table, int table_length) +{ + stbtt_uint8 *data = info->data + info->kern; + int k, length; + + // we only look at the first table. it must be 'horizontal' and format 0. + if (!info->kern) + return 0; + if (ttUSHORT(data+2) < 1) // number of tables, need at least 1 + return 0; + if (ttUSHORT(data+8) != 1) // horizontal flag must be set in format + return 0; + + length = ttUSHORT(data+10); + if (table_length < length) + length = table_length; + + for (k = 0; k < length; k++) + { + table[k].glyph1 = ttUSHORT(data+18+(k*6)); + table[k].glyph2 = ttUSHORT(data+20+(k*6)); + table[k].advance = ttSHORT(data+22+(k*6)); + } + + return length; +} + +static int stbtt__GetGlyphKernInfoAdvance(const stbtt_fontinfo *info, int glyph1, int glyph2) +{ + stbtt_uint8 *data = info->data + info->kern; + stbtt_uint32 needle, straw; + int l, r, m; + + // we only look at the first table. it must be 'horizontal' and format 0. + if (!info->kern) + return 0; + if (ttUSHORT(data+2) < 1) // number of tables, need at least 1 + return 0; + if (ttUSHORT(data+8) != 1) // horizontal flag must be set in format + return 0; + + l = 0; + r = ttUSHORT(data+10) - 1; + needle = glyph1 << 16 | glyph2; + while (l <= r) { + m = (l + r) >> 1; + straw = ttULONG(data+18+(m*6)); // note: unaligned read + if (needle < straw) + r = m - 1; + else if (needle > straw) + l = m + 1; + else + return ttSHORT(data+22+(m*6)); + } + return 0; +} + +static stbtt_int32 stbtt__GetCoverageIndex(stbtt_uint8 *coverageTable, int glyph) +{ + stbtt_uint16 coverageFormat = ttUSHORT(coverageTable); + switch (coverageFormat) { + case 1: { + stbtt_uint16 glyphCount = ttUSHORT(coverageTable + 2); + + // Binary search. + stbtt_int32 l=0, r=glyphCount-1, m; + int straw, needle=glyph; + while (l <= r) { + stbtt_uint8 *glyphArray = coverageTable + 4; + stbtt_uint16 glyphID; + m = (l + r) >> 1; + glyphID = ttUSHORT(glyphArray + 2 * m); + straw = glyphID; + if (needle < straw) + r = m - 1; + else if (needle > straw) + l = m + 1; + else { + return m; + } + } + break; + } + + case 2: { + stbtt_uint16 rangeCount = ttUSHORT(coverageTable + 2); + stbtt_uint8 *rangeArray = coverageTable + 4; + + // Binary search. + stbtt_int32 l=0, r=rangeCount-1, m; + int strawStart, strawEnd, needle=glyph; + while (l <= r) { + stbtt_uint8 *rangeRecord; + m = (l + r) >> 1; + rangeRecord = rangeArray + 6 * m; + strawStart = ttUSHORT(rangeRecord); + strawEnd = ttUSHORT(rangeRecord + 2); + if (needle < strawStart) + r = m - 1; + else if (needle > strawEnd) + l = m + 1; + else { + stbtt_uint16 startCoverageIndex = ttUSHORT(rangeRecord + 4); + return startCoverageIndex + glyph - strawStart; + } + } + break; + } + + default: return -1; // unsupported + } + + return -1; +} + +static stbtt_int32 stbtt__GetGlyphClass(stbtt_uint8 *classDefTable, int glyph) +{ + stbtt_uint16 classDefFormat = ttUSHORT(classDefTable); + switch (classDefFormat) + { + case 1: { + stbtt_uint16 startGlyphID = ttUSHORT(classDefTable + 2); + stbtt_uint16 glyphCount = ttUSHORT(classDefTable + 4); + stbtt_uint8 *classDef1ValueArray = classDefTable + 6; + + if (glyph >= startGlyphID && glyph < startGlyphID + glyphCount) + return (stbtt_int32)ttUSHORT(classDef1ValueArray + 2 * (glyph - startGlyphID)); + break; + } + + case 2: { + stbtt_uint16 classRangeCount = ttUSHORT(classDefTable + 2); + stbtt_uint8 *classRangeRecords = classDefTable + 4; + + // Binary search. + stbtt_int32 l=0, r=classRangeCount-1, m; + int strawStart, strawEnd, needle=glyph; + while (l <= r) { + stbtt_uint8 *classRangeRecord; + m = (l + r) >> 1; + classRangeRecord = classRangeRecords + 6 * m; + strawStart = ttUSHORT(classRangeRecord); + strawEnd = ttUSHORT(classRangeRecord + 2); + if (needle < strawStart) + r = m - 1; + else if (needle > strawEnd) + l = m + 1; + else + return (stbtt_int32)ttUSHORT(classRangeRecord + 4); + } + break; + } + + default: + return -1; // Unsupported definition type, return an error. + } + + // "All glyphs not assigned to a class fall into class 0". (OpenType spec) + return 0; +} + +// Define to STBTT_assert(x) if you want to break on unimplemented formats. +#define STBTT_GPOS_TODO_assert(x) + +static stbtt_int32 stbtt__GetGlyphGPOSInfoAdvance(const stbtt_fontinfo *info, int glyph1, int glyph2) +{ + stbtt_uint16 lookupListOffset; + stbtt_uint8 *lookupList; + stbtt_uint16 lookupCount; + stbtt_uint8 *data; + stbtt_int32 i, sti; + + if (!info->gpos) return 0; + + data = info->data + info->gpos; + + if (ttUSHORT(data+0) != 1) return 0; // Major version 1 + if (ttUSHORT(data+2) != 0) return 0; // Minor version 0 + + lookupListOffset = ttUSHORT(data+8); + lookupList = data + lookupListOffset; + lookupCount = ttUSHORT(lookupList); + + for (i=0; i= pairSetCount) return 0; + + needle=glyph2; + r=pairValueCount-1; + l=0; + + // Binary search. + while (l <= r) { + stbtt_uint16 secondGlyph; + stbtt_uint8 *pairValue; + m = (l + r) >> 1; + pairValue = pairValueArray + (2 + valueRecordPairSizeInBytes) * m; + secondGlyph = ttUSHORT(pairValue); + straw = secondGlyph; + if (needle < straw) + r = m - 1; + else if (needle > straw) + l = m + 1; + else { + stbtt_int16 xAdvance = ttSHORT(pairValue + 2); + return xAdvance; + } + } + } else + return 0; + break; + } + + case 2: { + stbtt_uint16 valueFormat1 = ttUSHORT(table + 4); + stbtt_uint16 valueFormat2 = ttUSHORT(table + 6); + if (valueFormat1 == 4 && valueFormat2 == 0) { // Support more formats? + stbtt_uint16 classDef1Offset = ttUSHORT(table + 8); + stbtt_uint16 classDef2Offset = ttUSHORT(table + 10); + int glyph1class = stbtt__GetGlyphClass(table + classDef1Offset, glyph1); + int glyph2class = stbtt__GetGlyphClass(table + classDef2Offset, glyph2); + + stbtt_uint16 class1Count = ttUSHORT(table + 12); + stbtt_uint16 class2Count = ttUSHORT(table + 14); + stbtt_uint8 *class1Records, *class2Records; + stbtt_int16 xAdvance; + + if (glyph1class < 0 || glyph1class >= class1Count) return 0; // malformed + if (glyph2class < 0 || glyph2class >= class2Count) return 0; // malformed + + class1Records = table + 16; + class2Records = class1Records + 2 * (glyph1class * class2Count); + xAdvance = ttSHORT(class2Records + 2 * glyph2class); + return xAdvance; + } else + return 0; + break; + } + + default: + return 0; // Unsupported position format + } + } + } + + return 0; +} + +STBTT_DEF int stbtt_GetGlyphKernAdvance(const stbtt_fontinfo *info, int g1, int g2) +{ + int xAdvance = 0; + + if (info->gpos) + xAdvance += stbtt__GetGlyphGPOSInfoAdvance(info, g1, g2); + else if (info->kern) + xAdvance += stbtt__GetGlyphKernInfoAdvance(info, g1, g2); + + return xAdvance; +} + +STBTT_DEF int stbtt_GetCodepointKernAdvance(const stbtt_fontinfo *info, int ch1, int ch2) +{ + if (!info->kern && !info->gpos) // if no kerning table, don't waste time looking up both codepoint->glyphs + return 0; + return stbtt_GetGlyphKernAdvance(info, stbtt_FindGlyphIndex(info,ch1), stbtt_FindGlyphIndex(info,ch2)); +} + +STBTT_DEF void stbtt_GetCodepointHMetrics(const stbtt_fontinfo *info, int codepoint, int *advanceWidth, int *leftSideBearing) +{ + stbtt_GetGlyphHMetrics(info, stbtt_FindGlyphIndex(info,codepoint), advanceWidth, leftSideBearing); +} + +STBTT_DEF void stbtt_GetFontVMetrics(const stbtt_fontinfo *info, int *ascent, int *descent, int *lineGap) +{ + if (ascent ) *ascent = ttSHORT(info->data+info->hhea + 4); + if (descent) *descent = ttSHORT(info->data+info->hhea + 6); + if (lineGap) *lineGap = ttSHORT(info->data+info->hhea + 8); +} + +STBTT_DEF int stbtt_GetFontVMetricsOS2(const stbtt_fontinfo *info, int *typoAscent, int *typoDescent, int *typoLineGap) +{ + int tab = stbtt__find_table(info->data, info->fontstart, "OS/2"); + if (!tab) + return 0; + if (typoAscent ) *typoAscent = ttSHORT(info->data+tab + 68); + if (typoDescent) *typoDescent = ttSHORT(info->data+tab + 70); + if (typoLineGap) *typoLineGap = ttSHORT(info->data+tab + 72); + return 1; +} + +STBTT_DEF void stbtt_GetFontBoundingBox(const stbtt_fontinfo *info, int *x0, int *y0, int *x1, int *y1) +{ + *x0 = ttSHORT(info->data + info->head + 36); + *y0 = ttSHORT(info->data + info->head + 38); + *x1 = ttSHORT(info->data + info->head + 40); + *y1 = ttSHORT(info->data + info->head + 42); +} + +STBTT_DEF float stbtt_ScaleForPixelHeight(const stbtt_fontinfo *info, float height) +{ + int fheight = ttSHORT(info->data + info->hhea + 4) - ttSHORT(info->data + info->hhea + 6); + return (float) height / fheight; +} + +STBTT_DEF float stbtt_ScaleForMappingEmToPixels(const stbtt_fontinfo *info, float pixels) +{ + int unitsPerEm = ttUSHORT(info->data + info->head + 18); + return pixels / unitsPerEm; +} + +STBTT_DEF void stbtt_FreeShape(const stbtt_fontinfo *info, stbtt_vertex *v) +{ + STBTT_free(v, info->userdata); +} + +STBTT_DEF stbtt_uint8 *stbtt_FindSVGDoc(const stbtt_fontinfo *info, int gl) +{ + int i; + stbtt_uint8 *data = info->data; + stbtt_uint8 *svg_doc_list = data + stbtt__get_svg((stbtt_fontinfo *) info); + + int numEntries = ttUSHORT(svg_doc_list); + stbtt_uint8 *svg_docs = svg_doc_list + 2; + + for(i=0; i= ttUSHORT(svg_doc)) && (gl <= ttUSHORT(svg_doc + 2))) + return svg_doc; + } + return 0; +} + +STBTT_DEF int stbtt_GetGlyphSVG(const stbtt_fontinfo *info, int gl, const char **svg) +{ + stbtt_uint8 *data = info->data; + stbtt_uint8 *svg_doc; + + if (info->svg == 0) + return 0; + + svg_doc = stbtt_FindSVGDoc(info, gl); + if (svg_doc != NULL) { + *svg = (char *) data + info->svg + ttULONG(svg_doc + 4); + return ttULONG(svg_doc + 8); + } else { + return 0; + } +} + +STBTT_DEF int stbtt_GetCodepointSVG(const stbtt_fontinfo *info, int unicode_codepoint, const char **svg) +{ + return stbtt_GetGlyphSVG(info, stbtt_FindGlyphIndex(info, unicode_codepoint), svg); +} + +////////////////////////////////////////////////////////////////////////////// +// +// antialiasing software rasterizer +// + +STBTT_DEF void stbtt_GetGlyphBitmapBoxSubpixel(const stbtt_fontinfo *font, int glyph, float scale_x, float scale_y,float shift_x, float shift_y, int *ix0, int *iy0, int *ix1, int *iy1) +{ + int x0=0,y0=0,x1,y1; // =0 suppresses compiler warning + if (!stbtt_GetGlyphBox(font, glyph, &x0,&y0,&x1,&y1)) { + // e.g. space character + if (ix0) *ix0 = 0; + if (iy0) *iy0 = 0; + if (ix1) *ix1 = 0; + if (iy1) *iy1 = 0; + } else { + // move to integral bboxes (treating pixels as little squares, what pixels get touched)? + if (ix0) *ix0 = STBTT_ifloor( x0 * scale_x + shift_x); + if (iy0) *iy0 = STBTT_ifloor(-y1 * scale_y + shift_y); + if (ix1) *ix1 = STBTT_iceil ( x1 * scale_x + shift_x); + if (iy1) *iy1 = STBTT_iceil (-y0 * scale_y + shift_y); + } +} + +STBTT_DEF void stbtt_GetGlyphBitmapBox(const stbtt_fontinfo *font, int glyph, float scale_x, float scale_y, int *ix0, int *iy0, int *ix1, int *iy1) +{ + stbtt_GetGlyphBitmapBoxSubpixel(font, glyph, scale_x, scale_y,0.0f,0.0f, ix0, iy0, ix1, iy1); +} + +STBTT_DEF void stbtt_GetCodepointBitmapBoxSubpixel(const stbtt_fontinfo *font, int codepoint, float scale_x, float scale_y, float shift_x, float shift_y, int *ix0, int *iy0, int *ix1, int *iy1) +{ + stbtt_GetGlyphBitmapBoxSubpixel(font, stbtt_FindGlyphIndex(font,codepoint), scale_x, scale_y,shift_x,shift_y, ix0,iy0,ix1,iy1); +} + +STBTT_DEF void stbtt_GetCodepointBitmapBox(const stbtt_fontinfo *font, int codepoint, float scale_x, float scale_y, int *ix0, int *iy0, int *ix1, int *iy1) +{ + stbtt_GetCodepointBitmapBoxSubpixel(font, codepoint, scale_x, scale_y,0.0f,0.0f, ix0,iy0,ix1,iy1); +} + +////////////////////////////////////////////////////////////////////////////// +// +// Rasterizer + +typedef struct stbtt__hheap_chunk +{ + struct stbtt__hheap_chunk *next; +} stbtt__hheap_chunk; + +typedef struct stbtt__hheap +{ + struct stbtt__hheap_chunk *head; + void *first_free; + int num_remaining_in_head_chunk; +} stbtt__hheap; + +static void *stbtt__hheap_alloc(stbtt__hheap *hh, size_t size, void *userdata) +{ + if (hh->first_free) { + void *p = hh->first_free; + hh->first_free = * (void **) p; + return p; + } else { + if (hh->num_remaining_in_head_chunk == 0) { + int count = (size < 32 ? 2000 : size < 128 ? 800 : 100); + stbtt__hheap_chunk *c = (stbtt__hheap_chunk *) STBTT_malloc(sizeof(stbtt__hheap_chunk) + size * count, userdata); + if (c == NULL) + return NULL; + c->next = hh->head; + hh->head = c; + hh->num_remaining_in_head_chunk = count; + } + --hh->num_remaining_in_head_chunk; + return (char *) (hh->head) + sizeof(stbtt__hheap_chunk) + size * hh->num_remaining_in_head_chunk; + } +} + +static void stbtt__hheap_free(stbtt__hheap *hh, void *p) +{ + *(void **) p = hh->first_free; + hh->first_free = p; +} + +static void stbtt__hheap_cleanup(stbtt__hheap *hh, void *userdata) +{ + stbtt__hheap_chunk *c = hh->head; + while (c) { + stbtt__hheap_chunk *n = c->next; + STBTT_free(c, userdata); + c = n; + } +} + +typedef struct stbtt__edge { + float x0,y0, x1,y1; + int invert; +} stbtt__edge; + + +typedef struct stbtt__active_edge +{ + struct stbtt__active_edge *next; + #if STBTT_RASTERIZER_VERSION==1 + int x,dx; + float ey; + int direction; + #elif STBTT_RASTERIZER_VERSION==2 + float fx,fdx,fdy; + float direction; + float sy; + float ey; + #else + #error "Unrecognized value of STBTT_RASTERIZER_VERSION" + #endif +} stbtt__active_edge; + +#if STBTT_RASTERIZER_VERSION == 1 +#define STBTT_FIXSHIFT 10 +#define STBTT_FIX (1 << STBTT_FIXSHIFT) +#define STBTT_FIXMASK (STBTT_FIX-1) + +static stbtt__active_edge *stbtt__new_active(stbtt__hheap *hh, stbtt__edge *e, int off_x, float start_point, void *userdata) +{ + stbtt__active_edge *z = (stbtt__active_edge *) stbtt__hheap_alloc(hh, sizeof(*z), userdata); + float dxdy = (e->x1 - e->x0) / (e->y1 - e->y0); + STBTT_assert(z != NULL); + if (!z) return z; + + // round dx down to avoid overshooting + if (dxdy < 0) + z->dx = -STBTT_ifloor(STBTT_FIX * -dxdy); + else + z->dx = STBTT_ifloor(STBTT_FIX * dxdy); + + z->x = STBTT_ifloor(STBTT_FIX * e->x0 + z->dx * (start_point - e->y0)); // use z->dx so when we offset later it's by the same amount + z->x -= off_x * STBTT_FIX; + + z->ey = e->y1; + z->next = 0; + z->direction = e->invert ? 1 : -1; + return z; +} +#elif STBTT_RASTERIZER_VERSION == 2 +static stbtt__active_edge *stbtt__new_active(stbtt__hheap *hh, stbtt__edge *e, int off_x, float start_point, void *userdata) +{ + stbtt__active_edge *z = (stbtt__active_edge *) stbtt__hheap_alloc(hh, sizeof(*z), userdata); + float dxdy = (e->x1 - e->x0) / (e->y1 - e->y0); + STBTT_assert(z != NULL); + //STBTT_assert(e->y0 <= start_point); + if (!z) return z; + z->fdx = dxdy; + z->fdy = dxdy != 0.0f ? (1.0f/dxdy) : 0.0f; + z->fx = e->x0 + dxdy * (start_point - e->y0); + z->fx -= off_x; + z->direction = e->invert ? 1.0f : -1.0f; + z->sy = e->y0; + z->ey = e->y1; + z->next = 0; + return z; +} +#else +#error "Unrecognized value of STBTT_RASTERIZER_VERSION" +#endif + +#if STBTT_RASTERIZER_VERSION == 1 +// 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 +static void stbtt__fill_active_edges(unsigned char *scanline, int len, stbtt__active_edge *e, int max_weight) +{ + // non-zero winding fill + int x0=0, w=0; + + while (e) { + if (w == 0) { + // if we're currently at zero, we need to record the edge start point + x0 = e->x; w += e->direction; + } else { + int x1 = e->x; w += e->direction; + // if we went to zero, we need to draw + if (w == 0) { + int i = x0 >> STBTT_FIXSHIFT; + int j = x1 >> STBTT_FIXSHIFT; + + if (i < len && j >= 0) { + if (i == j) { + // x0,x1 are the same pixel, so compute combined coverage + scanline[i] = scanline[i] + (stbtt_uint8) ((x1 - x0) * max_weight >> STBTT_FIXSHIFT); + } else { + if (i >= 0) // add antialiasing for x0 + scanline[i] = scanline[i] + (stbtt_uint8) (((STBTT_FIX - (x0 & STBTT_FIXMASK)) * max_weight) >> STBTT_FIXSHIFT); + else + i = -1; // clip + + if (j < len) // add antialiasing for x1 + scanline[j] = scanline[j] + (stbtt_uint8) (((x1 & STBTT_FIXMASK) * max_weight) >> STBTT_FIXSHIFT); + else + j = len; // clip + + for (++i; i < j; ++i) // fill pixels between x0 and x1 + scanline[i] = scanline[i] + (stbtt_uint8) max_weight; + } + } + } + } + + e = e->next; + } +} + +static void stbtt__rasterize_sorted_edges(stbtt__bitmap *result, stbtt__edge *e, int n, int vsubsample, int off_x, int off_y, void *userdata) +{ + stbtt__hheap hh = { 0, 0, 0 }; + stbtt__active_edge *active = NULL; + int y,j=0; + int max_weight = (255 / vsubsample); // weight per vertical scanline + int s; // vertical subsample index + unsigned char scanline_data[512], *scanline; + + if (result->w > 512) + scanline = (unsigned char *) STBTT_malloc(result->w, userdata); + else + scanline = scanline_data; + + y = off_y * vsubsample; + e[n].y0 = (off_y + result->h) * (float) vsubsample + 1; + + while (j < result->h) { + STBTT_memset(scanline, 0, result->w); + for (s=0; s < vsubsample; ++s) { + // find center of pixel for this scanline + float scan_y = y + 0.5f; + stbtt__active_edge **step = &active; + + // update all active edges; + // remove all active edges that terminate before the center of this scanline + while (*step) { + stbtt__active_edge * z = *step; + if (z->ey <= scan_y) { + *step = z->next; // delete from list + STBTT_assert(z->direction); + z->direction = 0; + stbtt__hheap_free(&hh, 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) { + stbtt__active_edge *t = *step; + stbtt__active_edge *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->y0 <= scan_y) { + if (e->y1 > scan_y) { + stbtt__active_edge *z = stbtt__new_active(&hh, e, off_x, scan_y, userdata); + if (z != NULL) { + // find insertion point + if (active == NULL) + active = z; + else if (z->x < active->x) { + // insert at front + z->next = active; + active = z; + } else { + // find thing to insert AFTER + stbtt__active_edge *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 XOR fashion + if (active) + stbtt__fill_active_edges(scanline, result->w, active, max_weight); + + ++y; + } + STBTT_memcpy(result->pixels + j * result->stride, scanline, result->w); + ++j; + } + + stbtt__hheap_cleanup(&hh, userdata); + + if (scanline != scanline_data) + STBTT_free(scanline, userdata); +} + +#elif STBTT_RASTERIZER_VERSION == 2 + +// the edge passed in here does not cross the vertical line at x or the vertical line at x+1 +// (i.e. it has already been clipped to those) +static void stbtt__handle_clipped_edge(float *scanline, int x, stbtt__active_edge *e, float x0, float y0, float x1, float y1) +{ + if (y0 == y1) return; + STBTT_assert(y0 < y1); + STBTT_assert(e->sy <= e->ey); + if (y0 > e->ey) return; + if (y1 < e->sy) return; + if (y0 < e->sy) { + x0 += (x1-x0) * (e->sy - y0) / (y1-y0); + y0 = e->sy; + } + if (y1 > e->ey) { + x1 += (x1-x0) * (e->ey - y1) / (y1-y0); + y1 = e->ey; + } + + if (x0 == x) + STBTT_assert(x1 <= x+1); + else if (x0 == x+1) + STBTT_assert(x1 >= x); + else if (x0 <= x) + STBTT_assert(x1 <= x); + else if (x0 >= x+1) + STBTT_assert(x1 >= x+1); + else + STBTT_assert(x1 >= x && x1 <= x+1); + + if (x0 <= x && x1 <= x) + scanline[x] += e->direction * (y1-y0); + else if (x0 >= x+1 && x1 >= x+1) + ; + else { + STBTT_assert(x0 >= x && x0 <= x+1 && x1 >= x && x1 <= x+1); + scanline[x] += e->direction * (y1-y0) * (1-((x0-x)+(x1-x))/2); // coverage = 1 - average x position + } +} + +static float stbtt__sized_trapezoid_area(float height, float top_width, float bottom_width) +{ + STBTT_assert(top_width >= 0); + STBTT_assert(bottom_width >= 0); + return (top_width + bottom_width) / 2.0f * height; +} + +static float stbtt__position_trapezoid_area(float height, float tx0, float tx1, float bx0, float bx1) +{ + return stbtt__sized_trapezoid_area(height, tx1 - tx0, bx1 - bx0); +} + +static float stbtt__sized_triangle_area(float height, float width) +{ + return height * width / 2; +} + +static void stbtt__fill_active_edges_new(float *scanline, float *scanline_fill, int len, stbtt__active_edge *e, float y_top) +{ + float y_bottom = y_top+1; + + while (e) { + // brute force every pixel + + // compute intersection points with top & bottom + STBTT_assert(e->ey >= y_top); + + if (e->fdx == 0) { + float x0 = e->fx; + if (x0 < len) { + if (x0 >= 0) { + stbtt__handle_clipped_edge(scanline,(int) x0,e, x0,y_top, x0,y_bottom); + stbtt__handle_clipped_edge(scanline_fill-1,(int) x0+1,e, x0,y_top, x0,y_bottom); + } else { + stbtt__handle_clipped_edge(scanline_fill-1,0,e, x0,y_top, x0,y_bottom); + } + } + } else { + float x0 = e->fx; + float dx = e->fdx; + float xb = x0 + dx; + float x_top, x_bottom; + float sy0,sy1; + float dy = e->fdy; + STBTT_assert(e->sy <= y_bottom && e->ey >= y_top); + + // compute endpoints of line segment clipped to this scanline (if the + // line segment starts on this scanline. x0 is the intersection of the + // line with y_top, but that may be off the line segment. + if (e->sy > y_top) { + x_top = x0 + dx * (e->sy - y_top); + sy0 = e->sy; + } else { + x_top = x0; + sy0 = y_top; + } + if (e->ey < y_bottom) { + x_bottom = x0 + dx * (e->ey - y_top); + sy1 = e->ey; + } else { + x_bottom = xb; + sy1 = y_bottom; + } + + if (x_top >= 0 && x_bottom >= 0 && x_top < len && x_bottom < len) { + // from here on, we don't have to range check x values + + if ((int) x_top == (int) x_bottom) { + float height; + // simple case, only spans one pixel + int x = (int) x_top; + height = (sy1 - sy0) * e->direction; + STBTT_assert(x >= 0 && x < len); + scanline[x] += stbtt__position_trapezoid_area(height, x_top, x+1.0f, x_bottom, x+1.0f); + scanline_fill[x] += height; // everything right of this pixel is filled + } else { + int x,x1,x2; + float y_crossing, y_final, step, sign, area; + // covers 2+ pixels + if (x_top > x_bottom) { + // flip scanline vertically; signed area is the same + float t; + sy0 = y_bottom - (sy0 - y_top); + sy1 = y_bottom - (sy1 - y_top); + t = sy0, sy0 = sy1, sy1 = t; + t = x_bottom, x_bottom = x_top, x_top = t; + dx = -dx; + dy = -dy; + t = x0, x0 = xb, xb = t; + } + STBTT_assert(dy >= 0); + STBTT_assert(dx >= 0); + + x1 = (int) x_top; + x2 = (int) x_bottom; + // compute intersection with y axis at x1+1 + y_crossing = y_top + dy * (x1+1 - x0); + + // compute intersection with y axis at x2 + y_final = y_top + dy * (x2 - x0); + + // x1 x_top x2 x_bottom + // y_top +------|-----+------------+------------+--------|---+------------+ + // | | | | | | + // | | | | | | + // sy0 | Txxxxx|............|............|............|............| + // y_crossing | *xxxxx.......|............|............|............| + // | | xxxxx..|............|............|............| + // | | /- xx*xxxx........|............|............| + // | | dy < | xxxxxx..|............|............| + // y_final | | \- | xx*xxx.........|............| + // sy1 | | | | xxxxxB...|............| + // | | | | | | + // | | | | | | + // y_bottom +------------+------------+------------+------------+------------+ + // + // goal is to measure the area covered by '.' in each pixel + + // if x2 is right at the right edge of x1, y_crossing can blow up, github #1057 + // @TODO: maybe test against sy1 rather than y_bottom? + if (y_crossing > y_bottom) + y_crossing = y_bottom; + + sign = e->direction; + + // area of the rectangle covered from sy0..y_crossing + area = sign * (y_crossing-sy0); + + // area of the triangle (x_top,sy0), (x1+1,sy0), (x1+1,y_crossing) + scanline[x1] += stbtt__sized_triangle_area(area, x1+1 - x_top); + + // check if final y_crossing is blown up; no test case for this + if (y_final > y_bottom) { + y_final = y_bottom; + dy = (y_final - y_crossing ) / (x2 - (x1+1)); // if denom=0, y_final = y_crossing, so y_final <= y_bottom + } + + // in second pixel, area covered by line segment found in first pixel + // is always a rectangle 1 wide * the height of that line segment; this + // is exactly what the variable 'area' stores. it also gets a contribution + // from the line segment within it. the THIRD pixel will get the first + // pixel's rectangle contribution, the second pixel's rectangle contribution, + // and its own contribution. the 'own contribution' is the same in every pixel except + // the leftmost and rightmost, a trapezoid that slides down in each pixel. + // the second pixel's contribution to the third pixel will be the + // rectangle 1 wide times the height change in the second pixel, which is dy. + + step = sign * dy * 1; // dy is dy/dx, change in y for every 1 change in x, + // which multiplied by 1-pixel-width is how much pixel area changes for each step in x + // so the area advances by 'step' every time + + for (x = x1+1; x < x2; ++x) { + scanline[x] += area + step/2; // area of trapezoid is 1*step/2 + area += step; + } + STBTT_assert(STBTT_fabs(area) <= 1.01f); // accumulated error from area += step unless we round step down + STBTT_assert(sy1 > y_final-0.01f); + + // area covered in the last pixel is the rectangle from all the pixels to the left, + // plus the trapezoid filled by the line segment in this pixel all the way to the right edge + scanline[x2] += area + sign * stbtt__position_trapezoid_area(sy1-y_final, (float) x2, x2+1.0f, x_bottom, x2+1.0f); + + // the rest of the line is filled based on the total height of the line segment in this pixel + scanline_fill[x2] += sign * (sy1-sy0); + } + } else { + // if edge goes outside of box we're drawing, we require + // clipping logic. since this does not match the intended use + // of this library, we use a different, very slow brute + // force implementation + // note though that this does happen some of the time because + // x_top and x_bottom can be extrapolated at the top & bottom of + // the shape and actually lie outside the bounding box + int x; + for (x=0; x < len; ++x) { + // cases: + // + // there can be up to two intersections with the pixel. any intersection + // with left or right edges can be handled by splitting into two (or three) + // regions. intersections with top & bottom do not necessitate case-wise logic. + // + // the old way of doing this found the intersections with the left & right edges, + // then used some simple logic to produce up to three segments in sorted order + // from top-to-bottom. however, this had a problem: if an x edge was epsilon + // across the x border, then the corresponding y position might not be distinct + // from the other y segment, and it might ignored as an empty segment. to avoid + // that, we need to explicitly produce segments based on x positions. + + // rename variables to clearly-defined pairs + float y0 = y_top; + float x1 = (float) (x); + float x2 = (float) (x+1); + float x3 = xb; + float y3 = y_bottom; + + // x = e->x + e->dx * (y-y_top) + // (y-y_top) = (x - e->x) / e->dx + // y = (x - e->x) / e->dx + y_top + float y1 = (x - x0) / dx + y_top; + float y2 = (x+1 - x0) / dx + y_top; + + if (x0 < x1 && x3 > x2) { // three segments descending down-right + stbtt__handle_clipped_edge(scanline,x,e, x0,y0, x1,y1); + stbtt__handle_clipped_edge(scanline,x,e, x1,y1, x2,y2); + stbtt__handle_clipped_edge(scanline,x,e, x2,y2, x3,y3); + } else if (x3 < x1 && x0 > x2) { // three segments descending down-left + stbtt__handle_clipped_edge(scanline,x,e, x0,y0, x2,y2); + stbtt__handle_clipped_edge(scanline,x,e, x2,y2, x1,y1); + stbtt__handle_clipped_edge(scanline,x,e, x1,y1, x3,y3); + } else if (x0 < x1 && x3 > x1) { // two segments across x, down-right + stbtt__handle_clipped_edge(scanline,x,e, x0,y0, x1,y1); + stbtt__handle_clipped_edge(scanline,x,e, x1,y1, x3,y3); + } else if (x3 < x1 && x0 > x1) { // two segments across x, down-left + stbtt__handle_clipped_edge(scanline,x,e, x0,y0, x1,y1); + stbtt__handle_clipped_edge(scanline,x,e, x1,y1, x3,y3); + } else if (x0 < x2 && x3 > x2) { // two segments across x+1, down-right + stbtt__handle_clipped_edge(scanline,x,e, x0,y0, x2,y2); + stbtt__handle_clipped_edge(scanline,x,e, x2,y2, x3,y3); + } else if (x3 < x2 && x0 > x2) { // two segments across x+1, down-left + stbtt__handle_clipped_edge(scanline,x,e, x0,y0, x2,y2); + stbtt__handle_clipped_edge(scanline,x,e, x2,y2, x3,y3); + } else { // one segment + stbtt__handle_clipped_edge(scanline,x,e, x0,y0, x3,y3); + } + } + } + } + e = e->next; + } +} + +// directly AA rasterize edges w/o supersampling +static void stbtt__rasterize_sorted_edges(stbtt__bitmap *result, stbtt__edge *e, int n, int vsubsample, int off_x, int off_y, void *userdata) +{ + stbtt__hheap hh = { 0, 0, 0 }; + stbtt__active_edge *active = NULL; + int y,j=0, i; + float scanline_data[129], *scanline, *scanline2; + + STBTT__NOTUSED(vsubsample); + + if (result->w > 64) + scanline = (float *) STBTT_malloc((result->w*2+1) * sizeof(float), userdata); + else + scanline = scanline_data; + + scanline2 = scanline + result->w; + + y = off_y; + e[n].y0 = (float) (off_y + result->h) + 1; + + while (j < result->h) { + // find center of pixel for this scanline + float scan_y_top = y + 0.0f; + float scan_y_bottom = y + 1.0f; + stbtt__active_edge **step = &active; + + STBTT_memset(scanline , 0, result->w*sizeof(scanline[0])); + STBTT_memset(scanline2, 0, (result->w+1)*sizeof(scanline[0])); + + // update all active edges; + // remove all active edges that terminate before the top of this scanline + while (*step) { + stbtt__active_edge * z = *step; + if (z->ey <= scan_y_top) { + *step = z->next; // delete from list + STBTT_assert(z->direction); + z->direction = 0; + stbtt__hheap_free(&hh, z); + } else { + step = &((*step)->next); // advance through list + } + } + + // insert all edges that start before the bottom of this scanline + while (e->y0 <= scan_y_bottom) { + if (e->y0 != e->y1) { + stbtt__active_edge *z = stbtt__new_active(&hh, e, off_x, scan_y_top, userdata); + if (z != NULL) { + if (j == 0 && off_y != 0) { + if (z->ey < scan_y_top) { + // this can happen due to subpixel positioning and some kind of fp rounding error i think + z->ey = scan_y_top; + } + } + STBTT_assert(z->ey >= scan_y_top); // if we get really unlucky a tiny bit of an edge can be out of bounds + // insert at front + z->next = active; + active = z; + } + } + ++e; + } + + // now process all active edges + if (active) + stbtt__fill_active_edges_new(scanline, scanline2+1, result->w, active, scan_y_top); + + { + float sum = 0; + for (i=0; i < result->w; ++i) { + float k; + int m; + sum += scanline2[i]; + k = scanline[i] + sum; + k = (float) STBTT_fabs(k)*255 + 0.5f; + m = (int) k; + if (m > 255) m = 255; + result->pixels[j*result->stride + i] = (unsigned char) m; + } + } + // advance all the edges + step = &active; + while (*step) { + stbtt__active_edge *z = *step; + z->fx += z->fdx; // advance to position for current scanline + step = &((*step)->next); // advance through list + } + + ++y; + ++j; + } + + stbtt__hheap_cleanup(&hh, userdata); + + if (scanline != scanline_data) + STBTT_free(scanline, userdata); +} +#else +#error "Unrecognized value of STBTT_RASTERIZER_VERSION" +#endif + +#define STBTT__COMPARE(a,b) ((a)->y0 < (b)->y0) + +static void stbtt__sort_edges_ins_sort(stbtt__edge *p, int n) +{ + int i,j; + for (i=1; i < n; ++i) { + stbtt__edge t = p[i], *a = &t; + j = i; + while (j > 0) { + stbtt__edge *b = &p[j-1]; + int c = STBTT__COMPARE(a,b); + if (!c) break; + p[j] = p[j-1]; + --j; + } + if (i != j) + p[j] = t; + } +} + +static void stbtt__sort_edges_quicksort(stbtt__edge *p, int n) +{ + /* threshold for transitioning to insertion sort */ + while (n > 12) { + stbtt__edge t; + int c01,c12,c,m,i,j; + + /* compute median of three */ + m = n >> 1; + c01 = STBTT__COMPARE(&p[0],&p[m]); + c12 = STBTT__COMPARE(&p[m],&p[n-1]); + /* if 0 >= mid >= end, or 0 < mid < end, then use mid */ + if (c01 != c12) { + /* otherwise, we'll need to swap something else to middle */ + int z; + c = STBTT__COMPARE(&p[0],&p[n-1]); + /* 0>mid && midn => n; 0 0 */ + /* 0n: 0>n => 0; 0 n */ + z = (c == c12) ? 0 : n-1; + t = p[z]; + p[z] = p[m]; + p[m] = t; + } + /* now p[m] is the median-of-three */ + /* swap it to the beginning so it won't move around */ + t = p[0]; + p[0] = p[m]; + p[m] = t; + + /* partition loop */ + i=1; + j=n-1; + for(;;) { + /* handling of equality is crucial here */ + /* for sentinels & efficiency with duplicates */ + for (;;++i) { + if (!STBTT__COMPARE(&p[i], &p[0])) break; + } + for (;;--j) { + if (!STBTT__COMPARE(&p[0], &p[j])) break; + } + /* make sure we haven't crossed */ + if (i >= j) break; + t = p[i]; + p[i] = p[j]; + p[j] = t; + + ++i; + --j; + } + /* recurse on smaller side, iterate on larger */ + if (j < (n-i)) { + stbtt__sort_edges_quicksort(p,j); + p = p+i; + n = n-i; + } else { + stbtt__sort_edges_quicksort(p+i, n-i); + n = j; + } + } +} + +static void stbtt__sort_edges(stbtt__edge *p, int n) +{ + stbtt__sort_edges_quicksort(p, n); + stbtt__sort_edges_ins_sort(p, n); +} + +typedef struct +{ + float x,y; +} stbtt__point; + +static void stbtt__rasterize(stbtt__bitmap *result, stbtt__point *pts, int *wcount, int windings, float scale_x, float scale_y, float shift_x, float shift_y, int off_x, int off_y, int invert, void *userdata) +{ + float y_scale_inv = invert ? -scale_y : scale_y; + stbtt__edge *e; + int n,i,j,k,m; +#if STBTT_RASTERIZER_VERSION == 1 + int vsubsample = result->h < 8 ? 15 : 5; +#elif STBTT_RASTERIZER_VERSION == 2 + int vsubsample = 1; +#else + #error "Unrecognized value of STBTT_RASTERIZER_VERSION" +#endif + // vsubsample should divide 255 evenly; otherwise we won't reach full opacity + + // now we have to blow out the windings into explicit edge lists + n = 0; + for (i=0; i < windings; ++i) + n += wcount[i]; + + e = (stbtt__edge *) STBTT_malloc(sizeof(*e) * (n+1), userdata); // add an extra one as a sentinel + if (e == 0) return; + n = 0; + + m=0; + for (i=0; i < windings; ++i) { + stbtt__point *p = pts + m; + m += wcount[i]; + j = wcount[i]-1; + for (k=0; k < wcount[i]; j=k++) { + int a=k,b=j; + // skip the edge if horizontal + if (p[j].y == p[k].y) + continue; + // add edge from j to k to the list + e[n].invert = 0; + if (invert ? p[j].y > p[k].y : p[j].y < p[k].y) { + e[n].invert = 1; + a=j,b=k; + } + e[n].x0 = p[a].x * scale_x + shift_x; + e[n].y0 = (p[a].y * y_scale_inv + shift_y) * vsubsample; + e[n].x1 = p[b].x * scale_x + shift_x; + e[n].y1 = (p[b].y * y_scale_inv + shift_y) * vsubsample; + ++n; + } + } + + // now sort the edges by their highest point (should snap to integer, and then by x) + //STBTT_sort(e, n, sizeof(e[0]), stbtt__edge_compare); + stbtt__sort_edges(e, n); + + // now, traverse the scanlines and find the intersections on each scanline, use xor winding rule + stbtt__rasterize_sorted_edges(result, e, n, vsubsample, off_x, off_y, userdata); + + STBTT_free(e, userdata); +} + +static void stbtt__add_point(stbtt__point *points, int n, float x, float y) +{ + if (!points) return; // during first pass, it's unallocated + points[n].x = x; + points[n].y = y; +} + +// tessellate until threshold p is happy... @TODO warped to compensate for non-linear stretching +static int stbtt__tesselate_curve(stbtt__point *points, int *num_points, float x0, float y0, float x1, float y1, float x2, float y2, float objspace_flatness_squared, int n) +{ + // midpoint + float mx = (x0 + 2*x1 + x2)/4; + float my = (y0 + 2*y1 + y2)/4; + // versus directly drawn line + float dx = (x0+x2)/2 - mx; + float dy = (y0+y2)/2 - my; + if (n > 16) // 65536 segments on one curve better be enough! + return 1; + if (dx*dx+dy*dy > objspace_flatness_squared) { // half-pixel error allowed... need to be smaller if AA + stbtt__tesselate_curve(points, num_points, x0,y0, (x0+x1)/2.0f,(y0+y1)/2.0f, mx,my, objspace_flatness_squared,n+1); + stbtt__tesselate_curve(points, num_points, mx,my, (x1+x2)/2.0f,(y1+y2)/2.0f, x2,y2, objspace_flatness_squared,n+1); + } else { + stbtt__add_point(points, *num_points,x2,y2); + *num_points = *num_points+1; + } + return 1; +} + +static void stbtt__tesselate_cubic(stbtt__point *points, int *num_points, float x0, float y0, float x1, float y1, float x2, float y2, float x3, float y3, float objspace_flatness_squared, int n) +{ + // @TODO this "flatness" calculation is just made-up nonsense that seems to work well enough + float dx0 = x1-x0; + float dy0 = y1-y0; + float dx1 = x2-x1; + float dy1 = y2-y1; + float dx2 = x3-x2; + float dy2 = y3-y2; + float dx = x3-x0; + float dy = y3-y0; + float longlen = (float) (STBTT_sqrt(dx0*dx0+dy0*dy0)+STBTT_sqrt(dx1*dx1+dy1*dy1)+STBTT_sqrt(dx2*dx2+dy2*dy2)); + float shortlen = (float) STBTT_sqrt(dx*dx+dy*dy); + float flatness_squared = longlen*longlen-shortlen*shortlen; + + if (n > 16) // 65536 segments on one curve better be enough! + return; + + if (flatness_squared > objspace_flatness_squared) { + float x01 = (x0+x1)/2; + float y01 = (y0+y1)/2; + float x12 = (x1+x2)/2; + float y12 = (y1+y2)/2; + float x23 = (x2+x3)/2; + float y23 = (y2+y3)/2; + + float xa = (x01+x12)/2; + float ya = (y01+y12)/2; + float xb = (x12+x23)/2; + float yb = (y12+y23)/2; + + float mx = (xa+xb)/2; + float my = (ya+yb)/2; + + stbtt__tesselate_cubic(points, num_points, x0,y0, x01,y01, xa,ya, mx,my, objspace_flatness_squared,n+1); + stbtt__tesselate_cubic(points, num_points, mx,my, xb,yb, x23,y23, x3,y3, objspace_flatness_squared,n+1); + } else { + stbtt__add_point(points, *num_points,x3,y3); + *num_points = *num_points+1; + } +} + +// returns number of contours +static stbtt__point *stbtt_FlattenCurves(stbtt_vertex *vertices, int num_verts, float objspace_flatness, int **contour_lengths, int *num_contours, void *userdata) +{ + stbtt__point *points=0; + int num_points=0; + + float objspace_flatness_squared = objspace_flatness * objspace_flatness; + int i,n=0,start=0, pass; + + // count how many "moves" there are to get the contour count + for (i=0; i < num_verts; ++i) + if (vertices[i].type == STBTT_vmove) + ++n; + + *num_contours = n; + if (n == 0) return 0; + + *contour_lengths = (int *) STBTT_malloc(sizeof(**contour_lengths) * n, userdata); + + if (*contour_lengths == 0) { + *num_contours = 0; + return 0; + } + + // make two passes through the points so we don't need to realloc + for (pass=0; pass < 2; ++pass) { + float x=0,y=0; + if (pass == 1) { + points = (stbtt__point *) STBTT_malloc(num_points * sizeof(points[0]), userdata); + if (points == NULL) goto error; + } + num_points = 0; + n= -1; + for (i=0; i < num_verts; ++i) { + switch (vertices[i].type) { + case STBTT_vmove: + // start the next contour + if (n >= 0) + (*contour_lengths)[n] = num_points - start; + ++n; + start = num_points; + + x = vertices[i].x, y = vertices[i].y; + stbtt__add_point(points, num_points++, x,y); + break; + case STBTT_vline: + x = vertices[i].x, y = vertices[i].y; + stbtt__add_point(points, num_points++, x, y); + break; + case STBTT_vcurve: + stbtt__tesselate_curve(points, &num_points, x,y, + vertices[i].cx, vertices[i].cy, + vertices[i].x, vertices[i].y, + objspace_flatness_squared, 0); + x = vertices[i].x, y = vertices[i].y; + break; + case STBTT_vcubic: + stbtt__tesselate_cubic(points, &num_points, x,y, + vertices[i].cx, vertices[i].cy, + vertices[i].cx1, vertices[i].cy1, + vertices[i].x, vertices[i].y, + objspace_flatness_squared, 0); + x = vertices[i].x, y = vertices[i].y; + break; + } + } + (*contour_lengths)[n] = num_points - start; + } + + return points; +error: + STBTT_free(points, userdata); + STBTT_free(*contour_lengths, userdata); + *contour_lengths = 0; + *num_contours = 0; + return NULL; +} + +STBTT_DEF void stbtt_Rasterize(stbtt__bitmap *result, float flatness_in_pixels, stbtt_vertex *vertices, int num_verts, float scale_x, float scale_y, float shift_x, float shift_y, int x_off, int y_off, int invert, void *userdata) +{ + float scale = scale_x > scale_y ? scale_y : scale_x; + int winding_count = 0; + int *winding_lengths = NULL; + stbtt__point *windings = stbtt_FlattenCurves(vertices, num_verts, flatness_in_pixels / scale, &winding_lengths, &winding_count, userdata); + if (windings) { + stbtt__rasterize(result, windings, winding_lengths, winding_count, scale_x, scale_y, shift_x, shift_y, x_off, y_off, invert, userdata); + STBTT_free(winding_lengths, userdata); + STBTT_free(windings, userdata); + } +} + +STBTT_DEF void stbtt_FreeBitmap(unsigned char *bitmap, void *userdata) +{ + STBTT_free(bitmap, userdata); +} + +STBTT_DEF unsigned char *stbtt_GetGlyphBitmapSubpixel(const stbtt_fontinfo *info, float scale_x, float scale_y, float shift_x, float shift_y, int glyph, int *width, int *height, int *xoff, int *yoff) +{ + int ix0,iy0,ix1,iy1; + stbtt__bitmap gbm; + stbtt_vertex *vertices; + int num_verts = stbtt_GetGlyphShape(info, glyph, &vertices); + + if (scale_x == 0) scale_x = scale_y; + if (scale_y == 0) { + if (scale_x == 0) { + STBTT_free(vertices, info->userdata); + return NULL; + } + scale_y = scale_x; + } + + stbtt_GetGlyphBitmapBoxSubpixel(info, glyph, scale_x, scale_y, shift_x, shift_y, &ix0,&iy0,&ix1,&iy1); + + // now we get the size + gbm.w = (ix1 - ix0); + gbm.h = (iy1 - iy0); + gbm.pixels = NULL; // in case we error + + if (width ) *width = gbm.w; + if (height) *height = gbm.h; + if (xoff ) *xoff = ix0; + if (yoff ) *yoff = iy0; + + if (gbm.w && gbm.h) { + gbm.pixels = (unsigned char *) STBTT_malloc(gbm.w * gbm.h, info->userdata); + if (gbm.pixels) { + gbm.stride = gbm.w; + + stbtt_Rasterize(&gbm, 0.35f, vertices, num_verts, scale_x, scale_y, shift_x, shift_y, ix0, iy0, 1, info->userdata); + } + } + STBTT_free(vertices, info->userdata); + return gbm.pixels; +} + +STBTT_DEF unsigned char *stbtt_GetGlyphBitmap(const stbtt_fontinfo *info, float scale_x, float scale_y, int glyph, int *width, int *height, int *xoff, int *yoff) +{ + return stbtt_GetGlyphBitmapSubpixel(info, scale_x, scale_y, 0.0f, 0.0f, glyph, width, height, xoff, yoff); +} + +STBTT_DEF void stbtt_MakeGlyphBitmapSubpixel(const stbtt_fontinfo *info, unsigned char *output, int out_w, int out_h, int out_stride, float scale_x, float scale_y, float shift_x, float shift_y, int glyph) +{ + int ix0,iy0; + stbtt_vertex *vertices; + int num_verts = stbtt_GetGlyphShape(info, glyph, &vertices); + stbtt__bitmap gbm; + + stbtt_GetGlyphBitmapBoxSubpixel(info, glyph, scale_x, scale_y, shift_x, shift_y, &ix0,&iy0,0,0); + gbm.pixels = output; + gbm.w = out_w; + gbm.h = out_h; + gbm.stride = out_stride; + + if (gbm.w && gbm.h) + stbtt_Rasterize(&gbm, 0.35f, vertices, num_verts, scale_x, scale_y, shift_x, shift_y, ix0,iy0, 1, info->userdata); + + STBTT_free(vertices, info->userdata); +} + +STBTT_DEF void stbtt_MakeGlyphBitmap(const stbtt_fontinfo *info, unsigned char *output, int out_w, int out_h, int out_stride, float scale_x, float scale_y, int glyph) +{ + stbtt_MakeGlyphBitmapSubpixel(info, output, out_w, out_h, out_stride, scale_x, scale_y, 0.0f,0.0f, glyph); +} + +STBTT_DEF unsigned char *stbtt_GetCodepointBitmapSubpixel(const stbtt_fontinfo *info, float scale_x, float scale_y, float shift_x, float shift_y, int codepoint, int *width, int *height, int *xoff, int *yoff) +{ + return stbtt_GetGlyphBitmapSubpixel(info, scale_x, scale_y,shift_x,shift_y, stbtt_FindGlyphIndex(info,codepoint), width,height,xoff,yoff); +} + +STBTT_DEF void stbtt_MakeCodepointBitmapSubpixelPrefilter(const stbtt_fontinfo *info, unsigned char *output, int out_w, int out_h, int out_stride, float scale_x, float scale_y, float shift_x, float shift_y, int oversample_x, int oversample_y, float *sub_x, float *sub_y, int codepoint) +{ + stbtt_MakeGlyphBitmapSubpixelPrefilter(info, output, out_w, out_h, out_stride, scale_x, scale_y, shift_x, shift_y, oversample_x, oversample_y, sub_x, sub_y, stbtt_FindGlyphIndex(info,codepoint)); +} + +STBTT_DEF void stbtt_MakeCodepointBitmapSubpixel(const stbtt_fontinfo *info, unsigned char *output, int out_w, int out_h, int out_stride, float scale_x, float scale_y, float shift_x, float shift_y, int codepoint) +{ + stbtt_MakeGlyphBitmapSubpixel(info, output, out_w, out_h, out_stride, scale_x, scale_y, shift_x, shift_y, stbtt_FindGlyphIndex(info,codepoint)); +} + +STBTT_DEF unsigned char *stbtt_GetCodepointBitmap(const stbtt_fontinfo *info, float scale_x, float scale_y, int codepoint, int *width, int *height, int *xoff, int *yoff) +{ + return stbtt_GetCodepointBitmapSubpixel(info, scale_x, scale_y, 0.0f,0.0f, codepoint, width,height,xoff,yoff); +} + +STBTT_DEF void stbtt_MakeCodepointBitmap(const stbtt_fontinfo *info, unsigned char *output, int out_w, int out_h, int out_stride, float scale_x, float scale_y, int codepoint) +{ + stbtt_MakeCodepointBitmapSubpixel(info, output, out_w, out_h, out_stride, scale_x, scale_y, 0.0f,0.0f, codepoint); +} + +////////////////////////////////////////////////////////////////////////////// +// +// bitmap baking +// +// This is SUPER-CRAPPY packing to keep source code small + +static int stbtt_BakeFontBitmap_internal(unsigned char *data, int offset, // font location (use offset=0 for plain .ttf) + float pixel_height, // height of font in pixels + unsigned char *pixels, int pw, int ph, // bitmap to be filled in + int first_char, int num_chars, // characters to bake + stbtt_bakedchar *chardata) +{ + float scale; + int x,y,bottom_y, i; + stbtt_fontinfo f; + f.userdata = NULL; + if (!stbtt_InitFont(&f, data, offset)) + return -1; + STBTT_memset(pixels, 0, pw*ph); // background of 0 around pixels + x=y=1; + bottom_y = 1; + + scale = stbtt_ScaleForPixelHeight(&f, pixel_height); + + for (i=0; i < num_chars; ++i) { + int advance, lsb, x0,y0,x1,y1,gw,gh; + int g = stbtt_FindGlyphIndex(&f, first_char + i); + stbtt_GetGlyphHMetrics(&f, g, &advance, &lsb); + stbtt_GetGlyphBitmapBox(&f, g, scale,scale, &x0,&y0,&x1,&y1); + gw = x1-x0; + gh = y1-y0; + if (x + gw + 1 >= pw) + y = bottom_y, x = 1; // advance to next row + if (y + gh + 1 >= ph) // check if it fits vertically AFTER potentially moving to next row + return -i; + STBTT_assert(x+gw < pw); + STBTT_assert(y+gh < ph); + stbtt_MakeGlyphBitmap(&f, pixels+x+y*pw, gw,gh,pw, scale,scale, g); + chardata[i].x0 = (stbtt_int16) x; + chardata[i].y0 = (stbtt_int16) y; + chardata[i].x1 = (stbtt_int16) (x + gw); + chardata[i].y1 = (stbtt_int16) (y + gh); + chardata[i].xadvance = scale * advance; + chardata[i].xoff = (float) x0; + chardata[i].yoff = (float) y0; + x = x + gw + 1; + if (y+gh+1 > bottom_y) + bottom_y = y+gh+1; + } + return bottom_y; +} + +STBTT_DEF void stbtt_GetBakedQuad(const stbtt_bakedchar *chardata, int pw, int ph, int char_index, float *xpos, float *ypos, stbtt_aligned_quad *q, int opengl_fillrule) +{ + float d3d_bias = opengl_fillrule ? 0 : -0.5f; + float ipw = 1.0f / pw, iph = 1.0f / ph; + const stbtt_bakedchar *b = chardata + char_index; + int round_x = STBTT_ifloor((*xpos + b->xoff) + 0.5f); + int round_y = STBTT_ifloor((*ypos + b->yoff) + 0.5f); + + q->x0 = round_x + d3d_bias; + q->y0 = round_y + d3d_bias; + q->x1 = round_x + b->x1 - b->x0 + d3d_bias; + q->y1 = round_y + b->y1 - b->y0 + d3d_bias; + + q->s0 = b->x0 * ipw; + q->t0 = b->y0 * iph; + q->s1 = b->x1 * ipw; + q->t1 = b->y1 * iph; + + *xpos += b->xadvance; +} + +////////////////////////////////////////////////////////////////////////////// +// +// rectangle packing replacement routines if you don't have stb_rect_pack.h +// + +#ifndef STB_RECT_PACK_VERSION + +typedef int stbrp_coord; + +//////////////////////////////////////////////////////////////////////////////////// +// // +// // +// COMPILER WARNING ?!?!? // +// // +// // +// if you get a compile warning due to these symbols being defined more than // +// once, move #include "stb_rect_pack.h" before #include "stb_truetype.h" // +// // +//////////////////////////////////////////////////////////////////////////////////// + +typedef struct +{ + int width,height; + int x,y,bottom_y; +} stbrp_context; + +typedef struct +{ + unsigned char x; +} stbrp_node; + +struct stbrp_rect +{ + stbrp_coord x,y; + int id,w,h,was_packed; +}; + +static void stbrp_init_target(stbrp_context *con, int pw, int ph, stbrp_node *nodes, int num_nodes) +{ + con->width = pw; + con->height = ph; + con->x = 0; + con->y = 0; + con->bottom_y = 0; + STBTT__NOTUSED(nodes); + STBTT__NOTUSED(num_nodes); +} + +static void stbrp_pack_rects(stbrp_context *con, stbrp_rect *rects, int num_rects) +{ + int i; + for (i=0; i < num_rects; ++i) { + if (con->x + rects[i].w > con->width) { + con->x = 0; + con->y = con->bottom_y; + } + if (con->y + rects[i].h > con->height) + break; + rects[i].x = con->x; + rects[i].y = con->y; + rects[i].was_packed = 1; + con->x += rects[i].w; + if (con->y + rects[i].h > con->bottom_y) + con->bottom_y = con->y + rects[i].h; + } + for ( ; i < num_rects; ++i) + rects[i].was_packed = 0; +} +#endif + +////////////////////////////////////////////////////////////////////////////// +// +// bitmap baking +// +// This is SUPER-AWESOME (tm Ryan Gordon) packing using stb_rect_pack.h. If +// stb_rect_pack.h isn't available, it uses the BakeFontBitmap strategy. + +STBTT_DEF int stbtt_PackBegin(stbtt_pack_context *spc, unsigned char *pixels, int pw, int ph, int stride_in_bytes, int padding, void *alloc_context) +{ + stbrp_context *context = (stbrp_context *) STBTT_malloc(sizeof(*context) ,alloc_context); + int num_nodes = pw - padding; + stbrp_node *nodes = (stbrp_node *) STBTT_malloc(sizeof(*nodes ) * num_nodes,alloc_context); + + if (context == NULL || nodes == NULL) { + if (context != NULL) STBTT_free(context, alloc_context); + if (nodes != NULL) STBTT_free(nodes , alloc_context); + return 0; + } + + spc->user_allocator_context = alloc_context; + spc->width = pw; + spc->height = ph; + spc->pixels = pixels; + spc->pack_info = context; + spc->nodes = nodes; + spc->padding = padding; + spc->stride_in_bytes = stride_in_bytes != 0 ? stride_in_bytes : pw; + spc->h_oversample = 1; + spc->v_oversample = 1; + spc->skip_missing = 0; + + stbrp_init_target(context, pw-padding, ph-padding, nodes, num_nodes); + + if (pixels) + STBTT_memset(pixels, 0, pw*ph); // background of 0 around pixels + + return 1; +} + +STBTT_DEF void stbtt_PackEnd (stbtt_pack_context *spc) +{ + STBTT_free(spc->nodes , spc->user_allocator_context); + STBTT_free(spc->pack_info, spc->user_allocator_context); +} + +STBTT_DEF void stbtt_PackSetOversampling(stbtt_pack_context *spc, unsigned int h_oversample, unsigned int v_oversample) +{ + STBTT_assert(h_oversample <= STBTT_MAX_OVERSAMPLE); + STBTT_assert(v_oversample <= STBTT_MAX_OVERSAMPLE); + if (h_oversample <= STBTT_MAX_OVERSAMPLE) + spc->h_oversample = h_oversample; + if (v_oversample <= STBTT_MAX_OVERSAMPLE) + spc->v_oversample = v_oversample; +} + +STBTT_DEF void stbtt_PackSetSkipMissingCodepoints(stbtt_pack_context *spc, int skip) +{ + spc->skip_missing = skip; +} + +#define STBTT__OVER_MASK (STBTT_MAX_OVERSAMPLE-1) + +static void stbtt__h_prefilter(unsigned char *pixels, int w, int h, int stride_in_bytes, unsigned int kernel_width) +{ + unsigned char buffer[STBTT_MAX_OVERSAMPLE]; + int safe_w = w - kernel_width; + int j; + STBTT_memset(buffer, 0, STBTT_MAX_OVERSAMPLE); // suppress bogus warning from VS2013 -analyze + for (j=0; j < h; ++j) { + int i; + unsigned int total; + STBTT_memset(buffer, 0, kernel_width); + + total = 0; + + // make kernel_width a constant in common cases so compiler can optimize out the divide + switch (kernel_width) { + case 2: + for (i=0; i <= safe_w; ++i) { + total += pixels[i] - buffer[i & STBTT__OVER_MASK]; + buffer[(i+kernel_width) & STBTT__OVER_MASK] = pixels[i]; + pixels[i] = (unsigned char) (total / 2); + } + break; + case 3: + for (i=0; i <= safe_w; ++i) { + total += pixels[i] - buffer[i & STBTT__OVER_MASK]; + buffer[(i+kernel_width) & STBTT__OVER_MASK] = pixels[i]; + pixels[i] = (unsigned char) (total / 3); + } + break; + case 4: + for (i=0; i <= safe_w; ++i) { + total += pixels[i] - buffer[i & STBTT__OVER_MASK]; + buffer[(i+kernel_width) & STBTT__OVER_MASK] = pixels[i]; + pixels[i] = (unsigned char) (total / 4); + } + break; + case 5: + for (i=0; i <= safe_w; ++i) { + total += pixels[i] - buffer[i & STBTT__OVER_MASK]; + buffer[(i+kernel_width) & STBTT__OVER_MASK] = pixels[i]; + pixels[i] = (unsigned char) (total / 5); + } + break; + default: + for (i=0; i <= safe_w; ++i) { + total += pixels[i] - buffer[i & STBTT__OVER_MASK]; + buffer[(i+kernel_width) & STBTT__OVER_MASK] = pixels[i]; + pixels[i] = (unsigned char) (total / kernel_width); + } + break; + } + + for (; i < w; ++i) { + STBTT_assert(pixels[i] == 0); + total -= buffer[i & STBTT__OVER_MASK]; + pixels[i] = (unsigned char) (total / kernel_width); + } + + pixels += stride_in_bytes; + } +} + +static void stbtt__v_prefilter(unsigned char *pixels, int w, int h, int stride_in_bytes, unsigned int kernel_width) +{ + unsigned char buffer[STBTT_MAX_OVERSAMPLE]; + int safe_h = h - kernel_width; + int j; + STBTT_memset(buffer, 0, STBTT_MAX_OVERSAMPLE); // suppress bogus warning from VS2013 -analyze + for (j=0; j < w; ++j) { + int i; + unsigned int total; + STBTT_memset(buffer, 0, kernel_width); + + total = 0; + + // make kernel_width a constant in common cases so compiler can optimize out the divide + switch (kernel_width) { + case 2: + for (i=0; i <= safe_h; ++i) { + total += pixels[i*stride_in_bytes] - buffer[i & STBTT__OVER_MASK]; + buffer[(i+kernel_width) & STBTT__OVER_MASK] = pixels[i*stride_in_bytes]; + pixels[i*stride_in_bytes] = (unsigned char) (total / 2); + } + break; + case 3: + for (i=0; i <= safe_h; ++i) { + total += pixels[i*stride_in_bytes] - buffer[i & STBTT__OVER_MASK]; + buffer[(i+kernel_width) & STBTT__OVER_MASK] = pixels[i*stride_in_bytes]; + pixels[i*stride_in_bytes] = (unsigned char) (total / 3); + } + break; + case 4: + for (i=0; i <= safe_h; ++i) { + total += pixels[i*stride_in_bytes] - buffer[i & STBTT__OVER_MASK]; + buffer[(i+kernel_width) & STBTT__OVER_MASK] = pixels[i*stride_in_bytes]; + pixels[i*stride_in_bytes] = (unsigned char) (total / 4); + } + break; + case 5: + for (i=0; i <= safe_h; ++i) { + total += pixels[i*stride_in_bytes] - buffer[i & STBTT__OVER_MASK]; + buffer[(i+kernel_width) & STBTT__OVER_MASK] = pixels[i*stride_in_bytes]; + pixels[i*stride_in_bytes] = (unsigned char) (total / 5); + } + break; + default: + for (i=0; i <= safe_h; ++i) { + total += pixels[i*stride_in_bytes] - buffer[i & STBTT__OVER_MASK]; + buffer[(i+kernel_width) & STBTT__OVER_MASK] = pixels[i*stride_in_bytes]; + pixels[i*stride_in_bytes] = (unsigned char) (total / kernel_width); + } + break; + } + + for (; i < h; ++i) { + STBTT_assert(pixels[i*stride_in_bytes] == 0); + total -= buffer[i & STBTT__OVER_MASK]; + pixels[i*stride_in_bytes] = (unsigned char) (total / kernel_width); + } + + pixels += 1; + } +} + +static float stbtt__oversample_shift(int oversample) +{ + if (!oversample) + return 0.0f; + + // The prefilter is a box filter of width "oversample", + // which shifts phase by (oversample - 1)/2 pixels in + // oversampled space. We want to shift in the opposite + // direction to counter this. + return (float)-(oversample - 1) / (2.0f * (float)oversample); +} + +// rects array must be big enough to accommodate all characters in the given ranges +STBTT_DEF int stbtt_PackFontRangesGatherRects(stbtt_pack_context *spc, const stbtt_fontinfo *info, stbtt_pack_range *ranges, int num_ranges, stbrp_rect *rects) +{ + int i,j,k; + int missing_glyph_added = 0; + + k=0; + for (i=0; i < num_ranges; ++i) { + float fh = ranges[i].font_size; + float scale = fh > 0 ? stbtt_ScaleForPixelHeight(info, fh) : stbtt_ScaleForMappingEmToPixels(info, -fh); + ranges[i].h_oversample = (unsigned char) spc->h_oversample; + ranges[i].v_oversample = (unsigned char) spc->v_oversample; + for (j=0; j < ranges[i].num_chars; ++j) { + int x0,y0,x1,y1; + int codepoint = ranges[i].array_of_unicode_codepoints == NULL ? ranges[i].first_unicode_codepoint_in_range + j : ranges[i].array_of_unicode_codepoints[j]; + int glyph = stbtt_FindGlyphIndex(info, codepoint); + if (glyph == 0 && (spc->skip_missing || missing_glyph_added)) { + rects[k].w = rects[k].h = 0; + } else { + stbtt_GetGlyphBitmapBoxSubpixel(info,glyph, + scale * spc->h_oversample, + scale * spc->v_oversample, + 0,0, + &x0,&y0,&x1,&y1); + rects[k].w = (stbrp_coord) (x1-x0 + spc->padding + spc->h_oversample-1); + rects[k].h = (stbrp_coord) (y1-y0 + spc->padding + spc->v_oversample-1); + if (glyph == 0) + missing_glyph_added = 1; + } + ++k; + } + } + + return k; +} + +STBTT_DEF void stbtt_MakeGlyphBitmapSubpixelPrefilter(const stbtt_fontinfo *info, unsigned char *output, int out_w, int out_h, int out_stride, float scale_x, float scale_y, float shift_x, float shift_y, int prefilter_x, int prefilter_y, float *sub_x, float *sub_y, int glyph) +{ + stbtt_MakeGlyphBitmapSubpixel(info, + output, + out_w - (prefilter_x - 1), + out_h - (prefilter_y - 1), + out_stride, + scale_x, + scale_y, + shift_x, + shift_y, + glyph); + + if (prefilter_x > 1) + stbtt__h_prefilter(output, out_w, out_h, out_stride, prefilter_x); + + if (prefilter_y > 1) + stbtt__v_prefilter(output, out_w, out_h, out_stride, prefilter_y); + + *sub_x = stbtt__oversample_shift(prefilter_x); + *sub_y = stbtt__oversample_shift(prefilter_y); +} + +// rects array must be big enough to accommodate all characters in the given ranges +STBTT_DEF int stbtt_PackFontRangesRenderIntoRects(stbtt_pack_context *spc, const stbtt_fontinfo *info, stbtt_pack_range *ranges, int num_ranges, stbrp_rect *rects) +{ + int i,j,k, missing_glyph = -1, return_value = 1; + + // save current values + int old_h_over = spc->h_oversample; + int old_v_over = spc->v_oversample; + + k = 0; + for (i=0; i < num_ranges; ++i) { + float fh = ranges[i].font_size; + float scale = fh > 0 ? stbtt_ScaleForPixelHeight(info, fh) : stbtt_ScaleForMappingEmToPixels(info, -fh); + float recip_h,recip_v,sub_x,sub_y; + spc->h_oversample = ranges[i].h_oversample; + spc->v_oversample = ranges[i].v_oversample; + recip_h = 1.0f / spc->h_oversample; + recip_v = 1.0f / spc->v_oversample; + sub_x = stbtt__oversample_shift(spc->h_oversample); + sub_y = stbtt__oversample_shift(spc->v_oversample); + for (j=0; j < ranges[i].num_chars; ++j) { + stbrp_rect *r = &rects[k]; + if (r->was_packed && r->w != 0 && r->h != 0) { + stbtt_packedchar *bc = &ranges[i].chardata_for_range[j]; + int advance, lsb, x0,y0,x1,y1; + int codepoint = ranges[i].array_of_unicode_codepoints == NULL ? ranges[i].first_unicode_codepoint_in_range + j : ranges[i].array_of_unicode_codepoints[j]; + int glyph = stbtt_FindGlyphIndex(info, codepoint); + stbrp_coord pad = (stbrp_coord) spc->padding; + + // pad on left and top + r->x += pad; + r->y += pad; + r->w -= pad; + r->h -= pad; + stbtt_GetGlyphHMetrics(info, glyph, &advance, &lsb); + stbtt_GetGlyphBitmapBox(info, glyph, + scale * spc->h_oversample, + scale * spc->v_oversample, + &x0,&y0,&x1,&y1); + stbtt_MakeGlyphBitmapSubpixel(info, + spc->pixels + r->x + r->y*spc->stride_in_bytes, + r->w - spc->h_oversample+1, + r->h - spc->v_oversample+1, + spc->stride_in_bytes, + scale * spc->h_oversample, + scale * spc->v_oversample, + 0,0, + glyph); + + if (spc->h_oversample > 1) + stbtt__h_prefilter(spc->pixels + r->x + r->y*spc->stride_in_bytes, + r->w, r->h, spc->stride_in_bytes, + spc->h_oversample); + + if (spc->v_oversample > 1) + stbtt__v_prefilter(spc->pixels + r->x + r->y*spc->stride_in_bytes, + r->w, r->h, spc->stride_in_bytes, + spc->v_oversample); + + bc->x0 = (stbtt_int16) r->x; + bc->y0 = (stbtt_int16) r->y; + bc->x1 = (stbtt_int16) (r->x + r->w); + bc->y1 = (stbtt_int16) (r->y + r->h); + bc->xadvance = scale * advance; + bc->xoff = (float) x0 * recip_h + sub_x; + bc->yoff = (float) y0 * recip_v + sub_y; + bc->xoff2 = (x0 + r->w) * recip_h + sub_x; + bc->yoff2 = (y0 + r->h) * recip_v + sub_y; + + if (glyph == 0) + missing_glyph = j; + } else if (spc->skip_missing) { + return_value = 0; + } else if (r->was_packed && r->w == 0 && r->h == 0 && missing_glyph >= 0) { + ranges[i].chardata_for_range[j] = ranges[i].chardata_for_range[missing_glyph]; + } else { + return_value = 0; // if any fail, report failure + } + + ++k; + } + } + + // restore original values + spc->h_oversample = old_h_over; + spc->v_oversample = old_v_over; + + return return_value; +} + +STBTT_DEF void stbtt_PackFontRangesPackRects(stbtt_pack_context *spc, stbrp_rect *rects, int num_rects) +{ + stbrp_pack_rects((stbrp_context *) spc->pack_info, rects, num_rects); +} + +STBTT_DEF int stbtt_PackFontRanges(stbtt_pack_context *spc, const unsigned char *fontdata, int font_index, stbtt_pack_range *ranges, int num_ranges) +{ + stbtt_fontinfo info; + int i,j,n, return_value = 1; + //stbrp_context *context = (stbrp_context *) spc->pack_info; + stbrp_rect *rects; + + // flag all characters as NOT packed + for (i=0; i < num_ranges; ++i) + for (j=0; j < ranges[i].num_chars; ++j) + ranges[i].chardata_for_range[j].x0 = + ranges[i].chardata_for_range[j].y0 = + ranges[i].chardata_for_range[j].x1 = + ranges[i].chardata_for_range[j].y1 = 0; + + n = 0; + for (i=0; i < num_ranges; ++i) + n += ranges[i].num_chars; + + rects = (stbrp_rect *) STBTT_malloc(sizeof(*rects) * n, spc->user_allocator_context); + if (rects == NULL) + return 0; + + info.userdata = spc->user_allocator_context; + stbtt_InitFont(&info, fontdata, stbtt_GetFontOffsetForIndex(fontdata,font_index)); + + n = stbtt_PackFontRangesGatherRects(spc, &info, ranges, num_ranges, rects); + + stbtt_PackFontRangesPackRects(spc, rects, n); + + return_value = stbtt_PackFontRangesRenderIntoRects(spc, &info, ranges, num_ranges, rects); + + STBTT_free(rects, spc->user_allocator_context); + return return_value; +} + +STBTT_DEF int stbtt_PackFontRange(stbtt_pack_context *spc, const unsigned char *fontdata, int font_index, float font_size, + int first_unicode_codepoint_in_range, int num_chars_in_range, stbtt_packedchar *chardata_for_range) +{ + stbtt_pack_range range; + range.first_unicode_codepoint_in_range = first_unicode_codepoint_in_range; + range.array_of_unicode_codepoints = NULL; + range.num_chars = num_chars_in_range; + range.chardata_for_range = chardata_for_range; + range.font_size = font_size; + return stbtt_PackFontRanges(spc, fontdata, font_index, &range, 1); +} + +STBTT_DEF void stbtt_GetScaledFontVMetrics(const unsigned char *fontdata, int index, float size, float *ascent, float *descent, float *lineGap) +{ + int i_ascent, i_descent, i_lineGap; + float scale; + stbtt_fontinfo info; + stbtt_InitFont(&info, fontdata, stbtt_GetFontOffsetForIndex(fontdata, index)); + scale = size > 0 ? stbtt_ScaleForPixelHeight(&info, size) : stbtt_ScaleForMappingEmToPixels(&info, -size); + stbtt_GetFontVMetrics(&info, &i_ascent, &i_descent, &i_lineGap); + *ascent = (float) i_ascent * scale; + *descent = (float) i_descent * scale; + *lineGap = (float) i_lineGap * scale; +} + +STBTT_DEF void stbtt_GetPackedQuad(const stbtt_packedchar *chardata, int pw, int ph, int char_index, float *xpos, float *ypos, stbtt_aligned_quad *q, int align_to_integer) +{ + float ipw = 1.0f / pw, iph = 1.0f / ph; + const stbtt_packedchar *b = chardata + char_index; + + if (align_to_integer) { + float x = (float) STBTT_ifloor((*xpos + b->xoff) + 0.5f); + float y = (float) STBTT_ifloor((*ypos + b->yoff) + 0.5f); + q->x0 = x; + q->y0 = y; + q->x1 = x + b->xoff2 - b->xoff; + q->y1 = y + b->yoff2 - b->yoff; + } else { + q->x0 = *xpos + b->xoff; + q->y0 = *ypos + b->yoff; + q->x1 = *xpos + b->xoff2; + q->y1 = *ypos + b->yoff2; + } + + q->s0 = b->x0 * ipw; + q->t0 = b->y0 * iph; + q->s1 = b->x1 * ipw; + q->t1 = b->y1 * iph; + + *xpos += b->xadvance; +} + +////////////////////////////////////////////////////////////////////////////// +// +// sdf computation +// + +#define STBTT_min(a,b) ((a) < (b) ? (a) : (b)) +#define STBTT_max(a,b) ((a) < (b) ? (b) : (a)) + +static int stbtt__ray_intersect_bezier(float orig[2], float ray[2], float q0[2], float q1[2], float q2[2], float hits[2][2]) +{ + float q0perp = q0[1]*ray[0] - q0[0]*ray[1]; + float q1perp = q1[1]*ray[0] - q1[0]*ray[1]; + float q2perp = q2[1]*ray[0] - q2[0]*ray[1]; + float roperp = orig[1]*ray[0] - orig[0]*ray[1]; + + float a = q0perp - 2*q1perp + q2perp; + float b = q1perp - q0perp; + float c = q0perp - roperp; + + float s0 = 0., s1 = 0.; + int num_s = 0; + + if (a != 0.0) { + float discr = b*b - a*c; + if (discr > 0.0) { + float rcpna = -1 / a; + float d = (float) STBTT_sqrt(discr); + s0 = (b+d) * rcpna; + s1 = (b-d) * rcpna; + if (s0 >= 0.0 && s0 <= 1.0) + num_s = 1; + if (d > 0.0 && s1 >= 0.0 && s1 <= 1.0) { + if (num_s == 0) s0 = s1; + ++num_s; + } + } + } else { + // 2*b*s + c = 0 + // s = -c / (2*b) + s0 = c / (-2 * b); + if (s0 >= 0.0 && s0 <= 1.0) + num_s = 1; + } + + if (num_s == 0) + return 0; + else { + float rcp_len2 = 1 / (ray[0]*ray[0] + ray[1]*ray[1]); + float rayn_x = ray[0] * rcp_len2, rayn_y = ray[1] * rcp_len2; + + float q0d = q0[0]*rayn_x + q0[1]*rayn_y; + float q1d = q1[0]*rayn_x + q1[1]*rayn_y; + float q2d = q2[0]*rayn_x + q2[1]*rayn_y; + float rod = orig[0]*rayn_x + orig[1]*rayn_y; + + float q10d = q1d - q0d; + float q20d = q2d - q0d; + float q0rd = q0d - rod; + + hits[0][0] = q0rd + s0*(2.0f - 2.0f*s0)*q10d + s0*s0*q20d; + hits[0][1] = a*s0+b; + + if (num_s > 1) { + hits[1][0] = q0rd + s1*(2.0f - 2.0f*s1)*q10d + s1*s1*q20d; + hits[1][1] = a*s1+b; + return 2; + } else { + return 1; + } + } +} + +static int equal(float *a, float *b) +{ + return (a[0] == b[0] && a[1] == b[1]); +} + +static int stbtt__compute_crossings_x(float x, float y, int nverts, stbtt_vertex *verts) +{ + int i; + float orig[2], ray[2] = { 1, 0 }; + float y_frac; + int winding = 0; + + // make sure y never passes through a vertex of the shape + y_frac = (float) STBTT_fmod(y, 1.0f); + if (y_frac < 0.01f) + y += 0.01f; + else if (y_frac > 0.99f) + y -= 0.01f; + + orig[0] = x; + orig[1] = y; + + // test a ray from (-infinity,y) to (x,y) + for (i=0; i < nverts; ++i) { + if (verts[i].type == STBTT_vline) { + int x0 = (int) verts[i-1].x, y0 = (int) verts[i-1].y; + int x1 = (int) verts[i ].x, y1 = (int) verts[i ].y; + if (y > STBTT_min(y0,y1) && y < STBTT_max(y0,y1) && x > STBTT_min(x0,x1)) { + float x_inter = (y - y0) / (y1 - y0) * (x1-x0) + x0; + if (x_inter < x) + winding += (y0 < y1) ? 1 : -1; + } + } + if (verts[i].type == STBTT_vcurve) { + int x0 = (int) verts[i-1].x , y0 = (int) verts[i-1].y ; + int x1 = (int) verts[i ].cx, y1 = (int) verts[i ].cy; + int x2 = (int) verts[i ].x , y2 = (int) verts[i ].y ; + int ax = STBTT_min(x0,STBTT_min(x1,x2)), ay = STBTT_min(y0,STBTT_min(y1,y2)); + int by = STBTT_max(y0,STBTT_max(y1,y2)); + if (y > ay && y < by && x > ax) { + float q0[2],q1[2],q2[2]; + float hits[2][2]; + q0[0] = (float)x0; + q0[1] = (float)y0; + q1[0] = (float)x1; + q1[1] = (float)y1; + q2[0] = (float)x2; + q2[1] = (float)y2; + if (equal(q0,q1) || equal(q1,q2)) { + x0 = (int)verts[i-1].x; + y0 = (int)verts[i-1].y; + x1 = (int)verts[i ].x; + y1 = (int)verts[i ].y; + if (y > STBTT_min(y0,y1) && y < STBTT_max(y0,y1) && x > STBTT_min(x0,x1)) { + float x_inter = (y - y0) / (y1 - y0) * (x1-x0) + x0; + if (x_inter < x) + winding += (y0 < y1) ? 1 : -1; + } + } else { + int num_hits = stbtt__ray_intersect_bezier(orig, ray, q0, q1, q2, hits); + if (num_hits >= 1) + if (hits[0][0] < 0) + winding += (hits[0][1] < 0 ? -1 : 1); + if (num_hits >= 2) + if (hits[1][0] < 0) + winding += (hits[1][1] < 0 ? -1 : 1); + } + } + } + } + return winding; +} + +static float stbtt__cuberoot( float x ) +{ + if (x<0) + return -(float) STBTT_pow(-x,1.0f/3.0f); + else + return (float) STBTT_pow( x,1.0f/3.0f); +} + +// x^3 + a*x^2 + b*x + c = 0 +static int stbtt__solve_cubic(float a, float b, float c, float* r) +{ + float s = -a / 3; + float p = b - a*a / 3; + float q = a * (2*a*a - 9*b) / 27 + c; + float p3 = p*p*p; + float d = q*q + 4*p3 / 27; + if (d >= 0) { + float z = (float) STBTT_sqrt(d); + float u = (-q + z) / 2; + float v = (-q - z) / 2; + u = stbtt__cuberoot(u); + v = stbtt__cuberoot(v); + r[0] = s + u + v; + return 1; + } else { + float u = (float) STBTT_sqrt(-p/3); + float v = (float) STBTT_acos(-STBTT_sqrt(-27/p3) * q / 2) / 3; // p3 must be negative, since d is negative + float m = (float) STBTT_cos(v); + float n = (float) STBTT_cos(v-3.141592/2)*1.732050808f; + r[0] = s + u * 2 * m; + r[1] = s - u * (m + n); + r[2] = s - u * (m - n); + + //STBTT_assert( STBTT_fabs(((r[0]+a)*r[0]+b)*r[0]+c) < 0.05f); // these asserts may not be safe at all scales, though they're in bezier t parameter units so maybe? + //STBTT_assert( STBTT_fabs(((r[1]+a)*r[1]+b)*r[1]+c) < 0.05f); + //STBTT_assert( STBTT_fabs(((r[2]+a)*r[2]+b)*r[2]+c) < 0.05f); + return 3; + } +} + +STBTT_DEF unsigned char * stbtt_GetGlyphSDF(const stbtt_fontinfo *info, float scale, int glyph, int padding, unsigned char onedge_value, float pixel_dist_scale, int *width, int *height, int *xoff, int *yoff) +{ + float scale_x = scale, scale_y = scale; + int ix0,iy0,ix1,iy1; + int w,h; + unsigned char *data; + + if (scale == 0) return NULL; + + stbtt_GetGlyphBitmapBoxSubpixel(info, glyph, scale, scale, 0.0f,0.0f, &ix0,&iy0,&ix1,&iy1); + + // if empty, return NULL + if (ix0 == ix1 || iy0 == iy1) + return NULL; + + ix0 -= padding; + iy0 -= padding; + ix1 += padding; + iy1 += padding; + + w = (ix1 - ix0); + h = (iy1 - iy0); + + if (width ) *width = w; + if (height) *height = h; + if (xoff ) *xoff = ix0; + if (yoff ) *yoff = iy0; + + // invert for y-downwards bitmaps + scale_y = -scale_y; + + { + int x,y,i,j; + float *precompute; + stbtt_vertex *verts; + int num_verts = stbtt_GetGlyphShape(info, glyph, &verts); + data = (unsigned char *) STBTT_malloc(w * h, info->userdata); + precompute = (float *) STBTT_malloc(num_verts * sizeof(float), info->userdata); + + for (i=0,j=num_verts-1; i < num_verts; j=i++) { + if (verts[i].type == STBTT_vline) { + float x0 = verts[i].x*scale_x, y0 = verts[i].y*scale_y; + float x1 = verts[j].x*scale_x, y1 = verts[j].y*scale_y; + float dist = (float) STBTT_sqrt((x1-x0)*(x1-x0) + (y1-y0)*(y1-y0)); + precompute[i] = (dist == 0) ? 0.0f : 1.0f / dist; + } else if (verts[i].type == STBTT_vcurve) { + float x2 = verts[j].x *scale_x, y2 = verts[j].y *scale_y; + float x1 = verts[i].cx*scale_x, y1 = verts[i].cy*scale_y; + float x0 = verts[i].x *scale_x, y0 = verts[i].y *scale_y; + float bx = x0 - 2*x1 + x2, by = y0 - 2*y1 + y2; + float len2 = bx*bx + by*by; + if (len2 != 0.0f) + precompute[i] = 1.0f / (bx*bx + by*by); + else + precompute[i] = 0.0f; + } else + precompute[i] = 0.0f; + } + + for (y=iy0; y < iy1; ++y) { + for (x=ix0; x < ix1; ++x) { + float val; + float min_dist = 999999.0f; + float sx = (float) x + 0.5f; + float sy = (float) y + 0.5f; + float x_gspace = (sx / scale_x); + float y_gspace = (sy / scale_y); + + int winding = stbtt__compute_crossings_x(x_gspace, y_gspace, num_verts, verts); // @OPTIMIZE: this could just be a rasterization, but needs to be line vs. non-tesselated curves so a new path + + for (i=0; i < num_verts; ++i) { + float x0 = verts[i].x*scale_x, y0 = verts[i].y*scale_y; + + if (verts[i].type == STBTT_vline && precompute[i] != 0.0f) { + float x1 = verts[i-1].x*scale_x, y1 = verts[i-1].y*scale_y; + + float dist,dist2 = (x0-sx)*(x0-sx) + (y0-sy)*(y0-sy); + if (dist2 < min_dist*min_dist) + min_dist = (float) STBTT_sqrt(dist2); + + // coarse culling against bbox + //if (sx > STBTT_min(x0,x1)-min_dist && sx < STBTT_max(x0,x1)+min_dist && + // sy > STBTT_min(y0,y1)-min_dist && sy < STBTT_max(y0,y1)+min_dist) + dist = (float) STBTT_fabs((x1-x0)*(y0-sy) - (y1-y0)*(x0-sx)) * precompute[i]; + STBTT_assert(i != 0); + if (dist < min_dist) { + // check position along line + // x' = x0 + t*(x1-x0), y' = y0 + t*(y1-y0) + // minimize (x'-sx)*(x'-sx)+(y'-sy)*(y'-sy) + float dx = x1-x0, dy = y1-y0; + float px = x0-sx, py = y0-sy; + // minimize (px+t*dx)^2 + (py+t*dy)^2 = px*px + 2*px*dx*t + t^2*dx*dx + py*py + 2*py*dy*t + t^2*dy*dy + // derivative: 2*px*dx + 2*py*dy + (2*dx*dx+2*dy*dy)*t, set to 0 and solve + float t = -(px*dx + py*dy) / (dx*dx + dy*dy); + if (t >= 0.0f && t <= 1.0f) + min_dist = dist; + } + } else if (verts[i].type == STBTT_vcurve) { + float x2 = verts[i-1].x *scale_x, y2 = verts[i-1].y *scale_y; + float x1 = verts[i ].cx*scale_x, y1 = verts[i ].cy*scale_y; + float box_x0 = STBTT_min(STBTT_min(x0,x1),x2); + float box_y0 = STBTT_min(STBTT_min(y0,y1),y2); + float box_x1 = STBTT_max(STBTT_max(x0,x1),x2); + float box_y1 = STBTT_max(STBTT_max(y0,y1),y2); + // coarse culling against bbox to avoid computing cubic unnecessarily + if (sx > box_x0-min_dist && sx < box_x1+min_dist && sy > box_y0-min_dist && sy < box_y1+min_dist) { + int num=0; + float ax = x1-x0, ay = y1-y0; + float bx = x0 - 2*x1 + x2, by = y0 - 2*y1 + y2; + float mx = x0 - sx, my = y0 - sy; + float res[3] = {0.f,0.f,0.f}; + float px,py,t,it,dist2; + float a_inv = precompute[i]; + if (a_inv == 0.0) { // if a_inv is 0, it's 2nd degree so use quadratic formula + float a = 3*(ax*bx + ay*by); + float b = 2*(ax*ax + ay*ay) + (mx*bx+my*by); + float c = mx*ax+my*ay; + if (a == 0.0) { // if a is 0, it's linear + if (b != 0.0) { + res[num++] = -c/b; + } + } else { + float discriminant = b*b - 4*a*c; + if (discriminant < 0) + num = 0; + else { + float root = (float) STBTT_sqrt(discriminant); + res[0] = (-b - root)/(2*a); + res[1] = (-b + root)/(2*a); + num = 2; // don't bother distinguishing 1-solution case, as code below will still work + } + } + } else { + float b = 3*(ax*bx + ay*by) * a_inv; // could precompute this as it doesn't depend on sample point + float c = (2*(ax*ax + ay*ay) + (mx*bx+my*by)) * a_inv; + float d = (mx*ax+my*ay) * a_inv; + num = stbtt__solve_cubic(b, c, d, res); + } + dist2 = (x0-sx)*(x0-sx) + (y0-sy)*(y0-sy); + if (dist2 < min_dist*min_dist) + min_dist = (float) STBTT_sqrt(dist2); + + if (num >= 1 && res[0] >= 0.0f && res[0] <= 1.0f) { + t = res[0], it = 1.0f - t; + px = it*it*x0 + 2*t*it*x1 + t*t*x2; + py = it*it*y0 + 2*t*it*y1 + t*t*y2; + dist2 = (px-sx)*(px-sx) + (py-sy)*(py-sy); + if (dist2 < min_dist * min_dist) + min_dist = (float) STBTT_sqrt(dist2); + } + if (num >= 2 && res[1] >= 0.0f && res[1] <= 1.0f) { + t = res[1], it = 1.0f - t; + px = it*it*x0 + 2*t*it*x1 + t*t*x2; + py = it*it*y0 + 2*t*it*y1 + t*t*y2; + dist2 = (px-sx)*(px-sx) + (py-sy)*(py-sy); + if (dist2 < min_dist * min_dist) + min_dist = (float) STBTT_sqrt(dist2); + } + if (num >= 3 && res[2] >= 0.0f && res[2] <= 1.0f) { + t = res[2], it = 1.0f - t; + px = it*it*x0 + 2*t*it*x1 + t*t*x2; + py = it*it*y0 + 2*t*it*y1 + t*t*y2; + dist2 = (px-sx)*(px-sx) + (py-sy)*(py-sy); + if (dist2 < min_dist * min_dist) + min_dist = (float) STBTT_sqrt(dist2); + } + } + } + } + if (winding == 0) + min_dist = -min_dist; // if outside the shape, value is negative + val = onedge_value + pixel_dist_scale * min_dist; + if (val < 0) + val = 0; + else if (val > 255) + val = 255; + data[(y-iy0)*w+(x-ix0)] = (unsigned char) val; + } + } + STBTT_free(precompute, info->userdata); + STBTT_free(verts, info->userdata); + } + return data; +} + +STBTT_DEF unsigned char * stbtt_GetCodepointSDF(const stbtt_fontinfo *info, float scale, int codepoint, int padding, unsigned char onedge_value, float pixel_dist_scale, int *width, int *height, int *xoff, int *yoff) +{ + return stbtt_GetGlyphSDF(info, scale, stbtt_FindGlyphIndex(info, codepoint), padding, onedge_value, pixel_dist_scale, width, height, xoff, yoff); +} + +STBTT_DEF void stbtt_FreeSDF(unsigned char *bitmap, void *userdata) +{ + STBTT_free(bitmap, userdata); +} + +////////////////////////////////////////////////////////////////////////////// +// +// font name matching -- recommended not to use this +// + +// check if a utf8 string contains a prefix which is the utf16 string; if so return length of matching utf8 string +static stbtt_int32 stbtt__CompareUTF8toUTF16_bigendian_prefix(stbtt_uint8 *s1, stbtt_int32 len1, stbtt_uint8 *s2, stbtt_int32 len2) +{ + stbtt_int32 i=0; + + // convert utf16 to utf8 and compare the results while converting + while (len2) { + stbtt_uint16 ch = s2[0]*256 + s2[1]; + if (ch < 0x80) { + if (i >= len1) return -1; + if (s1[i++] != ch) return -1; + } else if (ch < 0x800) { + if (i+1 >= len1) return -1; + if (s1[i++] != 0xc0 + (ch >> 6)) return -1; + if (s1[i++] != 0x80 + (ch & 0x3f)) return -1; + } else if (ch >= 0xd800 && ch < 0xdc00) { + stbtt_uint32 c; + stbtt_uint16 ch2 = s2[2]*256 + s2[3]; + if (i+3 >= len1) return -1; + c = ((ch - 0xd800) << 10) + (ch2 - 0xdc00) + 0x10000; + if (s1[i++] != 0xf0 + (c >> 18)) return -1; + if (s1[i++] != 0x80 + ((c >> 12) & 0x3f)) return -1; + if (s1[i++] != 0x80 + ((c >> 6) & 0x3f)) return -1; + if (s1[i++] != 0x80 + ((c ) & 0x3f)) return -1; + s2 += 2; // plus another 2 below + len2 -= 2; + } else if (ch >= 0xdc00 && ch < 0xe000) { + return -1; + } else { + if (i+2 >= len1) return -1; + if (s1[i++] != 0xe0 + (ch >> 12)) return -1; + if (s1[i++] != 0x80 + ((ch >> 6) & 0x3f)) return -1; + if (s1[i++] != 0x80 + ((ch ) & 0x3f)) return -1; + } + s2 += 2; + len2 -= 2; + } + return i; +} + +static int stbtt_CompareUTF8toUTF16_bigendian_internal(char *s1, int len1, char *s2, int len2) +{ + return len1 == stbtt__CompareUTF8toUTF16_bigendian_prefix((stbtt_uint8*) s1, len1, (stbtt_uint8*) s2, len2); +} + +// returns results in whatever encoding you request... but note that 2-byte encodings +// will be BIG-ENDIAN... use stbtt_CompareUTF8toUTF16_bigendian() to compare +STBTT_DEF const char *stbtt_GetFontNameString(const stbtt_fontinfo *font, int *length, int platformID, int encodingID, int languageID, int nameID) +{ + stbtt_int32 i,count,stringOffset; + stbtt_uint8 *fc = font->data; + stbtt_uint32 offset = font->fontstart; + stbtt_uint32 nm = stbtt__find_table(fc, offset, "name"); + if (!nm) return NULL; + + count = ttUSHORT(fc+nm+2); + stringOffset = nm + ttUSHORT(fc+nm+4); + for (i=0; i < count; ++i) { + stbtt_uint32 loc = nm + 6 + 12 * i; + if (platformID == ttUSHORT(fc+loc+0) && encodingID == ttUSHORT(fc+loc+2) + && languageID == ttUSHORT(fc+loc+4) && nameID == ttUSHORT(fc+loc+6)) { + *length = ttUSHORT(fc+loc+8); + return (const char *) (fc+stringOffset+ttUSHORT(fc+loc+10)); + } + } + return NULL; +} + +static int stbtt__matchpair(stbtt_uint8 *fc, stbtt_uint32 nm, stbtt_uint8 *name, stbtt_int32 nlen, stbtt_int32 target_id, stbtt_int32 next_id) +{ + stbtt_int32 i; + stbtt_int32 count = ttUSHORT(fc+nm+2); + stbtt_int32 stringOffset = nm + ttUSHORT(fc+nm+4); + + for (i=0; i < count; ++i) { + stbtt_uint32 loc = nm + 6 + 12 * i; + stbtt_int32 id = ttUSHORT(fc+loc+6); + if (id == target_id) { + // find the encoding + stbtt_int32 platform = ttUSHORT(fc+loc+0), encoding = ttUSHORT(fc+loc+2), language = ttUSHORT(fc+loc+4); + + // is this a Unicode encoding? + if (platform == 0 || (platform == 3 && encoding == 1) || (platform == 3 && encoding == 10)) { + stbtt_int32 slen = ttUSHORT(fc+loc+8); + stbtt_int32 off = ttUSHORT(fc+loc+10); + + // check if there's a prefix match + stbtt_int32 matchlen = stbtt__CompareUTF8toUTF16_bigendian_prefix(name, nlen, fc+stringOffset+off,slen); + if (matchlen >= 0) { + // check for target_id+1 immediately following, with same encoding & language + if (i+1 < count && ttUSHORT(fc+loc+12+6) == next_id && ttUSHORT(fc+loc+12) == platform && ttUSHORT(fc+loc+12+2) == encoding && ttUSHORT(fc+loc+12+4) == language) { + slen = ttUSHORT(fc+loc+12+8); + off = ttUSHORT(fc+loc+12+10); + if (slen == 0) { + if (matchlen == nlen) + return 1; + } else if (matchlen < nlen && name[matchlen] == ' ') { + ++matchlen; + if (stbtt_CompareUTF8toUTF16_bigendian_internal((char*) (name+matchlen), nlen-matchlen, (char*)(fc+stringOffset+off),slen)) + return 1; + } + } else { + // if nothing immediately following + if (matchlen == nlen) + return 1; + } + } + } + + // @TODO handle other encodings + } + } + return 0; +} + +static int stbtt__matches(stbtt_uint8 *fc, stbtt_uint32 offset, stbtt_uint8 *name, stbtt_int32 flags) +{ + stbtt_int32 nlen = (stbtt_int32) STBTT_strlen((char *) name); + stbtt_uint32 nm,hd; + if (!stbtt__isfont(fc+offset)) return 0; + + // check italics/bold/underline flags in macStyle... + if (flags) { + hd = stbtt__find_table(fc, offset, "head"); + if ((ttUSHORT(fc+hd+44) & 7) != (flags & 7)) return 0; + } + + nm = stbtt__find_table(fc, offset, "name"); + if (!nm) return 0; + + if (flags) { + // if we checked the macStyle flags, then just check the family and ignore the subfamily + if (stbtt__matchpair(fc, nm, name, nlen, 16, -1)) return 1; + if (stbtt__matchpair(fc, nm, name, nlen, 1, -1)) return 1; + if (stbtt__matchpair(fc, nm, name, nlen, 3, -1)) return 1; + } else { + if (stbtt__matchpair(fc, nm, name, nlen, 16, 17)) return 1; + if (stbtt__matchpair(fc, nm, name, nlen, 1, 2)) return 1; + if (stbtt__matchpair(fc, nm, name, nlen, 3, -1)) return 1; + } + + return 0; +} + +static int stbtt_FindMatchingFont_internal(unsigned char *font_collection, char *name_utf8, stbtt_int32 flags) +{ + stbtt_int32 i; + for (i=0;;++i) { + stbtt_int32 off = stbtt_GetFontOffsetForIndex(font_collection, i); + if (off < 0) return off; + if (stbtt__matches((stbtt_uint8 *) font_collection, off, (stbtt_uint8*) name_utf8, flags)) + return off; + } +} + +#if defined(__GNUC__) || defined(__clang__) +#pragma GCC diagnostic push +#pragma GCC diagnostic ignored "-Wcast-qual" +#endif + +STBTT_DEF int stbtt_BakeFontBitmap(const unsigned char *data, int offset, + float pixel_height, unsigned char *pixels, int pw, int ph, + int first_char, int num_chars, stbtt_bakedchar *chardata) +{ + return stbtt_BakeFontBitmap_internal((unsigned char *) data, offset, pixel_height, pixels, pw, ph, first_char, num_chars, chardata); +} + +STBTT_DEF int stbtt_GetFontOffsetForIndex(const unsigned char *data, int index) +{ + return stbtt_GetFontOffsetForIndex_internal((unsigned char *) data, index); +} + +STBTT_DEF int stbtt_GetNumberOfFonts(const unsigned char *data) +{ + return stbtt_GetNumberOfFonts_internal((unsigned char *) data); +} + +STBTT_DEF int stbtt_InitFont(stbtt_fontinfo *info, const unsigned char *data, int offset) +{ + return stbtt_InitFont_internal(info, (unsigned char *) data, offset); +} + +STBTT_DEF int stbtt_FindMatchingFont(const unsigned char *fontdata, const char *name, int flags) +{ + return stbtt_FindMatchingFont_internal((unsigned char *) fontdata, (char *) name, flags); +} + +STBTT_DEF int stbtt_CompareUTF8toUTF16_bigendian(const char *s1, int len1, const char *s2, int len2) +{ + return stbtt_CompareUTF8toUTF16_bigendian_internal((char *) s1, len1, (char *) s2, len2); +} + +#if defined(__GNUC__) || defined(__clang__) +#pragma GCC diagnostic pop +#endif + +#endif // STB_TRUETYPE_IMPLEMENTATION + + +// FULL VERSION HISTORY +// +// 1.25 (2021-07-11) many fixes +// 1.24 (2020-02-05) fix warning +// 1.23 (2020-02-02) query SVG data for glyphs; query whole kerning table (but only kern not GPOS) +// 1.22 (2019-08-11) minimize missing-glyph duplication; fix kerning if both 'GPOS' and 'kern' are defined +// 1.21 (2019-02-25) fix warning +// 1.20 (2019-02-07) PackFontRange skips missing codepoints; GetScaleFontVMetrics() +// 1.19 (2018-02-11) OpenType GPOS kerning (horizontal only), STBTT_fmod +// 1.18 (2018-01-29) add missing function +// 1.17 (2017-07-23) make more arguments const; doc fix +// 1.16 (2017-07-12) SDF support +// 1.15 (2017-03-03) make more arguments const +// 1.14 (2017-01-16) num-fonts-in-TTC function +// 1.13 (2017-01-02) support OpenType fonts, certain Apple fonts +// 1.12 (2016-10-25) suppress warnings about casting away const with -Wcast-qual +// 1.11 (2016-04-02) fix unused-variable warning +// 1.10 (2016-04-02) allow user-defined fabs() replacement +// fix memory leak if fontsize=0.0 +// fix warning from duplicate typedef +// 1.09 (2016-01-16) warning fix; avoid crash on outofmem; use alloc userdata for PackFontRanges +// 1.08 (2015-09-13) document stbtt_Rasterize(); fixes for vertical & horizontal edges +// 1.07 (2015-08-01) allow PackFontRanges to accept arrays of sparse codepoints; +// allow PackFontRanges to pack and render in separate phases; +// fix stbtt_GetFontOFfsetForIndex (never worked for non-0 input?); +// fixed an assert() bug in the new rasterizer +// replace assert() with STBTT_assert() in new rasterizer +// 1.06 (2015-07-14) performance improvements (~35% faster on x86 and x64 on test machine) +// also more precise AA rasterizer, except if shapes overlap +// remove need for STBTT_sort +// 1.05 (2015-04-15) fix misplaced definitions for STBTT_STATIC +// 1.04 (2015-04-15) typo in example +// 1.03 (2015-04-12) STBTT_STATIC, fix memory leak in new packing, various fixes +// 1.02 (2014-12-10) fix various warnings & compile issues w/ stb_rect_pack, C++ +// 1.01 (2014-12-08) fix subpixel position when oversampling to exactly match +// non-oversampled; STBTT_POINT_SIZE for packed case only +// 1.00 (2014-12-06) add new PackBegin etc. API, w/ support for oversampling +// 0.99 (2014-09-18) fix multiple bugs with subpixel rendering (ryg) +// 0.9 (2014-08-07) support certain mac/iOS fonts without an MS platformID +// 0.8b (2014-07-07) fix a warning +// 0.8 (2014-05-25) fix a few more warnings +// 0.7 (2013-09-25) bugfix: subpixel glyph bug fixed in 0.5 had come back +// 0.6c (2012-07-24) improve documentation +// 0.6b (2012-07-20) fix a few more warnings +// 0.6 (2012-07-17) fix warnings; added stbtt_ScaleForMappingEmToPixels, +// stbtt_GetFontBoundingBox, stbtt_IsGlyphEmpty +// 0.5 (2011-12-09) bugfixes: +// subpixel glyph renderer computed wrong bounding box +// first vertex of shape can be off-curve (FreeSans) +// 0.4b (2011-12-03) fixed an error in the font baking example +// 0.4 (2011-12-01) kerning, subpixel rendering (tor) +// bugfixes for: +// codepoint-to-glyph conversion using table fmt=12 +// codepoint-to-glyph conversion using table fmt=4 +// stbtt_GetBakedQuad with non-square texture (Zer) +// updated Hello World! sample to use kerning and subpixel +// fixed some warnings +// 0.3 (2009-06-24) cmap fmt=12, compound shapes (MM) +// userdata, malloc-from-userdata, non-zero fill (stb) +// 0.2 (2009-03-11) Fix unsigned/signed char warnings +// 0.1 (2009-03-09) First public release +// + +/* +------------------------------------------------------------------------------ +This software is available under 2 licenses -- choose whichever you prefer. +------------------------------------------------------------------------------ +ALTERNATIVE A - MIT License +Copyright (c) 2017 Sean Barrett +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. +------------------------------------------------------------------------------ +ALTERNATIVE B - Public Domain (www.unlicense.org) +This is free and unencumbered software released into the public domain. +Anyone is free to copy, modify, publish, use, compile, sell, or distribute this +software, either in source code form or as a compiled binary, for any purpose, +commercial or non-commercial, and by any means. +In jurisdictions that recognize copyright laws, the author or authors of this +software dedicate any and all copyright interest in the software to the public +domain. We make this dedication for the benefit of the public at large and to +the detriment of our heirs and successors. We intend this dedication to be an +overt act of relinquishment in perpetuity of all present and future rights to +this software under copyright law. +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 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. +------------------------------------------------------------------------------ +*/ diff --git a/libmui/mui/stb_ttc.h b/libmui/mui/stb_ttc.h new file mode 100644 index 0000000..6d828be --- /dev/null +++ b/libmui/mui/stb_ttc.h @@ -0,0 +1,660 @@ +/* + * stb_ttc.h + * + * Copyright (C) 2020 Michel Pollet + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#ifndef STB_TTC_H_ +#define STB_TTC_H_ + +#include "stb_truetype.h" + +#define STB_TTC_PACKED __attribute__((packed)) +/* Number of bins in the hash table(s) */ +#define STB_TTC_BINCOUNT 16 +/* Number of structures we prealocate when resizing arrays */ +#define STB_TTC_PAGESIZE 16 + +/* + * This is used in the hash table mapping the glyph/scale pair to the + * primary array of cached glyphs. + */ +typedef struct stb_ttc_index { + unsigned int intscale, glyph, index; +} STB_TTC_PACKED stb_ttc_index; + +/* + * This is used by MeasureText. Returns the number os glyphs in the string, + * the ascent and descent /of that text/ and the bounding box of that text. + * Note that x0 can be negative, if the first character is for example, a 'j' + */ +typedef struct stb_ttc_measure { + short glyph_count; + short ascent, descent; + short x0,y0,x1,y1; +} STB_TTC_PACKED stb_ttc_measure; + +/* + * A glyph info for a certain scale. You can have the same glyph at multiple + * scales in one cache. + */ +typedef struct stb_ttc_g { + unsigned int intscale; // for comparison purpose + float scale; + unsigned int glyph; + int advance, lsb; + short x0,y0,x1,y1; + // position in the pixel cache, or 0xffff if not yet cached. + unsigned short p_x, p_y; +} STB_TTC_PACKED stb_ttc_g; + +/* + * Used to cache the codepoints to glyphs map in a hash table + */ +typedef struct stb_ttc_cp_gl { + unsigned int cp; + unsigned int glyph; +} STB_TTC_PACKED stb_ttc_cp_gl; + +/* + * Used to cache the kerning between cp1 and cp2 in a hash + */ +typedef struct stb_ttc_cp_kern { + unsigned int hash; + unsigned int cp1, cp2; + int kern; +} STB_TTC_PACKED stb_ttc_cp_kern; + +// to remap codepoint->glyph without calling back into stb +struct _stb_ttc_cp_bin { + unsigned int cp_count; + stb_ttc_cp_gl * cp_gl; +}; +// to remap codepoint pairs to associated kerning +struct _stb_ttc_kn_bin { + unsigned int kn_count; + stb_ttc_cp_kern * cp_kn; +}; +// for glyph -> position in the glyph array +struct _stb_ttc_g_bin { + unsigned int i_count; + stb_ttc_index * index; +}; + +/* + * Main STB TrueType Cache structure. + */ +typedef struct stb_ttc_info { + stbtt_fontinfo font; + unsigned int font_size; + unsigned font_mmap: 1; // is font from a file + int ascent, descent; + // hash to remap codepoint->glyph without calling back into stb + struct _stb_ttc_cp_bin cp_bin[STB_TTC_BINCOUNT]; + // hash to remap codepoint pairs to associated kerning + struct _stb_ttc_kn_bin kn_bin[STB_TTC_BINCOUNT]; + // hash that for glyph -> position in the glyph array + struct _stb_ttc_g_bin g_bin[STB_TTC_BINCOUNT]; + // glyph array, contains glyph dimensions, and x,y of pixel in cache + unsigned int g_count; + stb_ttc_g * glyph; + // pixels cache for glyphs + unsigned int p_stride; + unsigned int p_height; // current total height + unsigned int p_line_height; // current glyph line height + unsigned int p_line_x, p_line_y; // current x,y position in pixels + unsigned char * pixels; +} stb_ttc_info; + + +#ifdef STBTTC_STATIC +#define STBTTC_DEF static +#else +#define STBTTC_DEF extern +#endif + +/* + * Preload a range of codepoints into the glyph cache. This doesn't add the + * pixels, it just load the glyph measurement, advance, lsb into the cache + */ +STBTTC_DEF int +stb_ttc_CacheCodepointRange( + struct stb_ttc_info * ttc, + unsigned int cp, + unsigned int count, + float scale ); +/* + * This walks the cache for glyphs that haven't been pre-rendered and + * render them in the cache. Also, it attempts to sort the glyphs by height + * into the cache to save a little bit of memory + * Note: This call is optional, the glyph renderer will populate the pixel + * cache lazily on the fly. + */ +STBTTC_DEF int +stb_ttc_RenderAllCachedGlyphs( + struct stb_ttc_info * ttc ); + +STBTTC_DEF int +stb_ttc_MeasureText( + struct stb_ttc_info * ttc, + float scale, + const char * text, + stb_ttc_measure *out); + +STBTTC_DEF int +stb_ttc_DrawText( + struct stb_ttc_info * ttc, + float scale, + const char * text, + unsigned int dx, + unsigned int base_dy, + unsigned char * pixels, + unsigned int p_w, + unsigned int p_h, + unsigned int p_stride); +// load a font from a file +STBTTC_DEF int +stb_ttc_MapFont( + struct stb_ttc_info * ttc, + const char * font_file); +// load a font from memory +STBTTC_DEF int +stb_ttc_LoadFont( + struct stb_ttc_info * ttc, + const void * font_data, + unsigned int font_size); +STBTTC_DEF void +stb_ttc_Free( + struct stb_ttc_info * ttc); + +#ifdef STB_TTC_IMPLEMENTATION +/* + * return font glyph index from codepoint, cache the result if it wasn't. + */ +static int +stb_ttc__CodepointGetGlyph( + struct stb_ttc_info *fi, + unsigned int cp ) +{ + struct _stb_ttc_cp_bin *b = &fi->cp_bin[cp & (STB_TTC_BINCOUNT - 1)]; + int di = 0; + for (unsigned int gi = 0; gi < b->cp_count; gi++, di++) + if (b->cp_gl[gi].cp == cp) + return b->cp_gl[gi].glyph; + else if (b->cp_gl[gi].cp > cp) + break; + int gl = stbtt_FindGlyphIndex(&fi->font, cp); + if (gl == 0) + return -1; + if (!(b->cp_count % STB_TTC_PAGESIZE)) + b->cp_gl = realloc(b->cp_gl, + sizeof(b->cp_gl[0]) * (b->cp_count + STB_TTC_PAGESIZE)); + memmove(b->cp_gl + di + 1, b->cp_gl + di, + sizeof(b->cp_gl[0]) * (b->cp_count - di)); + b->cp_count++; + b->cp_gl[di].cp = cp; + b->cp_gl[di].glyph = gl; + return gl; +} + +/* + * return kerning for 2 codepoints, cache the result. + */ +static int +stb_ttc__CodepointsGetKerning( + struct stb_ttc_info *fi, + unsigned int cp1, + unsigned int cp2 ) +{ + unsigned int hash = cp1 + ((cp1 * 100) * cp2); + struct _stb_ttc_kn_bin *b = &fi->kn_bin[hash & (STB_TTC_BINCOUNT - 1)]; + int di = 0; + for (unsigned int gi = 0; gi < b->kn_count; gi++, di++) + if (b->cp_kn[gi].hash == hash && + b->cp_kn[gi].cp1 == cp1 && + b->cp_kn[gi].cp2 == cp2) + return b->cp_kn[gi].kern; + else if (b->cp_kn[gi].hash > hash) + break; + int kern = stbtt_GetCodepointKernAdvance(&fi->font, cp1, cp2); + + if (!(b->kn_count % STB_TTC_PAGESIZE)) + b->cp_kn = realloc(b->cp_kn, + sizeof(b->cp_kn[0]) * (b->kn_count + STB_TTC_PAGESIZE)); + memmove(b->cp_kn + di + 1, b->cp_kn + di, + sizeof(b->cp_kn[0]) * (b->kn_count - di)); + stb_ttc_cp_kern k = { + .hash = hash, + .cp1 = cp1, + .cp2 = cp2, + .kern = kern, + }; + b->kn_count++; + b->cp_kn[di] = k; + return kern; +} + +/* + * returns the index of glyph in the glyph table, or -1 + */ +static int +stb_ttc__ScaledGlyphGetOffset( + struct stb_ttc_info *fi, + unsigned int glyph, + float scale ) +{ + unsigned int intscale = 1.0f / scale * 1000; + unsigned int hash = glyph + (glyph * intscale); + struct _stb_ttc_g_bin *b = &fi->g_bin[hash & (STB_TTC_BINCOUNT - 1)]; + for (unsigned int gi = 0; gi < b->i_count; gi++) + if (b->index[gi].intscale == intscale && + b->index[gi].glyph == glyph) + return b->index[gi].index; + else if (b->index[gi].glyph > glyph) + break; + return -1; +} + +static struct stb_ttc_g * +stb_ttc__ScaledGlyphGetCache( + struct stb_ttc_info *ttc, + unsigned int glyph, + float scale ) +{ + if (glyph == (unsigned int)-1) + return NULL; + int cached_index = stb_ttc__ScaledGlyphGetOffset(ttc, glyph, scale); + if (cached_index != -1) + return &ttc->glyph[cached_index]; + + stb_ttc_g gc = { }; + + gc.glyph = glyph; + gc.intscale = 1.0f / scale * 1000; + gc.scale = scale; + gc.p_x = gc.p_y = -1; // not initialised yet + // we use locals, as we are storing values in shorter types + { + int advance, lsb, x0, y0, x1, y1; + stbtt_GetGlyphHMetrics(&ttc->font, glyph, &advance, &lsb); + stbtt_GetGlyphBitmapBox(&ttc->font, glyph, scale, scale, &x0, &y0, &x1, + &y1); + gc.advance = advance; + gc.lsb = lsb; + gc.x0 = x0; + gc.y0 = y0; + gc.x1 = x1; + gc.y1 = y1; + } + if (!(ttc->g_count % STB_TTC_PAGESIZE)) + ttc->glyph = realloc(ttc->glyph, + sizeof(ttc->glyph[0]) * (ttc->g_count + STB_TTC_PAGESIZE)); + stb_ttc_index gh = { + .intscale = gc.intscale, + .glyph = glyph, + .index = ttc->g_count + }; + ttc->g_count++; + + unsigned int hash = glyph + (glyph * gc.intscale); + struct _stb_ttc_g_bin *b = &ttc->g_bin[hash & (STB_TTC_BINCOUNT - 1)]; + + if (!(b->i_count % STB_TTC_PAGESIZE)) + b->index = realloc(b->index, + sizeof(b->index[0]) * (b->i_count + STB_TTC_PAGESIZE)); + unsigned int di = b->i_count; + for (unsigned int i = 0; i < b->i_count; i++) + if (b->index[i].glyph > glyph) { + di = i; + break; + } + if (b->i_count - di) + memmove(&b->index[di + 1], &b->index[di], + sizeof(b->index[0]) * (b->i_count - di)); + b->index[di] = gh; + b->i_count++; + ttc->glyph[gh.index] = gc; + return &ttc->glyph[gh.index]; +} + +static void +stb_ttc__ScaledGlyphRenderToCache( + struct stb_ttc_info *fi, + struct stb_ttc_g *g ) +{ + int wt = g->x1 - g->x0; + wt = (wt + 3) & ~3; + unsigned int ht = g->y1 - g->y0; + // find the new horizontal position for glyph, "wrap" next line if needed + if (fi->p_line_x + wt > fi->p_stride) { + fi->p_line_x = 0; + fi->p_line_y += fi->p_line_height; + fi->p_line_height = 0; + } + if (ht > fi->p_line_height) + fi->p_line_height = ht; + // reallocate the pixel cache to accommodate more lines, if needed + if (fi->p_line_y + ht > fi->p_height) { + int add = fi->p_line_y + ht - fi->p_height; + fi->pixels = realloc(fi->pixels, (fi->p_height + add) * fi->p_stride); + memset(fi->pixels + (fi->p_height * fi->p_stride), 0xff, + add * fi->p_stride); + fi->p_height += add; + } + g->p_x = fi->p_line_x; + g->p_y = fi->p_line_y; + stbtt_MakeGlyphBitmap(&fi->font, + fi->pixels + (g->p_y * fi->p_stride) + g->p_x, g->x1 - g->x0, + g->y1 - g->y0, fi->p_stride, g->scale, g->scale, g->glyph); + fi->p_line_x += wt; +} + +static void +stb_ttc__GlyphRenderFromCache( + struct stb_ttc_info * fi, + struct stb_ttc_g * g, + unsigned int dx, + unsigned int base_dy, + unsigned char * pixels, + unsigned int p_w, + unsigned int p_h, + unsigned int p_stride) +{ + int _dy = base_dy + g->y0; + int _sy = 0; + + if (_dy >= (int)p_h || dx >= p_w || (int)base_dy + (g->y1 - g->y0) < 0) + return; + unsigned char * src = fi->pixels + (g->p_y * fi->p_stride); + + // skip lines that would be over the top of dst pixels + if (_dy < 0) { + _sy -= _dy; + src += (-_dy) * fi->p_stride; + _dy = 0; + } + unsigned char * dst = pixels + (_dy * p_stride); // beginning of line in pixels + while (_dy < (int)p_h && _sy < (g->y1 - g->y0)) { + int rw = g->x1 - g->x0; // remaining width of glyph line + int line_dx = (int)dx - g->x0; + unsigned char * src_p = src + g->p_x; + // skip what would be before the left border of pixels + if (line_dx < 0) { + src_p += -line_dx; + rw += line_dx; + line_dx = 0; + } + if (line_dx + rw >= (int)p_w) + rw = p_w - line_dx; + unsigned char * dst_p = dst + line_dx; + for (; rw > 0; rw--) { + unsigned short s = *dst_p + *src_p; + unsigned char d = -(s >> 8) | (unsigned char)s; + *dst_p++ = d; + src_p++; + } + dst += p_stride; + src += fi->p_stride; + _dy++; + _sy++; + } +} + +/* + * Preload a range of codepoints into the glyph cache. This doesn't add the + * pixels, it just load the glyph measurement, advance, lsb into the cache + */ +STBTTC_DEF int +stb_ttc_CacheCodepointRange( + struct stb_ttc_info * ttc, + unsigned int cp, + unsigned int count, + float scale ) +{ + int res = 0; + for (unsigned int c = cp; c < cp + count; c++) { + int gl = stb_ttc__CodepointGetGlyph(ttc, c); + if (gl == -1) + continue; + stb_ttc__ScaledGlyphGetCache(ttc, gl, scale); + res++; + } + return res; +} + +static stb_ttc_info * _ttc; +static int +_compare( + const void *a, + const void *b) +{ + stb_ttc_info *ttc = _ttc; + int g1 = *((int*)a), g2 = *((int*)b); + return (((ttc->glyph[g1].y1 - ttc->glyph[g1].y0) * 500) + ttc->glyph[g1].glyph) - + (((ttc->glyph[g2].y1 - ttc->glyph[g2].y0) * 500) + ttc->glyph[g2].glyph); +} + +/* + * This walks the cache for glyphs that haven't been pre-rendered and + * render them in the cache. Also, it attempts to sort the glyphs by height + * into the cache to save a little bit of memory + * Note: This call is optional, the glyph renderer will populate the pixel + * cache lazily on the fly. + */ +STBTTC_DEF int +stb_ttc_RenderAllCachedGlyphs( + struct stb_ttc_info * ttc ) +{ + // go on and sort the glyph indexes by height + int * to_sort = malloc(ttc->g_count * sizeof(int)); + for (int i = 0; i < (int)ttc->g_count; i++) + to_sort[i] = i; + _ttc = ttc; + qsort(to_sort, ttc->g_count, sizeof(int), _compare); + int count = 0; + // now load all the glyph pixels to the pixel cache + for (int i = 0; i < (int)ttc->g_count; i++) { + stb_ttc_g * g = &ttc->glyph[to_sort[i]]; + if (g->p_y == (unsigned short)-1) { + stb_ttc__ScaledGlyphRenderToCache(ttc, &ttc->glyph[to_sort[i]]); + count++; + } + } + free(to_sort); + return count; +} + +// Copyright (c) 2008-2010 Bjoern Hoehrmann +// See http://bjoern.hoehrmann.de/utf-8/decoder/dfa/ for details. +#define UTF8_ACCEPT 0 +#define UTF8_REJECT 12 + +static inline unsigned int +stb_ttc__UTF8_Decode( + unsigned int* state, + unsigned int* codep, + unsigned char byte) +{ + static const unsigned char 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, + }; + unsigned int type = utf8d[byte]; + *codep = (*state != UTF8_ACCEPT) ? + (byte & 0x3fu) | (*codep << 6) : + (0xff >> type) & (byte); + *state = utf8d[256 + *state + type]; + return *state; +} + +STBTTC_DEF int +stb_ttc_MeasureText( + struct stb_ttc_info * ttc, + float scale, + const char * text, + stb_ttc_measure *out) +{ + unsigned int state = 0; + int xpos = 0; + unsigned int last = 0; + stb_ttc_measure m = { .ascent = ttc->ascent * scale, + .descent = ttc->descent * scale}; + unsigned int cp = 0; + + for (int ch = 0; text[ch]; ch++) { + if (stb_ttc__UTF8_Decode(&state, &cp, text[ch]) != UTF8_ACCEPT) + continue; + if (last) { + int kern = scale * stb_ttc__CodepointsGetKerning(ttc, last, cp); + xpos += kern; + } + last = cp; + int gl = stb_ttc__CodepointGetGlyph(ttc, cp); + if (gl == -1) + continue; + stb_ttc_g *gc = stb_ttc__ScaledGlyphGetCache(ttc, gl, scale); + if (!gc) + continue; + if (m.glyph_count == 0) + m.x0 = gc->x0; + if (gc->y0 < m.y0) m.y0 = gc->y0; + if (gc->y1 > m.y1) m.y1 = gc->y1; + m.glyph_count++; + xpos += gc->advance; + } + m.x1 = xpos * scale; + if (out) *out = m; + return m.x1 - m.x0; +} + +STBTTC_DEF int +stb_ttc_DrawText( + struct stb_ttc_info * ttc, + float scale, + const char * text, + unsigned int dx, + unsigned int base_dy, + unsigned char * pixels, + unsigned int p_w, + unsigned int p_h, + unsigned int p_stride) +{ + unsigned int state = 0; + int xpos = dx / scale; + unsigned int last = 0; + int glyph_count = 0; + unsigned int cp = 0; + + for (int ch = 0; text[ch]; ch++) { + if (stb_ttc__UTF8_Decode(&state, &cp, text[ch]) != UTF8_ACCEPT) + continue; + if (last) { + int kern = scale * stb_ttc__CodepointsGetKerning(ttc, last, cp); + xpos += kern; + } + glyph_count++; + last = cp; + int gl = stb_ttc__CodepointGetGlyph(ttc, cp); + if (gl == -1) + continue; + stb_ttc_g *gc = stb_ttc__ScaledGlyphGetCache(ttc, gl, scale); + if (!gc) + continue; + // if (glyph_count == 1) + // xpos += -gc->x0 / scale; + if (gc->p_y == (unsigned short) -1) + stb_ttc__ScaledGlyphRenderToCache(ttc, gc); + + int pxpos = gc->x0 + ((xpos + gc->lsb) * scale); + stb_ttc__GlyphRenderFromCache(ttc, gc, pxpos, base_dy, + pixels, p_w, p_h, p_stride); + xpos += gc->advance; + } + return glyph_count; +} + +#include +#include +#include +#include +#include + +STBTTC_DEF int +stb_ttc_MapFont( + struct stb_ttc_info * ttc, + const char * font_file) +{ + int fd = open(font_file, O_RDONLY); + if (fd == -1) + return -1; + struct stat st; + if (fstat(fd, &st) == -1) { + close(fd); + return -1; + } + ttc->font_size = st.st_size; + unsigned char *map = mmap(NULL, ttc->font_size, PROT_READ, MAP_PRIVATE, fd, 0); + close(fd); + if (map == MAP_FAILED) + return -1; + ttc->font_mmap = 1; + ttc->p_stride = 100; // default value; + // this make the code work for both ttf and ttc files + int offset = stbtt_GetFontOffsetForIndex(map, 0); + stbtt_InitFont(&ttc->font, map, offset); + stbtt_GetFontVMetrics(&ttc->font, &ttc->ascent, &ttc->descent, 0); + return 0; +} + +STBTTC_DEF int +stb_ttc_LoadFont( + struct stb_ttc_info * ttc, + const void * font_data, + unsigned int font_size) +{ + ttc->font_size = font_size; + unsigned char *map = (unsigned char *)font_data; + ttc->font_mmap = 0; + ttc->p_stride = 100; // default value; + // this make the code work for both ttf and ttc files + int offset = stbtt_GetFontOffsetForIndex(map, 0); + stbtt_InitFont(&ttc->font, map, offset); + stbtt_GetFontVMetrics(&ttc->font, &ttc->ascent, &ttc->descent, 0); + return 0; +} + +STBTTC_DEF void +stb_ttc_Free( + struct stb_ttc_info * ttc) +{ + for (int i = 0; i < STB_TTC_BINCOUNT; i++) { + free(ttc->cp_bin[i].cp_gl); + free(ttc->kn_bin[i].cp_kn); + free(ttc->g_bin[i].index); + } + free(ttc->pixels); + free(ttc->glyph); + if (ttc->font_mmap) + munmap(ttc->font.data, ttc->font_size); +} +#endif /* STB_TTC_IMPLEMENTATION */ + +#endif /* STB_TTC_H_ */ diff --git a/libmui/mui/xft.c b/libmui/mui/xft.c new file mode 100644 index 0000000..714c85f --- /dev/null +++ b/libmui/mui/xft.c @@ -0,0 +1,2874 @@ +/* + * xft.c + * + * Copyright(c) 2007-2023 Jianjun Jiang <8192542@qq.com> + * Mobile phone: +86-18665388956 + * QQ: 8192542 + * + * 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. + * + */ + +#include "xft.h" + +/* + * type + */ +typedef int64_t XCG_FT_Int64; +typedef uint64_t XCG_FT_UInt64; +typedef int32_t XCG_FT_Int32; +typedef uint32_t XCG_FT_UInt32; + +#define XCG_FT_BOOL(x) ((XCG_FT_Bool)(x)) +#ifndef TRUE +#define TRUE 1 +#endif +#ifndef FALSE +#define FALSE 0 +#endif + +/* + * math + */ +#define XCG_FT_MIN(a, b) ((a) < (b) ? (a) : (b)) +#define XCG_FT_MAX(a, b) ((a) > (b) ? (a) : (b)) +#define XCG_FT_ABS(a) ((a) < 0 ? -(a) : (a)) +#define XCG_FT_HYPOT(x, y) (x = XCG_FT_ABS(x), y = XCG_FT_ABS(y), x > y ? x + (3 * y >> 3) : y + (3 * x >> 3)) +#define XCG_FT_ANGLE_PI (180L << 16) +#define XCG_FT_ANGLE_2PI (XCG_FT_ANGLE_PI * 2) +#define XCG_FT_ANGLE_PI2 (XCG_FT_ANGLE_PI / 2) +#define XCG_FT_ANGLE_PI4 (XCG_FT_ANGLE_PI / 4) + +#define XCG_FT_MSB(x) (31 - __builtin_clz(x)) +#define XCG_FT_PAD_FLOOR(x, n) ((x) & ~((n)-1)) +#define XCG_FT_PAD_ROUND(x, n) XCG_FT_PAD_FLOOR((x) + ((n) / 2), n) +#define XCG_FT_PAD_CEIL(x, n) XCG_FT_PAD_FLOOR((x) + ((n) - 1), n) + +#define XCG_FT_MOVE_SIGN(x, s) \ + do { \ + if(x < 0) { \ + x = -x; \ + s = -s; \ + } \ + } while(0) + +XCG_FT_Long XCG_FT_MulFix(XCG_FT_Long a, XCG_FT_Long b) +{ + XCG_FT_Int s = 1; + XCG_FT_Long c; + + XCG_FT_MOVE_SIGN(a, s); + XCG_FT_MOVE_SIGN(b, s); + c = (XCG_FT_Long)(((XCG_FT_Int64)a * b + 0x8000L) >> 16); + + return (s > 0) ? c : -c; +} + +XCG_FT_Long XCG_FT_MulDiv(XCG_FT_Long a, XCG_FT_Long b, XCG_FT_Long c) +{ + XCG_FT_Int s = 1; + XCG_FT_Long d; + + XCG_FT_MOVE_SIGN(a, s); + XCG_FT_MOVE_SIGN(b, s); + XCG_FT_MOVE_SIGN(c, s); + d = (XCG_FT_Long)(c > 0 ? ((XCG_FT_Int64)a * b + (c >> 1)) / c : 0x7FFFFFFFL); + return (s > 0) ? d : -d; +} + +XCG_FT_Long XCG_FT_DivFix(XCG_FT_Long a, XCG_FT_Long b) +{ + XCG_FT_Int s = 1; + XCG_FT_Long q; + + XCG_FT_MOVE_SIGN(a, s); + XCG_FT_MOVE_SIGN(b, s); + q = (XCG_FT_Long)(b > 0 ? (((XCG_FT_UInt64)a << 16) + (b >> 1)) / b : 0x7FFFFFFFL); + return (s < 0 ? -q : q); +} + +#define XCG_FT_TRIG_SCALE (0xDBD95B16UL) +#define XCG_FT_TRIG_SAFE_MSB (29) +#define XCG_FT_TRIG_MAX_ITERS (23) + +static const XCG_FT_Fixed ft_trig_arctan_table[] = { + 1740967L, 919879L, 466945L, 234379L, 117304L, 58666L, 29335L, 14668L, + 7334L, 3667L, 1833L, 917L, 458L, 229L, 115L, 57L, + 29L, 14L, 7L, 4L, 2L, 1L +}; + +static XCG_FT_Fixed ft_trig_downscale(XCG_FT_Fixed val) +{ + XCG_FT_Fixed s; + XCG_FT_Int64 v; + + s = val; + val = XCG_FT_ABS(val); + v = (val * (XCG_FT_Int64)XCG_FT_TRIG_SCALE) + 0x100000000UL; + val = (XCG_FT_Fixed)(v >> 32); + return (s >= 0) ? val : -val; +} + +static XCG_FT_Int ft_trig_prenorm(XCG_FT_Vector *vec) +{ + XCG_FT_Pos x, y; + XCG_FT_Int shift; + + x = vec->x; + y = vec->y; + + shift = XCG_FT_MSB(XCG_FT_ABS(x) | XCG_FT_ABS(y)); + + if(shift <= XCG_FT_TRIG_SAFE_MSB) + { + shift = XCG_FT_TRIG_SAFE_MSB - shift; + vec->x = (XCG_FT_Pos)((XCG_FT_ULong)x << shift); + vec->y = (XCG_FT_Pos)((XCG_FT_ULong)y << shift); + } + else + { + shift -= XCG_FT_TRIG_SAFE_MSB; + vec->x = x >> shift; + vec->y = y >> shift; + shift = -shift; + } + return shift; +} + +static void ft_trig_pseudo_rotate(XCG_FT_Vector *vec, XCG_FT_Angle theta) +{ + XCG_FT_Int i; + XCG_FT_Fixed x, y, xtemp, b; + const XCG_FT_Fixed *arctanptr; + + x = vec->x; + y = vec->y; + while(theta < -XCG_FT_ANGLE_PI4) + { + xtemp = y; + y = -x; + x = xtemp; + theta += XCG_FT_ANGLE_PI2; + } + while(theta > XCG_FT_ANGLE_PI4) + { + xtemp = -y; + y = x; + x = xtemp; + theta -= XCG_FT_ANGLE_PI2; + } + arctanptr = ft_trig_arctan_table; + for(i = 1, b = 1; i < XCG_FT_TRIG_MAX_ITERS; b <<= 1, i++) + { + XCG_FT_Fixed v1 = ((y + b) >> i); + XCG_FT_Fixed v2 = ((x + b) >> i); + if(theta < 0) + { + xtemp = x + v1; + y = y - v2; + x = xtemp; + theta += *arctanptr++; + } + else + { + xtemp = x - v1; + y = y + v2; + x = xtemp; + theta -= *arctanptr++; + } + } + vec->x = x; + vec->y = y; +} + +static void ft_trig_pseudo_polarize(XCG_FT_Vector *vec) +{ + XCG_FT_Angle theta; + XCG_FT_Int i; + XCG_FT_Fixed x, y, xtemp, b; + const XCG_FT_Fixed *arctanptr; + + x = vec->x; + y = vec->y; + if(y > x) + { + if(y > -x) + { + theta = XCG_FT_ANGLE_PI2; + xtemp = y; + y = -x; + x = xtemp; + } + else + { + theta = y > 0 ? XCG_FT_ANGLE_PI : -XCG_FT_ANGLE_PI; + x = -x; + y = -y; + } + } + else + { + if(y < -x) + { + theta = -XCG_FT_ANGLE_PI2; + xtemp = -y; + y = x; + x = xtemp; + } + else + { + theta = 0; + } + } + arctanptr = ft_trig_arctan_table; + for(i = 1, b = 1; i < XCG_FT_TRIG_MAX_ITERS; b <<= 1, i++) + { + XCG_FT_Fixed v1 = ((y + b) >> i); + XCG_FT_Fixed v2 = ((x + b) >> i); + if(y > 0) + { + xtemp = x + v1; + y = y - v2; + x = xtemp; + theta += *arctanptr++; + } + else + { + xtemp = x - v1; + y = y + v2; + x = xtemp; + theta -= *arctanptr++; + } + } + if(theta >= 0) + theta = XCG_FT_PAD_ROUND(theta, 32); + else + theta = -XCG_FT_PAD_ROUND(-theta, 32); + vec->x = x; + vec->y = theta; +} + +XCG_FT_Fixed XCG_FT_Cos(XCG_FT_Angle angle) +{ + XCG_FT_Vector v; + + v.x = XCG_FT_TRIG_SCALE >> 8; + v.y = 0; + ft_trig_pseudo_rotate(&v, angle); + return (v.x + 0x80L) >> 8; +} + +XCG_FT_Fixed XCG_FT_Sin(XCG_FT_Angle angle) +{ + return XCG_FT_Cos(XCG_FT_ANGLE_PI2 - angle); +} + +XCG_FT_Fixed XCG_FT_Tan(XCG_FT_Angle angle) +{ + XCG_FT_Vector v; + + v.x = XCG_FT_TRIG_SCALE >> 8; + v.y = 0; + ft_trig_pseudo_rotate(&v, angle); + return XCG_FT_DivFix(v.y, v.x); +} + +XCG_FT_Angle XCG_FT_Atan2(XCG_FT_Fixed dx, XCG_FT_Fixed dy) +{ + XCG_FT_Vector v; + + if(dx == 0 && dy == 0) + return 0; + v.x = dx; + v.y = dy; + ft_trig_prenorm(&v); + ft_trig_pseudo_polarize(&v); + return v.y; +} + +void XCG_FT_Vector_Unit(XCG_FT_Vector *vec, XCG_FT_Angle angle) +{ + vec->x = XCG_FT_TRIG_SCALE >> 8; + vec->y = 0; + ft_trig_pseudo_rotate(vec, angle); + vec->x = (vec->x + 0x80L) >> 8; + vec->y = (vec->y + 0x80L) >> 8; +} + +void XCG_FT_Vector_Rotate(XCG_FT_Vector *vec, XCG_FT_Angle angle) +{ + XCG_FT_Int shift; + XCG_FT_Vector v = *vec; + + if(v.x == 0 && v.y == 0) + return; + shift = ft_trig_prenorm(&v); + ft_trig_pseudo_rotate(&v, angle); + v.x = ft_trig_downscale(v.x); + v.y = ft_trig_downscale(v.y); + + if(shift > 0) + { + XCG_FT_Int32 half = (XCG_FT_Int32)1L << (shift - 1); + + vec->x = (v.x + half - (v.x < 0)) >> shift; + vec->y = (v.y + half - (v.y < 0)) >> shift; + } + else + { + shift = -shift; + vec->x = (XCG_FT_Pos)((XCG_FT_ULong)v.x << shift); + vec->y = (XCG_FT_Pos)((XCG_FT_ULong)v.y << shift); + } +} + +XCG_FT_Fixed XCG_FT_Vector_Length(XCG_FT_Vector *vec) +{ + XCG_FT_Int shift; + XCG_FT_Vector v; + + v = *vec; + if(v.x == 0) + { + return XCG_FT_ABS(v.y); + } + else if(v.y == 0) + { + return XCG_FT_ABS(v.x); + } + + shift = ft_trig_prenorm(&v); + ft_trig_pseudo_polarize(&v); + v.x = ft_trig_downscale(v.x); + if(shift > 0) + return (v.x + (1 << (shift - 1))) >> shift; + + return (XCG_FT_Fixed)((XCG_FT_UInt32)v.x << -shift); +} + +void XCG_FT_Vector_Polarize(XCG_FT_Vector *vec, XCG_FT_Fixed *length, XCG_FT_Angle *angle) +{ + XCG_FT_Int shift; + XCG_FT_Vector v; + + v = *vec; + if(v.x == 0 && v.y == 0) + return; + shift = ft_trig_prenorm(&v); + ft_trig_pseudo_polarize(&v); + v.x = ft_trig_downscale(v.x); + + *length = (shift >= 0) ? (v.x >> shift) : (XCG_FT_Fixed)((XCG_FT_UInt32)v.x << -shift); + *angle = v.y; +} + +void XCG_FT_Vector_From_Polar(XCG_FT_Vector *vec, XCG_FT_Fixed length, XCG_FT_Angle angle) +{ + vec->x = length; + vec->y = 0; + + XCG_FT_Vector_Rotate(vec, angle); +} + +XCG_FT_Angle XCG_FT_Angle_Diff(XCG_FT_Angle angle1, XCG_FT_Angle angle2) +{ + XCG_FT_Angle delta = angle2 - angle1; + + while(delta <= -XCG_FT_ANGLE_PI) + delta += XCG_FT_ANGLE_2PI; + + while(delta > XCG_FT_ANGLE_PI) + delta -= XCG_FT_ANGLE_2PI; + + return delta; +} + +/* + * raster + */ +typedef long TCoord; +typedef long TPos; +typedef long TArea; +typedef ptrdiff_t XCG_FT_PtrDist; + +#define xcg_ft_setjmp setjmp +#define xcg_ft_longjmp longjmp +#define xcg_ft_jmp_buf jmp_buf + +#define ErrRaster_Invalid_Mode -2 +#define ErrRaster_Invalid_Outline -1 +#define ErrRaster_Invalid_Argument -3 +#define ErrRaster_Memory_Overflow -4 +#define ErrRaster_OutOfMemory -6 + +#define XCG_FT_MINIMUM_POOL_SIZE 8192 +#define XCG_FT_MAX_GRAY_SPANS 256 + +#define RAS_ARG PWorker worker +#define RAS_ARG_ PWorker worker, +#define RAS_VAR worker +#define RAS_VAR_ worker, +#define ras (*worker) + +#define PIXEL_BITS 8 +#define ONE_PIXEL (1L << PIXEL_BITS) +#define TRUNC(x) (TCoord)((x) >> PIXEL_BITS) +#define FRACT(x) (TCoord)((x) & (ONE_PIXEL - 1)) +#if PIXEL_BITS >= 6 +#define UPSCALE(x) ((x) * (ONE_PIXEL >> 6)) +#define DOWNSCALE(x) ((x) >> (PIXEL_BITS - 6)) +#else +#define UPSCALE(x) ((x) >> (6 - PIXEL_BITS)) +#define DOWNSCALE(x) ((x) * (64 >> PIXEL_BITS)) +#endif + +#define XCG_FT_DIV_MOD(type, dividend, divisor, quotient, remainder) \ + do { \ + (quotient) = (type)((dividend) / (divisor)); \ + (remainder) = (type)((dividend) % (divisor)); \ + if((remainder) < 0) \ + { \ + (quotient)--; \ + (remainder) += (type)(divisor); \ + } \ + } while(0) + +typedef struct TCell_ * PCell; +typedef struct TCell_ { + int x; + int cover; + TArea area; + PCell next; +} TCell; + +typedef struct TWorker_ { + TCoord ex, ey; + TPos min_ex, max_ex; + TPos min_ey, max_ey; + TPos count_ex, count_ey; + TArea area; + int cover; + int invalid; + PCell cells; + XCG_FT_PtrDist max_cells; + XCG_FT_PtrDist num_cells; + TPos x, y; + XCG_FT_Outline outline; + XCG_FT_BBox clip_box; + XCG_FT_Span gray_spans[XCG_FT_MAX_GRAY_SPANS]; + int num_gray_spans; + int skip_spans; + XCG_FT_Raster_Span_Func render_span; + void *render_span_data; + int band_size; + int band_shoot; + xcg_ft_jmp_buf jump_buffer; + void *buffer; + long buffer_size; + PCell *ycells; + TPos ycount; +} TWorker, *PWorker; + +static void gray_init_cells(RAS_ARG_ void* buffer, long byte_size) +{ + ras.buffer = buffer; + ras.buffer_size = byte_size; + ras.ycells = (PCell*)buffer; + ras.cells = NULL; + ras.max_cells = 0; + ras.num_cells = 0; + ras.area = 0; + ras.cover = 0; + ras.invalid = 1; +} + +static void gray_compute_cbox( RAS_ARG) +{ + XCG_FT_Outline *outline = &ras.outline; + XCG_FT_Vector *vec = outline->points; + XCG_FT_Vector *limit = vec + outline->n_points; + + if(outline->n_points <= 0) + { + ras.min_ex = ras.max_ex = 0; + ras.min_ey = ras.max_ey = 0; + return; + } + ras.min_ex = ras.max_ex = vec->x; + ras.min_ey = ras.max_ey = vec->y; + vec++; + for(; vec < limit; vec++) + { + TPos x = vec->x; + TPos y = vec->y; + if(x < ras.min_ex) + ras.min_ex = x; + if(x > ras.max_ex) + ras.max_ex = x; + if(y < ras.min_ey) + ras.min_ey = y; + if(y > ras.max_ey) + ras.max_ey = y; + } + ras.min_ex = ras.min_ex >> 6; + ras.min_ey = ras.min_ey >> 6; + ras.max_ex = (ras.max_ex + 63) >> 6; + ras.max_ey = (ras.max_ey + 63) >> 6; +} + +static PCell gray_find_cell(RAS_ARG) +{ + PCell *pcell, cell; + TPos x = ras.ex; + + if(x > ras.count_ex) + x = ras.count_ex; + pcell = &ras.ycells[ras.ey]; + for(;;) + { + cell = *pcell; + if(cell == NULL || cell->x > x) + break; + if(cell->x == x) + goto Exit; + pcell = &cell->next; + } + if(ras.num_cells >= ras.max_cells) + xcg_ft_longjmp(ras.jump_buffer, 1); + cell = ras.cells + ras.num_cells++; + cell->x = x; + cell->area = 0; + cell->cover = 0; + cell->next = *pcell; + *pcell = cell; +Exit: + return cell; +} + +static void gray_record_cell( RAS_ARG) +{ + if( ras.area | ras.cover) + { + PCell cell = gray_find_cell( RAS_VAR); + cell->area += ras.area; + cell->cover += ras.cover; + } +} + +static void gray_set_cell( RAS_ARG_ TCoord ex, TCoord ey) +{ + ey -= ras.min_ey; + if(ex > ras.max_ex) + ex = ras.max_ex; + ex -= ras.min_ex; + if(ex < 0) + ex = -1; + if(ex != ras.ex || ey != ras.ey) + { + if(!ras.invalid) + gray_record_cell( RAS_VAR); + ras.area = 0; + ras.cover = 0; + ras.ex = ex; + ras.ey = ey; + } + ras.invalid = ((unsigned int)ey >= (unsigned int)ras.count_ey || ex >= ras.count_ex); +} + +static void gray_start_cell( RAS_ARG_ TCoord ex, TCoord ey) +{ + if(ex > ras.max_ex) + ex = (TCoord)( ras.max_ex); + if(ex < ras.min_ex) + ex = (TCoord)( ras.min_ex - 1); + ras.area = 0; + ras.cover = 0; + ras.ex = ex - ras.min_ex; + ras.ey = ey - ras.min_ey; + ras.invalid = 0; + gray_set_cell( RAS_VAR_ ex, ey); +} + +static void gray_render_scanline( RAS_ARG_ TCoord ey, TPos x1, TCoord y1, TPos x2, TCoord y2) +{ + TCoord ex1, ex2, fx1, fx2, first, dy, delta, mod; + TPos p, dx; + int incr; + + ex1 = TRUNC(x1); + ex2 = TRUNC(x2); + if(y1 == y2) + { + gray_set_cell( RAS_VAR_ ex2, ey); + return; + } + fx1 = FRACT(x1); + fx2 = FRACT(x2); + if(ex1 == ex2) + goto End; + dx = x2 - x1; + dy = y2 - y1; + if(dx > 0) + { + p = ( ONE_PIXEL - fx1) * dy; + first = ONE_PIXEL; + incr = 1; + } + else + { + p = fx1 * dy; + first = 0; + incr = -1; + dx = -dx; + } + XCG_FT_DIV_MOD(TCoord, p, dx, delta, mod); + ras.area += (TArea)(fx1 + first) * delta; + ras.cover += delta; + y1 += delta; + ex1 += incr; + gray_set_cell( RAS_VAR_ ex1, ey); + if(ex1 != ex2) + { + TCoord lift, rem; + p = ONE_PIXEL * dy; + XCG_FT_DIV_MOD(TCoord, p, dx, lift, rem); + do { + delta = lift; + mod += rem; + if(mod >= (TCoord)dx) + { + mod -= (TCoord)dx; + delta++; + } + ras.area += (TArea)( ONE_PIXEL * delta); + ras.cover += delta; + y1 += delta; + ex1 += incr; + gray_set_cell( RAS_VAR_ ex1, ey); + } while(ex1 != ex2); + } + fx1 = ONE_PIXEL - first; +End: + dy = y2 - y1; + ras.area += (TArea)((fx1 + fx2) * dy); + ras.cover += dy; +} + +static void gray_render_line( RAS_ARG_ TPos to_x, TPos to_y) +{ + TCoord ey1, ey2, fy1, fy2, first, delta, mod; + TPos p, dx, dy, x, x2; + int incr; + + ey1 = TRUNC(ras.y); + ey2 = TRUNC(to_y); + if((ey1 >= ras.max_ey && ey2 >= ras.max_ey) || (ey1 < ras.min_ey && ey2 < ras.min_ey)) + goto End; + fy1 = FRACT(ras.y); + fy2 = FRACT(to_y); + if(ey1 == ey2) + { + gray_render_scanline( RAS_VAR_ ey1, ras.x, fy1, to_x, fy2); + goto End; + } + dx = to_x - ras.x; + dy = to_y - ras.y; + if(dx == 0) + { + TCoord ex = TRUNC(ras.x); + TCoord two_fx = FRACT( ras.x ) << 1; + TPos area, max_ey1; + + if(dy > 0) + { + first = ONE_PIXEL; + } + else + { + first = 0; + } + delta = first - fy1; + ras.area += (TArea)two_fx * delta; + ras.cover += delta; + delta = first + first - ONE_PIXEL; + area = (TArea)two_fx * delta; + max_ey1 = ras.count_ey + ras.min_ey; + if(dy < 0) + { + if(ey1 > max_ey1) + { + ey1 = (max_ey1 > ey2) ? max_ey1 : ey2; + gray_set_cell(&ras, ex, ey1); + } + else + { + ey1--; + gray_set_cell(&ras, ex, ey1); + } + while(ey1 > ey2 && ey1 >= ras.min_ey) + { + ras.area += area; + ras.cover += delta; + ey1--; + + gray_set_cell(&ras, ex, ey1); + } + if(ey1 != ey2) + { + ey1 = ey2; + gray_set_cell(&ras, ex, ey1); + } + } + else + { + if(ey1 < ras.min_ey) + { + ey1 = (ras.min_ey < ey2) ? ras.min_ey : ey2; + gray_set_cell(&ras, ex, ey1); + } + else + { + ey1++; + gray_set_cell(&ras, ex, ey1); + } + while(ey1 < ey2 && ey1 < max_ey1) + { + ras.area += area; + ras.cover += delta; + ey1++; + + gray_set_cell(&ras, ex, ey1); + } + if(ey1 != ey2) + { + ey1 = ey2; + gray_set_cell(&ras, ex, ey1); + } + } + delta = (int)(fy2 - ONE_PIXEL + first); + ras.area += (TArea)two_fx * delta; + ras.cover += delta; + goto End; + } + if(dy > 0) + { + p = ( ONE_PIXEL - fy1) * dx; + first = ONE_PIXEL; + incr = 1; + } + else + { + p = fy1 * dx; + first = 0; + incr = -1; + dy = -dy; + } + XCG_FT_DIV_MOD(TCoord, p, dy, delta, mod); + x = ras.x + delta; + gray_render_scanline( RAS_VAR_ ey1, ras.x, fy1, x, (TCoord)first); + ey1 += incr; + gray_set_cell( RAS_VAR_ TRUNC( x ), ey1); + if(ey1 != ey2) + { + TCoord lift, rem; + p = ONE_PIXEL * dx; + XCG_FT_DIV_MOD(TCoord, p, dy, lift, rem); + do { + delta = lift; + mod += rem; + if(mod >= (TCoord)dy) + { + mod -= (TCoord)dy; + delta++; + } + x2 = x + delta; + gray_render_scanline( RAS_VAR_ ey1, x, ONE_PIXEL - first, x2, first); + x = x2; + ey1 += incr; + gray_set_cell( RAS_VAR_ TRUNC( x ), ey1); + } while(ey1 != ey2); + } + gray_render_scanline( RAS_VAR_ ey1, x, ONE_PIXEL - first, to_x, fy2); +End: + ras.x = to_x; + ras.y = to_y; +} + +static void gray_split_conic(XCG_FT_Vector * base) +{ + TPos a, b; + + base[4].x = base[2].x; + b = base[1].x; + a = base[3].x = (base[2].x + b) / 2; + b = base[1].x = (base[0].x + b) / 2; + base[2].x = (a + b) / 2; + + base[4].y = base[2].y; + b = base[1].y; + a = base[3].y = (base[2].y + b) / 2; + b = base[1].y = (base[0].y + b) / 2; + base[2].y = (a + b) / 2; +} + +static void gray_render_conic( RAS_ARG_ const XCG_FT_Vector * control, const XCG_FT_Vector * to) +{ + XCG_FT_Vector bez_stack[16 * 2 + 1]; + XCG_FT_Vector *arc = bez_stack; + TPos dx, dy; + int draw, split; + + arc[0].x = UPSCALE(to->x); + arc[0].y = UPSCALE(to->y); + arc[1].x = UPSCALE(control->x); + arc[1].y = UPSCALE(control->y); + arc[2].x = ras.x; + arc[2].y = ras.y; + + if(( TRUNC( arc[0].y ) >= ras.max_ey && + TRUNC( arc[1].y ) >= ras.max_ey && + TRUNC( arc[2].y ) >= ras.max_ey) || ( TRUNC( arc[0].y ) < ras.min_ey && + TRUNC( arc[1].y ) < ras.min_ey && + TRUNC( arc[2].y ) < ras.min_ey)) + { + ras.x = arc[0].x; + ras.y = arc[0].y; + return; + } + dx = XCG_FT_ABS(arc[2].x + arc[0].x - 2 * arc[1].x); + dy = XCG_FT_ABS(arc[2].y + arc[0].y - 2 * arc[1].y); + if(dx < dy) + dx = dy; + draw = 1; + while(dx > ONE_PIXEL / 4) + { + dx >>= 2; + draw <<= 1; + } + do { + split = 1; + while((draw & split) == 0) + { + gray_split_conic(arc); + arc += 2; + split <<= 1; + } + gray_render_line( RAS_VAR_ arc[0].x, arc[0].y); + arc -= 2; + } while(--draw); +} + +static void gray_split_cubic(XCG_FT_Vector * base) +{ + TPos a, b, c, d; + + base[6].x = base[3].x; + c = base[1].x; + d = base[2].x; + base[1].x = a = (base[0].x + c) / 2; + base[5].x = b = (base[3].x + d) / 2; + c = (c + d) / 2; + base[2].x = a = (a + c) / 2; + base[4].x = b = (b + c) / 2; + base[3].x = (a + b) / 2; + + base[6].y = base[3].y; + c = base[1].y; + d = base[2].y; + base[1].y = a = (base[0].y + c) / 2; + base[5].y = b = (base[3].y + d) / 2; + c = (c + d) / 2; + base[2].y = a = (a + c) / 2; + base[4].y = b = (b + c) / 2; + base[3].y = (a + b) / 2; +} + +static void gray_render_cubic(RAS_ARG_ const XCG_FT_Vector * control1, const XCG_FT_Vector * control2, const XCG_FT_Vector * to) +{ + XCG_FT_Vector bez_stack[16 * 3 + 1]; + XCG_FT_Vector * arc = bez_stack; + TPos dx, dy, dx_, dy_; + TPos dx1, dy1, dx2, dy2; + TPos L, s, s_limit; + + arc[0].x = UPSCALE(to->x); + arc[0].y = UPSCALE(to->y); + arc[1].x = UPSCALE(control2->x); + arc[1].y = UPSCALE(control2->y); + arc[2].x = UPSCALE(control1->x); + arc[2].y = UPSCALE(control1->y); + arc[3].x = ras.x; + arc[3].y = ras.y; + + if(( TRUNC( arc[0].y ) >= ras.max_ey && + TRUNC( arc[1].y ) >= ras.max_ey && + TRUNC( arc[2].y ) >= ras.max_ey && + TRUNC( arc[3].y ) >= ras.max_ey) || ( TRUNC( arc[0].y ) < ras.min_ey && + TRUNC( arc[1].y ) < ras.min_ey && + TRUNC( arc[2].y ) < ras.min_ey && + TRUNC( arc[3].y ) < ras.min_ey)) + { + ras.x = arc[0].x; + ras.y = arc[0].y; + return; + } + for(;;) + { + dx = dx_ = arc[3].x - arc[0].x; + dy = dy_ = arc[3].y - arc[0].y; + L = XCG_FT_HYPOT(dx_, dy_); + if(L >= (1 << 23)) + goto Split; + s_limit = L * (TPos)( ONE_PIXEL / 6); + dx1 = arc[1].x - arc[0].x; + dy1 = arc[1].y - arc[0].y; + s = XCG_FT_ABS(dy * dx1 - dx * dy1); + if(s > s_limit) + goto Split; + dx2 = arc[2].x - arc[0].x; + dy2 = arc[2].y - arc[0].y; + s = XCG_FT_ABS(dy * dx2 - dx * dy2); + if(s > s_limit) + goto Split; + if(dx1 * (dx1 - dx) + dy1 * (dy1 - dy) > 0 || dx2 * (dx2 - dx) + dy2 * (dy2 - dy) > 0) + goto Split; + gray_render_line( RAS_VAR_ arc[0].x, arc[0].y); + if(arc == bez_stack) + return; + arc -= 3; + continue; +Split: + gray_split_cubic(arc); + arc += 3; + } +} + +static int gray_move_to(const XCG_FT_Vector *to, PWorker worker) +{ + TPos x, y; + + if(!ras.invalid) + gray_record_cell(worker); + x = UPSCALE(to->x); + y = UPSCALE(to->y); + gray_start_cell(worker, TRUNC(x), TRUNC(y)); + ras.x = x; + ras.y = y; + return 0; +} + +static void gray_hline( RAS_ARG_ TCoord x, TCoord y, TPos area, int acount) +{ + int coverage; + + coverage = (int)(area >> ( PIXEL_BITS * 2 + 1 - 8)); + if(coverage < 0) + coverage = -coverage; + if( ras.outline.flags & XCG_FT_OUTLINE_EVEN_ODD_FILL) + { + coverage &= 511; + if(coverage > 256) + coverage = 512 - coverage; + else if(coverage == 256) + coverage = 255; + } + else + { + if(coverage >= 256) + coverage = 255; + } + y += (TCoord)ras.min_ey; + x += (TCoord)ras.min_ex; + if(x >= (1 << 23)) + x = (1 << 23) - 1; + if(y >= (1 << 23)) + y = (1 << 23) - 1; + if(coverage) + { + XCG_FT_Span *span; + int count; + int skip; + count = ras.num_gray_spans; + span = ras.gray_spans + count - 1; + if(count > 0 && span->y == y && span->x + span->len == x && span->coverage == coverage) + { + span->len = span->len + acount; + return; + } + if(count >= XCG_FT_MAX_GRAY_SPANS) + { + if( ras.render_span && count > ras.skip_spans) + { + skip = ras.skip_spans > 0 ? ras.skip_spans : 0; + ras.render_span( ras.num_gray_spans - skip, + ras.gray_spans + skip, + ras.render_span_data); + } + ras.skip_spans -= ras.num_gray_spans; + ras.num_gray_spans = 0; + span = ras.gray_spans; + } + else + span++; + span->x = x; + span->len = acount; + span->y = y; + span->coverage = (unsigned char)coverage; + ras.num_gray_spans++; + } +} + +static void gray_sweep(RAS_ARG) +{ + int yindex; + + if( ras.num_cells == 0) + return; + for(yindex = 0; yindex < ras.ycount; yindex++) + { + PCell cell = ras.ycells[yindex]; + TCoord cover = 0; + TCoord x = 0; + + for(; cell != NULL; cell = cell->next) + { + TArea area; + if(cell->x > x && cover != 0) + gray_hline( RAS_VAR_ x, yindex, cover * ( ONE_PIXEL * 2), cell->x - x); + cover += cell->cover; + area = cover * ( ONE_PIXEL * 2) - cell->area; + if(area != 0 && cell->x >= 0) + gray_hline( RAS_VAR_ cell->x, yindex, area, 1); + x = cell->x + 1; + } + if( ras.count_ex > x && cover != 0) + gray_hline( RAS_VAR_ x, yindex, cover * ( ONE_PIXEL * 2), ras.count_ex - x); + } +} + +static int XCG_FT_Outline_Decompose(const XCG_FT_Outline * outline, void * user) +{ +#undef SCALED +#define SCALED( x ) (x) + XCG_FT_Vector v_last; + XCG_FT_Vector v_control; + XCG_FT_Vector v_start; + XCG_FT_Vector * point; + XCG_FT_Vector * limit; + char * tags; + int n; + int first; + int error; + char tag; + + if(!outline) + return ErrRaster_Invalid_Outline; + first = 0; + for(n = 0; n < outline->n_contours; n++) + { + int last; + last = outline->contours[n]; + if(last < 0) + goto Invalid_Outline; + limit = outline->points + last; + v_start = outline->points[first]; + v_start.x = SCALED(v_start.x); + v_start.y = SCALED(v_start.y); + v_last = outline->points[last]; + v_last.x = SCALED(v_last.x); + v_last.y = SCALED(v_last.y); + v_control = v_start; + point = outline->points + first; + tags = outline->tags + first; + tag = XCG_FT_CURVE_TAG(tags[0]); + if(tag == XCG_FT_CURVE_TAG_CUBIC) + goto Invalid_Outline; + if(tag == XCG_FT_CURVE_TAG_CONIC) + { + if( XCG_FT_CURVE_TAG( outline->tags[last] ) == XCG_FT_CURVE_TAG_ON) + { + v_start = v_last; + limit--; + } + else + { + v_start.x = (v_start.x + v_last.x) / 2; + v_start.y = (v_start.y + v_last.y) / 2; + v_last = v_start; + } + point--; + tags--; + } + error = gray_move_to(&v_start, user); + if(error) + goto Exit; + while(point < limit) + { + point++; + tags++; + tag = XCG_FT_CURVE_TAG(tags[0]); + switch(tag) + { + case XCG_FT_CURVE_TAG_ON: + { + XCG_FT_Vector vec; + vec.x = SCALED(point->x); + vec.y = SCALED(point->y); + gray_render_line(user, UPSCALE(vec.x), UPSCALE(vec.y)); + continue; + } + case XCG_FT_CURVE_TAG_CONIC: + { + v_control.x = SCALED(point->x); + v_control.y = SCALED(point->y); +Do_Conic: + if(point < limit) + { + XCG_FT_Vector vec; + XCG_FT_Vector v_middle; + point++; + tags++; + tag = XCG_FT_CURVE_TAG(tags[0]); + vec.x = SCALED(point->x); + vec.y = SCALED(point->y); + if(tag == XCG_FT_CURVE_TAG_ON) + { + gray_render_conic(user, &v_control, &vec); + continue; + } + if(tag != XCG_FT_CURVE_TAG_CONIC) + goto Invalid_Outline; + v_middle.x = (v_control.x + vec.x) / 2; + v_middle.y = (v_control.y + vec.y) / 2; + gray_render_conic(user, &v_control, &v_middle); + v_control = vec; + goto Do_Conic; + } + gray_render_conic(user, &v_control, &v_start); + goto Close; + } + default: + { + XCG_FT_Vector vec1, vec2; + if(point + 1 > limit || + XCG_FT_CURVE_TAG( tags[1] ) != XCG_FT_CURVE_TAG_CUBIC) + goto Invalid_Outline; + point += 2; + tags += 2; + vec1.x = SCALED(point[-2].x); + vec1.y = SCALED(point[-2].y); + vec2.x = SCALED(point[-1].x); + vec2.y = SCALED(point[-1].y); + if(point <= limit) + { + XCG_FT_Vector vec; + vec.x = SCALED(point->x); + vec.y = SCALED(point->y); + gray_render_cubic(user, &vec1, &vec2, &vec); + continue; + } + gray_render_cubic(user, &vec1, &vec2, &v_start); + goto Close; + } + } + } + gray_render_line(user, UPSCALE(v_start.x), UPSCALE(v_start.y)); +Close: + first = last + 1; + } + return 0; +Exit: + return error; +Invalid_Outline: + return ErrRaster_Invalid_Outline; +} + +typedef struct TBand_ { + TPos min, max; +} TBand; + +static int gray_convert_glyph_inner(RAS_ARG) +{ + volatile int error = 0; + + if( xcg_ft_setjmp( ras.jump_buffer) == 0) + { + error = XCG_FT_Outline_Decompose(&ras.outline, &ras); + if(!ras.invalid) + gray_record_cell( RAS_VAR); + } + else + { + error = ErrRaster_Memory_Overflow; + } + return error; +} + +static int gray_convert_glyph(RAS_ARG) +{ + TBand bands[40]; + TBand *volatile band; + int volatile n, num_bands; + TPos volatile min, max, max_y; + XCG_FT_BBox *clip; + int skip; + + ras.num_gray_spans = 0; + gray_compute_cbox( RAS_VAR); + clip = &ras.clip_box; + if( ras.max_ex <= clip->xMin || ras.min_ex >= clip->xMax || ras.max_ey <= clip->yMin || ras.min_ey >= clip->yMax) + return 0; + if( ras.min_ex < clip->xMin) + ras.min_ex = clip->xMin; + if( ras.min_ey < clip->yMin) + ras.min_ey = clip->yMin; + if( ras.max_ex > clip->xMax) + ras.max_ex = clip->xMax; + if( ras.max_ey > clip->yMax) + ras.max_ey = clip->yMax; + ras.count_ex = ras.max_ex - ras.min_ex; + ras.count_ey = ras.max_ey - ras.min_ey; + num_bands = (int)(( ras.max_ey - ras.min_ey) / ras.band_size); + if(num_bands == 0) + num_bands = 1; + if(num_bands >= 39) + num_bands = 39; + ras.band_shoot = 0; + min = ras.min_ey; + max_y = ras.max_ey; + for(n = 0; n < num_bands; n++, min = max) + { + max = min + ras.band_size; + if(n == num_bands - 1 || max > max_y) + max = max_y; + bands[0].min = min; + bands[0].max = max; + band = bands; + while(band >= bands) + { + TPos bottom, top, middle; + int error; + { + PCell cells_max; + int yindex; + int cell_start, cell_end, cell_mod; + ras.ycells = (PCell*)ras.buffer; + ras.ycount = band->max - band->min; + cell_start = sizeof(PCell) * ras.ycount; + cell_mod = cell_start % sizeof(TCell); + if(cell_mod > 0) + cell_start += sizeof(TCell) - cell_mod; + cell_end = ras.buffer_size; + cell_end -= cell_end % sizeof(TCell); + cells_max = (PCell)((char*)ras.buffer + cell_end); + ras.cells = (PCell)((char*)ras.buffer + cell_start); + if( ras.cells >= cells_max) + goto ReduceBands; + ras.max_cells = (int)(cells_max - ras.cells); + if( ras.max_cells < 2) + goto ReduceBands; + for(yindex = 0; yindex < ras.ycount; yindex++) + ras.ycells[yindex] = NULL; + } + ras.num_cells = 0; + ras.invalid = 1; + ras.min_ey = band->min; + ras.max_ey = band->max; + ras.count_ey = band->max - band->min; + error = gray_convert_glyph_inner( RAS_VAR); + if(!error) + { + gray_sweep( RAS_VAR); + band--; + continue; + } + else if(error != ErrRaster_Memory_Overflow) + return 1; +ReduceBands: + bottom = band->min; + top = band->max; + middle = bottom + ((top - bottom) >> 1); + if(middle == bottom) + { + return ErrRaster_OutOfMemory; + } + if(bottom - top >= ras.band_size) + ras.band_shoot++; + band[1].min = bottom; + band[1].max = middle; + band[0].min = middle; + band[0].max = top; + band++; + } + } + if( ras.render_span && ras.num_gray_spans > ras.skip_spans) + { + skip = ras.skip_spans > 0 ? ras.skip_spans : 0; + ras.render_span( ras.num_gray_spans - skip, + ras.gray_spans + skip, + ras.render_span_data); + } + ras.skip_spans -= ras.num_gray_spans; + if( ras.band_shoot > 8 && ras.band_size > 16) + ras.band_size = ras.band_size / 2; + return 0; +} + +static int gray_raster_render(RAS_ARG_ void * buffer, long buffer_size, const XCG_FT_Raster_Params * params) +{ + const XCG_FT_Outline *outline = (const XCG_FT_Outline*)params->source; + if(outline == NULL) + return ErrRaster_Invalid_Outline; + if(outline->n_points == 0 || outline->n_contours <= 0) + return 0; + if(!outline->contours || !outline->points) + return ErrRaster_Invalid_Outline; + if(outline->n_points != outline->contours[outline->n_contours - 1] + 1) + return ErrRaster_Invalid_Outline; + if(!(params->flags & XCG_FT_RASTER_FLAG_AA)) + return ErrRaster_Invalid_Mode; + if(!(params->flags & XCG_FT_RASTER_FLAG_DIRECT)) + return ErrRaster_Invalid_Mode; + if(params->flags & XCG_FT_RASTER_FLAG_CLIP) + { + ras.clip_box = params->clip_box; + } + else + { + ras.clip_box.xMin = -(1 << 23); + ras.clip_box.yMin = -(1 << 23); + ras.clip_box.xMax = (1 << 23) - 1; + ras.clip_box.yMax = (1 << 23) - 1; + } + gray_init_cells( RAS_VAR_ buffer, buffer_size); + ras.outline = *outline; + ras.num_cells = 0; + ras.invalid = 1; + ras.band_size = (int)(buffer_size / (long)(sizeof(TCell) * 8)); + ras.render_span = (XCG_FT_Raster_Span_Func)params->gray_spans; + ras.render_span_data = params->user; + return gray_convert_glyph( RAS_VAR); +} + +void XCG_FT_Raster_Render(const XCG_FT_Raster_Params * params) +{ + char stack[XCG_FT_MINIMUM_POOL_SIZE]; + size_t length = XCG_FT_MINIMUM_POOL_SIZE; + + TWorker worker; + worker.skip_spans = 0; + int rendered_spans = 0; + int error = gray_raster_render(&worker, stack, length, params); + while(error == ErrRaster_OutOfMemory) + { + if(worker.skip_spans < 0) + rendered_spans += -worker.skip_spans; + worker.skip_spans = rendered_spans; + length *= 2; + void *heap = malloc(length); + error = gray_raster_render(&worker, heap, length, params); + free(heap); + } +} + +/* + * stroker + */ +#define XCG_FT_SMALL_CONIC_THRESHOLD (XCG_FT_ANGLE_PI / 6) +#define XCG_FT_SMALL_CUBIC_THRESHOLD (XCG_FT_ANGLE_PI / 8) +#define XCG_FT_IS_SMALL(x) ((x) > -2 && (x) < 2) + +static XCG_FT_Pos ft_pos_abs(XCG_FT_Pos x) +{ + return x >= 0 ? x : -x; +} + +static void ft_conic_split(XCG_FT_Vector *base) +{ + XCG_FT_Pos a, b; + + base[4].x = base[2].x; + a = base[0].x + base[1].x; + b = base[1].x + base[2].x; + base[3].x = b >> 1; + base[2].x = (a + b) >> 2; + base[1].x = a >> 1; + + base[4].y = base[2].y; + a = base[0].y + base[1].y; + b = base[1].y + base[2].y; + base[3].y = b >> 1; + base[2].y = (a + b) >> 2; + base[1].y = a >> 1; +} + +static XCG_FT_Bool ft_conic_is_small_enough(XCG_FT_Vector *base, XCG_FT_Angle *angle_in, XCG_FT_Angle *angle_out) +{ + XCG_FT_Vector d1, d2; + XCG_FT_Angle theta; + XCG_FT_Int close1, close2; + + d1.x = base[1].x - base[2].x; + d1.y = base[1].y - base[2].y; + d2.x = base[0].x - base[1].x; + d2.y = base[0].y - base[1].y; + close1 = XCG_FT_IS_SMALL(d1.x) && XCG_FT_IS_SMALL(d1.y); + close2 = XCG_FT_IS_SMALL(d2.x) && XCG_FT_IS_SMALL(d2.y); + if(close1) + { + if(close2) + { + } + else + { + *angle_in = *angle_out = XCG_FT_Atan2(d2.x, d2.y); + } + } + else + { + if(close2) + { + *angle_in = *angle_out = XCG_FT_Atan2(d1.x, d1.y); + } + else + { + *angle_in = XCG_FT_Atan2(d1.x, d1.y); + *angle_out = XCG_FT_Atan2(d2.x, d2.y); + } + } + theta = ft_pos_abs(XCG_FT_Angle_Diff(*angle_in, *angle_out)); + return XCG_FT_BOOL(theta < XCG_FT_SMALL_CONIC_THRESHOLD); +} + +static void ft_cubic_split(XCG_FT_Vector * base) +{ + XCG_FT_Pos a, b, c; + + base[6].x = base[3].x; + a = base[0].x + base[1].x; + b = base[1].x + base[2].x; + c = base[2].x + base[3].x; + base[5].x = c >> 1; + c += b; + base[4].x = c >> 2; + base[1].x = a >> 1; + a += b; + base[2].x = a >> 2; + base[3].x = (a + c) >> 3; + + base[6].y = base[3].y; + a = base[0].y + base[1].y; + b = base[1].y + base[2].y; + c = base[2].y + base[3].y; + base[5].y = c >> 1; + c += b; + base[4].y = c >> 2; + base[1].y = a >> 1; + a += b; + base[2].y = a >> 2; + base[3].y = (a + c) >> 3; +} + +static XCG_FT_Angle ft_angle_mean(XCG_FT_Angle angle1, XCG_FT_Angle angle2) +{ + return angle1 + XCG_FT_Angle_Diff(angle1, angle2) / 2; +} + +static XCG_FT_Bool ft_cubic_is_small_enough(XCG_FT_Vector *base, XCG_FT_Angle *angle_in, XCG_FT_Angle *angle_mid, XCG_FT_Angle *angle_out) +{ + XCG_FT_Vector d1, d2, d3; + XCG_FT_Angle theta1, theta2; + XCG_FT_Int close1, close2, close3; + + d1.x = base[2].x - base[3].x; + d1.y = base[2].y - base[3].y; + d2.x = base[1].x - base[2].x; + d2.y = base[1].y - base[2].y; + d3.x = base[0].x - base[1].x; + d3.y = base[0].y - base[1].y; + close1 = XCG_FT_IS_SMALL(d1.x) && XCG_FT_IS_SMALL(d1.y); + close2 = XCG_FT_IS_SMALL(d2.x) && XCG_FT_IS_SMALL(d2.y); + close3 = XCG_FT_IS_SMALL(d3.x) && XCG_FT_IS_SMALL(d3.y); + + if(close1) + { + if(close2) + { + if(close3) + { + } + else + { + *angle_in = *angle_mid = *angle_out = XCG_FT_Atan2(d3.x, d3.y); + } + } + else + { + if(close3) + { + *angle_in = *angle_mid = *angle_out = XCG_FT_Atan2(d2.x, d2.y); + } + else + { + *angle_in = *angle_mid = XCG_FT_Atan2(d2.x, d2.y); + *angle_out = XCG_FT_Atan2(d3.x, d3.y); + } + } + } + else + { + if(close2) + { + if(close3) + { + *angle_in = *angle_mid = *angle_out = XCG_FT_Atan2(d1.x, d1.y); + } + else + { + *angle_in = XCG_FT_Atan2(d1.x, d1.y); + *angle_out = XCG_FT_Atan2(d3.x, d3.y); + *angle_mid = ft_angle_mean(*angle_in, *angle_out); + } + } + else + { + if(close3) + { + *angle_in = XCG_FT_Atan2(d1.x, d1.y); + *angle_mid = *angle_out = XCG_FT_Atan2(d2.x, d2.y); + } + else + { + *angle_in = XCG_FT_Atan2(d1.x, d1.y); + *angle_mid = XCG_FT_Atan2(d2.x, d2.y); + *angle_out = XCG_FT_Atan2(d3.x, d3.y); + } + } + } + theta1 = ft_pos_abs(XCG_FT_Angle_Diff(*angle_in, *angle_mid)); + theta2 = ft_pos_abs(XCG_FT_Angle_Diff(*angle_mid, *angle_out)); + return XCG_FT_BOOL(theta1 < XCG_FT_SMALL_CUBIC_THRESHOLD && theta2 < XCG_FT_SMALL_CUBIC_THRESHOLD); +} + +typedef enum XCG_FT_StrokeTags_ { + XCG_FT_STROKE_TAG_ON = 1, + XCG_FT_STROKE_TAG_CUBIC = 2, + XCG_FT_STROKE_TAG_BEGIN = 4, + XCG_FT_STROKE_TAG_END = 8, +} XCG_FT_StrokeTags; + +#define XCG_FT_STROKE_TAG_BEGIN_END (XCG_FT_STROKE_TAG_BEGIN | XCG_FT_STROKE_TAG_END) + +typedef struct XCG_FT_StrokeBorderRec_ { + XCG_FT_UInt num_points; + XCG_FT_UInt max_points; + XCG_FT_Vector * points; + XCG_FT_Byte * tags; + XCG_FT_Bool movable; + XCG_FT_Int start; + XCG_FT_Bool valid; +} XCG_FT_StrokeBorderRec, *XCG_FT_StrokeBorder; + +XCG_FT_Error XCG_FT_Outline_Check(XCG_FT_Outline * outline) +{ + if(outline) + { + XCG_FT_Int n_points = outline->n_points; + XCG_FT_Int n_contours = outline->n_contours; + XCG_FT_Int end0, end; + XCG_FT_Int n; + + if(n_points == 0 && n_contours == 0) + return 0; + if(n_points <= 0 || n_contours <= 0) + goto Bad; + end0 = end = -1; + for(n = 0; n < n_contours; n++) + { + end = outline->contours[n]; + if(end <= end0 || end >= n_points) + goto Bad; + end0 = end; + } + if(end != n_points - 1) + goto Bad; + return 0; + } +Bad: + return -1; +} + +void XCG_FT_Outline_Get_CBox(const XCG_FT_Outline * outline, XCG_FT_BBox * acbox) +{ + XCG_FT_Pos xMin, yMin, xMax, yMax; + + if(outline && acbox) + { + if(outline->n_points == 0) + { + xMin = 0; + yMin = 0; + xMax = 0; + yMax = 0; + } + else + { + XCG_FT_Vector *vec = outline->points; + XCG_FT_Vector *limit = vec + outline->n_points; + xMin = xMax = vec->x; + yMin = yMax = vec->y; + vec++; + for(; vec < limit; vec++) + { + XCG_FT_Pos x, y; + x = vec->x; + if(x < xMin) + xMin = x; + if(x > xMax) + xMax = x; + y = vec->y; + if(y < yMin) + yMin = y; + if(y > yMax) + yMax = y; + } + } + acbox->xMin = xMin; + acbox->xMax = xMax; + acbox->yMin = yMin; + acbox->yMax = yMax; + } +} + +static XCG_FT_Error ft_stroke_border_grow(XCG_FT_StrokeBorder border, XCG_FT_UInt new_points) +{ + XCG_FT_UInt old_max = border->max_points; + XCG_FT_UInt new_max = border->num_points + new_points; + XCG_FT_Error error = 0; + + if(new_max > old_max) + { + XCG_FT_UInt cur_max = old_max; + while(cur_max < new_max) + cur_max += (cur_max >> 1) + 16; + border->points = (XCG_FT_Vector*)realloc(border->points, cur_max * sizeof(XCG_FT_Vector)); + border->tags = (XCG_FT_Byte*)realloc(border->tags, cur_max * sizeof(XCG_FT_Byte)); + if(!border->points || !border->tags) + goto Exit; + border->max_points = cur_max; + } +Exit: + return error; +} + +static void ft_stroke_border_close(XCG_FT_StrokeBorder border, XCG_FT_Bool reverse) +{ + XCG_FT_UInt start = border->start; + XCG_FT_UInt count = border->num_points; + + if(count <= start + 1U) + border->num_points = start; + else + { + border->num_points = --count; + border->points[start] = border->points[count]; + border->tags[start] = border->tags[count]; + if(reverse) + { + { + XCG_FT_Vector *vec1 = border->points + start + 1; + XCG_FT_Vector *vec2 = border->points + count - 1; + for(; vec1 < vec2; vec1++, vec2--) + { + XCG_FT_Vector tmp; + tmp = *vec1; + *vec1 = *vec2; + *vec2 = tmp; + } + } + { + XCG_FT_Byte *tag1 = border->tags + start + 1; + XCG_FT_Byte *tag2 = border->tags + count - 1; + + for(; tag1 < tag2; tag1++, tag2--) + { + XCG_FT_Byte tmp; + tmp = *tag1; + *tag1 = *tag2; + *tag2 = tmp; + } + } + } + border->tags[start] |= XCG_FT_STROKE_TAG_BEGIN; + border->tags[count - 1] |= XCG_FT_STROKE_TAG_END; + } + border->start = -1; + border->movable = FALSE; +} + +static XCG_FT_Error ft_stroke_border_lineto(XCG_FT_StrokeBorder border, XCG_FT_Vector * to, XCG_FT_Bool movable) +{ + XCG_FT_Error error = 0; + + if(border->movable) + { + border->points[border->num_points - 1] = *to; + } + else + { + if(border->num_points > 0&& + XCG_FT_IS_SMALL(border->points[border->num_points - 1].x - to->x) && + XCG_FT_IS_SMALL(border->points[border->num_points - 1].y - to->y)) + return error; + error = ft_stroke_border_grow(border, 1); + if(!error) + { + XCG_FT_Vector *vec = border->points + border->num_points; + XCG_FT_Byte *tag = border->tags + border->num_points; + vec[0] = *to; + tag[0] = XCG_FT_STROKE_TAG_ON; + border->num_points += 1; + } + } + border->movable = movable; + return error; +} + +static XCG_FT_Error ft_stroke_border_conicto(XCG_FT_StrokeBorder border, XCG_FT_Vector * control, XCG_FT_Vector * to) +{ + XCG_FT_Error error; + + error = ft_stroke_border_grow(border, 2); + if(!error) + { + XCG_FT_Vector *vec = border->points + border->num_points; + XCG_FT_Byte *tag = border->tags + border->num_points; + + vec[0] = *control; + vec[1] = *to; + tag[0] = 0; + tag[1] = XCG_FT_STROKE_TAG_ON; + border->num_points += 2; + } + border->movable = FALSE; + return error; +} + +static XCG_FT_Error ft_stroke_border_cubicto(XCG_FT_StrokeBorder border, XCG_FT_Vector * control1, XCG_FT_Vector * control2, XCG_FT_Vector * to) +{ + XCG_FT_Error error; + + error = ft_stroke_border_grow(border, 3); + if(!error) + { + XCG_FT_Vector *vec = border->points + border->num_points; + XCG_FT_Byte *tag = border->tags + border->num_points; + + vec[0] = *control1; + vec[1] = *control2; + vec[2] = *to; + tag[0] = XCG_FT_STROKE_TAG_CUBIC; + tag[1] = XCG_FT_STROKE_TAG_CUBIC; + tag[2] = XCG_FT_STROKE_TAG_ON; + border->num_points += 3; + } + border->movable = FALSE; + return error; +} + +#define XCG_FT_ARC_CUBIC_ANGLE (XCG_FT_ANGLE_PI / 2) + +static XCG_FT_Error ft_stroke_border_arcto(XCG_FT_StrokeBorder border, XCG_FT_Vector *center, XCG_FT_Fixed radius, XCG_FT_Angle angle_start, XCG_FT_Angle angle_diff) +{ + XCG_FT_Fixed coef; + XCG_FT_Vector a0, a1, a2, a3; + XCG_FT_Int i, arcs = 1; + XCG_FT_Error error = 0; + + while(angle_diff > XCG_FT_ARC_CUBIC_ANGLE * arcs || -angle_diff > XCG_FT_ARC_CUBIC_ANGLE * arcs) + arcs++; + coef = XCG_FT_Tan(angle_diff / (4 * arcs)); + coef += coef / 3; + XCG_FT_Vector_From_Polar(&a0, radius, angle_start); + a1.x = XCG_FT_MulFix(-a0.y, coef); + a1.y = XCG_FT_MulFix(a0.x, coef); + a0.x += center->x; + a0.y += center->y; + a1.x += a0.x; + a1.y += a0.y; + for(i = 1; i <= arcs; i++) + { + XCG_FT_Vector_From_Polar(&a3, radius, angle_start + i * angle_diff / arcs); + a2.x = XCG_FT_MulFix(a3.y, coef); + a2.y = XCG_FT_MulFix(-a3.x, coef); + a3.x += center->x; + a3.y += center->y; + a2.x += a3.x; + a2.y += a3.y; + error = ft_stroke_border_cubicto(border, &a1, &a2, &a3); + if(error) + break; + a1.x = a3.x - a2.x + a3.x; + a1.y = a3.y - a2.y + a3.y; + } + return error; +} + +static XCG_FT_Error ft_stroke_border_moveto(XCG_FT_StrokeBorder border, XCG_FT_Vector * to) +{ + if(border->start >= 0) + ft_stroke_border_close(border, FALSE); + border->start = border->num_points; + border->movable = FALSE; + return ft_stroke_border_lineto(border, to, FALSE); +} + +static void ft_stroke_border_init(XCG_FT_StrokeBorder border) +{ + border->points = NULL; + border->tags = NULL; + border->num_points = 0; + border->max_points = 0; + border->start = -1; + border->valid = FALSE; +} + +static void ft_stroke_border_reset(XCG_FT_StrokeBorder border) +{ + border->num_points = 0; + border->start = -1; + border->valid = FALSE; +} + +static void ft_stroke_border_done(XCG_FT_StrokeBorder border) +{ + free(border->points); + free(border->tags); + + border->num_points = 0; + border->max_points = 0; + border->start = -1; + border->valid = FALSE; +} + +static XCG_FT_Error ft_stroke_border_get_counts(XCG_FT_StrokeBorder border, XCG_FT_UInt * anum_points, XCG_FT_UInt * anum_contours) +{ + XCG_FT_Error error = 0; + XCG_FT_UInt num_points = 0; + XCG_FT_UInt num_contours = 0; + XCG_FT_UInt count = border->num_points; + XCG_FT_Vector *point = border->points; + XCG_FT_Byte *tags = border->tags; + XCG_FT_Int in_contour = 0; + + for(; count > 0; count--, num_points++, point++, tags++) + { + if(tags[0] & XCG_FT_STROKE_TAG_BEGIN) + { + if(in_contour != 0) + goto Fail; + in_contour = 1; + } + else if(in_contour == 0) + goto Fail; + if(tags[0] & XCG_FT_STROKE_TAG_END) + { + in_contour = 0; + num_contours++; + } + } + if(in_contour != 0) + goto Fail; + border->valid = TRUE; +Exit: + *anum_points = num_points; + *anum_contours = num_contours; + return error; +Fail: + num_points = 0; + num_contours = 0; + goto Exit; +} + +static void ft_stroke_border_export(XCG_FT_StrokeBorder border, XCG_FT_Outline * outline) +{ + if(outline->points != NULL && border->points != NULL) + memcpy(outline->points + outline->n_points, border->points, border->num_points * sizeof(XCG_FT_Vector)); + if(outline->tags) + { + XCG_FT_UInt count = border->num_points; + XCG_FT_Byte *read = border->tags; + XCG_FT_Byte *write = (XCG_FT_Byte*)outline->tags + outline->n_points; + for(; count > 0; count--, read++, write++) + { + if(*read & XCG_FT_STROKE_TAG_ON) + *write = XCG_FT_CURVE_TAG_ON; + else if(*read & XCG_FT_STROKE_TAG_CUBIC) + *write = XCG_FT_CURVE_TAG_CUBIC; + else + *write = XCG_FT_CURVE_TAG_CONIC; + } + } + if(outline->contours) + { + XCG_FT_UInt count = border->num_points; + XCG_FT_Byte *tags = border->tags; + XCG_FT_Int *write = outline->contours + outline->n_contours; + XCG_FT_Int idx = (XCG_FT_Int)outline->n_points; + for(; count > 0; count--, tags++, idx++) + { + if(*tags & XCG_FT_STROKE_TAG_END) + { + *write++ = idx; + outline->n_contours++; + } + } + } + outline->n_points = (int)(outline->n_points + border->num_points); + XCG_FT_Outline_Check(outline); +} + +#define XCG_FT_SIDE_TO_ROTATE(s) (XCG_FT_ANGLE_PI2 - (s) * XCG_FT_ANGLE_PI) + +typedef struct XCG_FT_StrokerRec_ { + XCG_FT_Angle angle_in; + XCG_FT_Angle angle_out; + XCG_FT_Vector center; + XCG_FT_Fixed line_length; + XCG_FT_Bool first_point; + XCG_FT_Bool subpath_open; + XCG_FT_Angle subpath_angle; + XCG_FT_Vector subpath_start; + XCG_FT_Fixed subpath_line_length; + XCG_FT_Bool handle_wide_strokes; + XCG_FT_Stroker_LineCap line_cap; + XCG_FT_Stroker_LineJoin line_join; + XCG_FT_Stroker_LineJoin line_join_saved; + XCG_FT_Fixed miter_limit; + XCG_FT_Fixed radius; + XCG_FT_StrokeBorderRec borders[2]; +} XCG_FT_StrokerRec; + +XCG_FT_Error XCG_FT_Stroker_New(XCG_FT_Stroker * astroker) +{ + XCG_FT_Error error = 0; + XCG_FT_Stroker stroker = NULL; + stroker = (XCG_FT_StrokerRec*)calloc(1, sizeof(XCG_FT_StrokerRec)); + if(stroker) + { + ft_stroke_border_init(&stroker->borders[0]); + ft_stroke_border_init(&stroker->borders[1]); + } + *astroker = stroker; + return error; +} + +void XCG_FT_Stroker_Rewind(XCG_FT_Stroker stroker) +{ + if(stroker) + { + ft_stroke_border_reset(&stroker->borders[0]); + ft_stroke_border_reset(&stroker->borders[1]); + } +} + +void XCG_FT_Stroker_Set(XCG_FT_Stroker stroker, XCG_FT_Fixed radius, XCG_FT_Stroker_LineCap line_cap, XCG_FT_Stroker_LineJoin line_join, XCG_FT_Fixed miter_limit) +{ + stroker->radius = radius; + stroker->line_cap = line_cap; + stroker->line_join = line_join; + stroker->miter_limit = miter_limit; + if(stroker->miter_limit < 0x10000) + stroker->miter_limit = 0x10000; + stroker->line_join_saved = line_join; + XCG_FT_Stroker_Rewind(stroker); +} + +void XCG_FT_Stroker_Done(XCG_FT_Stroker stroker) +{ + if(stroker) + { + ft_stroke_border_done(&stroker->borders[0]); + ft_stroke_border_done(&stroker->borders[1]); + free(stroker); + } +} + +static XCG_FT_Error ft_stroker_arcto(XCG_FT_Stroker stroker, XCG_FT_Int side) +{ + XCG_FT_Angle total, rotate; + XCG_FT_Fixed radius = stroker->radius; + XCG_FT_Error error = 0; + XCG_FT_StrokeBorder border = stroker->borders + side; + + rotate = XCG_FT_SIDE_TO_ROTATE(side); + total = XCG_FT_Angle_Diff(stroker->angle_in, stroker->angle_out); + if(total == XCG_FT_ANGLE_PI) + total = -rotate * 2; + error = ft_stroke_border_arcto(border, &stroker->center, radius, stroker->angle_in + rotate, total); + border->movable = FALSE; + return error; +} + +static XCG_FT_Error ft_stroker_cap(XCG_FT_Stroker stroker, XCG_FT_Angle angle, XCG_FT_Int side) +{ + XCG_FT_Error error = 0; + + if(stroker->line_cap == XCG_FT_STROKER_LINECAP_ROUND) + { + stroker->angle_in = angle; + stroker->angle_out = angle + XCG_FT_ANGLE_PI; + error = ft_stroker_arcto(stroker, side); + } + else + { + XCG_FT_Vector middle, delta; + XCG_FT_Fixed radius = stroker->radius; + XCG_FT_StrokeBorder border = stroker->borders + side; + XCG_FT_Vector_From_Polar(&middle, radius, angle); + delta.x = side ? middle.y : -middle.y; + delta.y = side ? -middle.x : middle.x; + if(stroker->line_cap == XCG_FT_STROKER_LINECAP_SQUARE) + { + middle.x += stroker->center.x; + middle.y += stroker->center.y; + } + else + { + middle.x = stroker->center.x; + middle.y = stroker->center.y; + } + delta.x += middle.x; + delta.y += middle.y; + error = ft_stroke_border_lineto(border, &delta, FALSE); + if(error) + goto Exit; + delta.x = middle.x - delta.x + middle.x; + delta.y = middle.y - delta.y + middle.y; + error = ft_stroke_border_lineto(border, &delta, FALSE); + } +Exit: + return error; +} + +static XCG_FT_Error ft_stroker_inside(XCG_FT_Stroker stroker, XCG_FT_Int side, XCG_FT_Fixed line_length) +{ + XCG_FT_StrokeBorder border = stroker->borders + side; + XCG_FT_Angle phi, theta, rotate; + XCG_FT_Fixed length; + XCG_FT_Vector sigma = { 0, 0 }; + XCG_FT_Vector delta; + XCG_FT_Error error = 0; + XCG_FT_Bool intersect; + + rotate = XCG_FT_SIDE_TO_ROTATE(side); + theta = XCG_FT_Angle_Diff(stroker->angle_in, stroker->angle_out) / 2; + if(!border->movable || line_length == 0 || theta > 0x59C000 || theta < -0x59C000) + intersect = FALSE; + else + { + XCG_FT_Fixed min_length; + XCG_FT_Vector_Unit(&sigma, theta); + min_length = ft_pos_abs(XCG_FT_MulDiv(stroker->radius, sigma.y, sigma.x)); + intersect = XCG_FT_BOOL(min_length && stroker->line_length >= min_length && line_length >= min_length); + } + if(!intersect) + { + XCG_FT_Vector_From_Polar(&delta, stroker->radius, stroker->angle_out + rotate); + delta.x += stroker->center.x; + delta.y += stroker->center.y; + border->movable = FALSE; + } + else + { + phi = stroker->angle_in + theta + rotate; + length = XCG_FT_DivFix(stroker->radius, sigma.x); + XCG_FT_Vector_From_Polar(&delta, length, phi); + delta.x += stroker->center.x; + delta.y += stroker->center.y; + } + error = ft_stroke_border_lineto(border, &delta, FALSE); + return error; +} + +static XCG_FT_Error ft_stroker_outside(XCG_FT_Stroker stroker, XCG_FT_Int side, XCG_FT_Fixed line_length) +{ + XCG_FT_StrokeBorder border = stroker->borders + side; + XCG_FT_Error error; + XCG_FT_Angle rotate; + + if(stroker->line_join == XCG_FT_STROKER_LINEJOIN_ROUND) + error = ft_stroker_arcto(stroker, side); + else + { + XCG_FT_Fixed radius = stroker->radius; + XCG_FT_Vector sigma = { 0, 0 }; + XCG_FT_Angle theta = 0, phi = 0; + XCG_FT_Bool bevel, fixed_bevel; + + rotate = XCG_FT_SIDE_TO_ROTATE(side); + bevel = XCG_FT_BOOL(stroker->line_join == XCG_FT_STROKER_LINEJOIN_BEVEL); + fixed_bevel = XCG_FT_BOOL(stroker->line_join != XCG_FT_STROKER_LINEJOIN_MITER_VARIABLE); + +if(!bevel) + { + theta = XCG_FT_Angle_Diff(stroker->angle_in, stroker->angle_out) / 2; + if(theta == XCG_FT_ANGLE_PI2) + theta = -rotate; + phi = stroker->angle_in + theta + rotate; + XCG_FT_Vector_From_Polar(&sigma, stroker->miter_limit, theta); + if(sigma.x < 0x10000L) + { + if(fixed_bevel || ft_pos_abs(theta) > 57) + bevel = TRUE; + } + } + if(bevel) + { + if(fixed_bevel) + { + XCG_FT_Vector delta; + XCG_FT_Vector_From_Polar(&delta, radius, stroker->angle_out + rotate); + delta.x += stroker->center.x; + delta.y += stroker->center.y; + border->movable = FALSE; + error = ft_stroke_border_lineto(border, &delta, FALSE); + } + else + { + XCG_FT_Vector middle, delta; + XCG_FT_Fixed coef; + + XCG_FT_Vector_From_Polar(&middle, XCG_FT_MulFix(radius, stroker->miter_limit), phi); + coef = XCG_FT_DivFix(0x10000L - sigma.x, sigma.y); + delta.x = XCG_FT_MulFix(middle.y, coef); + delta.y = XCG_FT_MulFix(-middle.x, coef); + middle.x += stroker->center.x; + middle.y += stroker->center.y; + delta.x += middle.x; + delta.y += middle.y; + error = ft_stroke_border_lineto(border, &delta, FALSE); + if(error) + goto Exit; + delta.x = middle.x - delta.x + middle.x; + delta.y = middle.y - delta.y + middle.y; + error = ft_stroke_border_lineto(border, &delta, FALSE); + if(error) + goto Exit; + if(line_length == 0) + { + XCG_FT_Vector_From_Polar(&delta, radius, stroker->angle_out + rotate); + delta.x += stroker->center.x; + delta.y += stroker->center.y; + error = ft_stroke_border_lineto(border, &delta, FALSE); + } + } + } + else + { + XCG_FT_Fixed length; + XCG_FT_Vector delta; + length = XCG_FT_MulDiv(stroker->radius, stroker->miter_limit, sigma.x); + XCG_FT_Vector_From_Polar(&delta, length, phi); + delta.x += stroker->center.x; + delta.y += stroker->center.y; + error = ft_stroke_border_lineto(border, &delta, FALSE); + if(error) + goto Exit; + if(line_length == 0) + { + XCG_FT_Vector_From_Polar(&delta, stroker->radius, stroker->angle_out + rotate); + delta.x += stroker->center.x; + delta.y += stroker->center.y; + error = ft_stroke_border_lineto(border, &delta, FALSE); + } + } + } +Exit: + return error; +} + +static XCG_FT_Error ft_stroker_process_corner(XCG_FT_Stroker stroker, XCG_FT_Fixed line_length) +{ + XCG_FT_Error error = 0; + XCG_FT_Angle turn; + XCG_FT_Int inside_side; + + turn = XCG_FT_Angle_Diff(stroker->angle_in, stroker->angle_out); + if(turn == 0) + goto Exit; + inside_side = 0; + if(turn < 0) + inside_side = 1; + error = ft_stroker_inside(stroker, inside_side, line_length); + if(error) + goto Exit; + error = ft_stroker_outside(stroker, 1 - inside_side, line_length); +Exit: + return error; +} + +static XCG_FT_Error ft_stroker_subpath_start(XCG_FT_Stroker stroker, XCG_FT_Angle start_angle, XCG_FT_Fixed line_length) +{ + XCG_FT_Vector delta; + XCG_FT_Vector point; + XCG_FT_Error error; + XCG_FT_StrokeBorder border; + + XCG_FT_Vector_From_Polar(&delta, stroker->radius, start_angle + XCG_FT_ANGLE_PI2); + point.x = stroker->center.x + delta.x; + point.y = stroker->center.y + delta.y; + border = stroker->borders; + error = ft_stroke_border_moveto(border, &point); + if(error) + goto Exit; + point.x = stroker->center.x - delta.x; + point.y = stroker->center.y - delta.y; + border++; + error = ft_stroke_border_moveto(border, &point); + stroker->subpath_angle = start_angle; + stroker->first_point = FALSE; + stroker->subpath_line_length = line_length; +Exit: + return error; +} + +XCG_FT_Error XCG_FT_Stroker_LineTo(XCG_FT_Stroker stroker, XCG_FT_Vector * to) +{ + XCG_FT_Error error = 0; + XCG_FT_StrokeBorder border; + XCG_FT_Vector delta; + XCG_FT_Angle angle; + XCG_FT_Int side; + XCG_FT_Fixed line_length; + + delta.x = to->x - stroker->center.x; + delta.y = to->y - stroker->center.y; + if(delta.x == 0 && delta.y == 0) + goto Exit; + line_length = XCG_FT_Vector_Length(&delta); + angle = XCG_FT_Atan2(delta.x, delta.y); + XCG_FT_Vector_From_Polar(&delta, stroker->radius, angle + XCG_FT_ANGLE_PI2); + if(stroker->first_point) + { + error = ft_stroker_subpath_start(stroker, angle, line_length); + if(error) + goto Exit; + } + else + { + stroker->angle_out = angle; + error = ft_stroker_process_corner(stroker, line_length); + if(error) + goto Exit; + } + for(border = stroker->borders, side = 1; side >= 0; side--, border++) + { + XCG_FT_Vector point; + point.x = to->x + delta.x; + point.y = to->y + delta.y; + error = ft_stroke_border_lineto(border, &point, TRUE); + if(error) + goto Exit; + delta.x = -delta.x; + delta.y = -delta.y; + } + stroker->angle_in = angle; + stroker->center = *to; + stroker->line_length = line_length; +Exit: + return error; +} + +XCG_FT_Error XCG_FT_Stroker_ConicTo(XCG_FT_Stroker stroker, XCG_FT_Vector * control, XCG_FT_Vector * to) +{ + XCG_FT_Error error = 0; + XCG_FT_Vector bez_stack[34]; + XCG_FT_Vector *arc; + XCG_FT_Vector *limit = bez_stack + 30; + XCG_FT_Bool first_arc = TRUE; + + if(XCG_FT_IS_SMALL(stroker->center.x - control->x) && + XCG_FT_IS_SMALL(stroker->center.y - control->y) && + XCG_FT_IS_SMALL(control->x - to->x) && + XCG_FT_IS_SMALL(control->y - to->y)) + { + stroker->center = *to; + goto Exit; + } + arc = bez_stack; + arc[0] = *to; + arc[1] = *control; + arc[2] = stroker->center; + while(arc >= bez_stack) + { + XCG_FT_Angle angle_in, angle_out; + angle_in = angle_out = stroker->angle_in; + if(arc < limit && !ft_conic_is_small_enough(arc, &angle_in, &angle_out)) + { + if(stroker->first_point) + stroker->angle_in = angle_in; + ft_conic_split(arc); + arc += 2; + continue; + } + if(first_arc) + { + first_arc = FALSE; + if(stroker->first_point) + error = ft_stroker_subpath_start(stroker, angle_in, 0); + else + { + stroker->angle_out = angle_in; + error = ft_stroker_process_corner(stroker, 0); + } + } + else if(ft_pos_abs(XCG_FT_Angle_Diff(stroker->angle_in, angle_in)) > XCG_FT_SMALL_CONIC_THRESHOLD / 4) + { + stroker->center = arc[2]; + stroker->angle_out = angle_in; + stroker->line_join = XCG_FT_STROKER_LINEJOIN_ROUND; + error = ft_stroker_process_corner(stroker, 0); + stroker->line_join = stroker->line_join_saved; + } + if(error) + goto Exit; + { + XCG_FT_Vector ctrl, end; + XCG_FT_Angle theta, phi, rotate, alpha0 = 0; + XCG_FT_Fixed length; + XCG_FT_StrokeBorder border; + XCG_FT_Int side; + + theta = XCG_FT_Angle_Diff(angle_in, angle_out) / 2; + phi = angle_in + theta; + length = XCG_FT_DivFix(stroker->radius, XCG_FT_Cos(theta)); + if(stroker->handle_wide_strokes) + alpha0 = XCG_FT_Atan2(arc[0].x - arc[2].x, arc[0].y - arc[2].y); + for(border = stroker->borders, side = 0; side <= 1; side++, border++) + { + rotate = XCG_FT_SIDE_TO_ROTATE(side); + XCG_FT_Vector_From_Polar(&ctrl, length, phi + rotate); + ctrl.x += arc[1].x; + ctrl.y += arc[1].y; + XCG_FT_Vector_From_Polar(&end, stroker->radius, angle_out + rotate); + end.x += arc[0].x; + end.y += arc[0].y; + + if(stroker->handle_wide_strokes) + { + XCG_FT_Vector start; + XCG_FT_Angle alpha1; + + start = border->points[border->num_points - 1]; + alpha1 = XCG_FT_Atan2(end.x - start.x, end.y - start.y); + if(ft_pos_abs(XCG_FT_Angle_Diff(alpha0, alpha1)) > + XCG_FT_ANGLE_PI / 2) + { + XCG_FT_Angle beta, gamma; + XCG_FT_Vector bvec, delta; + XCG_FT_Fixed blen, sinA, sinB, alen; + + beta = XCG_FT_Atan2(arc[2].x - start.x, arc[2].y - start.y); + gamma = XCG_FT_Atan2(arc[0].x - end.x, arc[0].y - end.y); + + bvec.x = end.x - start.x; + bvec.y = end.y - start.y; + blen = XCG_FT_Vector_Length(&bvec); + sinA = ft_pos_abs(XCG_FT_Sin(alpha1 - gamma)); + sinB = ft_pos_abs(XCG_FT_Sin(beta - gamma)); + alen = XCG_FT_MulDiv(blen, sinA, sinB); + XCG_FT_Vector_From_Polar(&delta, alen, beta); + delta.x += start.x; + delta.y += start.y; + border->movable = FALSE; + error = ft_stroke_border_lineto(border, &delta, FALSE); + if(error) + goto Exit; + error = ft_stroke_border_lineto(border, &end, FALSE); + if(error) + goto Exit; + error = ft_stroke_border_conicto(border, &ctrl, &start); + if(error) + goto Exit; + error = ft_stroke_border_lineto(border, &end, FALSE); + if(error) + goto Exit; + continue; + } + } + error = ft_stroke_border_conicto(border, &ctrl, &end); + if(error) + goto Exit; + } + } + arc -= 2; + stroker->angle_in = angle_out; + } + stroker->center = *to; + stroker->line_length = 0; +Exit: + return error; +} + +XCG_FT_Error XCG_FT_Stroker_CubicTo(XCG_FT_Stroker stroker, XCG_FT_Vector *control1, XCG_FT_Vector *control2, XCG_FT_Vector *to) +{ + XCG_FT_Error error = 0; + XCG_FT_Vector bez_stack[37]; + XCG_FT_Vector *arc; + XCG_FT_Vector *limit = bez_stack + 32; + XCG_FT_Bool first_arc = TRUE; + + if(XCG_FT_IS_SMALL(stroker->center.x - control1->x) && + XCG_FT_IS_SMALL(stroker->center.y - control1->y) && + XCG_FT_IS_SMALL(control1->x - control2->x) && + XCG_FT_IS_SMALL(control1->y - control2->y) && + XCG_FT_IS_SMALL(control2->x - to->x) && + XCG_FT_IS_SMALL(control2->y - to->y)) + { + stroker->center = *to; + goto Exit; + } + arc = bez_stack; + arc[0] = *to; + arc[1] = *control2; + arc[2] = *control1; + arc[3] = stroker->center; + while(arc >= bez_stack) + { + XCG_FT_Angle angle_in, angle_mid, angle_out; + angle_in = angle_out = angle_mid = stroker->angle_in; + if(arc < limit && !ft_cubic_is_small_enough(arc, &angle_in, &angle_mid, &angle_out)) + { + if(stroker->first_point) + stroker->angle_in = angle_in; + ft_cubic_split(arc); + arc += 3; + continue; + } + if(first_arc) + { + first_arc = FALSE; + if(stroker->first_point) + error = ft_stroker_subpath_start(stroker, angle_in, 0); + else + { + stroker->angle_out = angle_in; + error = ft_stroker_process_corner(stroker, 0); + } + } + else if(ft_pos_abs(XCG_FT_Angle_Diff(stroker->angle_in, angle_in)) > XCG_FT_SMALL_CUBIC_THRESHOLD / 4) + { + stroker->center = arc[3]; + stroker->angle_out = angle_in; + stroker->line_join = XCG_FT_STROKER_LINEJOIN_ROUND; + error = ft_stroker_process_corner(stroker, 0); + stroker->line_join = stroker->line_join_saved; + } + if(error) + goto Exit; + { + XCG_FT_Vector ctrl1, ctrl2, end; + XCG_FT_Angle theta1, phi1, theta2, phi2, rotate, alpha0 = 0; + XCG_FT_Fixed length1, length2; + XCG_FT_StrokeBorder border; + XCG_FT_Int side; + + theta1 = XCG_FT_Angle_Diff(angle_in, angle_mid) / 2; + theta2 = XCG_FT_Angle_Diff(angle_mid, angle_out) / 2; + phi1 = ft_angle_mean(angle_in, angle_mid); + phi2 = ft_angle_mean(angle_mid, angle_out); + length1 = XCG_FT_DivFix(stroker->radius, XCG_FT_Cos(theta1)); + length2 = XCG_FT_DivFix(stroker->radius, XCG_FT_Cos(theta2)); + if(stroker->handle_wide_strokes) + alpha0 = XCG_FT_Atan2(arc[0].x - arc[3].x, arc[0].y - arc[3].y); + for(border = stroker->borders, side = 0; side <= 1; side++, border++) + { + rotate = XCG_FT_SIDE_TO_ROTATE(side); + XCG_FT_Vector_From_Polar(&ctrl1, length1, phi1 + rotate); + ctrl1.x += arc[2].x; + ctrl1.y += arc[2].y; + XCG_FT_Vector_From_Polar(&ctrl2, length2, phi2 + rotate); + ctrl2.x += arc[1].x; + ctrl2.y += arc[1].y; + XCG_FT_Vector_From_Polar(&end, stroker->radius, angle_out + rotate); + end.x += arc[0].x; + end.y += arc[0].y; + if(stroker->handle_wide_strokes) + { + XCG_FT_Vector start; + XCG_FT_Angle alpha1; + start = border->points[border->num_points - 1]; + alpha1 = XCG_FT_Atan2(end.x - start.x, end.y - start.y); + if(ft_pos_abs(XCG_FT_Angle_Diff(alpha0, alpha1)) > + XCG_FT_ANGLE_PI / 2) + { + XCG_FT_Angle beta, gamma; + XCG_FT_Vector bvec, delta; + XCG_FT_Fixed blen, sinA, sinB, alen; + beta = XCG_FT_Atan2(arc[3].x - start.x, arc[3].y - start.y); + gamma = XCG_FT_Atan2(arc[0].x - end.x, arc[0].y - end.y); + bvec.x = end.x - start.x; + bvec.y = end.y - start.y; + blen = XCG_FT_Vector_Length(&bvec); + sinA = ft_pos_abs(XCG_FT_Sin(alpha1 - gamma)); + sinB = ft_pos_abs(XCG_FT_Sin(beta - gamma)); + alen = XCG_FT_MulDiv(blen, sinA, sinB); + XCG_FT_Vector_From_Polar(&delta, alen, beta); + delta.x += start.x; + delta.y += start.y; + border->movable = FALSE; + error = ft_stroke_border_lineto(border, &delta, FALSE); + if(error) + goto Exit; + error = ft_stroke_border_lineto(border, &end, FALSE); + if(error) + goto Exit; + error = ft_stroke_border_cubicto(border, &ctrl2, &ctrl1, &start); + if(error) + goto Exit; + error = ft_stroke_border_lineto(border, &end, FALSE); + if(error) + goto Exit; + continue; + } + } + error = ft_stroke_border_cubicto(border, &ctrl1, &ctrl2, &end); + if(error) + goto Exit; + } + } + arc -= 3; + stroker->angle_in = angle_out; + } + stroker->center = *to; + stroker->line_length = 0; +Exit: + return error; +} + +XCG_FT_Error XCG_FT_Stroker_BeginSubPath(XCG_FT_Stroker stroker, XCG_FT_Vector * to, XCG_FT_Bool open) +{ + stroker->first_point = TRUE; + stroker->center = *to; + stroker->subpath_open = open; + stroker->handle_wide_strokes = XCG_FT_BOOL(stroker->line_join != XCG_FT_STROKER_LINEJOIN_ROUND || (stroker->subpath_open && stroker->line_cap == XCG_FT_STROKER_LINECAP_BUTT)); + stroker->subpath_start = *to; + stroker->angle_in = 0; + return 0; +} + +static XCG_FT_Error ft_stroker_add_reverse_left(XCG_FT_Stroker stroker, XCG_FT_Bool open) +{ + XCG_FT_StrokeBorder right = stroker->borders + 0; + XCG_FT_StrokeBorder left = stroker->borders + 1; + XCG_FT_Int new_points; + XCG_FT_Error error = 0; + + new_points = left->num_points - left->start; + if(new_points > 0) + { + error = ft_stroke_border_grow(right, (XCG_FT_UInt)new_points); + if(error) + goto Exit; + { + XCG_FT_Vector *dst_point = right->points + right->num_points; + XCG_FT_Byte *dst_tag = right->tags + right->num_points; + XCG_FT_Vector *src_point = left->points + left->num_points - 1; + XCG_FT_Byte *src_tag = left->tags + left->num_points - 1; + while(src_point >= left->points + left->start) + { + *dst_point = *src_point; + *dst_tag = *src_tag; + if(open) + dst_tag[0] &= ~XCG_FT_STROKE_TAG_BEGIN_END; + else + { + XCG_FT_Byte ttag = (XCG_FT_Byte)(dst_tag[0] & XCG_FT_STROKE_TAG_BEGIN_END); + if(ttag == XCG_FT_STROKE_TAG_BEGIN || ttag == XCG_FT_STROKE_TAG_END) + dst_tag[0] ^= XCG_FT_STROKE_TAG_BEGIN_END; + } + src_point--; + src_tag--; + dst_point++; + dst_tag++; + } + } + left->num_points = left->start; + right->num_points += new_points; + right->movable = FALSE; + left->movable = FALSE; + } +Exit: + return error; +} + +XCG_FT_Error XCG_FT_Stroker_EndSubPath(XCG_FT_Stroker stroker) +{ + XCG_FT_Error error = 0; + + if(stroker->subpath_open) + { + XCG_FT_StrokeBorder right = stroker->borders; + error = ft_stroker_cap(stroker, stroker->angle_in, 0); + if(error) + goto Exit; + error = ft_stroker_add_reverse_left(stroker, TRUE); + if(error) + goto Exit; + stroker->center = stroker->subpath_start; + error = ft_stroker_cap(stroker, stroker->subpath_angle + XCG_FT_ANGLE_PI, 0); + if(error) + goto Exit; + ft_stroke_border_close(right, FALSE); + } + else + { + XCG_FT_Angle turn; + XCG_FT_Int inside_side; + if(stroker->center.x != stroker->subpath_start.x || stroker->center.y != stroker->subpath_start.y) + { + error = XCG_FT_Stroker_LineTo(stroker, &stroker->subpath_start); + if(error) + goto Exit; + } + stroker->angle_out = stroker->subpath_angle; + turn = XCG_FT_Angle_Diff(stroker->angle_in, stroker->angle_out); + if(turn != 0) + { + inside_side = 0; + if(turn < 0) + inside_side = 1; + error = ft_stroker_inside(stroker, inside_side, stroker->subpath_line_length); + if(error) + goto Exit; + error = ft_stroker_outside(stroker, 1 - inside_side, stroker->subpath_line_length); + if(error) + goto Exit; + } + ft_stroke_border_close(stroker->borders + 0, FALSE); + ft_stroke_border_close(stroker->borders + 1, TRUE); + } +Exit: + return error; +} + +XCG_FT_Error XCG_FT_Stroker_GetBorderCounts(XCG_FT_Stroker stroker, XCG_FT_StrokerBorder border, XCG_FT_UInt * anum_points, XCG_FT_UInt * anum_contours) +{ + XCG_FT_UInt num_points = 0, num_contours = 0; + XCG_FT_Error error; + + if(!stroker || border > 1) + { + error = -1; + goto Exit; + } + error = ft_stroke_border_get_counts(stroker->borders + border, &num_points, &num_contours); +Exit: + if(anum_points) + *anum_points = num_points; + if(anum_contours) + *anum_contours = num_contours; + return error; +} + +XCG_FT_Error XCG_FT_Stroker_GetCounts(XCG_FT_Stroker stroker, XCG_FT_UInt * anum_points, XCG_FT_UInt * anum_contours) +{ + XCG_FT_UInt count1, count2, num_points = 0; + XCG_FT_UInt count3, count4, num_contours = 0; + XCG_FT_Error error; + + error = ft_stroke_border_get_counts(stroker->borders + 0, &count1, &count2); + if(error) + goto Exit; + error = ft_stroke_border_get_counts(stroker->borders + 1, &count3, &count4); + if(error) + goto Exit; + num_points = count1 + count3; + num_contours = count2 + count4; +Exit: + *anum_points = num_points; + *anum_contours = num_contours; + return error; +} + +void XCG_FT_Stroker_ExportBorder(XCG_FT_Stroker stroker, XCG_FT_StrokerBorder border, XCG_FT_Outline * outline) +{ + if(border == XCG_FT_STROKER_BORDER_LEFT || border == XCG_FT_STROKER_BORDER_RIGHT) + { + XCG_FT_StrokeBorder sborder = &stroker->borders[border]; + if(sborder->valid) + ft_stroke_border_export(sborder, outline); + } +} + +void XCG_FT_Stroker_Export(XCG_FT_Stroker stroker, XCG_FT_Outline * outline) +{ + XCG_FT_Stroker_ExportBorder(stroker, XCG_FT_STROKER_BORDER_LEFT, outline); + XCG_FT_Stroker_ExportBorder(stroker, XCG_FT_STROKER_BORDER_RIGHT, outline); +} + +XCG_FT_Error XCG_FT_Stroker_ParseOutline(XCG_FT_Stroker stroker, const XCG_FT_Outline *outline) +{ + XCG_FT_Vector v_last; + XCG_FT_Vector v_control; + XCG_FT_Vector v_start; + XCG_FT_Vector * point; + XCG_FT_Vector * limit; + char * tags; + XCG_FT_Error error; + XCG_FT_Int n; + XCG_FT_UInt first; + XCG_FT_Int tag; + + if(!outline || !stroker) + return -1; + XCG_FT_Stroker_Rewind(stroker); + first = 0; + for(n = 0; n < outline->n_contours; n++) + { + XCG_FT_UInt last; + last = outline->contours[n]; + limit = outline->points + last; + if(last <= first) + { + first = last + 1; + continue; + } + v_start = outline->points[first]; + v_last = outline->points[last]; + v_control = v_start; + point = outline->points + first; + tags = outline->tags + first; + tag = XCG_FT_CURVE_TAG(tags[0]); + if(tag == XCG_FT_CURVE_TAG_CUBIC) + goto Invalid_Outline; + if(tag == XCG_FT_CURVE_TAG_CONIC) + { + if(XCG_FT_CURVE_TAG(outline->tags[last]) == XCG_FT_CURVE_TAG_ON) + { + v_start = v_last; + limit--; + } + else + { + v_start.x = (v_start.x + v_last.x) / 2; + v_start.y = (v_start.y + v_last.y) / 2; + } + point--; + tags--; + } + error = XCG_FT_Stroker_BeginSubPath(stroker, &v_start, outline->contours_flag[n]); + if(error) + goto Exit; + while(point < limit) + { + point++; + tags++; + tag = XCG_FT_CURVE_TAG(tags[0]); + switch(tag) + { + case XCG_FT_CURVE_TAG_ON: + { + XCG_FT_Vector vec; + vec.x = point->x; + vec.y = point->y; + error = XCG_FT_Stroker_LineTo(stroker, &vec); + if(error) + goto Exit; + continue; + } + + case XCG_FT_CURVE_TAG_CONIC: + v_control.x = point->x; + v_control.y = point->y; +Do_Conic: + if(point < limit) + { + XCG_FT_Vector vec; + XCG_FT_Vector v_middle; + point++; + tags++; + tag = XCG_FT_CURVE_TAG(tags[0]); + vec = point[0]; + if(tag == XCG_FT_CURVE_TAG_ON) + { + error = XCG_FT_Stroker_ConicTo(stroker, &v_control, &vec); + if(error) + goto Exit; + continue; + } + if(tag != XCG_FT_CURVE_TAG_CONIC) + goto Invalid_Outline; + v_middle.x = (v_control.x + vec.x) / 2; + v_middle.y = (v_control.y + vec.y) / 2; + error = XCG_FT_Stroker_ConicTo(stroker, &v_control, &v_middle); + if(error) + goto Exit; + v_control = vec; + goto Do_Conic; + } + error = XCG_FT_Stroker_ConicTo(stroker, &v_control, &v_start); + goto Close; + + default: + { + XCG_FT_Vector vec1, vec2; + if(point + 1 > limit || + XCG_FT_CURVE_TAG(tags[1]) != XCG_FT_CURVE_TAG_CUBIC) + goto Invalid_Outline; + point += 2; + tags += 2; + vec1 = point[-2]; + vec2 = point[-1]; + if(point <= limit) + { + XCG_FT_Vector vec; + vec = point[0]; + error = XCG_FT_Stroker_CubicTo(stroker, &vec1, &vec2, &vec); + if(error) + goto Exit; + continue; + } + error = XCG_FT_Stroker_CubicTo(stroker, &vec1, &vec2, &v_start); + goto Close; + } + } + } +Close: + if(error) + goto Exit; + if(stroker->first_point) + { + stroker->subpath_open = TRUE; + error = ft_stroker_subpath_start(stroker, 0, 0); + if(error) + goto Exit; + } + error = XCG_FT_Stroker_EndSubPath(stroker); + if(error) + goto Exit; + first = last + 1; + } + return 0; +Exit: + return error; +Invalid_Outline: + return -2; +} diff --git a/libmui/mui/xft.h b/libmui/mui/xft.h new file mode 100644 index 0000000..df9cc68 --- /dev/null +++ b/libmui/mui/xft.h @@ -0,0 +1,141 @@ +#ifndef __XFT_H__ +#define __XFT_H__ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +/* + * type + */ +typedef signed long XCG_FT_Fixed; +typedef signed int XCG_FT_Int; +typedef unsigned int XCG_FT_UInt; +typedef signed long XCG_FT_Long; +typedef unsigned long XCG_FT_ULong; +typedef signed short XCG_FT_Short; +typedef unsigned char XCG_FT_Byte; +typedef unsigned char XCG_FT_Bool; +typedef int XCG_FT_Error; +typedef signed long XCG_FT_Pos; + +typedef struct XCG_FT_Vector_ { + XCG_FT_Pos x; + XCG_FT_Pos y; +} XCG_FT_Vector; + +/* + * math + */ +typedef XCG_FT_Fixed XCG_FT_Angle; + +XCG_FT_Long XCG_FT_MulFix(XCG_FT_Long a, XCG_FT_Long b); +XCG_FT_Long XCG_FT_MulDiv(XCG_FT_Long a, XCG_FT_Long b, XCG_FT_Long c); +XCG_FT_Long XCG_FT_DivFix(XCG_FT_Long a, XCG_FT_Long b); +XCG_FT_Fixed XCG_FT_Sin(XCG_FT_Angle angle); +XCG_FT_Fixed XCG_FT_Cos(XCG_FT_Angle angle); +XCG_FT_Fixed XCG_FT_Tan(XCG_FT_Angle angle); +XCG_FT_Angle XCG_FT_Atan2(XCG_FT_Fixed x, XCG_FT_Fixed y); +XCG_FT_Angle XCG_FT_Angle_Diff(XCG_FT_Angle angle1, XCG_FT_Angle angle2); +void XCG_FT_Vector_Unit(XCG_FT_Vector *vec, XCG_FT_Angle angle); +void XCG_FT_Vector_Rotate(XCG_FT_Vector *vec, XCG_FT_Angle angle); +XCG_FT_Fixed XCG_FT_Vector_Length(XCG_FT_Vector *vec); +void XCG_FT_Vector_Polarize(XCG_FT_Vector *vec, XCG_FT_Fixed *length, XCG_FT_Angle *angle); +void XCG_FT_Vector_From_Polar(XCG_FT_Vector *vec, XCG_FT_Fixed length, XCG_FT_Angle angle); + +/* + * raster + */ +typedef struct XCG_FT_BBox_ { + XCG_FT_Pos xMin, yMin; + XCG_FT_Pos xMax, yMax; +} XCG_FT_BBox; + +typedef struct XCG_FT_Outline_ { + int n_contours; + int n_points; + XCG_FT_Vector * points; + char * tags; + int * contours; + char * contours_flag; + int flags; +} XCG_FT_Outline; + +#define XCG_FT_OUTLINE_NONE 0x0 +#define XCG_FT_OUTLINE_OWNER 0x1 +#define XCG_FT_OUTLINE_EVEN_ODD_FILL 0x2 +#define XCG_FT_OUTLINE_REVERSE_FILL 0x4 +#define XCG_FT_CURVE_TAG(flag) (flag & 3) +#define XCG_FT_CURVE_TAG_ON 1 +#define XCG_FT_CURVE_TAG_CONIC 0 +#define XCG_FT_CURVE_TAG_CUBIC 2 +#define XCG_FT_Curve_Tag_On XCG_FT_CURVE_TAG_ON +#define XCG_FT_Curve_Tag_Conic XCG_FT_CURVE_TAG_CONIC +#define XCG_FT_Curve_Tag_Cubic XCG_FT_CURVE_TAG_CUBIC + +typedef struct XCG_FT_Span_ { + int x; + int len; + int y; + unsigned char coverage; +} XCG_FT_Span; + +typedef void (*XCG_FT_SpanFunc)(int count, const XCG_FT_Span * spans, void * user); +#define XCG_FT_Raster_Span_Func XCG_FT_SpanFunc + +#define XCG_FT_RASTER_FLAG_DEFAULT 0x0 +#define XCG_FT_RASTER_FLAG_AA 0x1 +#define XCG_FT_RASTER_FLAG_DIRECT 0x2 +#define XCG_FT_RASTER_FLAG_CLIP 0x4 + +typedef struct XCG_FT_Raster_Params_ { + const void * source; + int flags; + XCG_FT_SpanFunc gray_spans; + void * user; + XCG_FT_BBox clip_box; +} XCG_FT_Raster_Params; + +XCG_FT_Error XCG_FT_Outline_Check(XCG_FT_Outline * outline); +void XCG_FT_Outline_Get_CBox(const XCG_FT_Outline * outline, XCG_FT_BBox * acbox); +void XCG_FT_Raster_Render(const XCG_FT_Raster_Params * params); + +/* + * stroker + */ +typedef struct XCG_FT_StrokerRec_ * XCG_FT_Stroker; + +typedef enum XCG_FT_Stroker_LineJoin_ { + XCG_FT_STROKER_LINEJOIN_ROUND = 0, + XCG_FT_STROKER_LINEJOIN_BEVEL = 1, + XCG_FT_STROKER_LINEJOIN_MITER_VARIABLE = 2, + XCG_FT_STROKER_LINEJOIN_MITER = XCG_FT_STROKER_LINEJOIN_MITER_VARIABLE, + XCG_FT_STROKER_LINEJOIN_MITER_FIXED = 3, +} XCG_FT_Stroker_LineJoin; + +typedef enum XCG_FT_Stroker_LineCap_ { + XCG_FT_STROKER_LINECAP_BUTT = 0, + XCG_FT_STROKER_LINECAP_ROUND = 1, + XCG_FT_STROKER_LINECAP_SQUARE = 2, +} XCG_FT_Stroker_LineCap; + +typedef enum XCG_FT_StrokerBorder_ { + XCG_FT_STROKER_BORDER_LEFT = 0, + XCG_FT_STROKER_BORDER_RIGHT = 1, +} XCG_FT_StrokerBorder; + +XCG_FT_Error XCG_FT_Stroker_New(XCG_FT_Stroker * astroker); +void XCG_FT_Stroker_Set(XCG_FT_Stroker stroker, XCG_FT_Fixed radius, XCG_FT_Stroker_LineCap line_cap, XCG_FT_Stroker_LineJoin line_join, XCG_FT_Fixed miter_limit); +XCG_FT_Error XCG_FT_Stroker_ParseOutline(XCG_FT_Stroker stroker, const XCG_FT_Outline * outline); +XCG_FT_Error XCG_FT_Stroker_GetCounts(XCG_FT_Stroker stroker, XCG_FT_UInt * anum_points, XCG_FT_UInt * anum_contours); +void XCG_FT_Stroker_Export(XCG_FT_Stroker stroker, XCG_FT_Outline * outline); +void XCG_FT_Stroker_Done(XCG_FT_Stroker stroker); + +#endif /* __XFT_H__ */ diff --git a/libmui/tests/mui_playground.c b/libmui/tests/mui_playground.c new file mode 100644 index 0000000..b7ed530 --- /dev/null +++ b/libmui/tests/mui_playground.c @@ -0,0 +1,616 @@ +/* + * mui_playground.c + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ +#define MUI_HAS_XCB 1 +#define MUI_HAS_XKB 1 + +#if MUI_HAS_XCB +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include + +#if MUI_HAS_XKB +#include +#include +#include +#include +#else +struct xkb_state; +#endif + +#include "mui.h" +#include "mui_plugin.h" + +typedef struct mui_xcb_t { + mui_t ui; + mui_plug_t * plug; + void * plug_data; + + float ui_scale_x, ui_scale_y; + c2_pt_t size; + xcb_connection_t * xcb; + xcb_shm_segment_info_t shm; + xcb_window_t window; + xcb_pixmap_t xcb_pix; + xcb_gcontext_t xcb_context; + struct xkb_state * xkb_state; + + int redraw; +} mui_xcb_t; + + +int +_mui_xcb_init_keyboard( + mui_xcb_t *ui) +{ +#if MUI_HAS_XKB + uint8_t xkb_event_base, xkb_error_base; + if (!xkb_x11_setup_xkb_extension (ui->xcb, + XKB_X11_MIN_MAJOR_XKB_VERSION, XKB_X11_MIN_MINOR_XKB_VERSION, + XKB_X11_SETUP_XKB_EXTENSION_NO_FLAGS, + NULL, NULL, + &xkb_event_base, &xkb_error_base)) { + fprintf(stderr, "%s needs version %d.%d or newer\n", __func__, + XKB_X11_MIN_MAJOR_XKB_VERSION, XKB_X11_MIN_MINOR_XKB_VERSION); + goto error; + } + struct xkb_context *ctx = + xkb_context_new(XKB_CONTEXT_NO_FLAGS); + int32_t device_id = + xkb_x11_get_core_keyboard_device_id(ui->xcb); + if (device_id == -1) { + fprintf(stderr, "%s xkb_x11_get_core_keyboard_device_id\n", __func__); + goto error; + } + struct xkb_keymap *keymap = + xkb_x11_keymap_new_from_device(ctx, ui->xcb, + device_id, XKB_KEYMAP_COMPILE_NO_FLAGS); + ui->xkb_state = + xkb_x11_state_new_from_device(keymap, ui->xcb, device_id); + xkb_context_unref(ctx); + return 0; +#endif +error: + printf("XCB Keyboard initialisation: %s\n", ui->xkb_state ? "OK" : "Failed"); + ui->xkb_state = NULL; + return -1; +} + +/* + * xmodmap -pke or -pk will print the list of keycodes + */ +static bool +_mui_xcb_convert_keycode( + mui_xcb_t *ui, + xkb_keysym_t sym, + mui_event_t *out ) +{ + switch (sym) { + case XKB_KEY_F1 ... XKB_KEY_F12: + out->key.key = MUI_KEY_F1 + (sym - XKB_KEY_F1); + break; + case XKB_KEY_Escape: out->key.key = MUI_KEY_ESCAPE; break; + case XKB_KEY_Left: out->key.key = MUI_KEY_LEFT; break; + case XKB_KEY_Up: out->key.key = MUI_KEY_UP; break; + case XKB_KEY_Right: out->key.key = MUI_KEY_RIGHT; break; + case XKB_KEY_Down: out->key.key = MUI_KEY_DOWN; break; + // XKB_KEY_Begin + case XKB_KEY_Insert: out->key.key = MUI_KEY_INSERT; break; + case XKB_KEY_Home: out->key.key = MUI_KEY_HOME; break; + case XKB_KEY_End: out->key.key = MUI_KEY_END; break; + case XKB_KEY_Page_Up: out->key.key = MUI_KEY_PAGEUP; break; + case XKB_KEY_Page_Down: out->key.key = MUI_KEY_PAGEDOWN; break; + + case XKB_KEY_Shift_R: out->key.key = MUI_KEY_RSHIFT; break; + case XKB_KEY_Shift_L: out->key.key = MUI_KEY_LSHIFT; break; + case XKB_KEY_Control_R: out->key.key = MUI_KEY_RCTRL; break; + case XKB_KEY_Control_L: out->key.key = MUI_KEY_LCTRL; break; + case XKB_KEY_Alt_L: out->key.key = MUI_KEY_LALT; break; + case XKB_KEY_Alt_R: out->key.key = MUI_KEY_RALT; break; + case XKB_KEY_Super_L: out->key.key = MUI_KEY_LSUPER; break; + case XKB_KEY_Super_R: out->key.key = MUI_KEY_RSUPER; break; + default: + out->key.key = sym & 0xff; + break; + } +// printf("%s %08x to %04x\n", __func__, sym, out->key.key); + return true; +} + +int +mui_xcb_list_physical_screens( + struct xcb_connection_t * xcb, + struct c2_rect_array_t *out) +{ + if (!xcb || !out) + return -1; + c2_rect_array_clear(out); + xcb_screen_t *screen = xcb_setup_roots_iterator( + xcb_get_setup(xcb)).data; + xcb_randr_get_screen_resources_current_reply_t *reply = + xcb_randr_get_screen_resources_current_reply( + xcb, + xcb_randr_get_screen_resources_current( + xcb, screen->root), + NULL); + xcb_timestamp_t timestamp = reply->config_timestamp; + int len = xcb_randr_get_screen_resources_current_outputs_length(reply); + xcb_randr_output_t *randr_outputs = + xcb_randr_get_screen_resources_current_outputs(reply); + for (int i = 0; i < len; i++) { + xcb_randr_get_output_info_reply_t *output = + xcb_randr_get_output_info_reply( + xcb, + xcb_randr_get_output_info( + xcb, randr_outputs[i], timestamp), + NULL); + if (!output || output->crtc == XCB_NONE || + output->connection == XCB_RANDR_CONNECTION_DISCONNECTED) + continue; + xcb_randr_get_crtc_info_reply_t *crtc = + xcb_randr_get_crtc_info_reply(xcb, + xcb_randr_get_crtc_info( + xcb, output->crtc, timestamp), + NULL); + c2_rect_t r = C2_RECT(crtc->x, crtc->y, + crtc->x +crtc->width, crtc->y + crtc->height); + c2_rect_array_add(out, r); + free(crtc); + free(output); + } + free(reply); + return 0; +} + +static bool +_cui_match_physical_screen( + xcb_connection_t *xcb, + c2_pt_t want_size, + c2_pt_p found_pos ) +{ + bool res = false; + c2_rect_array_t sc = {}; + + mui_xcb_list_physical_screens(xcb, &sc); + + for (unsigned int i = 0; i < sc.count; i++) { + if (c2_rect_width(&sc.e[i]) == want_size.x && + c2_rect_height(&sc.e[i]) == want_size.y) { + *found_pos = sc.e[i].tl; + res = true; + } + } + return res; +} + +struct mui_t * +mui_xcb_init( + struct mui_t *mui, + struct mui_pixmap_t * pix ) +{ + mui_xcb_t *ui = (mui_xcb_t *)mui; + + pix->size.y = 720; + pix->size.x = 1280; + ui->ui.screen_size = pix->size; + ui->ui_scale_x = ui->ui_scale_y = 1; + printf("XCB: Starting on %dx%d window\n", + pix->size.x, pix->size.y); + + pix->size.x *= ui->ui_scale_x; + pix->size.y *= ui->ui_scale_y; + ui->size = pix->size; + + uint32_t value_mask; + uint32_t value_list[6]; + + ui->xcb = xcb_connect(NULL, NULL); + + bool windowed = 1; + bool opaque = 1; + c2_pt_t found_position = {}; + bool has_position = !windowed && _cui_match_physical_screen( + ui->xcb, ui->size, &found_position); + + xcb_screen_iterator_t iter = xcb_setup_roots_iterator( + xcb_get_setup(ui->xcb)); + printf("%s %d screens\n", __func__, iter.rem); + xcb_screen_t *screen = NULL; + while (iter.rem) { + screen = iter.data; + printf("%s screen %d: width: %d, height: %d\n", __func__, + iter.index, + screen->width_in_pixels, screen->height_in_pixels); + xcb_screen_next(&iter); + } + printf("XCB Screen depth %d\n", screen->root_depth); + + /* + * This walks thru the 'visual', looking for a true colour *32 bits* one + * which means it handles ARGB colors, which we can draw into. Also find + * one which color bit masks matches libcui & libpixman. + */ + xcb_visualtype_t *argb_visual = NULL; + xcb_depth_iterator_t depth_iter = + xcb_screen_allowed_depths_iterator(screen); + for (; depth_iter.rem; xcb_depth_next(&depth_iter)) { + xcb_visualtype_iterator_t visual_iter = + xcb_depth_visuals_iterator(depth_iter.data); + // printf("XCB Depth %d\n", depth_iter.data->depth); + if (depth_iter.data->depth != 32) + continue; + for (; visual_iter.rem; xcb_visualtype_next(&visual_iter)) { + if (visual_iter.data->_class == XCB_VISUAL_CLASS_TRUE_COLOR + && visual_iter.data->red_mask == 0xff0000 + && visual_iter.data->green_mask == 0x00ff00 + && visual_iter.data->blue_mask == 0x0000ff) { + argb_visual = visual_iter.data; + break; + } + } + } + printf("XCB ARGB Transparency %s\n", + argb_visual ? "Supported" : "Not available"); + if (windowed || opaque) + argb_visual = NULL; + + xcb_shm_query_version_reply_t *xcb_shm_present; + xcb_shm_present = xcb_shm_query_version_reply( + ui->xcb, xcb_shm_query_version(ui->xcb), NULL); + if (!xcb_shm_present || !xcb_shm_present->shared_pixmaps) { + printf("xcb_shm error... %p\n", xcb_shm_present); + printf("If using nvidia driver, you need\n" + " Option \"AllowSHMPixmaps\" \"1\"\n" + " In your /etc/X11/xorg.conf file\n"); + exit(0); + } + printf("XCB Shared memory present\n"); + + _mui_xcb_init_keyboard(ui); + + value_mask = XCB_CW_BACK_PIXEL | + XCB_CW_BORDER_PIXEL | + XCB_CW_OVERRIDE_REDIRECT | + XCB_CW_EVENT_MASK; + + xcb_colormap_t cmap = xcb_generate_id(ui->xcb); + /* required for having transparent windows */ + if (argb_visual) { + xcb_create_colormap(ui->xcb, XCB_COLORMAP_ALLOC_NONE, cmap, + screen->root, argb_visual->visual_id); + value_mask |= XCB_CW_COLORMAP; + } + uint32_t w_mask[] = { + screen->black_pixel, + // Border Pixel; not really needed for anything, but needed + // for ARGB window otherwise it doesn't get created properly + 0x88888888, + // if we found a screen of the exact size, remove the border + has_position ? 1 : 0, + XCB_EVENT_MASK_EXPOSURE | XCB_EVENT_MASK_BUTTON_PRESS | + XCB_EVENT_MASK_BUTTON_RELEASE | XCB_EVENT_MASK_POINTER_MOTION | + XCB_EVENT_MASK_ENTER_WINDOW | XCB_EVENT_MASK_LEAVE_WINDOW | + XCB_EVENT_MASK_KEY_PRESS | XCB_EVENT_MASK_KEY_RELEASE, + cmap + }; + ui->window = xcb_generate_id(ui->xcb); + xcb_create_window( + ui->xcb, + argb_visual ? 32 : XCB_COPY_FROM_PARENT, + ui->window, screen->root, + found_position.x, found_position.y, + pix->size.x, pix->size.y, 0, + XCB_WINDOW_CLASS_INPUT_OUTPUT, + argb_visual ? argb_visual->visual_id : screen->root_visual, + value_mask, w_mask); + xcb_free_colormap(ui->xcb, cmap); + + const char * title = "MII UI Playground"; + xcb_change_property(ui->xcb, XCB_PROP_MODE_REPLACE, + ui->window, XCB_ATOM_WM_NAME, XCB_ATOM_STRING, 8, + strlen(title), title); + + // create a graphic context + value_mask = XCB_GC_FOREGROUND | XCB_GC_GRAPHICS_EXPOSURES; + value_list[0] = screen->white_pixel; + value_list[1] = 0; + ui->xcb_context = xcb_generate_id(ui->xcb); + xcb_create_gc( + ui->xcb, ui->xcb_context, ui->window, value_mask, value_list); + // map the window onto the screen + xcb_map_window(ui->xcb, ui->window); + // wont show unless I do this + xcb_flush(ui->xcb); + ui->shm.shmid = shmget(IPC_PRIVATE, + pix->size.x * pix->size.y * 4, IPC_CREAT | 0777); + ui->shm.shmaddr = shmat(ui->shm.shmid, 0, 0); + ui->shm.shmseg = xcb_generate_id(ui->xcb); + xcb_shm_attach(ui->xcb, ui->shm.shmseg, ui->shm.shmid, 0); + shmctl(ui->shm.shmid, IPC_RMID, 0); + + ui->xcb_pix = xcb_generate_id(ui->xcb); + xcb_shm_create_pixmap( + ui->xcb, ui->xcb_pix, ui->window, + pix->size.x, pix->size.y, + argb_visual ? 32 : screen->root_depth, + ui->shm.shmseg, 0); + + pix->pixels = ui->shm.shmaddr; + pix->row_bytes = pix->size.x * 4; +// printf("%s pix is %p\n", __func__, pix->pixels); + ui->redraw = 1; + return &ui->ui; +} + +int +mui_xcb_poll( + struct mui_t * mui, + bool redrawn) +{ + mui_xcb_t * ui = (mui_xcb_t *)mui; + int gameover = 0; + xcb_generic_event_t *event; + static bool buttondown = false; + static mui_event_t key_ev; + + while (!gameover && (event = xcb_poll_for_event(ui->xcb)) != NULL) { + switch (event->response_type & ~0x80) { + case XCB_KEY_RELEASE: { + xcb_key_release_event_t *key = + (xcb_key_press_event_t*) event; + xkb_state_update_key(ui->xkb_state, + key->detail, XKB_KEY_UP); + xkb_keysym_t keysym = xkb_state_key_get_one_sym( + ui->xkb_state, key->detail); + key_ev.type = MUI_EVENT_KEYUP; + key_ev.key.up = 1; + if (_mui_xcb_convert_keycode(ui, keysym, &key_ev)) { + if (key_ev.key.key >= MUI_KEY_MODIFIERS && + key_ev.key.key <= MUI_KEY_MODIFIERS_LAST) { + mui->modifier_keys &= ~(1 << (key_ev.key.key - MUI_KEY_MODIFIERS)); + } + key_ev.modifiers = mui->modifier_keys; + // key_ev.modifiers |= MUI_MODIFIER_EVENT_TRACE; + if (ui->plug && ui->plug->event) + ui->plug->event(mui, ui->plug_data, &key_ev); + } + } break; + case XCB_KEY_PRESS: { + xcb_key_press_event_t *key = + (xcb_key_press_event_t*) event; + if (key->same_screen == 0) // repeat key + break; + xkb_state_update_key(ui->xkb_state, + key->detail, XKB_KEY_DOWN); + xkb_keysym_t keysym = xkb_state_key_get_one_sym( + ui->xkb_state, key->detail); + key_ev.type = MUI_EVENT_KEYDOWN; + key_ev.key.up = 0; + // printf("%s %08x\n", __func__, keysym); + if (_mui_xcb_convert_keycode(ui, keysym, &key_ev)) { + if (key_ev.key.key >= MUI_KEY_MODIFIERS && + key_ev.key.key <= MUI_KEY_MODIFIERS_LAST) { + mui->modifier_keys |= (1 << (key_ev.key.key - MUI_KEY_MODIFIERS)); + } + key_ev.modifiers = mui->modifier_keys; + // key_ev.modifiers |= MUI_MODIFIER_EVENT_TRACE; + // gameover = key_ev.key.key == 'q'; + if (ui->plug && ui->plug->event) + ui->plug->event(mui, ui->plug_data, &key_ev); + } + } break; + case XCB_BUTTON_RELEASE: + buttondown = false; + // fall through + case XCB_BUTTON_PRESS: { + xcb_button_press_event_t *m = + (xcb_button_press_event_t *)event; +#if 0 + printf("%s %s %02x %d at %4dx%4d\n", __func__, + (event->response_type & ~0x80) == XCB_BUTTON_PRESS ? + "down" : "up", + event->response_type, + m->detail, m->event_x, m->event_y); +#endif + switch (m->detail) { + case XCB_BUTTON_INDEX_1: { + case XCB_BUTTON_INDEX_3: + buttondown = (event->response_type & ~0x80) == XCB_BUTTON_PRESS; + mui_event_t ev = { + .type = buttondown ? + MUI_EVENT_BUTTONDOWN : + MUI_EVENT_BUTTONUP, + .mouse.button = m->detail, + .mouse.where.x = (float)m->event_x / ui->ui_scale_x, + .mouse.where.y = (float)m->event_y / ui->ui_scale_y, + .modifiers = mui->modifier_keys, + }; + if (ui->plug && ui->plug->event) + ui->plug->event(mui, ui->plug_data, &ev); + } break; + case XCB_BUTTON_INDEX_4: + case XCB_BUTTON_INDEX_5: { + mui_event_t ev = { + .type = MUI_EVENT_WHEEL, + .wheel.delta = m->detail == XCB_BUTTON_INDEX_4 ? + -1 : 1, + .wheel.where.x = (float)m->event_x / ui->ui_scale_x, + .wheel.where.y = (float)m->event_y / ui->ui_scale_y, + .modifiers = mui->modifier_keys, + }; + if (ui->plug && ui->plug->event) + ui->plug->event(mui, ui->plug_data, &ev); + } break; + } + } break; + case XCB_MOTION_NOTIFY: { + xcb_motion_notify_event_t *m = + (xcb_motion_notify_event_t*)event; +// if (buttondown) { + // printf("x=%d y=%d\n", event.motion.x, event.motion.y); + mui_event_t ev = { + .type = MUI_EVENT_DRAG, + .mouse.button = buttondown ? 1 : 0, + .mouse.where.x = (float)m->event_x / ui->ui_scale_x, + .mouse.where.y = (float)m->event_y / ui->ui_scale_y, + .modifiers = mui->modifier_keys, + }; + if (ui->plug && ui->plug->event) + ui->plug->event(mui, ui->plug_data, &ev); +// } + } break; + case XCB_ENTER_NOTIFY: + case XCB_LEAVE_NOTIFY: + break; + case XCB_EXPOSE: { + // xcb_expose_event_t *expose_event = (xcb_expose_event_t*) event; + ui->redraw++; + } break; + default: + // Handle other events + break; + } + free(event); + } + if (redrawn || ui->redraw || pixman_region32_not_empty(&mui->redraw)) { + // Handle window refresh event + int rc = 0; + c2_rect_t whole = C2_RECT(0, 0, ui->size.x, ui->size.y); + c2_rect_t *ra = (c2_rect_t*)pixman_region32_rectangles(&mui->redraw, &rc); + if (ui->redraw) { + ui->redraw = 0; + rc = 1; + ra = &whole; + } + if (rc) { + // printf("XCB: %d rects to redraw\n", rc); + for (int i = 0; i < rc; i++) { + c2_rect_t r = ra[i]; + // printf("XCB: %d,%d %dx%d\n", r.l, r.t, c2_rect_width(&r), c2_rect_height(&r)); + xcb_copy_area( + ui->xcb, ui->xcb_pix, ui->window, ui->xcb_context, + r.l, r.t, r.l, r.t, c2_rect_width(&r), c2_rect_height(&r)); + } + } + pixman_region32_clear(&mui->redraw); + } + xcb_flush(ui->xcb); + return gameover; +} + +void +mui_xcb_terminate( + struct mui_t * mui) +{ + mui_xcb_t * ui = (mui_xcb_t *)mui; + xcb_shm_detach(ui->xcb, ui->shm.shmseg); + shmdt(ui->shm.shmaddr); + xcb_free_pixmap(ui->xcb, ui->xcb_pix); + xcb_destroy_window(ui->xcb, ui->window); + xcb_disconnect(ui->xcb); +} + +#endif + +#include +#include +#include +#include +#include +#include + +int main() +{ + mui_xcb_t xcb_ui = {}; + mui_drawable_t dr = {}; + + // note: the mui_t is *not* initialized yet, it will be done + // in the _init() of the plugin. This is just to get the + // init/dispose done in the plugin to check for leaks etc + mui_t *mui = mui_xcb_init((struct mui_t *)&xcb_ui, &dr.pix); + mui_xcb_t * ui = &xcb_ui; + void * dynload = NULL; + char * filename = "build-x86_64-linux-gnu/lib/ui_tests.so"; + struct stat st_current = {}, st = {}; + + mui_time_t stamp = 0; + do { + if (stat(filename, &st) == 0 && st.st_mtime != st_current.st_mtime) { + st_current = st; + if (dynload) { + if (ui->plug_data && ui->plug && ui->plug->dispose) { + ui->plug->dispose(ui->plug_data); + ui->plug = NULL; + ui->plug_data = NULL; + } + printf("Closed %s\n", filename); + dlclose(dynload); + dynload = NULL; + } + } + if (!dynload) { + dynload = dlopen(filename, RTLD_NOW); + printf("Loading %s\n", filename); + if (!dynload) { + printf("Failed to load %s : %s\n", filename, dlerror()); + perror(filename); + sleep(2); + continue; + } + ui->plug = dlsym(dynload, "mui_plug"); + if (!ui->plug) { + printf("Failed to find mui_plug in %s\n", filename); + dlclose(dynload); + dynload = NULL; + sleep(10); + continue; + } + if (ui->plug->init) { + ui->plug_data = ui->plug->init(mui, ui->plug, &dr); + if (!ui->plug_data) { + printf("Failed to init plugin %s\n", filename); + dlclose(dynload); + dynload = NULL; + sleep(10); + continue; + } + } + stamp = mui_get_time(); + } + bool draw = false; + mui_run(mui); + if (ui->plug && ui->plug->draw) + draw = ui->plug->draw(mui, ui->plug_data, &dr, false); + if (mui_xcb_poll(mui, draw)) + break; + mui_time_t now = mui_get_time(); + while (stamp < now) + stamp += (MUI_TIME_SECOND / 60); + usleep(stamp-now); + } while (1); + if (dynload) { + if (ui->plug_data && ui->plug && ui->plug->dispose) { + ui->plug->dispose(ui->plug_data); + ui->plug = NULL; + ui->plug_data = NULL; + } + printf("Closed %s\n", filename); + dlclose(dynload); + } + mui_drawable_dispose(&dr); + mui_xcb_terminate(mui); + return 0; +} \ No newline at end of file diff --git a/libmui/tests/mui_plugin.h b/libmui/tests/mui_plugin.h new file mode 100644 index 0000000..75d0224 --- /dev/null +++ b/libmui/tests/mui_plugin.h @@ -0,0 +1,33 @@ +/* + * mui_plugin.h + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ + +#pragma once + +#include +#include + +struct mui_t; +struct mui_drawable_t; + +typedef struct mui_plug_t { + void * (*init)( + struct mui_t * ui, + struct mui_plug_t * plug, + struct mui_drawable_t * dr ); + void (*dispose)( + void * plug ); + int (*draw)( + struct mui_t *ui, + void *param, + struct mui_drawable_t * dr, + uint16_t all ); + int (*event)( + struct mui_t *ui, + void *param, + struct mui_event_t * event ); +} mui_plug_t; diff --git a/libmui/tests/ui_tests.c b/libmui/tests/ui_tests.c new file mode 100644 index 0000000..7f07b5e --- /dev/null +++ b/libmui/tests/ui_tests.c @@ -0,0 +1,236 @@ +/* + * ui_tests.c + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ +#define _GNU_SOURCE // for asprintf +#include +#include +#include +#include +#include + +#include "mui.h" +#include "mui_plugin.h" +#include "c2_geometry.h" + +typedef struct cg_ui_t { + mui_t *ui; +} cg_ui_t; + +#define MII_MUI_MENUS_C +#include "mii_mui_menus.h" +#include "mii_mui_settings.h" + + +#ifndef UI_VERSION +#define UI_VERSION "0.0.0" +#endif + +static void +_test_show_about( + cg_ui_t *g) +{ + mui_window_t *w = mui_window_get_by_id(g->ui, FCC('a','b','o','t')); + if (w) { + mui_window_select(w); + return; + } + w = mui_alert(g->ui, C2_PT(0,0), + "About MII", + "Version " UI_VERSION "\n" + "Build " __DATE__ " " __TIME__, + MUI_ALERT_INFO); + mui_window_set_id(w, FCC('a','b','o','t')); +} + +static mii_machine_config_t g_machine_conf = {}; +static mii_loadbin_conf_t g_loadbin_conf = {}; + +/* this is heavily endian dependend, as is the FCC macro */ +#define FCC_INDEX(_fcc) (isdigit(_fcc>>24) ? ((_fcc >> 24) - '0') : 0) + +int +_test_menubar_action( + mui_window_t *win, + void * cb_param, + uint32_t what, + void * param) +{ + cg_ui_t *g = cb_param; + + printf("%s %4.4s\n", __func__, (char*)&what); + + static int video_mode = 0; + static int audio_mute = 0; + switch (what) { + case MUI_MENUBAR_ACTION_PREPARE: { + mui_menu_item_t * items = param; + if (items == m_video_menu) { + printf("%s prepare video %d\n", __func__, video_mode); + for (int i = 0; m_video_menu[i].title; i++) { + switch (m_video_menu[i].uid) { + case FCC('v','d','c','0'): + case FCC('v','d','c','1'): + case FCC('v','d','c','2'): { + int idx = FCC_INDEX(m_video_menu[i].uid); + if (video_mode == idx) + strcpy(m_video_menu[i].mark, MUI_GLYPH_TICK); + else + m_video_menu[i].mark[0] = 0; + } break; + } + } + } else if (items == m_audio_menu) { + printf("%s prepare audio %d\n", __func__, audio_mute); + for (int i = 0; m_audio_menu[i].title; i++) { + switch (m_audio_menu[i].uid) { + case FCC('a','u','d','0'): + if (audio_mute) + strcpy(m_audio_menu[i].mark, MUI_GLYPH_TICK); + else + m_audio_menu[i].mark[0] = 0; + break; + } + } + } else { + printf("%s prepare (%s)\n", __func__, items[0].title); + } + } break; + case MUI_MENUBAR_ACTION_SELECT: { + mui_menu_item_t * item = param; + printf("%s Selected %4.4s '%s'\n", __func__, + (char*)&item->uid, item->title); + switch (item->uid) { + case FCC('a','b','o','t'): { + _test_show_about(g); + } break; + case FCC('q','u','i','t'): { + printf("%s Quit\n", __func__); + } break; + case FCC('s','l','o','t'): { + mii_mui_configure_slots(g->ui, &g_machine_conf); + } break; + case FCC('l', 'r', 'u', 'n'): { + mii_mui_load_binary(g->ui, &g_loadbin_conf); + } break; + case FCC('a','u','d','0'): + audio_mute = !audio_mute; + break; + case FCC('v','d','C','l'): { +// printf("%s Cycle video\n", __func__); + video_mode = (video_mode + 1) % 3; + } break; + case FCC('v','d','c','0'): + case FCC('v','d','c','1'): + case FCC('v','d','c','2'): + video_mode = FCC_INDEX(item->uid); + break; + default: + printf("%s menu item %4.4s %s IGNORED\n", + __func__, (char*)&item->uid, item->title); + break; + } + } break; + default: + printf("%s %4.4s IGNORED?\n", __func__, (char*)&what); + break; + } + + return 0; +} + +static void * +_init( + struct mui_t * ui, + struct mui_plug_t * plug, + mui_drawable_t * pix) +{ + mui_init(ui); + ui->screen_size = pix->pix.size; + asprintf(&ui->pref_directory, "%s/.local/share/mii", getenv("HOME")); + + cg_ui_t *g = calloc(1, sizeof(*g)); + g->ui = ui; + printf("%s\n", __func__); + mui_window_t * mbar = mui_menubar_new(ui); + mui_window_set_action(mbar, _test_menubar_action, g); + + mui_menubar_add_simple(mbar, MUI_GLYPH_APPLE, + FCC('a','p','p','l'), + m_apple_menu); + mui_menubar_add_simple(mbar, "File", + FCC('f','i','l','e'), + m_file_menu); + mui_menubar_add_simple(mbar, "Machine", + FCC('m','a','c','h'), + m_machine_menu); + mui_menubar_add_simple(mbar, "CPU", + FCC('c','p','u','m'), + m_cpu_menu); + +// mii_mui_configure_slots(g->ui, &g_machine_conf); +// mii_mui_load_binary(g->ui, &g_loadbin_conf); +// mii_mui_load_1mbrom(g->ui, &g_machine_conf.slot[0].conf.rom1mb); +// mii_mui_load_2dsk(g->ui, +// &g_machine_conf.slot[0].conf.disk2, MII_2DSK_DISKII); +#if 0 + mui_alert(ui, C2_PT(0,0), + "Testing one Two", + "Do you really want the printer to catch fire?\n" + "This operation cannot be cancelled.", + MUI_ALERT_WARN); +#endif +#if 1 + mui_stdfile_get(ui, + C2_PT(0, 0), + "Select image for SmartPort card", + "\\.(hdv|po|2mg)$", + getenv("HOME")); +#endif + + return g; +} + +static void +_dispose( + void *_ui) +{ + cg_ui_t *ui = _ui; + mui_dispose(ui->ui); + free(ui); +} + +static int +_draw( + struct mui_t *ui, + void *param, + mui_drawable_t *dr, + uint16_t all) +{ +// cg_ui_t *g = param; + mui_draw(ui, dr, all); + return 1; +} + +static int +_event( + struct mui_t *ui, + void *param, + mui_event_t *event) +{ +// cg_ui_t *g = param; +// printf("%s %d\n", __func__, event->type); + mui_handle_event(ui, event); + return 0; +} + + +mui_plug_t mui_plug = { + .init = _init, + .dispose = _dispose, + .draw = _draw, + .event = _event, +}; \ No newline at end of file diff --git a/nuklear/mii_emu.c b/nuklear/mii_emu.c deleted file mode 100644 index 1f4a320..0000000 --- a/nuklear/mii_emu.c +++ /dev/null @@ -1,323 +0,0 @@ -/* - * mii_emu.c - * - * Copyright (C) 2023 Michel Pollet - * - * SPDX-License-Identifier: MIT - */ - -#include -#include -#include -#include -#include -#include -#include -#include -#include - -#define NK_INCLUDE_FIXED_TYPES -#define NK_INCLUDE_STANDARD_IO -#define NK_INCLUDE_STANDARD_VARARGS -#define NK_INCLUDE_DEFAULT_ALLOCATOR -#define NK_INCLUDE_VERTEX_BUFFER_OUTPUT -#define NK_INCLUDE_FONT_BAKING -#define NK_INCLUDE_DEFAULT_FONT -#define NK_BUTTON_TRIGGER_ON_RELEASE -#include "nuklear.h" - -#define NK_XLIB_GL3_IMPLEMENTATION -#define NK_XLIB_LOAD_OPENGL_EXTENSIONS -#include "nuklear_xlib_gl3.h" - -#include "mii.h" -#include "mish.h" - -#define WINDOW_WIDTH 1280 -#define WINDOW_HEIGHT 720 // (10 + (192) * 3) - -#define MAX_VERTEX_BUFFER 512 * 1024 -#define MAX_ELEMENT_BUFFER 128 * 1024 - -void -mii_nuklear( - mii_t *mii, - struct nk_context *ctx); -void -mii_nuklear_init( - mii_t *mii, - struct nk_context *ctx); -/* =============================================================== - * - * DEMO - * - * ===============================================================*/ -struct XWindow { - Display *dpy; - Window win; - XVisualInfo *vis; - Colormap cmap; - XSetWindowAttributes swa; - XWindowAttributes attr; - GLXFBConfig fbc; - Atom wm_delete_window; - int width, height; -}; -static int gl_err = nk_false; -static int gl_error_handler(Display *dpy, XErrorEvent *ev) -{NK_UNUSED(dpy); NK_UNUSED(ev); gl_err = nk_true;return 0;} - -static void -die(const char *fmt, ...) -{ - va_list ap; - va_start(ap, fmt); - vfprintf(stderr, fmt, ap); - va_end(ap); - fputs("\n", stderr); - exit(EXIT_FAILURE); -} - -static int -has_extension(const char *string, const char *ext) -{ - const char *start, *where, *term; - where = strchr(ext, ' '); - if (where || *ext == '\0') - return nk_false; - - for (start = string;;) { - where = strstr((const char*)start, ext); - if (!where) break; - term = where + strlen(ext); - if (where == start || *(where - 1) == ' ') { - if (*term == ' ' || *term == '\0') - return nk_true; - } - start = term; - } - return nk_false; -} - -struct nk_font *nk_main_font = NULL; - -/* global now, mish commands use it */ -mii_t g_mii = {}; - -int -main( - int argc, - const char *argv[]) -{ - /* Platform */ - int running = 1; - struct XWindow win; - GLXContext glContext; - struct nk_context *ctx; - struct nk_colorf bg; - - mii_init(&g_mii); - int idx = 1; - uint32_t flags = MII_INIT_DEFAULT; - int r = mii_argv_parse(&g_mii, argc, argv, &idx, &flags); - if (r == 0) { - printf("mii: Invalid argument %s, skipped\n", argv[idx]); - } else if (r == -1) - exit(1); - mii_prepare(&g_mii, flags); - mish_prepare(1); - mish_set_command_parameter(MISH_FCC('m','i','i',' '), &g_mii); - printf("MISH_TELNET_PORT = %s\n", getenv("MISH_TELNET_PORT")); - - memset(&win, 0, sizeof(win)); - win.dpy = XOpenDisplay(NULL); - if (!win.dpy) die("Failed to open X display\n"); - { - /* check glx version */ - int glx_major, glx_minor; - if (!glXQueryVersion(win.dpy, &glx_major, &glx_minor)) - die("[X11]: Error: Failed to query OpenGL version\n"); - if ((glx_major == 1 && glx_minor < 3) || (glx_major < 1)) - die("[X11]: Error: Invalid GLX version!\n"); - } - { - /* find and pick matching framebuffer visual */ - int fb_count; - static GLint attr[] = { - GLX_X_RENDERABLE, True, - GLX_DRAWABLE_TYPE, GLX_WINDOW_BIT, - GLX_RENDER_TYPE, GLX_RGBA_BIT, - GLX_X_VISUAL_TYPE, GLX_TRUE_COLOR, - GLX_RED_SIZE, 8, - GLX_GREEN_SIZE, 8, - GLX_BLUE_SIZE, 8, - GLX_ALPHA_SIZE, 8, - GLX_DEPTH_SIZE, 24, - GLX_STENCIL_SIZE, 8, - GLX_DOUBLEBUFFER, True, - None - }; - GLXFBConfig *fbc; - fbc = glXChooseFBConfig(win.dpy, DefaultScreen(win.dpy), attr, &fb_count); - if (!fbc) die("[X11]: Error: failed to retrieve framebuffer configuration\n"); - { - /* pick framebuffer with most samples per pixel */ - int i; - int fb_best = -1, best_num_samples = -1; - for (i = 0; i < fb_count; ++i) { - XVisualInfo *vi = glXGetVisualFromFBConfig(win.dpy, fbc[i]); - if (vi) { - int sample_buffer, samples; - glXGetFBConfigAttrib(win.dpy, fbc[i], GLX_SAMPLE_BUFFERS, &sample_buffer); - glXGetFBConfigAttrib(win.dpy, fbc[i], GLX_SAMPLES, &samples); - if ((fb_best < 0) || (sample_buffer && samples > best_num_samples)) - fb_best = i, best_num_samples = samples; - XFree(vi); - } - } - win.fbc = fbc[fb_best]; - XFree(fbc); - win.vis = glXGetVisualFromFBConfig(win.dpy, win.fbc); - } - } - { - /* create window */ - win.cmap = XCreateColormap(win.dpy, RootWindow(win.dpy, win.vis->screen), win.vis->visual, AllocNone); - win.swa.colormap = win.cmap; - win.swa.background_pixmap = None; - win.swa.border_pixel = 0; - win.swa.event_mask = - ExposureMask | KeyPressMask | KeyReleaseMask | - ButtonPress | ButtonReleaseMask| ButtonMotionMask | - Button1MotionMask | Button3MotionMask | Button4MotionMask | Button5MotionMask| - PointerMotionMask| StructureNotifyMask; - win.win = XCreateWindow(win.dpy, RootWindow(win.dpy, win.vis->screen), 0, 0, - WINDOW_WIDTH, WINDOW_HEIGHT, 0, win.vis->depth, InputOutput, - win.vis->visual, CWBorderPixel|CWColormap|CWEventMask, &win.swa); - if (!win.win) die("[X11]: Failed to create window\n"); - XFree(win.vis); - - char title[128]; - sprintf(title, "MII //e Emulator"); - char *telnet = getenv("MISH_TELNET_PORT"); - if (telnet) - sprintf(title + strlen(title), " (telnet port %s)", telnet); - else - sprintf(title + strlen(title), " (telnet disabled)"); - XStoreName(win.dpy, win.win, title); - XMapWindow(win.dpy, win.win); - win.wm_delete_window = XInternAtom(win.dpy, "WM_DELETE_WINDOW", False); - XSetWMProtocols(win.dpy, win.win, &win.wm_delete_window, 1); - } - { - /* create opengl context */ - typedef GLXContext(*glxCreateContext)(Display*, GLXFBConfig, GLXContext, Bool, const int*); - int(*old_handler)(Display*, XErrorEvent*) = XSetErrorHandler(gl_error_handler); - const char *extensions_str = glXQueryExtensionsString(win.dpy, DefaultScreen(win.dpy)); - glxCreateContext create_context = (glxCreateContext) - glXGetProcAddressARB((const GLubyte*)"glXCreateContextAttribsARB"); - - gl_err = nk_false; - if (!has_extension(extensions_str, "GLX_ARB_create_context") || !create_context) { - fprintf(stdout, "[X11]: glXCreateContextAttribARB() not found...\n"); - fprintf(stdout, "[X11]: ... using old-style GLX context\n"); - glContext = glXCreateNewContext(win.dpy, win.fbc, GLX_RGBA_TYPE, 0, True); - } else { - GLint attr[] = { - GLX_CONTEXT_MAJOR_VERSION_ARB, 3, - GLX_CONTEXT_MINOR_VERSION_ARB, 0, - None - }; - glContext = create_context(win.dpy, win.fbc, 0, True, attr); - XSync(win.dpy, False); - if (gl_err || !glContext) { - /* Could not create GL 3.0 context. Fallback to old 2.x context. - * If a version below 3.0 is requested, implementations will - * return the newest context version compatible with OpenGL - * version less than version 3.0.*/ - attr[1] = 1; attr[3] = 0; - gl_err = nk_false; - fprintf(stdout, "[X11] Failed to create OpenGL 3.0 context\n"); - fprintf(stdout, "[X11] ... using old-style GLX context!\n"); - glContext = create_context(win.dpy, win.fbc, 0, True, attr); - } - } - XSync(win.dpy, False); - XSetErrorHandler(old_handler); - if (gl_err || !glContext) - die("[X11]: Failed to create an OpenGL context\n"); - glXMakeCurrent(win.dpy, win.win, glContext); - } -//extern const unsigned char mii_proggy_data[]; -//extern const unsigned int mii_proggy_size; -extern const unsigned char mii_droid_data[]; -extern const unsigned int mii_droid_size; - - ctx = nk_x11_init(win.dpy, win.win); - /* Load Fonts: if none of these are loaded a default font will be used */ - { - struct nk_font_atlas *atlas; - nk_x11_font_stash_begin(&atlas); - struct nk_font_config cfg = nk_font_config(0); -#if 0 - nk_rune ranges[] = { - 0x0020, 0x007E, /* Ascii */ - 0x00A1, 0x00FF, /* Symbols + Umlaute */ - 0 - }; -#endif - cfg.range = nk_font_default_glyph_ranges(); - cfg.oversample_h = cfg.oversample_v = 1; - cfg.pixel_snap = true; - nk_main_font = nk_font_atlas_add_from_memory(atlas, - (void*)mii_droid_data, mii_droid_size, 20, &cfg); - nk_x11_font_stash_end(); - /*nk_style_load_all_cursors(ctx, atlas->cursors);*/ - if (nk_main_font) - nk_style_set_font(ctx, &nk_main_font->handle); - } - - mii_nuklear_init(&g_mii, ctx); - - bg.r = 0.0f; bg.g = 0.0f; bg.b = 0.0f; bg.a = 1.0f; -// bg.r = 0.8f; bg.g = 0.8f; bg.b = 0.8f; bg.a = 1.0f; - while (running) { - /* Input */ - XEvent evt; - nk_input_begin(ctx); - while (XPending(win.dpy)) { - XNextEvent(win.dpy, &evt); - if (evt.type == ClientMessage) goto cleanup; - if (XFilterEvent(&evt, win.win)) continue; - nk_x11_handle_event(&evt); - } - nk_input_end(ctx); - - /* GUI */ - mii_nuklear(&g_mii, ctx); - - /* Draw */ - XGetWindowAttributes(win.dpy, win.win, &win.attr); - glViewport(0, 0, win.width, win.height); - glClear(GL_COLOR_BUFFER_BIT); - glClearColor(bg.r, bg.g, bg.b, bg.a); - /* IMPORTANT: `nk_x11_render` modifies some global OpenGL state - * with blending, scissor, face culling, depth test and viewport and - * defaults everything back into a default state. - * Make sure to either a.) save and restore or b.) reset your own state after - * rendering the UI. */ - nk_x11_render(NK_ANTI_ALIASING_ON, MAX_VERTEX_BUFFER, MAX_ELEMENT_BUFFER); - glXSwapBuffers(win.dpy, win.win); - } - mii_dispose(&g_mii); -cleanup: - nk_x11_shutdown(); - glXMakeCurrent(win.dpy, 0, 0); - glXDestroyContext(win.dpy, glContext); - XUnmapWindow(win.dpy, win.win); - XFreeColormap(win.dpy, win.cmap); - XDestroyWindow(win.dpy, win.win); - XCloseDisplay(win.dpy); - return 0; - -} diff --git a/nuklear/mii_fonts.c b/nuklear/mii_fonts.c deleted file mode 100644 index c6cc735..0000000 --- a/nuklear/mii_fonts.c +++ /dev/null @@ -1,9 +0,0 @@ - - -#define INCBIN_STYLE INCBIN_STYLE_SNAKE -#define INCBIN_PREFIX mii_ -#include "incbin.h" - -INCBIN(proggy, "fonts/ProggyClean.ttf"); -INCBIN(droid, "fonts/DroidSans.ttf"); - diff --git a/nuklear/mii_nuklear.c b/nuklear/mii_nuklear.c deleted file mode 100644 index 28939dc..0000000 --- a/nuklear/mii_nuklear.c +++ /dev/null @@ -1,500 +0,0 @@ -/* - * mii_nuklear.c - * - * Copyright (C) 2023 Michel Pollet - * - * SPDX-License-Identifier: MIT - */ - -#include -#include -#include -#include -#include -#include - -#include "mii.h" -#include "mii_bank.h" -#include "mii_sw.h" -#include "mii_thread.h" - -#define NK_INCLUDE_FIXED_TYPES -#define NK_INCLUDE_STANDARD_IO -#define NK_INCLUDE_STANDARD_VARARGS -#define NK_INCLUDE_DEFAULT_ALLOCATOR -#define NK_INCLUDE_VERTEX_BUFFER_OUTPUT -#define NK_INCLUDE_FONT_BAKING -#define NK_INCLUDE_DEFAULT_FONT -#define NK_BUTTON_TRIGGER_ON_RELEASE -#include "nuklear.h" -#include "nuklear_xlib_gl3.h" -//#include "stb_image_write.h" -#include -#include - -#define _NK_RGBA(_r,_g,_b,_a) {.r=_r,.g=_g,.b=_b,.a=_a} -static const struct nk_color style[NK_COLOR_COUNT] = { - [NK_COLOR_TEXT] = _NK_RGBA(0, 0, 0, 255), - [NK_COLOR_WINDOW] = _NK_RGBA(175, 175, 175, 255), - [NK_COLOR_HEADER] = _NK_RGBA(175, 175, 175, 255), - [NK_COLOR_BORDER] = _NK_RGBA(0, 0, 0, 255), - [NK_COLOR_BUTTON] = _NK_RGBA(185, 185, 185, 255), - [NK_COLOR_BUTTON_HOVER] = _NK_RGBA(170, 170, 170, 255), - [NK_COLOR_BUTTON_ACTIVE] = _NK_RGBA(160, 160, 160, 255), - [NK_COLOR_TOGGLE] = _NK_RGBA(150, 150, 150, 255), - [NK_COLOR_TOGGLE_HOVER] = _NK_RGBA(120, 120, 120, 255), - [NK_COLOR_TOGGLE_CURSOR] = _NK_RGBA(175, 175, 175, 255), - [NK_COLOR_SELECT] = _NK_RGBA(190, 190, 190, 255), - [NK_COLOR_SELECT_ACTIVE] = _NK_RGBA(175, 175, 175, 255), - [NK_COLOR_SLIDER] = _NK_RGBA(190, 190, 190, 255), - [NK_COLOR_SLIDER_CURSOR] = _NK_RGBA(80, 80, 80, 255), - [NK_COLOR_SLIDER_CURSOR_HOVER] = _NK_RGBA(70, 70, 70, 255), - [NK_COLOR_SLIDER_CURSOR_ACTIVE] = _NK_RGBA(60, 60, 60, 255), - [NK_COLOR_PROPERTY] = _NK_RGBA(175, 175, 175, 255), - [NK_COLOR_EDIT] = _NK_RGBA(150, 150, 150, 255), - [NK_COLOR_EDIT_CURSOR] = _NK_RGBA(0, 0, 0, 255), - [NK_COLOR_COMBO] = _NK_RGBA(175, 175, 175, 255), - [NK_COLOR_CHART] = _NK_RGBA(160, 160, 160, 255), - [NK_COLOR_CHART_COLOR] = _NK_RGBA(45, 45, 45, 255), - [NK_COLOR_CHART_COLOR_HIGHLIGHT] = _NK_RGBA( 255, 0, 0, 255), - [NK_COLOR_SCROLLBAR] = _NK_RGBA(180, 180, 180, 255), - [NK_COLOR_SCROLLBAR_CURSOR] = _NK_RGBA(140, 140, 140, 255), - [NK_COLOR_SCROLLBAR_CURSOR_HOVER] = _NK_RGBA(150, 150, 150, 255), - [NK_COLOR_SCROLLBAR_CURSOR_ACTIVE] = _NK_RGBA(160, 160, 160, 255), - [NK_COLOR_TAB_HEADER] = _NK_RGBA(180, 180, 180, 255), -}; - -static GLuint screen_texture; -static struct nk_image screen_nk; - - -extern struct nk_font *nk_main_font; -struct nk_font *nk_mono_font = NULL; - -extern const unsigned char mii_proggy_data[]; -extern const unsigned int mii_proggy_size; -extern const unsigned char mii_droid_data[]; -extern const unsigned int mii_droid_size; - -void -mii_nuklear_init( - mii_t *mii, - struct nk_context *ctx) -{ - nk_style_from_table(ctx, style); - - { - struct nk_font_atlas *atlas; - nk_x11_font_stash_begin(&atlas); - struct nk_font_config cfg = nk_font_config(0); - #if 0 - nk_rune ranges[] = { - 0x0020, 0x007E, /* Ascii */ - 0x00A1, 0x00FF, /* Symbols + Umlaute */ - 0 - }; - #endif - cfg.range = nk_font_default_glyph_ranges(); - cfg.oversample_h = cfg.oversample_v = 1; - cfg.pixel_snap = true; - struct nk_font *nkf = nk_font_atlas_add_from_memory(atlas, - (void*)mii_proggy_data, mii_proggy_size, 20, &cfg); - nk_x11_font_stash_end(); - nk_mono_font = nkf; - } - - glGenTextures(1, &screen_texture); - glBindTexture(GL_TEXTURE_2D, screen_texture); - - glTexImage2D(GL_TEXTURE_2D, 0, 4, - MII_VRAM_WIDTH, - MII_VRAM_HEIGHT, 0, GL_RGBA, - GL_UNSIGNED_BYTE, - mii->video.pixels); - glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR); - glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR); -// printf("%s texture created %d\n", __func__, screen_texture); -// display opengl error - GLenum err = glGetError(); - if (err != GL_NO_ERROR) { - printf("Error creating texture: %d\n", err); - } - screen_nk = nk_subimage_id( - screen_texture, MII_VRAM_WIDTH, MII_VRAM_HEIGHT, - nk_rect(0, 0, MII_VIDEO_WIDTH, MII_VIDEO_HEIGHT)); - - /* start the CPU/emulator thread */ - mii_thread_start(mii); - -} - - -extern int disk2_debug; -int show_zero_page = 0; - -static void -mii_nuklear_handle_input( - mii_t *mii, - struct nk_context *ctx) -{ - struct nk_input *in = &ctx->input; - if (in->keyboard.text_len) { - // printf("INPUT %d %02x %s\n", - // in->keyboard.text_len, in->keyboard.text[0], in->keyboard.text); - if (in->keyboard.text_len < 4) { - mii_keypress(mii, in->keyboard.text[0]); - } else if (in->keyboard.text_len > 1) { - uint32_t *raw = ((uint32_t *) in->keyboard.text); - for (int ki = 0; ki < in->keyboard.text_len / 4; ki ++) { - uint32_t key = (raw[ki] >> 16) & 0xffff; - uint8_t down = raw[ki] & 0xff; - printf("KEY %04x %s\n", key, down ? "down" : "up"); - if (down) { - if (key == 0xffc9) { // F12 - if (nk_input_is_key_down(in, NK_KEY_CTRL)) { - mii_th_signal_t sig = { - .cmd = SIGNAL_RESET, - .data = nk_input_is_key_down(in, NK_KEY_SHIFT) - }; - mii_th_fifo_write(mii_thread_get_fifo(mii), sig); - printf("RESET\n"); - } - } else if (key == 0xffc8) { // F11 - if (nk_input_is_key_down(in, NK_KEY_CTRL)) { - mii_th_signal_t sig = { - .cmd = SIGNAL_STOP, - }; - mii_th_fifo_write(mii_thread_get_fifo(mii), sig); - printf("STOP\n"); - } - } else if (key == 0xffc7) { // F10 - if (nk_input_is_key_down(in, NK_KEY_CTRL)) { - mii_th_signal_t sig = { - .cmd = SIGNAL_STEP, - }; - mii_th_fifo_write(mii_thread_get_fifo(mii), sig); - printf("STEP\n"); - } - } else if (key == 0xffc6) { // F9 - if (nk_input_is_key_down(in, NK_KEY_CTRL)) { - mii_th_signal_t sig = { - .cmd = SIGNAL_RUN, - }; - mii_th_fifo_write(mii_thread_get_fifo(mii), sig); - printf("RUN\n"); - } - } else if (key == 0xffc2) { // F5 - mii->speed = 1.0; - printf("Speed Normal (1MHz)\n"); - } else if (key == 0xffc3) { // F6 - mii->speed = 4; - printf("Speed Fast (4MHz)\n"); - } - } - if (key == 0xffeb || key == 0xffec) { // super left/right - key -= 0xffeb; - mii_bank_t *bank = &mii->bank[MII_BANK_MAIN]; - uint8_t old = mii_bank_peek(bank, 0xc061 + key); - mii_bank_poke(bank, 0xc061 + key, down ? 0x80 : 0); - if (!!down != !!old) { - printf("Apple %s %s\n", key ? "Open" : "Close", - down ? "down" : "up"); - } - } - } - } - in->keyboard.text_len = 0; - } else { - mii_th_signal_t sig = {.cmd = -1 }; - if (nk_input_is_key_pressed(in, NK_KEY_ENTER)) - sig.data = 0x0d; - else if (nk_input_is_key_pressed(in, NK_KEY_BACKSPACE)) - sig.data = 0x08; - else if (nk_input_is_key_pressed(in, NK_KEY_DEL)) - sig.data = 0x7f; - else if (nk_input_is_key_pressed(in, NK_KEY_UP)) - sig.data = 'K' - 'A' + 1; - else if (nk_input_is_key_pressed(in, NK_KEY_DOWN)) - sig.data = 'J' - 'A' + 1; - else if (nk_input_is_key_pressed(in, NK_KEY_LEFT)) - sig.data = 'H' - 'A' + 1; - else if (nk_input_is_key_pressed(in, NK_KEY_RIGHT)) - sig.data = 'U' - 'A' + 1; - else if (nk_input_is_key_pressed(in, NK_KEY_ESCAPE)) - sig.data = 0x1b; - if (sig.data) { - // mii_th_fifo_write(mii_thread_get_fifo(mii), sig); - // printf("Key %d\n", sig.data); - mii_keypress(mii, sig.data); - } - } -} - -void -mii_nuklear( - mii_t *mii, - struct nk_context *ctx) -{ - if (mii->video.frame_count != mii->video.frame_drawn) { - mii->video.frame_drawn = mii->video.frame_count; - // update texture with new pixels; we only need 192 lines, the others - // are padding for the power of 2 texture - glBindTexture(GL_TEXTURE_2D, screen_texture); - glTexSubImage2D(GL_TEXTURE_2D, 0, 0, 0, - MII_VRAM_WIDTH, - MII_VIDEO_HEIGHT, GL_RGBA, - GL_UNSIGNED_BYTE, - mii->video.pixels); - } - mii_nuklear_handle_input(mii, ctx); - int height = 720 - 8; - int width = MII_VIDEO_WIDTH * (height / (float)MII_VIDEO_HEIGHT); - int xpos = 1280 / 2 - width / 2; - { - struct nk_style *s = &ctx->style; - nk_style_push_color(ctx, &s->window.background, - nk_rgba(0,0,0, 255)); - nk_style_push_style_item(ctx, &s->window.fixed_background, - nk_style_item_color(nk_rgba(0,0,0, 255))); - if (nk_begin(ctx, "Apple //e Enhanced", - nk_rect(xpos, 0, width + 10, height + 20), - NK_WINDOW_NO_SCROLLBAR)) { - nk_layout_row_static(ctx, height, width, 1); - static int was_in = -1; - if (nk_widget_is_hovered(ctx) && mii->mouse.enabled) { - if (was_in != 1) { - was_in = 1; - ctx->input.mouse.grab = 1; - // printf("IN\n"); - } - struct nk_rect bounds = nk_widget_bounds(ctx); - // normalize mouse coordinates - double x = ctx->input.mouse.pos.x - bounds.x; - double y = ctx->input.mouse.pos.y - bounds.y; - // get mouse button state - int button = ctx->input.mouse.buttons[NK_BUTTON_LEFT].down; - // clamp coordinates inside bounds - double vw = bounds.w; - double vh = bounds.h; - double mw = mii->mouse.max_x - mii->mouse.min_x; - double mh = mii->mouse.max_y - mii->mouse.min_y; - mii->mouse.x = mii->mouse.min_x + (x * mw / vw) + 0.5; - mii->mouse.y = mii->mouse.min_y + (y * mh / vh) + 0.5; - mii->mouse.button = button; - // printf("Mouse is %d %d\n", (int)mii->mouse.x, (int)mii->mouse.y); - } else { - if (was_in == 1) { - was_in = 0; - ctx->input.mouse.ungrab = 1; - // printf("OUT\n"); - } - } - nk_image(ctx, screen_nk); - nk_end(ctx); - nk_style_pop_color(ctx); - nk_style_pop_style_item(ctx); - } - } - - struct nk_rect bounds = { .x = 0, .y = 0, .w = xpos, .h = height }; -// nk_window_get_bounds(ctx); - bool in = nk_input_is_mouse_hovering_rect(&ctx->input, bounds); - static bool menu_open = false; - - if (in || menu_open) { - struct nk_style *s = &ctx->style; - nk_style_push_color(ctx, &s->window.background, - nk_rgba(175, 175, 175, 255)); - nk_style_push_style_item(ctx, &s->window.fixed_background, - nk_style_item_color(nk_rgba(175, 175, 175, 255))); - - if (nk_begin(ctx, "Left Bar", - nk_rect(0, 0, xpos, height + 20), - NK_WINDOW_NO_SCROLLBAR)) { -#if 0 - nk_menubar_begin(ctx); - nk_layout_row_begin(ctx, NK_STATIC, 25, 2); - nk_layout_row_push(ctx, 60); - bool menu = false; - if (nk_menu_begin_label(ctx, "MII", NK_TEXT_LEFT, - nk_vec2(140, 200))) { - static size_t prog = 40; - static float slider = 0.5; - static int check = nk_true; - nk_layout_row_dynamic(ctx, 25, 1); - if (nk_menu_item_label(ctx, "Hide", NK_TEXT_LEFT)) - ;//show_menu = nk_false; - if (nk_menu_item_label(ctx, "About", NK_TEXT_LEFT)) - ;//show_app_about = nk_true; - nk_progress(ctx, &prog, 100, NK_MODIFIABLE); - nk_slider_float(ctx, 0.01, &slider, 1.0, 0.05); - nk_checkbox_label(ctx, "Mute", &check); - nk_menu_end(ctx); - menu = true; - } - menu_open = menu; - nk_menubar_end(ctx); - nk_layout_space_end(ctx); -#endif - int rw = xpos - 8; - // nk_layout_row_dynamic(ctx, 4, 1); - // nk_spacing(ctx, 1); - - nk_layout_row_static(ctx, 30, rw, 1); - if (nk_button_label(ctx, "C-RESET")) - mii_reset(mii, false); - if (nk_button_label(ctx, "C-OA-RESET")) - mii_reset(mii, true); -// nk_layout_row_dynamic(ctx, 0, 1); - nk_layout_row_begin(ctx, NK_STATIC, 30, 2); - nk_layout_row_push(ctx, 20); - nk_label(ctx, "V:", NK_TEXT_LEFT); - nk_layout_row_push(ctx, rw - 20 - 4); - const char *video_modes[] = { - "Color", - "Green", - "Amber", - }; - mii->video.color_mode = nk_combo(ctx, - video_modes, NK_LEN(video_modes), - mii->video.color_mode, 25, nk_vec2(80,200)); - nk_layout_space_end(ctx); - - nk_layout_row_begin(ctx, NK_STATIC, 30, 1); - nk_layout_row_push(ctx, rw); - const char *speed[] = { - "1 MHz", - "Slow", - "Fast", - }; - nk_label(ctx, "Speed:", NK_TEXT_LEFT); - nk_layout_row_push(ctx, 100); - int spi = mii->speed > .95 && mii->speed < 1.05 ? 0 : - mii->speed < 0.9 ? 1 : 2; - spi = nk_combo(ctx, - speed, NK_LEN(speed), - spi, 25, nk_vec2(80,200)); - mii->speed = spi == 0 ? 1.0 : spi == 1 ? 0.2 : 4.0; - nk_layout_space_end(ctx); - } - nk_end(ctx); - nk_style_pop_color(ctx); - nk_style_pop_style_item(ctx); - } - if ( 0 && nk_begin(ctx, "Controls", - nk_rect(width, 0, 350, 10 + 192 * 3), - NK_WINDOW_NO_SCROLLBAR)) { - nk_layout_row_static(ctx, 30, 110, 2); - if (nk_button_label(ctx, "C-RESET")) - mii_reset(mii, false); - if (nk_button_label(ctx, "C-OA-RESET")) - mii_reset(mii, true); - #if 0 - if (nk_button_label(ctx, "Screenshot")) { - stbi_write_png("screen.png", - MII_VIDEO_WIDTH, MII_VIDEO_HEIGHT, 4, mii->video.pixels, - MII_VRAM_WIDTH * 4); - printf("Screenshot taken\n"); - } - #endif - nk_layout_row_dynamic(ctx, 30, 4); - nk_label(ctx, "Speed:", NK_TEXT_CENTERED); - if (nk_option_label(ctx, "Slow", mii->speed < 0.9)) mii->speed = 0.2; - if (nk_option_label(ctx, "1 MHz", mii->speed > .95 && mii->speed < 1.05)) mii->speed = 1.0; - if (nk_option_label(ctx, "Fast", mii->speed > 1.1 && mii->speed < 4.1)) mii->speed = 4.0; - - nk_layout_row_dynamic(ctx, 30, 4); - nk_label(ctx, "Video:", NK_TEXT_CENTERED); - if (nk_option_label(ctx, "Color", mii->video.color_mode == MII_VIDEO_COLOR)) - mii->video.color_mode = MII_VIDEO_COLOR; - if (nk_option_label(ctx, "Green", mii->video.color_mode == MII_VIDEO_GREEN)) - mii->video.color_mode = MII_VIDEO_GREEN; - if (nk_option_label(ctx, "Amber", mii->video.color_mode == MII_VIDEO_AMBER)) - mii->video.color_mode = MII_VIDEO_AMBER; - -#if 0 - nk_layout_row_dynamic(ctx, 20, 1); - nk_style_set_font(ctx, &nk_mono_font->handle); - struct nk_color save = ctx->style.window.background; - - ctx->style.window.background = (struct nk_color)_NK_RGBA(0, 0, 0, 255); - struct nk_color fore = (struct nk_color)_NK_RGBA(0, 255, 0, 255); - char label[64]; - mii_dasm_t _dasm = {}; - mii_dasm_t *dasm = &_dasm; - mii_dasm_init(dasm, mii, 0); - dasm->mii = mii; - // display the last few cycles up to the PC - for (int di = 0; di < 3; di++) { - int pci = (mii_trace.idx + MII_PC_LOG_SIZE - 3 + di) % MII_PC_LOG_SIZE; - dasm->pc = mii_trace.log[pci]; - mii_dasm_step(dasm, label, sizeof(label)); - if (di == 2) - label[0] = '*'; - nk_label_colored(ctx, label, NK_TEXT_LEFT, fore); - } - // and the (potentially) next instruction here - mii_dasm_step(dasm, label, sizeof(label)); - nk_label_colored(ctx, label, NK_TEXT_LEFT, fore); - - sprintf(label, "A:%02X X:%02X Y:%02X S:%02X", - mii->cpu.A, mii->cpu.X, mii->cpu.Y, mii->cpu.S); - nk_label_colored(ctx, label, NK_TEXT_CENTERED, fore); - char n[] = {'C','Z','I','D','B','V','N'}; - label[0] = 0; - sprintf(label, "%04x ", mii->cpu.PC); - for (int i = 0; i < 7; i++) - sprintf(label + strlen(label), "%c%d ", n[i], - mii->cpu.P.P[i]); - nk_label_colored(ctx, label, NK_TEXT_CENTERED, fore); - - ctx->style.window.background = save; - nk_style_set_font(ctx, &nk_main_font->handle); -#endif - nk_layout_row_static(ctx, 30, 30, 3); - - if (nk_button_symbol(ctx, NK_SYMBOL_RECT_SOLID)) { - mii_th_fifo_write(mii_thread_get_fifo(mii), (mii_th_signal_t){.cmd = SIGNAL_STOP}); - } - if (nk_button_symbol(ctx, NK_SYMBOL_PLUS)) { - mii_th_fifo_write(mii_thread_get_fifo(mii), (mii_th_signal_t){.cmd = SIGNAL_STEP}); - } - if (nk_button_symbol(ctx, NK_SYMBOL_TRIANGLE_RIGHT)) { - mii_th_fifo_write(mii_thread_get_fifo(mii), (mii_th_signal_t){.cmd = SIGNAL_RUN}); - } - nk_layout_row_dynamic(ctx, 20, 3); - { - char label[32]; - sprintf(label, "CPU: %.1fMHz", mii->speed_current); - nk_label(ctx, label, NK_TEXT_CENTERED); - } -#if 0 - nk_layout_row_dynamic(ctx, 20, 2); - nk_checkbox_label(ctx, "Disk II Debug", &disk2_debug); - nk_checkbox_label(ctx, "Slowmo", &mii_SLOW); -// nk_checkbox_label(ctx, "Zero Page", &show_zero_page); -#endif - nk_end(ctx); - } - if (show_zero_page) { - if (nk_begin(ctx, "Zero Page", - nk_rect(0, 10 + 192 * 3, 600, 10 + 16 * 20), - 0 )) { - nk_layout_row_dynamic(ctx, 20, 1); - nk_style_set_font(ctx, &nk_mono_font->handle); - struct nk_color fore = (struct nk_color)_NK_RGBA(0, 255, 0, 255); - uint8_t *zp = mii->bank[0].mem; - char label[128]; - for (int ri = 0; ri < (256 / 16); ri++) { - sprintf(label, "%02x: ", ri * 16); - for (int ci = 0; ci < 16; ci++) { - sprintf(label + strlen(label), "%02X ", - zp[ri * 16 + ci]); - } - nk_label_colored(ctx, label, NK_TEXT_LEFT, fore); - } - nk_style_set_font(ctx, &nk_main_font->handle); - nk_end(ctx); - } - } -} diff --git a/nuklear/mii_stb_implement.c b/nuklear/mii_stb_implement.c deleted file mode 100644 index 32fa1f8..0000000 --- a/nuklear/mii_stb_implement.c +++ /dev/null @@ -1,22 +0,0 @@ -/* - * mii_stb.h - * - * Copyright (C) 2023 Michel Pollet - * - * SPDX-License-Identifier: MIT - */ - - -#define NK_INCLUDE_FIXED_TYPES -#define NK_INCLUDE_STANDARD_IO -#define NK_INCLUDE_STANDARD_VARARGS -#define NK_INCLUDE_DEFAULT_ALLOCATOR -#define NK_INCLUDE_VERTEX_BUFFER_OUTPUT -#define NK_INCLUDE_FONT_BAKING -#define NK_INCLUDE_DEFAULT_FONT -#define NK_BUTTON_TRIGGER_ON_RELEASE -#define NK_IMPLEMENTATION -#include "nuklear.h" - -#define STB_IMAGE_WRITE_IMPLEMENTATION -#include "stb_image_write.h" diff --git a/nuklear/nuklear.h b/nuklear/nuklear.h deleted file mode 100644 index 4012fda..0000000 --- a/nuklear/nuklear.h +++ /dev/null @@ -1,30034 +0,0 @@ -/* -/// # Nuklear -/// ![](https://cloud.githubusercontent.com/assets/8057201/11761525/ae06f0ca-a0c6-11e5-819d-5610b25f6ef4.gif) -/// -/// ## Contents -/// 1. About section -/// 2. Highlights section -/// 3. Features section -/// 4. Usage section -/// 1. Flags section -/// 2. Constants section -/// 3. Dependencies section -/// 5. Example section -/// 6. API section -/// 1. Context section -/// 2. Input section -/// 3. Drawing section -/// 4. Window section -/// 5. Layouting section -/// 6. Groups section -/// 7. Tree section -/// 8. Properties section -/// 7. License section -/// 8. Changelog section -/// 9. Gallery section -/// 10. Credits section -/// -/// ## About -/// This is a minimal state immediate mode graphical user interface toolkit -/// written in ANSI C and licensed under public domain. It was designed as a simple -/// embeddable user interface for application and does not have any dependencies, -/// a default renderbackend or OS window and input handling but instead provides a very modular -/// library approach by using simple input state for input and draw -/// commands describing primitive shapes as output. So instead of providing a -/// layered library that tries to abstract over a number of platform and -/// render backends it only focuses on the actual UI. -/// -/// ## Highlights -/// - Graphical user interface toolkit -/// - Single header library -/// - Written in C89 (a.k.a. ANSI C or ISO C90) -/// - Small codebase (~18kLOC) -/// - Focus on portability, efficiency and simplicity -/// - No dependencies (not even the standard library if not wanted) -/// - Fully skinnable and customizable -/// - Low memory footprint with total memory control if needed or wanted -/// - UTF-8 support -/// - No global or hidden state -/// - Customizable library modules (you can compile and use only what you need) -/// - Optional font baker and vertex buffer output -/// - [Code available on github](https://github.com/Immediate-Mode-UI/Nuklear/) -/// -/// ## Features -/// - Absolutely no platform dependent code -/// - Memory management control ranging from/to -/// - Ease of use by allocating everything from standard library -/// - Control every byte of memory inside the library -/// - Font handling control ranging from/to -/// - Use your own font implementation for everything -/// - Use this libraries internal font baking and handling API -/// - Drawing output control ranging from/to -/// - Simple shapes for more high level APIs which already have drawing capabilities -/// - Hardware accessible anti-aliased vertex buffer output -/// - Customizable colors and properties ranging from/to -/// - Simple changes to color by filling a simple color table -/// - Complete control with ability to use skinning to decorate widgets -/// - Bendable UI library with widget ranging from/to -/// - Basic widgets like buttons, checkboxes, slider, ... -/// - Advanced widget like abstract comboboxes, contextual menus,... -/// - Compile time configuration to only compile what you need -/// - Subset which can be used if you do not want to link or use the standard library -/// - Can be easily modified to only update on user input instead of frame updates -/// -/// ## Usage -/// This library is self contained in one single header file and can be used either -/// in header only mode or in implementation mode. The header only mode is used -/// by default when included and allows including this header in other headers -/// and does not contain the actual implementation.

    -/// -/// The implementation mode requires to define the preprocessor macro -/// NK_IMPLEMENTATION in *one* .c/.cpp file before #including this file, e.g.: -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~C -/// #define NK_IMPLEMENTATION -/// #include "nuklear.h" -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Also optionally define the symbols listed in the section "OPTIONAL DEFINES" -/// below in header and implementation mode if you want to use additional functionality -/// or need more control over the library. -/// -/// !!! WARNING -/// Every time nuklear is included define the same compiler flags. This very important not doing so could lead to compiler errors or even worse stack corruptions. -/// -/// ### Flags -/// Flag | Description -/// --------------------------------|------------------------------------------ -/// NK_PRIVATE | If defined declares all functions as static, so they can only be accessed inside the file that contains the implementation -/// NK_INCLUDE_FIXED_TYPES | If defined it will include header `` for fixed sized types otherwise nuklear tries to select the correct type. If that fails it will throw a compiler error and you have to select the correct types yourself. -/// NK_INCLUDE_DEFAULT_ALLOCATOR | If defined it will include header `` and provide additional functions to use this library without caring for memory allocation control and therefore ease memory management. -/// NK_INCLUDE_STANDARD_IO | If defined it will include header `` and provide additional functions depending on file loading. -/// NK_INCLUDE_STANDARD_VARARGS | If defined it will include header and provide additional functions depending on file loading. -/// NK_INCLUDE_STANDARD_BOOL | If defined it will include header `` for nk_bool otherwise nuklear defines nk_bool as int. -/// NK_INCLUDE_VERTEX_BUFFER_OUTPUT | Defining this adds a vertex draw command list backend to this library, which allows you to convert queue commands into vertex draw commands. This is mainly if you need a hardware accessible format for OpenGL, DirectX, Vulkan, Metal,... -/// NK_INCLUDE_FONT_BAKING | Defining this adds `stb_truetype` and `stb_rect_pack` implementation to this library and provides font baking and rendering. If you already have font handling or do not want to use this font handler you don't have to define it. -/// NK_INCLUDE_DEFAULT_FONT | Defining this adds the default font: ProggyClean.ttf into this library which can be loaded into a font atlas and allows using this library without having a truetype font -/// NK_INCLUDE_COMMAND_USERDATA | Defining this adds a userdata pointer into each command. Can be useful for example if you want to provide custom shaders depending on the used widget. Can be combined with the style structures. -/// NK_BUTTON_TRIGGER_ON_RELEASE | Different platforms require button clicks occurring either on buttons being pressed (up to down) or released (down to up). By default this library will react on buttons being pressed, but if you define this it will only trigger if a button is released. -/// NK_ZERO_COMMAND_MEMORY | Defining this will zero out memory for each drawing command added to a drawing queue (inside nk_command_buffer_push). Zeroing command memory is very useful for fast checking (using memcmp) if command buffers are equal and avoid drawing frames when nothing on screen has changed since previous frame. -/// NK_UINT_DRAW_INDEX | Defining this will set the size of vertex index elements when using NK_VERTEX_BUFFER_OUTPUT to 32bit instead of the default of 16bit -/// NK_KEYSTATE_BASED_INPUT | Define this if your backend uses key state for each frame rather than key press/release events -/// -/// !!! WARNING -/// The following flags will pull in the standard C library: -/// - NK_INCLUDE_DEFAULT_ALLOCATOR -/// - NK_INCLUDE_STANDARD_IO -/// - NK_INCLUDE_STANDARD_VARARGS -/// -/// !!! WARNING -/// The following flags if defined need to be defined for both header and implementation: -/// - NK_INCLUDE_FIXED_TYPES -/// - NK_INCLUDE_DEFAULT_ALLOCATOR -/// - NK_INCLUDE_STANDARD_VARARGS -/// - NK_INCLUDE_STANDARD_BOOL -/// - NK_INCLUDE_VERTEX_BUFFER_OUTPUT -/// - NK_INCLUDE_FONT_BAKING -/// - NK_INCLUDE_DEFAULT_FONT -/// - NK_INCLUDE_STANDARD_VARARGS -/// - NK_INCLUDE_COMMAND_USERDATA -/// - NK_UINT_DRAW_INDEX -/// -/// ### Constants -/// Define | Description -/// --------------------------------|--------------------------------------- -/// NK_BUFFER_DEFAULT_INITIAL_SIZE | Initial buffer size allocated by all buffers while using the default allocator functions included by defining NK_INCLUDE_DEFAULT_ALLOCATOR. If you don't want to allocate the default 4k memory then redefine it. -/// NK_MAX_NUMBER_BUFFER | Maximum buffer size for the conversion buffer between float and string Under normal circumstances this should be more than sufficient. -/// NK_INPUT_MAX | Defines the max number of bytes which can be added as text input in one frame. Under normal circumstances this should be more than sufficient. -/// -/// !!! WARNING -/// The following constants if defined need to be defined for both header and implementation: -/// - NK_MAX_NUMBER_BUFFER -/// - NK_BUFFER_DEFAULT_INITIAL_SIZE -/// - NK_INPUT_MAX -/// -/// ### Dependencies -/// Function | Description -/// ------------|--------------------------------------------------------------- -/// NK_ASSERT | If you don't define this, nuklear will use with assert(). -/// NK_MEMSET | You can define this to 'memset' or your own memset implementation replacement. If not nuklear will use its own version. -/// NK_MEMCPY | You can define this to 'memcpy' or your own memcpy implementation replacement. If not nuklear will use its own version. -/// NK_INV_SQRT | You can define this to your own inverse sqrt implementation replacement. If not nuklear will use its own slow and not highly accurate version. -/// NK_SIN | You can define this to 'sinf' or your own sine implementation replacement. If not nuklear will use its own approximation implementation. -/// NK_COS | You can define this to 'cosf' or your own cosine implementation replacement. If not nuklear will use its own approximation implementation. -/// NK_STRTOD | You can define this to `strtod` or your own string to double conversion implementation replacement. If not defined nuklear will use its own imprecise and possibly unsafe version (does not handle nan or infinity!). -/// NK_DTOA | You can define this to `dtoa` or your own double to string conversion implementation replacement. If not defined nuklear will use its own imprecise and possibly unsafe version (does not handle nan or infinity!). -/// NK_VSNPRINTF| If you define `NK_INCLUDE_STANDARD_VARARGS` as well as `NK_INCLUDE_STANDARD_IO` and want to be safe define this to `vsnprintf` on compilers supporting later versions of C or C++. By default nuklear will check for your stdlib version in C as well as compiler version in C++. if `vsnprintf` is available it will define it to `vsnprintf` directly. If not defined and if you have older versions of C or C++ it will be defined to `vsprintf` which is unsafe. -/// -/// !!! WARNING -/// The following dependencies will pull in the standard C library if not redefined: -/// - NK_ASSERT -/// -/// !!! WARNING -/// The following dependencies if defined need to be defined for both header and implementation: -/// - NK_ASSERT -/// -/// !!! WARNING -/// The following dependencies if defined need to be defined only for the implementation part: -/// - NK_MEMSET -/// - NK_MEMCPY -/// - NK_SQRT -/// - NK_SIN -/// - NK_COS -/// - NK_STRTOD -/// - NK_DTOA -/// - NK_VSNPRINTF -/// -/// ## Example -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// // init gui state -/// enum {EASY, HARD}; -/// static int op = EASY; -/// static float value = 0.6f; -/// static int i = 20; -/// struct nk_context ctx; -/// -/// nk_init_fixed(&ctx, calloc(1, MAX_MEMORY), MAX_MEMORY, &font); -/// if (nk_begin(&ctx, "Show", nk_rect(50, 50, 220, 220), -/// NK_WINDOW_BORDER|NK_WINDOW_MOVABLE|NK_WINDOW_CLOSABLE)) { -/// // fixed widget pixel width -/// nk_layout_row_static(&ctx, 30, 80, 1); -/// if (nk_button_label(&ctx, "button")) { -/// // event handling -/// } -/// -/// // fixed widget window ratio width -/// nk_layout_row_dynamic(&ctx, 30, 2); -/// if (nk_option_label(&ctx, "easy", op == EASY)) op = EASY; -/// if (nk_option_label(&ctx, "hard", op == HARD)) op = HARD; -/// -/// // custom widget pixel width -/// nk_layout_row_begin(&ctx, NK_STATIC, 30, 2); -/// { -/// nk_layout_row_push(&ctx, 50); -/// nk_label(&ctx, "Volume:", NK_TEXT_LEFT); -/// nk_layout_row_push(&ctx, 110); -/// nk_slider_float(&ctx, 0, &value, 1.0f, 0.1f); -/// } -/// nk_layout_row_end(&ctx); -/// } -/// nk_end(&ctx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// ![](https://cloud.githubusercontent.com/assets/8057201/10187981/584ecd68-675c-11e5-897c-822ef534a876.png) -/// -/// ## API -/// -*/ -#ifndef NK_SINGLE_FILE - #define NK_SINGLE_FILE -#endif - -#ifndef NK_NUKLEAR_H_ -#define NK_NUKLEAR_H_ - -#ifdef __cplusplus -extern "C" { -#endif -/* - * ============================================================== - * - * CONSTANTS - * - * =============================================================== - */ -#define NK_UNDEFINED (-1.0f) -#define NK_UTF_INVALID 0xFFFD /* internal invalid utf8 rune */ -#define NK_UTF_SIZE 4 /* describes the number of bytes a glyph consists of*/ -#ifndef NK_INPUT_MAX - #define NK_INPUT_MAX 16 -#endif -#ifndef NK_MAX_NUMBER_BUFFER - #define NK_MAX_NUMBER_BUFFER 64 -#endif -#ifndef NK_SCROLLBAR_HIDING_TIMEOUT - #define NK_SCROLLBAR_HIDING_TIMEOUT 4.0f -#endif -/* - * ============================================================== - * - * HELPER - * - * =============================================================== - */ -#ifndef NK_API - #ifdef NK_PRIVATE - #if (defined(__STDC_VERSION__) && (__STDC_VERSION__ >= 199409L)) - #define NK_API static inline - #elif defined(__cplusplus) - #define NK_API static inline - #else - #define NK_API static - #endif - #else - #define NK_API extern - #endif -#endif -#ifndef NK_LIB - #ifdef NK_SINGLE_FILE - #define NK_LIB static - #else - #define NK_LIB extern - #endif -#endif - -#define NK_INTERN static -#define NK_STORAGE static -#define NK_GLOBAL static - -#define NK_FLAG(x) (1 << (x)) -#define NK_STRINGIFY(x) #x -#define NK_MACRO_STRINGIFY(x) NK_STRINGIFY(x) -#define NK_STRING_JOIN_IMMEDIATE(arg1, arg2) arg1 ## arg2 -#define NK_STRING_JOIN_DELAY(arg1, arg2) NK_STRING_JOIN_IMMEDIATE(arg1, arg2) -#define NK_STRING_JOIN(arg1, arg2) NK_STRING_JOIN_DELAY(arg1, arg2) - -#ifdef _MSC_VER - #define NK_UNIQUE_NAME(name) NK_STRING_JOIN(name,__COUNTER__) -#else - #define NK_UNIQUE_NAME(name) NK_STRING_JOIN(name,__LINE__) -#endif - -#ifndef NK_STATIC_ASSERT - #define NK_STATIC_ASSERT(exp) typedef char NK_UNIQUE_NAME(_dummy_array)[(exp)?1:-1] -#endif - -#ifndef NK_FILE_LINE -#ifdef _MSC_VER - #define NK_FILE_LINE __FILE__ ":" NK_MACRO_STRINGIFY(__COUNTER__) -#else - #define NK_FILE_LINE __FILE__ ":" NK_MACRO_STRINGIFY(__LINE__) -#endif -#endif - -#define NK_MIN(a,b) ((a) < (b) ? (a) : (b)) -#define NK_MAX(a,b) ((a) < (b) ? (b) : (a)) -#define NK_CLAMP(i,v,x) (NK_MAX(NK_MIN(v,x), i)) - -#ifdef NK_INCLUDE_STANDARD_VARARGS - #include - #if defined(_MSC_VER) && (_MSC_VER >= 1600) /* VS 2010 and above */ - #include - #define NK_PRINTF_FORMAT_STRING _Printf_format_string_ - #else - #define NK_PRINTF_FORMAT_STRING - #endif - #if defined(__GNUC__) - #define NK_PRINTF_VARARG_FUNC(fmtargnumber) __attribute__((format(__printf__, fmtargnumber, fmtargnumber+1))) - #define NK_PRINTF_VALIST_FUNC(fmtargnumber) __attribute__((format(__printf__, fmtargnumber, 0))) - #else - #define NK_PRINTF_VARARG_FUNC(fmtargnumber) - #define NK_PRINTF_VALIST_FUNC(fmtargnumber) - #endif -#endif - -/* - * =============================================================== - * - * BASIC - * - * =============================================================== - */ -#ifdef NK_INCLUDE_FIXED_TYPES - #include - #define NK_INT8 int8_t - #define NK_UINT8 uint8_t - #define NK_INT16 int16_t - #define NK_UINT16 uint16_t - #define NK_INT32 int32_t - #define NK_UINT32 uint32_t - #define NK_SIZE_TYPE uintptr_t - #define NK_POINTER_TYPE uintptr_t -#else - #ifndef NK_INT8 - #define NK_INT8 signed char - #endif - #ifndef NK_UINT8 - #define NK_UINT8 unsigned char - #endif - #ifndef NK_INT16 - #define NK_INT16 signed short - #endif - #ifndef NK_UINT16 - #define NK_UINT16 unsigned short - #endif - #ifndef NK_INT32 - #if defined(_MSC_VER) - #define NK_INT32 __int32 - #else - #define NK_INT32 signed int - #endif - #endif - #ifndef NK_UINT32 - #if defined(_MSC_VER) - #define NK_UINT32 unsigned __int32 - #else - #define NK_UINT32 unsigned int - #endif - #endif - #ifndef NK_SIZE_TYPE - #if defined(_WIN64) && defined(_MSC_VER) - #define NK_SIZE_TYPE unsigned __int64 - #elif (defined(_WIN32) || defined(WIN32)) && defined(_MSC_VER) - #define NK_SIZE_TYPE unsigned __int32 - #elif defined(__GNUC__) || defined(__clang__) - #if defined(__x86_64__) || defined(__ppc64__) || defined(__aarch64__) - #define NK_SIZE_TYPE unsigned long - #else - #define NK_SIZE_TYPE unsigned int - #endif - #else - #define NK_SIZE_TYPE unsigned long - #endif - #endif - #ifndef NK_POINTER_TYPE - #if defined(_WIN64) && defined(_MSC_VER) - #define NK_POINTER_TYPE unsigned __int64 - #elif (defined(_WIN32) || defined(WIN32)) && defined(_MSC_VER) - #define NK_POINTER_TYPE unsigned __int32 - #elif defined(__GNUC__) || defined(__clang__) - #if defined(__x86_64__) || defined(__ppc64__) || defined(__aarch64__) - #define NK_POINTER_TYPE unsigned long - #else - #define NK_POINTER_TYPE unsigned int - #endif - #else - #define NK_POINTER_TYPE unsigned long - #endif - #endif -#endif - -#ifndef NK_BOOL - #ifdef NK_INCLUDE_STANDARD_BOOL - #include - #define NK_BOOL bool - #else - #define NK_BOOL int /* could be char, use int for drop-in replacement backwards compatibility */ - #endif -#endif - -typedef NK_INT8 nk_char; -typedef NK_UINT8 nk_uchar; -typedef NK_UINT8 nk_byte; -typedef NK_INT16 nk_short; -typedef NK_UINT16 nk_ushort; -typedef NK_INT32 nk_int; -typedef NK_UINT32 nk_uint; -typedef NK_SIZE_TYPE nk_size; -typedef NK_POINTER_TYPE nk_ptr; -typedef NK_BOOL nk_bool; - -typedef nk_uint nk_hash; -typedef nk_uint nk_flags; -typedef nk_uint nk_rune; - -/* Make sure correct type size: - * This will fire with a negative subscript error if the type sizes - * are set incorrectly by the compiler, and compile out if not */ -NK_STATIC_ASSERT(sizeof(nk_short) == 2); -NK_STATIC_ASSERT(sizeof(nk_ushort) == 2); -NK_STATIC_ASSERT(sizeof(nk_uint) == 4); -NK_STATIC_ASSERT(sizeof(nk_int) == 4); -NK_STATIC_ASSERT(sizeof(nk_byte) == 1); -NK_STATIC_ASSERT(sizeof(nk_flags) >= 4); -NK_STATIC_ASSERT(sizeof(nk_rune) >= 4); -NK_STATIC_ASSERT(sizeof(nk_size) >= sizeof(void*)); -NK_STATIC_ASSERT(sizeof(nk_ptr) >= sizeof(void*)); -#ifdef NK_INCLUDE_STANDARD_BOOL -NK_STATIC_ASSERT(sizeof(nk_bool) == sizeof(bool)); -#else -NK_STATIC_ASSERT(sizeof(nk_bool) >= 2); -#endif - -/* ============================================================================ - * - * API - * - * =========================================================================== */ -struct nk_buffer; -struct nk_allocator; -struct nk_command_buffer; -struct nk_draw_command; -struct nk_convert_config; -struct nk_style_item; -struct nk_text_edit; -struct nk_draw_list; -struct nk_user_font; -struct nk_panel; -struct nk_context; -struct nk_draw_vertex_layout_element; -struct nk_style_button; -struct nk_style_toggle; -struct nk_style_selectable; -struct nk_style_slide; -struct nk_style_progress; -struct nk_style_scrollbar; -struct nk_style_edit; -struct nk_style_property; -struct nk_style_chart; -struct nk_style_combo; -struct nk_style_tab; -struct nk_style_window_header; -struct nk_style_window; - -enum {nk_false, nk_true}; -struct nk_color {nk_byte r,g,b,a;}; -struct nk_colorf {float r,g,b,a;}; -struct nk_vec2 {float x,y;}; -struct nk_vec2i {short x, y;}; -struct nk_rect {float x,y,w,h;}; -struct nk_recti {short x,y,w,h;}; -typedef char nk_glyph[NK_UTF_SIZE]; -typedef union {void *ptr; int id;} nk_handle; -struct nk_image {nk_handle handle; nk_handle unit; nk_ushort w, h; nk_ushort region[4];}; -struct nk_nine_slice {struct nk_image img; nk_ushort l, t, r, b;}; -struct nk_cursor {struct nk_image img; struct nk_vec2 size, offset;}; -struct nk_scroll {nk_uint x, y;}; - -enum nk_heading {NK_UP, NK_RIGHT, NK_DOWN, NK_LEFT}; -enum nk_button_behavior {NK_BUTTON_DEFAULT, NK_BUTTON_REPEATER}; -enum nk_modify {NK_FIXED = nk_false, NK_MODIFIABLE = nk_true}; -enum nk_orientation {NK_VERTICAL, NK_HORIZONTAL}; -enum nk_collapse_states {NK_MINIMIZED = nk_false, NK_MAXIMIZED = nk_true}; -enum nk_show_states {NK_HIDDEN = nk_false, NK_SHOWN = nk_true}; -enum nk_chart_type {NK_CHART_LINES, NK_CHART_COLUMN, NK_CHART_MAX}; -enum nk_chart_event {NK_CHART_HOVERING = 0x01, NK_CHART_CLICKED = 0x02}; -enum nk_color_format {NK_RGB, NK_RGBA}; -enum nk_popup_type {NK_POPUP_STATIC, NK_POPUP_DYNAMIC}; -enum nk_layout_format {NK_DYNAMIC, NK_STATIC}; -enum nk_tree_type {NK_TREE_NODE, NK_TREE_TAB}; - -typedef void*(*nk_plugin_alloc)(nk_handle, void *old, nk_size); -typedef void (*nk_plugin_free)(nk_handle, void *old); -typedef nk_bool(*nk_plugin_filter)(const struct nk_text_edit*, nk_rune unicode); -typedef void(*nk_plugin_paste)(nk_handle, struct nk_text_edit*); -typedef void(*nk_plugin_copy)(nk_handle, const char*, int len); - -struct nk_allocator { - nk_handle userdata; - nk_plugin_alloc alloc; - nk_plugin_free free; -}; -enum nk_symbol_type { - NK_SYMBOL_NONE, - NK_SYMBOL_X, - NK_SYMBOL_UNDERSCORE, - NK_SYMBOL_CIRCLE_SOLID, - NK_SYMBOL_CIRCLE_OUTLINE, - NK_SYMBOL_RECT_SOLID, - NK_SYMBOL_RECT_OUTLINE, - NK_SYMBOL_TRIANGLE_UP, - NK_SYMBOL_TRIANGLE_DOWN, - NK_SYMBOL_TRIANGLE_LEFT, - NK_SYMBOL_TRIANGLE_RIGHT, - NK_SYMBOL_PLUS, - NK_SYMBOL_MINUS, - NK_SYMBOL_MAX -}; -/* ============================================================================= - * - * CONTEXT - * - * =============================================================================*/ -/*/// ### Context -/// Contexts are the main entry point and the majestro of nuklear and contain all required state. -/// They are used for window, memory, input, style, stack, commands and time management and need -/// to be passed into all nuklear GUI specific functions. -/// -/// #### Usage -/// To use a context it first has to be initialized which can be achieved by calling -/// one of either `nk_init_default`, `nk_init_fixed`, `nk_init`, `nk_init_custom`. -/// Each takes in a font handle and a specific way of handling memory. Memory control -/// hereby ranges from standard library to just specifying a fixed sized block of memory -/// which nuklear has to manage itself from. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// struct nk_context ctx; -/// nk_init_xxx(&ctx, ...); -/// while (1) { -/// // [...] -/// nk_clear(&ctx); -/// } -/// nk_free(&ctx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// #### Reference -/// Function | Description -/// --------------------|------------------------------------------------------- -/// __nk_init_default__ | Initializes context with standard library memory allocation (malloc,free) -/// __nk_init_fixed__ | Initializes context from single fixed size memory block -/// __nk_init__ | Initializes context with memory allocator callbacks for alloc and free -/// __nk_init_custom__ | Initializes context from two buffers. One for draw commands the other for window/panel/table allocations -/// __nk_clear__ | Called at the end of the frame to reset and prepare the context for the next frame -/// __nk_free__ | Shutdown and free all memory allocated inside the context -/// __nk_set_user_data__| Utility function to pass user data to draw command - */ -#ifdef NK_INCLUDE_DEFAULT_ALLOCATOR -/*/// #### nk_init_default -/// Initializes a `nk_context` struct with a default standard library allocator. -/// Should be used if you don't want to be bothered with memory management in nuklear. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// nk_bool nk_init_default(struct nk_context *ctx, const struct nk_user_font *font); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|--------------------------------------------------------------- -/// __ctx__ | Must point to an either stack or heap allocated `nk_context` struct -/// __font__ | Must point to a previously initialized font handle for more info look at font documentation -/// -/// Returns either `false(0)` on failure or `true(1)` on success. -/// -*/ -NK_API nk_bool nk_init_default(struct nk_context*, const struct nk_user_font*); -#endif -/*/// #### nk_init_fixed -/// Initializes a `nk_context` struct from single fixed size memory block -/// Should be used if you want complete control over nuklear's memory management. -/// Especially recommended for system with little memory or systems with virtual memory. -/// For the later case you can just allocate for example 16MB of virtual memory -/// and only the required amount of memory will actually be committed. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// nk_bool nk_init_fixed(struct nk_context *ctx, void *memory, nk_size size, const struct nk_user_font *font); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// !!! Warning -/// make sure the passed memory block is aligned correctly for `nk_draw_commands`. -/// -/// Parameter | Description -/// ------------|-------------------------------------------------------------- -/// __ctx__ | Must point to an either stack or heap allocated `nk_context` struct -/// __memory__ | Must point to a previously allocated memory block -/// __size__ | Must contain the total size of __memory__ -/// __font__ | Must point to a previously initialized font handle for more info look at font documentation -/// -/// Returns either `false(0)` on failure or `true(1)` on success. -*/ -NK_API nk_bool nk_init_fixed(struct nk_context*, void *memory, nk_size size, const struct nk_user_font*); -/*/// #### nk_init -/// Initializes a `nk_context` struct with memory allocation callbacks for nuklear to allocate -/// memory from. Used internally for `nk_init_default` and provides a kitchen sink allocation -/// interface to nuklear. Can be useful for cases like monitoring memory consumption. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// nk_bool nk_init(struct nk_context *ctx, struct nk_allocator *alloc, const struct nk_user_font *font); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|--------------------------------------------------------------- -/// __ctx__ | Must point to an either stack or heap allocated `nk_context` struct -/// __alloc__ | Must point to a previously allocated memory allocator -/// __font__ | Must point to a previously initialized font handle for more info look at font documentation -/// -/// Returns either `false(0)` on failure or `true(1)` on success. -*/ -NK_API nk_bool nk_init(struct nk_context*, struct nk_allocator*, const struct nk_user_font*); -/*/// #### nk_init_custom -/// Initializes a `nk_context` struct from two different either fixed or growing -/// buffers. The first buffer is for allocating draw commands while the second buffer is -/// used for allocating windows, panels and state tables. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// nk_bool nk_init_custom(struct nk_context *ctx, struct nk_buffer *cmds, struct nk_buffer *pool, const struct nk_user_font *font); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|--------------------------------------------------------------- -/// __ctx__ | Must point to an either stack or heap allocated `nk_context` struct -/// __cmds__ | Must point to a previously initialized memory buffer either fixed or dynamic to store draw commands into -/// __pool__ | Must point to a previously initialized memory buffer either fixed or dynamic to store windows, panels and tables -/// __font__ | Must point to a previously initialized font handle for more info look at font documentation -/// -/// Returns either `false(0)` on failure or `true(1)` on success. -*/ -NK_API nk_bool nk_init_custom(struct nk_context*, struct nk_buffer *cmds, struct nk_buffer *pool, const struct nk_user_font*); -/*/// #### nk_clear -/// Resets the context state at the end of the frame. This includes mostly -/// garbage collector tasks like removing windows or table not called and therefore -/// used anymore. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_clear(struct nk_context *ctx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to a previously initialized `nk_context` struct -*/ -NK_API void nk_clear(struct nk_context*); -/*/// #### nk_free -/// Frees all memory allocated by nuklear. Not needed if context was -/// initialized with `nk_init_fixed`. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_free(struct nk_context *ctx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to a previously initialized `nk_context` struct -*/ -NK_API void nk_free(struct nk_context*); -#ifdef NK_INCLUDE_COMMAND_USERDATA -/*/// #### nk_set_user_data -/// Sets the currently passed userdata passed down into each draw command. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_set_user_data(struct nk_context *ctx, nk_handle data); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|-------------------------------------------------------------- -/// __ctx__ | Must point to a previously initialized `nk_context` struct -/// __data__ | Handle with either pointer or index to be passed into every draw commands -*/ -NK_API void nk_set_user_data(struct nk_context*, nk_handle handle); -#endif -/* ============================================================================= - * - * INPUT - * - * =============================================================================*/ -/*/// ### Input -/// The input API is responsible for holding the current input state composed of -/// mouse, key and text input states. -/// It is worth noting that no direct OS or window handling is done in nuklear. -/// Instead all input state has to be provided by platform specific code. This on one hand -/// expects more work from the user and complicates usage but on the other hand -/// provides simple abstraction over a big number of platforms, libraries and other -/// already provided functionality. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// nk_input_begin(&ctx); -/// while (GetEvent(&evt)) { -/// if (evt.type == MOUSE_MOVE) -/// nk_input_motion(&ctx, evt.motion.x, evt.motion.y); -/// else if (evt.type == [...]) { -/// // [...] -/// } -/// } nk_input_end(&ctx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// #### Usage -/// Input state needs to be provided to nuklear by first calling `nk_input_begin` -/// which resets internal state like delta mouse position and button transitions. -/// After `nk_input_begin` all current input state needs to be provided. This includes -/// mouse motion, button and key pressed and released, text input and scrolling. -/// Both event- or state-based input handling are supported by this API -/// and should work without problems. Finally after all input state has been -/// mirrored `nk_input_end` needs to be called to finish input process. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// struct nk_context ctx; -/// nk_init_xxx(&ctx, ...); -/// while (1) { -/// Event evt; -/// nk_input_begin(&ctx); -/// while (GetEvent(&evt)) { -/// if (evt.type == MOUSE_MOVE) -/// nk_input_motion(&ctx, evt.motion.x, evt.motion.y); -/// else if (evt.type == [...]) { -/// // [...] -/// } -/// } -/// nk_input_end(&ctx); -/// // [...] -/// nk_clear(&ctx); -/// } nk_free(&ctx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// #### Reference -/// Function | Description -/// --------------------|------------------------------------------------------- -/// __nk_input_begin__ | Begins the input mirroring process. Needs to be called before all other `nk_input_xxx` calls -/// __nk_input_motion__ | Mirrors mouse cursor position -/// __nk_input_key__ | Mirrors key state with either pressed or released -/// __nk_input_button__ | Mirrors mouse button state with either pressed or released -/// __nk_input_scroll__ | Mirrors mouse scroll values -/// __nk_input_char__ | Adds a single ASCII text character into an internal text buffer -/// __nk_input_glyph__ | Adds a single multi-byte UTF-8 character into an internal text buffer -/// __nk_input_unicode__| Adds a single unicode rune into an internal text buffer -/// __nk_input_end__ | Ends the input mirroring process by calculating state changes. Don't call any `nk_input_xxx` function referenced above after this call -*/ -enum nk_keys { - NK_KEY_NONE, - NK_KEY_SHIFT, - NK_KEY_CTRL, - NK_KEY_ALT, - NK_KEY_DEL, - NK_KEY_ENTER, - NK_KEY_TAB, - NK_KEY_BACKSPACE, - NK_KEY_ESCAPE, - NK_KEY_COPY, - NK_KEY_CUT, - NK_KEY_PASTE, - NK_KEY_UP, - NK_KEY_DOWN, - NK_KEY_LEFT, - NK_KEY_RIGHT, - /* Shortcuts: text field */ - NK_KEY_TEXT_INSERT_MODE, - NK_KEY_TEXT_REPLACE_MODE, - NK_KEY_TEXT_RESET_MODE, - NK_KEY_TEXT_LINE_START, - NK_KEY_TEXT_LINE_END, - NK_KEY_TEXT_START, - NK_KEY_TEXT_END, - NK_KEY_TEXT_UNDO, - NK_KEY_TEXT_REDO, - NK_KEY_TEXT_SELECT_ALL, - NK_KEY_TEXT_WORD_LEFT, - NK_KEY_TEXT_WORD_RIGHT, - /* Shortcuts: scrollbar */ - NK_KEY_SCROLL_START, - NK_KEY_SCROLL_END, - NK_KEY_SCROLL_DOWN, - NK_KEY_SCROLL_UP, - NK_KEY_MAX -}; -enum nk_buttons { - NK_BUTTON_LEFT, - NK_BUTTON_MIDDLE, - NK_BUTTON_RIGHT, - NK_BUTTON_DOUBLE, - NK_BUTTON_MAX -}; -/*/// #### nk_input_begin -/// Begins the input mirroring process by resetting text, scroll -/// mouse, previous mouse position and movement as well as key state transitions, -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_input_begin(struct nk_context*); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to a previously initialized `nk_context` struct -*/ -NK_API void nk_input_begin(struct nk_context*); -/*/// #### nk_input_motion -/// Mirrors current mouse position to nuklear -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_input_motion(struct nk_context *ctx, int x, int y); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to a previously initialized `nk_context` struct -/// __x__ | Must hold an integer describing the current mouse cursor x-position -/// __y__ | Must hold an integer describing the current mouse cursor y-position -*/ -NK_API void nk_input_motion(struct nk_context*, int x, int y); -/*/// #### nk_input_key -/// Mirrors the state of a specific key to nuklear -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_input_key(struct nk_context*, enum nk_keys key, nk_bool down); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to a previously initialized `nk_context` struct -/// __key__ | Must be any value specified in enum `nk_keys` that needs to be mirrored -/// __down__ | Must be 0 for key is up and 1 for key is down -*/ -NK_API void nk_input_key(struct nk_context*, enum nk_keys, nk_bool down); -/*/// #### nk_input_button -/// Mirrors the state of a specific mouse button to nuklear -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_input_button(struct nk_context *ctx, enum nk_buttons btn, int x, int y, nk_bool down); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to a previously initialized `nk_context` struct -/// __btn__ | Must be any value specified in enum `nk_buttons` that needs to be mirrored -/// __x__ | Must contain an integer describing mouse cursor x-position on click up/down -/// __y__ | Must contain an integer describing mouse cursor y-position on click up/down -/// __down__ | Must be 0 for key is up and 1 for key is down -*/ -NK_API void nk_input_button(struct nk_context*, enum nk_buttons, int x, int y, nk_bool down); -/*/// #### nk_input_scroll -/// Copies the last mouse scroll value to nuklear. Is generally -/// a scroll value. So does not have to come from mouse and could also originate -/// TODO finish this sentence -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_input_scroll(struct nk_context *ctx, struct nk_vec2 val); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to a previously initialized `nk_context` struct -/// __val__ | vector with both X- as well as Y-scroll value -*/ -NK_API void nk_input_scroll(struct nk_context*, struct nk_vec2 val); -/*/// #### nk_input_char -/// Copies a single ASCII character into an internal text buffer -/// This is basically a helper function to quickly push ASCII characters into -/// nuklear. -/// -/// !!! Note -/// Stores up to NK_INPUT_MAX bytes between `nk_input_begin` and `nk_input_end`. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_input_char(struct nk_context *ctx, char c); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to a previously initialized `nk_context` struct -/// __c__ | Must be a single ASCII character preferable one that can be printed -*/ -NK_API void nk_input_char(struct nk_context*, char); -/*/// #### nk_input_glyph -/// Converts an encoded unicode rune into UTF-8 and copies the result into an -/// internal text buffer. -/// -/// !!! Note -/// Stores up to NK_INPUT_MAX bytes between `nk_input_begin` and `nk_input_end`. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_input_glyph(struct nk_context *ctx, const nk_glyph g); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to a previously initialized `nk_context` struct -/// __g__ | UTF-32 unicode codepoint -*/ -NK_API void nk_input_glyph(struct nk_context*, const nk_glyph); -/*/// #### nk_input_unicode -/// Converts a unicode rune into UTF-8 and copies the result -/// into an internal text buffer. -/// !!! Note -/// Stores up to NK_INPUT_MAX bytes between `nk_input_begin` and `nk_input_end`. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_input_unicode(struct nk_context*, nk_rune rune); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to a previously initialized `nk_context` struct -/// __rune__ | UTF-32 unicode codepoint -*/ -NK_API void nk_input_unicode(struct nk_context*, nk_rune); -/*/// #### nk_input_end -/// End the input mirroring process by resetting mouse grabbing -/// state to ensure the mouse cursor is not grabbed indefinitely. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_input_end(struct nk_context *ctx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to a previously initialized `nk_context` struct -*/ -NK_API void nk_input_end(struct nk_context*); -/* ============================================================================= - * - * DRAWING - * - * =============================================================================*/ -/*/// ### Drawing -/// This library was designed to be render backend agnostic so it does -/// not draw anything to screen directly. Instead all drawn shapes, widgets -/// are made of, are buffered into memory and make up a command queue. -/// Each frame therefore fills the command buffer with draw commands -/// that then need to be executed by the user and his own render backend. -/// After that the command buffer needs to be cleared and a new frame can be -/// started. It is probably important to note that the command buffer is the main -/// drawing API and the optional vertex buffer API only takes this format and -/// converts it into a hardware accessible format. -/// -/// #### Usage -/// To draw all draw commands accumulated over a frame you need your own render -/// backend able to draw a number of 2D primitives. This includes at least -/// filled and stroked rectangles, circles, text, lines, triangles and scissors. -/// As soon as this criterion is met you can iterate over each draw command -/// and execute each draw command in a interpreter like fashion: -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// const struct nk_command *cmd = 0; -/// nk_foreach(cmd, &ctx) { -/// switch (cmd->type) { -/// case NK_COMMAND_LINE: -/// your_draw_line_function(...) -/// break; -/// case NK_COMMAND_RECT -/// your_draw_rect_function(...) -/// break; -/// case //...: -/// //[...] -/// } -/// } -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// In program flow context draw commands need to be executed after input has been -/// gathered and the complete UI with windows and their contained widgets have -/// been executed and before calling `nk_clear` which frees all previously -/// allocated draw commands. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// struct nk_context ctx; -/// nk_init_xxx(&ctx, ...); -/// while (1) { -/// Event evt; -/// nk_input_begin(&ctx); -/// while (GetEvent(&evt)) { -/// if (evt.type == MOUSE_MOVE) -/// nk_input_motion(&ctx, evt.motion.x, evt.motion.y); -/// else if (evt.type == [...]) { -/// [...] -/// } -/// } -/// nk_input_end(&ctx); -/// // -/// // [...] -/// // -/// const struct nk_command *cmd = 0; -/// nk_foreach(cmd, &ctx) { -/// switch (cmd->type) { -/// case NK_COMMAND_LINE: -/// your_draw_line_function(...) -/// break; -/// case NK_COMMAND_RECT -/// your_draw_rect_function(...) -/// break; -/// case ...: -/// // [...] -/// } -/// nk_clear(&ctx); -/// } -/// nk_free(&ctx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// You probably noticed that you have to draw all of the UI each frame which is -/// quite wasteful. While the actual UI updating loop is quite fast rendering -/// without actually needing it is not. So there are multiple things you could do. -/// -/// First is only update on input. This of course is only an option if your -/// application only depends on the UI and does not require any outside calculations. -/// If you actually only update on input make sure to update the UI two times each -/// frame and call `nk_clear` directly after the first pass and only draw in -/// the second pass. In addition it is recommended to also add additional timers -/// to make sure the UI is not drawn more than a fixed number of frames per second. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// struct nk_context ctx; -/// nk_init_xxx(&ctx, ...); -/// while (1) { -/// // [...wait for input ] -/// // [...do two UI passes ...] -/// do_ui(...) -/// nk_clear(&ctx); -/// do_ui(...) -/// // -/// // draw -/// const struct nk_command *cmd = 0; -/// nk_foreach(cmd, &ctx) { -/// switch (cmd->type) { -/// case NK_COMMAND_LINE: -/// your_draw_line_function(...) -/// break; -/// case NK_COMMAND_RECT -/// your_draw_rect_function(...) -/// break; -/// case ...: -/// //[...] -/// } -/// nk_clear(&ctx); -/// } -/// nk_free(&ctx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// The second probably more applicable trick is to only draw if anything changed. -/// It is not really useful for applications with continuous draw loop but -/// quite useful for desktop applications. To actually get nuklear to only -/// draw on changes you first have to define `NK_ZERO_COMMAND_MEMORY` and -/// allocate a memory buffer that will store each unique drawing output. -/// After each frame you compare the draw command memory inside the library -/// with your allocated buffer by memcmp. If memcmp detects differences -/// you have to copy the command buffer into the allocated buffer -/// and then draw like usual (this example uses fixed memory but you could -/// use dynamically allocated memory). -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// //[... other defines ...] -/// #define NK_ZERO_COMMAND_MEMORY -/// #include "nuklear.h" -/// // -/// // setup context -/// struct nk_context ctx; -/// void *last = calloc(1,64*1024); -/// void *buf = calloc(1,64*1024); -/// nk_init_fixed(&ctx, buf, 64*1024); -/// // -/// // loop -/// while (1) { -/// // [...input...] -/// // [...ui...] -/// void *cmds = nk_buffer_memory(&ctx.memory); -/// if (memcmp(cmds, last, ctx.memory.allocated)) { -/// memcpy(last,cmds,ctx.memory.allocated); -/// const struct nk_command *cmd = 0; -/// nk_foreach(cmd, &ctx) { -/// switch (cmd->type) { -/// case NK_COMMAND_LINE: -/// your_draw_line_function(...) -/// break; -/// case NK_COMMAND_RECT -/// your_draw_rect_function(...) -/// break; -/// case ...: -/// // [...] -/// } -/// } -/// } -/// nk_clear(&ctx); -/// } -/// nk_free(&ctx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Finally while using draw commands makes sense for higher abstracted platforms like -/// X11 and Win32 or drawing libraries it is often desirable to use graphics -/// hardware directly. Therefore it is possible to just define -/// `NK_INCLUDE_VERTEX_BUFFER_OUTPUT` which includes optional vertex output. -/// To access the vertex output you first have to convert all draw commands into -/// vertexes by calling `nk_convert` which takes in your preferred vertex format. -/// After successfully converting all draw commands just iterate over and execute all -/// vertex draw commands: -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// // fill configuration -/// struct your_vertex -/// { -/// float pos[2]; // important to keep it to 2 floats -/// float uv[2]; -/// unsigned char col[4]; -/// }; -/// struct nk_convert_config cfg = {}; -/// static const struct nk_draw_vertex_layout_element vertex_layout[] = { -/// {NK_VERTEX_POSITION, NK_FORMAT_FLOAT, NK_OFFSETOF(struct your_vertex, pos)}, -/// {NK_VERTEX_TEXCOORD, NK_FORMAT_FLOAT, NK_OFFSETOF(struct your_vertex, uv)}, -/// {NK_VERTEX_COLOR, NK_FORMAT_R8G8B8A8, NK_OFFSETOF(struct your_vertex, col)}, -/// {NK_VERTEX_LAYOUT_END} -/// }; -/// cfg.shape_AA = NK_ANTI_ALIASING_ON; -/// cfg.line_AA = NK_ANTI_ALIASING_ON; -/// cfg.vertex_layout = vertex_layout; -/// cfg.vertex_size = sizeof(struct your_vertex); -/// cfg.vertex_alignment = NK_ALIGNOF(struct your_vertex); -/// cfg.circle_segment_count = 22; -/// cfg.curve_segment_count = 22; -/// cfg.arc_segment_count = 22; -/// cfg.global_alpha = 1.0f; -/// cfg.tex_null = dev->tex_null; -/// // -/// // setup buffers and convert -/// struct nk_buffer cmds, verts, idx; -/// nk_buffer_init_default(&cmds); -/// nk_buffer_init_default(&verts); -/// nk_buffer_init_default(&idx); -/// nk_convert(&ctx, &cmds, &verts, &idx, &cfg); -/// // -/// // draw -/// nk_draw_foreach(cmd, &ctx, &cmds) { -/// if (!cmd->elem_count) continue; -/// //[...] -/// } -/// nk_buffer_free(&cms); -/// nk_buffer_free(&verts); -/// nk_buffer_free(&idx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// #### Reference -/// Function | Description -/// --------------------|------------------------------------------------------- -/// __nk__begin__ | Returns the first draw command in the context draw command list to be drawn -/// __nk__next__ | Increments the draw command iterator to the next command inside the context draw command list -/// __nk_foreach__ | Iterates over each draw command inside the context draw command list -/// __nk_convert__ | Converts from the abstract draw commands list into a hardware accessible vertex format -/// __nk_draw_begin__ | Returns the first vertex command in the context vertex draw list to be executed -/// __nk__draw_next__ | Increments the vertex command iterator to the next command inside the context vertex command list -/// __nk__draw_end__ | Returns the end of the vertex draw list -/// __nk_draw_foreach__ | Iterates over each vertex draw command inside the vertex draw list -*/ -enum nk_anti_aliasing {NK_ANTI_ALIASING_OFF, NK_ANTI_ALIASING_ON}; -enum nk_convert_result { - NK_CONVERT_SUCCESS = 0, - NK_CONVERT_INVALID_PARAM = 1, - NK_CONVERT_COMMAND_BUFFER_FULL = NK_FLAG(1), - NK_CONVERT_VERTEX_BUFFER_FULL = NK_FLAG(2), - NK_CONVERT_ELEMENT_BUFFER_FULL = NK_FLAG(3) -}; -struct nk_draw_null_texture { - nk_handle texture; /* texture handle to a texture with a white pixel */ - struct nk_vec2 uv; /* coordinates to a white pixel in the texture */ -}; -struct nk_convert_config { - float global_alpha; /* global alpha value */ - enum nk_anti_aliasing line_AA; /* line anti-aliasing flag can be turned off if you are tight on memory */ - enum nk_anti_aliasing shape_AA; /* shape anti-aliasing flag can be turned off if you are tight on memory */ - unsigned circle_segment_count; /* number of segments used for circles: default to 22 */ - unsigned arc_segment_count; /* number of segments used for arcs: default to 22 */ - unsigned curve_segment_count; /* number of segments used for curves: default to 22 */ - struct nk_draw_null_texture tex_null; /* handle to texture with a white pixel for shape drawing */ - const struct nk_draw_vertex_layout_element *vertex_layout; /* describes the vertex output format and packing */ - nk_size vertex_size; /* sizeof one vertex for vertex packing */ - nk_size vertex_alignment; /* vertex alignment: Can be obtained by NK_ALIGNOF */ -}; -/*/// #### nk__begin -/// Returns a draw command list iterator to iterate all draw -/// commands accumulated over one frame. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// const struct nk_command* nk__begin(struct nk_context*); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | must point to an previously initialized `nk_context` struct at the end of a frame -/// -/// Returns draw command pointer pointing to the first command inside the draw command list -*/ -NK_API const struct nk_command* nk__begin(struct nk_context*); -/*/// #### nk__next -/// Returns draw command pointer pointing to the next command inside the draw command list -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// const struct nk_command* nk__next(struct nk_context*, const struct nk_command*); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct at the end of a frame -/// __cmd__ | Must point to an previously a draw command either returned by `nk__begin` or `nk__next` -/// -/// Returns draw command pointer pointing to the next command inside the draw command list -*/ -NK_API const struct nk_command* nk__next(struct nk_context*, const struct nk_command*); -/*/// #### nk_foreach -/// Iterates over each draw command inside the context draw command list -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// #define nk_foreach(c, ctx) -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct at the end of a frame -/// __cmd__ | Command pointer initialized to NULL -/// -/// Iterates over each draw command inside the context draw command list -*/ -#define nk_foreach(c, ctx) for((c) = nk__begin(ctx); (c) != 0; (c) = nk__next(ctx,c)) -#ifdef NK_INCLUDE_VERTEX_BUFFER_OUTPUT -/*/// #### nk_convert -/// Converts all internal draw commands into vertex draw commands and fills -/// three buffers with vertexes, vertex draw commands and vertex indices. The vertex format -/// as well as some other configuration values have to be configured by filling out a -/// `nk_convert_config` struct. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// nk_flags nk_convert(struct nk_context *ctx, struct nk_buffer *cmds, -/// struct nk_buffer *vertices, struct nk_buffer *elements, const struct nk_convert_config*); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct at the end of a frame -/// __cmds__ | Must point to a previously initialized buffer to hold converted vertex draw commands -/// __vertices__| Must point to a previously initialized buffer to hold all produced vertices -/// __elements__| Must point to a previously initialized buffer to hold all produced vertex indices -/// __config__ | Must point to a filled out `nk_config` struct to configure the conversion process -/// -/// Returns one of enum nk_convert_result error codes -/// -/// Parameter | Description -/// --------------------------------|----------------------------------------------------------- -/// NK_CONVERT_SUCCESS | Signals a successful draw command to vertex buffer conversion -/// NK_CONVERT_INVALID_PARAM | An invalid argument was passed in the function call -/// NK_CONVERT_COMMAND_BUFFER_FULL | The provided buffer for storing draw commands is full or failed to allocate more memory -/// NK_CONVERT_VERTEX_BUFFER_FULL | The provided buffer for storing vertices is full or failed to allocate more memory -/// NK_CONVERT_ELEMENT_BUFFER_FULL | The provided buffer for storing indices is full or failed to allocate more memory -*/ -NK_API nk_flags nk_convert(struct nk_context*, struct nk_buffer *cmds, struct nk_buffer *vertices, struct nk_buffer *elements, const struct nk_convert_config*); -/*/// #### nk__draw_begin -/// Returns a draw vertex command buffer iterator to iterate over the vertex draw command buffer -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// const struct nk_draw_command* nk__draw_begin(const struct nk_context*, const struct nk_buffer*); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct at the end of a frame -/// __buf__ | Must point to an previously by `nk_convert` filled out vertex draw command buffer -/// -/// Returns vertex draw command pointer pointing to the first command inside the vertex draw command buffer -*/ -NK_API const struct nk_draw_command* nk__draw_begin(const struct nk_context*, const struct nk_buffer*); -/*/// #### nk__draw_end -/// Returns the vertex draw command at the end of the vertex draw command buffer -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// const struct nk_draw_command* nk__draw_end(const struct nk_context *ctx, const struct nk_buffer *buf); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct at the end of a frame -/// __buf__ | Must point to an previously by `nk_convert` filled out vertex draw command buffer -/// -/// Returns vertex draw command pointer pointing to the end of the last vertex draw command inside the vertex draw command buffer -*/ -NK_API const struct nk_draw_command* nk__draw_end(const struct nk_context*, const struct nk_buffer*); -/*/// #### nk__draw_next -/// Increments the vertex draw command buffer iterator -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// const struct nk_draw_command* nk__draw_next(const struct nk_draw_command*, const struct nk_buffer*, const struct nk_context*); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __cmd__ | Must point to an previously either by `nk__draw_begin` or `nk__draw_next` returned vertex draw command -/// __buf__ | Must point to an previously by `nk_convert` filled out vertex draw command buffer -/// __ctx__ | Must point to an previously initialized `nk_context` struct at the end of a frame -/// -/// Returns vertex draw command pointer pointing to the end of the last vertex draw command inside the vertex draw command buffer -*/ -NK_API const struct nk_draw_command* nk__draw_next(const struct nk_draw_command*, const struct nk_buffer*, const struct nk_context*); -/*/// #### nk_draw_foreach -/// Iterates over each vertex draw command inside a vertex draw command buffer -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// #define nk_draw_foreach(cmd,ctx, b) -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __cmd__ | `nk_draw_command`iterator set to NULL -/// __buf__ | Must point to an previously by `nk_convert` filled out vertex draw command buffer -/// __ctx__ | Must point to an previously initialized `nk_context` struct at the end of a frame -*/ -#define nk_draw_foreach(cmd,ctx, b) for((cmd)=nk__draw_begin(ctx, b); (cmd)!=0; (cmd)=nk__draw_next(cmd, b, ctx)) -#endif -/* ============================================================================= - * - * WINDOW - * - * ============================================================================= -/// ### Window -/// Windows are the main persistent state used inside nuklear and are life time -/// controlled by simply "retouching" (i.e. calling) each window each frame. -/// All widgets inside nuklear can only be added inside the function pair `nk_begin_xxx` -/// and `nk_end`. Calling any widgets outside these two functions will result in an -/// assert in debug or no state change in release mode.

    -/// -/// Each window holds frame persistent state like position, size, flags, state tables, -/// and some garbage collected internal persistent widget state. Each window -/// is linked into a window stack list which determines the drawing and overlapping -/// order. The topmost window thereby is the currently active window.

    -/// -/// To change window position inside the stack occurs either automatically by -/// user input by being clicked on or programmatically by calling `nk_window_focus`. -/// Windows by default are visible unless explicitly being defined with flag -/// `NK_WINDOW_HIDDEN`, the user clicked the close button on windows with flag -/// `NK_WINDOW_CLOSABLE` or if a window was explicitly hidden by calling -/// `nk_window_show`. To explicitly close and destroy a window call `nk_window_close`.

    -/// -/// #### Usage -/// To create and keep a window you have to call one of the two `nk_begin_xxx` -/// functions to start window declarations and `nk_end` at the end. Furthermore it -/// is recommended to check the return value of `nk_begin_xxx` and only process -/// widgets inside the window if the value is not 0. Either way you have to call -/// `nk_end` at the end of window declarations. Furthermore, do not attempt to -/// nest `nk_begin_xxx` calls which will hopefully result in an assert or if not -/// in a segmentation fault. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// if (nk_begin_xxx(...) { -/// // [... widgets ...] -/// } -/// nk_end(ctx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// In the grand concept window and widget declarations need to occur after input -/// handling and before drawing to screen. Not doing so can result in higher -/// latency or at worst invalid behavior. Furthermore make sure that `nk_clear` -/// is called at the end of the frame. While nuklear's default platform backends -/// already call `nk_clear` for you if you write your own backend not calling -/// `nk_clear` can cause asserts or even worse undefined behavior. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// struct nk_context ctx; -/// nk_init_xxx(&ctx, ...); -/// while (1) { -/// Event evt; -/// nk_input_begin(&ctx); -/// while (GetEvent(&evt)) { -/// if (evt.type == MOUSE_MOVE) -/// nk_input_motion(&ctx, evt.motion.x, evt.motion.y); -/// else if (evt.type == [...]) { -/// nk_input_xxx(...); -/// } -/// } -/// nk_input_end(&ctx); -/// -/// if (nk_begin_xxx(...) { -/// //[...] -/// } -/// nk_end(ctx); -/// -/// const struct nk_command *cmd = 0; -/// nk_foreach(cmd, &ctx) { -/// case NK_COMMAND_LINE: -/// your_draw_line_function(...) -/// break; -/// case NK_COMMAND_RECT -/// your_draw_rect_function(...) -/// break; -/// case //...: -/// //[...] -/// } -/// nk_clear(&ctx); -/// } -/// nk_free(&ctx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// #### Reference -/// Function | Description -/// ------------------------------------|---------------------------------------- -/// nk_begin | Starts a new window; needs to be called every frame for every window (unless hidden) or otherwise the window gets removed -/// nk_begin_titled | Extended window start with separated title and identifier to allow multiple windows with same name but not title -/// nk_end | Needs to be called at the end of the window building process to process scaling, scrollbars and general cleanup -// -/// nk_window_find | Finds and returns the window with give name -/// nk_window_get_bounds | Returns a rectangle with screen position and size of the currently processed window. -/// nk_window_get_position | Returns the position of the currently processed window -/// nk_window_get_size | Returns the size with width and height of the currently processed window -/// nk_window_get_width | Returns the width of the currently processed window -/// nk_window_get_height | Returns the height of the currently processed window -/// nk_window_get_panel | Returns the underlying panel which contains all processing state of the current window -/// nk_window_get_content_region | Returns the position and size of the currently visible and non-clipped space inside the currently processed window -/// nk_window_get_content_region_min | Returns the upper rectangle position of the currently visible and non-clipped space inside the currently processed window -/// nk_window_get_content_region_max | Returns the upper rectangle position of the currently visible and non-clipped space inside the currently processed window -/// nk_window_get_content_region_size | Returns the size of the currently visible and non-clipped space inside the currently processed window -/// nk_window_get_canvas | Returns the draw command buffer. Can be used to draw custom widgets -/// nk_window_get_scroll | Gets the scroll offset of the current window -/// nk_window_has_focus | Returns if the currently processed window is currently active -/// nk_window_is_collapsed | Returns if the window with given name is currently minimized/collapsed -/// nk_window_is_closed | Returns if the currently processed window was closed -/// nk_window_is_hidden | Returns if the currently processed window was hidden -/// nk_window_is_active | Same as nk_window_has_focus for some reason -/// nk_window_is_hovered | Returns if the currently processed window is currently being hovered by mouse -/// nk_window_is_any_hovered | Return if any window currently hovered -/// nk_item_is_any_active | Returns if any window or widgets is currently hovered or active -// -/// nk_window_set_bounds | Updates position and size of the currently processed window -/// nk_window_set_position | Updates position of the currently process window -/// nk_window_set_size | Updates the size of the currently processed window -/// nk_window_set_focus | Set the currently processed window as active window -/// nk_window_set_scroll | Sets the scroll offset of the current window -// -/// nk_window_close | Closes the window with given window name which deletes the window at the end of the frame -/// nk_window_collapse | Collapses the window with given window name -/// nk_window_collapse_if | Collapses the window with given window name if the given condition was met -/// nk_window_show | Hides a visible or reshows a hidden window -/// nk_window_show_if | Hides/shows a window depending on condition -*/ -/* -/// #### nk_panel_flags -/// Flag | Description -/// ----------------------------|---------------------------------------- -/// NK_WINDOW_BORDER | Draws a border around the window to visually separate window from the background -/// NK_WINDOW_MOVABLE | The movable flag indicates that a window can be moved by user input or by dragging the window header -/// NK_WINDOW_SCALABLE | The scalable flag indicates that a window can be scaled by user input by dragging a scaler icon at the button of the window -/// NK_WINDOW_CLOSABLE | Adds a closable icon into the header -/// NK_WINDOW_MINIMIZABLE | Adds a minimize icon into the header -/// NK_WINDOW_NO_SCROLLBAR | Removes the scrollbar from the window -/// NK_WINDOW_TITLE | Forces a header at the top at the window showing the title -/// NK_WINDOW_SCROLL_AUTO_HIDE | Automatically hides the window scrollbar if no user interaction: also requires delta time in `nk_context` to be set each frame -/// NK_WINDOW_BACKGROUND | Always keep window in the background -/// NK_WINDOW_SCALE_LEFT | Puts window scaler in the left-bottom corner instead right-bottom -/// NK_WINDOW_NO_INPUT | Prevents window of scaling, moving or getting focus -/// -/// #### nk_collapse_states -/// State | Description -/// ----------------|----------------------------------------------------------- -/// __NK_MINIMIZED__| UI section is collased and not visible until maximized -/// __NK_MAXIMIZED__| UI section is extended and visible until minimized -///

    -*/ -enum nk_panel_flags { - NK_WINDOW_BORDER = NK_FLAG(0), - NK_WINDOW_MOVABLE = NK_FLAG(1), - NK_WINDOW_SCALABLE = NK_FLAG(2), - NK_WINDOW_CLOSABLE = NK_FLAG(3), - NK_WINDOW_MINIMIZABLE = NK_FLAG(4), - NK_WINDOW_NO_SCROLLBAR = NK_FLAG(5), - NK_WINDOW_TITLE = NK_FLAG(6), - NK_WINDOW_SCROLL_AUTO_HIDE = NK_FLAG(7), - NK_WINDOW_BACKGROUND = NK_FLAG(8), - NK_WINDOW_SCALE_LEFT = NK_FLAG(9), - NK_WINDOW_NO_INPUT = NK_FLAG(10) -}; -/*/// #### nk_begin -/// Starts a new window; needs to be called every frame for every -/// window (unless hidden) or otherwise the window gets removed -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// nk_bool nk_begin(struct nk_context *ctx, const char *title, struct nk_rect bounds, nk_flags flags); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __title__ | Window title and identifier. Needs to be persistent over frames to identify the window -/// __bounds__ | Initial position and window size. However if you do not define `NK_WINDOW_SCALABLE` or `NK_WINDOW_MOVABLE` you can set window position and size every frame -/// __flags__ | Window flags defined in the nk_panel_flags section with a number of different window behaviors -/// -/// Returns `true(1)` if the window can be filled up with widgets from this point -/// until `nk_end` or `false(0)` otherwise for example if minimized -*/ -NK_API nk_bool nk_begin(struct nk_context *ctx, const char *title, struct nk_rect bounds, nk_flags flags); -/*/// #### nk_begin_titled -/// Extended window start with separated title and identifier to allow multiple -/// windows with same title but not name -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// nk_bool nk_begin_titled(struct nk_context *ctx, const char *name, const char *title, struct nk_rect bounds, nk_flags flags); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __name__ | Window identifier. Needs to be persistent over frames to identify the window -/// __title__ | Window title displayed inside header if flag `NK_WINDOW_TITLE` or either `NK_WINDOW_CLOSABLE` or `NK_WINDOW_MINIMIZED` was set -/// __bounds__ | Initial position and window size. However if you do not define `NK_WINDOW_SCALABLE` or `NK_WINDOW_MOVABLE` you can set window position and size every frame -/// __flags__ | Window flags defined in the nk_panel_flags section with a number of different window behaviors -/// -/// Returns `true(1)` if the window can be filled up with widgets from this point -/// until `nk_end` or `false(0)` otherwise for example if minimized -*/ -NK_API nk_bool nk_begin_titled(struct nk_context *ctx, const char *name, const char *title, struct nk_rect bounds, nk_flags flags); -/*/// #### nk_end -/// Needs to be called at the end of the window building process to process scaling, scrollbars and general cleanup. -/// All widget calls after this functions will result in asserts or no state changes -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_end(struct nk_context *ctx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -*/ -NK_API void nk_end(struct nk_context *ctx); -/*/// #### nk_window_find -/// Finds and returns a window from passed name -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// struct nk_window *nk_window_find(struct nk_context *ctx, const char *name); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __name__ | Window identifier -/// -/// Returns a `nk_window` struct pointing to the identified window or NULL if -/// no window with the given name was found -*/ -NK_API struct nk_window *nk_window_find(struct nk_context *ctx, const char *name); -/*/// #### nk_window_get_bounds -/// Returns a rectangle with screen position and size of the currently processed window -/// -/// !!! WARNING -/// Only call this function between calls `nk_begin_xxx` and `nk_end` -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// struct nk_rect nk_window_get_bounds(const struct nk_context *ctx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// -/// Returns a `nk_rect` struct with window upper left window position and size -*/ -NK_API struct nk_rect nk_window_get_bounds(const struct nk_context *ctx); -/*/// #### nk_window_get_position -/// Returns the position of the currently processed window. -/// -/// !!! WARNING -/// Only call this function between calls `nk_begin_xxx` and `nk_end` -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// struct nk_vec2 nk_window_get_position(const struct nk_context *ctx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// -/// Returns a `nk_vec2` struct with window upper left position -*/ -NK_API struct nk_vec2 nk_window_get_position(const struct nk_context *ctx); -/*/// #### nk_window_get_size -/// Returns the size with width and height of the currently processed window. -/// -/// !!! WARNING -/// Only call this function between calls `nk_begin_xxx` and `nk_end` -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// struct nk_vec2 nk_window_get_size(const struct nk_context *ctx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// -/// Returns a `nk_vec2` struct with window width and height -*/ -NK_API struct nk_vec2 nk_window_get_size(const struct nk_context*); -/*/// #### nk_window_get_width -/// Returns the width of the currently processed window. -/// -/// !!! WARNING -/// Only call this function between calls `nk_begin_xxx` and `nk_end` -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// float nk_window_get_width(const struct nk_context *ctx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// -/// Returns the current window width -*/ -NK_API float nk_window_get_width(const struct nk_context*); -/*/// #### nk_window_get_height -/// Returns the height of the currently processed window. -/// -/// !!! WARNING -/// Only call this function between calls `nk_begin_xxx` and `nk_end` -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// float nk_window_get_height(const struct nk_context *ctx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// -/// Returns the current window height -*/ -NK_API float nk_window_get_height(const struct nk_context*); -/*/// #### nk_window_get_panel -/// Returns the underlying panel which contains all processing state of the current window. -/// -/// !!! WARNING -/// Only call this function between calls `nk_begin_xxx` and `nk_end` -/// !!! WARNING -/// Do not keep the returned panel pointer around, it is only valid until `nk_end` -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// struct nk_panel* nk_window_get_panel(struct nk_context *ctx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// -/// Returns a pointer to window internal `nk_panel` state. -*/ -NK_API struct nk_panel* nk_window_get_panel(struct nk_context*); -/*/// #### nk_window_get_content_region -/// Returns the position and size of the currently visible and non-clipped space -/// inside the currently processed window. -/// -/// !!! WARNING -/// Only call this function between calls `nk_begin_xxx` and `nk_end` -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// struct nk_rect nk_window_get_content_region(struct nk_context *ctx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// -/// Returns `nk_rect` struct with screen position and size (no scrollbar offset) -/// of the visible space inside the current window -*/ -NK_API struct nk_rect nk_window_get_content_region(struct nk_context*); -/*/// #### nk_window_get_content_region_min -/// Returns the upper left position of the currently visible and non-clipped -/// space inside the currently processed window. -/// -/// !!! WARNING -/// Only call this function between calls `nk_begin_xxx` and `nk_end` -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// struct nk_vec2 nk_window_get_content_region_min(struct nk_context *ctx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// -/// returns `nk_vec2` struct with upper left screen position (no scrollbar offset) -/// of the visible space inside the current window -*/ -NK_API struct nk_vec2 nk_window_get_content_region_min(struct nk_context*); -/*/// #### nk_window_get_content_region_max -/// Returns the lower right screen position of the currently visible and -/// non-clipped space inside the currently processed window. -/// -/// !!! WARNING -/// Only call this function between calls `nk_begin_xxx` and `nk_end` -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// struct nk_vec2 nk_window_get_content_region_max(struct nk_context *ctx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// -/// Returns `nk_vec2` struct with lower right screen position (no scrollbar offset) -/// of the visible space inside the current window -*/ -NK_API struct nk_vec2 nk_window_get_content_region_max(struct nk_context*); -/*/// #### nk_window_get_content_region_size -/// Returns the size of the currently visible and non-clipped space inside the -/// currently processed window -/// -/// !!! WARNING -/// Only call this function between calls `nk_begin_xxx` and `nk_end` -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// struct nk_vec2 nk_window_get_content_region_size(struct nk_context *ctx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// -/// Returns `nk_vec2` struct with size the visible space inside the current window -*/ -NK_API struct nk_vec2 nk_window_get_content_region_size(struct nk_context*); -/*/// #### nk_window_get_canvas -/// Returns the draw command buffer. Can be used to draw custom widgets -/// !!! WARNING -/// Only call this function between calls `nk_begin_xxx` and `nk_end` -/// !!! WARNING -/// Do not keep the returned command buffer pointer around it is only valid until `nk_end` -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// struct nk_command_buffer* nk_window_get_canvas(struct nk_context *ctx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// -/// Returns a pointer to window internal `nk_command_buffer` struct used as -/// drawing canvas. Can be used to do custom drawing. -*/ -NK_API struct nk_command_buffer* nk_window_get_canvas(struct nk_context*); -/*/// #### nk_window_get_scroll -/// Gets the scroll offset for the current window -/// !!! WARNING -/// Only call this function between calls `nk_begin_xxx` and `nk_end` -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_window_get_scroll(struct nk_context *ctx, nk_uint *offset_x, nk_uint *offset_y); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// -------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __offset_x__ | A pointer to the x offset output (or NULL to ignore) -/// __offset_y__ | A pointer to the y offset output (or NULL to ignore) -*/ -NK_API void nk_window_get_scroll(struct nk_context*, nk_uint *offset_x, nk_uint *offset_y); -/*/// #### nk_window_has_focus -/// Returns if the currently processed window is currently active -/// !!! WARNING -/// Only call this function between calls `nk_begin_xxx` and `nk_end` -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// nk_bool nk_window_has_focus(const struct nk_context *ctx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// -/// Returns `false(0)` if current window is not active or `true(1)` if it is -*/ -NK_API nk_bool nk_window_has_focus(const struct nk_context*); -/*/// #### nk_window_is_hovered -/// Return if the current window is being hovered -/// !!! WARNING -/// Only call this function between calls `nk_begin_xxx` and `nk_end` -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// nk_bool nk_window_is_hovered(struct nk_context *ctx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// -/// Returns `true(1)` if current window is hovered or `false(0)` otherwise -*/ -NK_API nk_bool nk_window_is_hovered(struct nk_context*); -/*/// #### nk_window_is_collapsed -/// Returns if the window with given name is currently minimized/collapsed -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// nk_bool nk_window_is_collapsed(struct nk_context *ctx, const char *name); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __name__ | Identifier of window you want to check if it is collapsed -/// -/// Returns `true(1)` if current window is minimized and `false(0)` if window not -/// found or is not minimized -*/ -NK_API nk_bool nk_window_is_collapsed(struct nk_context *ctx, const char *name); -/*/// #### nk_window_is_closed -/// Returns if the window with given name was closed by calling `nk_close` -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// nk_bool nk_window_is_closed(struct nk_context *ctx, const char *name); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __name__ | Identifier of window you want to check if it is closed -/// -/// Returns `true(1)` if current window was closed or `false(0)` window not found or not closed -*/ -NK_API nk_bool nk_window_is_closed(struct nk_context*, const char*); -/*/// #### nk_window_is_hidden -/// Returns if the window with given name is hidden -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// nk_bool nk_window_is_hidden(struct nk_context *ctx, const char *name); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __name__ | Identifier of window you want to check if it is hidden -/// -/// Returns `true(1)` if current window is hidden or `false(0)` window not found or visible -*/ -NK_API nk_bool nk_window_is_hidden(struct nk_context*, const char*); -/*/// #### nk_window_is_active -/// Same as nk_window_has_focus for some reason -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// nk_bool nk_window_is_active(struct nk_context *ctx, const char *name); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __name__ | Identifier of window you want to check if it is active -/// -/// Returns `true(1)` if current window is active or `false(0)` window not found or not active -*/ -NK_API nk_bool nk_window_is_active(struct nk_context*, const char*); -/*/// #### nk_window_is_any_hovered -/// Returns if the any window is being hovered -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// nk_bool nk_window_is_any_hovered(struct nk_context*); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// -/// Returns `true(1)` if any window is hovered or `false(0)` otherwise -*/ -NK_API nk_bool nk_window_is_any_hovered(struct nk_context*); -/*/// #### nk_item_is_any_active -/// Returns if the any window is being hovered or any widget is currently active. -/// Can be used to decide if input should be processed by UI or your specific input handling. -/// Example could be UI and 3D camera to move inside a 3D space. -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// nk_bool nk_item_is_any_active(struct nk_context*); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// -/// Returns `true(1)` if any window is hovered or any item is active or `false(0)` otherwise -*/ -NK_API nk_bool nk_item_is_any_active(struct nk_context*); -/*/// #### nk_window_set_bounds -/// Updates position and size of window with passed in name -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_window_set_bounds(struct nk_context*, const char *name, struct nk_rect bounds); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __name__ | Identifier of the window to modify both position and size -/// __bounds__ | Must point to a `nk_rect` struct with the new position and size -*/ -NK_API void nk_window_set_bounds(struct nk_context*, const char *name, struct nk_rect bounds); -/*/// #### nk_window_set_position -/// Updates position of window with passed name -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_window_set_position(struct nk_context*, const char *name, struct nk_vec2 pos); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __name__ | Identifier of the window to modify both position -/// __pos__ | Must point to a `nk_vec2` struct with the new position -*/ -NK_API void nk_window_set_position(struct nk_context*, const char *name, struct nk_vec2 pos); -/*/// #### nk_window_set_size -/// Updates size of window with passed in name -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_window_set_size(struct nk_context*, const char *name, struct nk_vec2); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __name__ | Identifier of the window to modify both window size -/// __size__ | Must point to a `nk_vec2` struct with new window size -*/ -NK_API void nk_window_set_size(struct nk_context*, const char *name, struct nk_vec2); -/*/// #### nk_window_set_focus -/// Sets the window with given name as active -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_window_set_focus(struct nk_context*, const char *name); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __name__ | Identifier of the window to set focus on -*/ -NK_API void nk_window_set_focus(struct nk_context*, const char *name); -/*/// #### nk_window_set_scroll -/// Sets the scroll offset for the current window -/// !!! WARNING -/// Only call this function between calls `nk_begin_xxx` and `nk_end` -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_window_set_scroll(struct nk_context *ctx, nk_uint offset_x, nk_uint offset_y); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// -------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __offset_x__ | The x offset to scroll to -/// __offset_y__ | The y offset to scroll to -*/ -NK_API void nk_window_set_scroll(struct nk_context*, nk_uint offset_x, nk_uint offset_y); -/*/// #### nk_window_close -/// Closes a window and marks it for being freed at the end of the frame -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_window_close(struct nk_context *ctx, const char *name); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __name__ | Identifier of the window to close -*/ -NK_API void nk_window_close(struct nk_context *ctx, const char *name); -/*/// #### nk_window_collapse -/// Updates collapse state of a window with given name -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_window_collapse(struct nk_context*, const char *name, enum nk_collapse_states state); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __name__ | Identifier of the window to close -/// __state__ | value out of nk_collapse_states section -*/ -NK_API void nk_window_collapse(struct nk_context*, const char *name, enum nk_collapse_states state); -/*/// #### nk_window_collapse_if -/// Updates collapse state of a window with given name if given condition is met -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_window_collapse_if(struct nk_context*, const char *name, enum nk_collapse_states, int cond); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __name__ | Identifier of the window to either collapse or maximize -/// __state__ | value out of nk_collapse_states section the window should be put into -/// __cond__ | condition that has to be met to actually commit the collapse state change -*/ -NK_API void nk_window_collapse_if(struct nk_context*, const char *name, enum nk_collapse_states, int cond); -/*/// #### nk_window_show -/// updates visibility state of a window with given name -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_window_show(struct nk_context*, const char *name, enum nk_show_states); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __name__ | Identifier of the window to either collapse or maximize -/// __state__ | state with either visible or hidden to modify the window with -*/ -NK_API void nk_window_show(struct nk_context*, const char *name, enum nk_show_states); -/*/// #### nk_window_show_if -/// Updates visibility state of a window with given name if a given condition is met -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_window_show_if(struct nk_context*, const char *name, enum nk_show_states, int cond); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __name__ | Identifier of the window to either hide or show -/// __state__ | state with either visible or hidden to modify the window with -/// __cond__ | condition that has to be met to actually commit the visbility state change -*/ -NK_API void nk_window_show_if(struct nk_context*, const char *name, enum nk_show_states, int cond); -/* ============================================================================= - * - * LAYOUT - * - * ============================================================================= -/// ### Layouting -/// Layouting in general describes placing widget inside a window with position and size. -/// While in this particular implementation there are five different APIs for layouting -/// each with different trade offs between control and ease of use.

    -/// -/// All layouting methods in this library are based around the concept of a row. -/// A row has a height the window content grows by and a number of columns and each -/// layouting method specifies how each widget is placed inside the row. -/// After a row has been allocated by calling a layouting functions and then -/// filled with widgets will advance an internal pointer over the allocated row.

    -/// -/// To actually define a layout you just call the appropriate layouting function -/// and each subsequent widget call will place the widget as specified. Important -/// here is that if you define more widgets then columns defined inside the layout -/// functions it will allocate the next row without you having to make another layouting

    -/// call. -/// -/// Biggest limitation with using all these APIs outside the `nk_layout_space_xxx` API -/// is that you have to define the row height for each. However the row height -/// often depends on the height of the font.

    -/// -/// To fix that internally nuklear uses a minimum row height that is set to the -/// height plus padding of currently active font and overwrites the row height -/// value if zero.

    -/// -/// If you manually want to change the minimum row height then -/// use nk_layout_set_min_row_height, and use nk_layout_reset_min_row_height to -/// reset it back to be derived from font height.

    -/// -/// Also if you change the font in nuklear it will automatically change the minimum -/// row height for you and. This means if you change the font but still want -/// a minimum row height smaller than the font you have to repush your value.

    -/// -/// For actually more advanced UI I would even recommend using the `nk_layout_space_xxx` -/// layouting method in combination with a cassowary constraint solver (there are -/// some versions on github with permissive license model) to take over all control over widget -/// layouting yourself. However for quick and dirty layouting using all the other layouting -/// functions should be fine. -/// -/// #### Usage -/// 1. __nk_layout_row_dynamic__

    -/// The easiest layouting function is `nk_layout_row_dynamic`. It provides each -/// widgets with same horizontal space inside the row and dynamically grows -/// if the owning window grows in width. So the number of columns dictates -/// the size of each widget dynamically by formula: -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// widget_width = (window_width - padding - spacing) * (1/colum_count) -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Just like all other layouting APIs if you define more widget than columns this -/// library will allocate a new row and keep all layouting parameters previously -/// defined. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// if (nk_begin_xxx(...) { -/// // first row with height: 30 composed of two widgets -/// nk_layout_row_dynamic(&ctx, 30, 2); -/// nk_widget(...); -/// nk_widget(...); -/// // -/// // second row with same parameter as defined above -/// nk_widget(...); -/// nk_widget(...); -/// // -/// // third row uses 0 for height which will use auto layouting -/// nk_layout_row_dynamic(&ctx, 0, 2); -/// nk_widget(...); -/// nk_widget(...); -/// } -/// nk_end(...); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// 2. __nk_layout_row_static__

    -/// Another easy layouting function is `nk_layout_row_static`. It provides each -/// widget with same horizontal pixel width inside the row and does not grow -/// if the owning window scales smaller or bigger. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// if (nk_begin_xxx(...) { -/// // first row with height: 30 composed of two widgets with width: 80 -/// nk_layout_row_static(&ctx, 30, 80, 2); -/// nk_widget(...); -/// nk_widget(...); -/// // -/// // second row with same parameter as defined above -/// nk_widget(...); -/// nk_widget(...); -/// // -/// // third row uses 0 for height which will use auto layouting -/// nk_layout_row_static(&ctx, 0, 80, 2); -/// nk_widget(...); -/// nk_widget(...); -/// } -/// nk_end(...); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// 3. __nk_layout_row_xxx__

    -/// A little bit more advanced layouting API are functions `nk_layout_row_begin`, -/// `nk_layout_row_push` and `nk_layout_row_end`. They allow to directly -/// specify each column pixel or window ratio in a row. It supports either -/// directly setting per column pixel width or widget window ratio but not -/// both. Furthermore it is a immediate mode API so each value is directly -/// pushed before calling a widget. Therefore the layout is not automatically -/// repeating like the last two layouting functions. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// if (nk_begin_xxx(...) { -/// // first row with height: 25 composed of two widgets with width 60 and 40 -/// nk_layout_row_begin(ctx, NK_STATIC, 25, 2); -/// nk_layout_row_push(ctx, 60); -/// nk_widget(...); -/// nk_layout_row_push(ctx, 40); -/// nk_widget(...); -/// nk_layout_row_end(ctx); -/// // -/// // second row with height: 25 composed of two widgets with window ratio 0.25 and 0.75 -/// nk_layout_row_begin(ctx, NK_DYNAMIC, 25, 2); -/// nk_layout_row_push(ctx, 0.25f); -/// nk_widget(...); -/// nk_layout_row_push(ctx, 0.75f); -/// nk_widget(...); -/// nk_layout_row_end(ctx); -/// // -/// // third row with auto generated height: composed of two widgets with window ratio 0.25 and 0.75 -/// nk_layout_row_begin(ctx, NK_DYNAMIC, 0, 2); -/// nk_layout_row_push(ctx, 0.25f); -/// nk_widget(...); -/// nk_layout_row_push(ctx, 0.75f); -/// nk_widget(...); -/// nk_layout_row_end(ctx); -/// } -/// nk_end(...); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// 4. __nk_layout_row__

    -/// The array counterpart to API nk_layout_row_xxx is the single nk_layout_row -/// functions. Instead of pushing either pixel or window ratio for every widget -/// it allows to define it by array. The trade of for less control is that -/// `nk_layout_row` is automatically repeating. Otherwise the behavior is the -/// same. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// if (nk_begin_xxx(...) { -/// // two rows with height: 30 composed of two widgets with width 60 and 40 -/// const float ratio[] = {60,40}; -/// nk_layout_row(ctx, NK_STATIC, 30, 2, ratio); -/// nk_widget(...); -/// nk_widget(...); -/// nk_widget(...); -/// nk_widget(...); -/// // -/// // two rows with height: 30 composed of two widgets with window ratio 0.25 and 0.75 -/// const float ratio[] = {0.25, 0.75}; -/// nk_layout_row(ctx, NK_DYNAMIC, 30, 2, ratio); -/// nk_widget(...); -/// nk_widget(...); -/// nk_widget(...); -/// nk_widget(...); -/// // -/// // two rows with auto generated height composed of two widgets with window ratio 0.25 and 0.75 -/// const float ratio[] = {0.25, 0.75}; -/// nk_layout_row(ctx, NK_DYNAMIC, 30, 2, ratio); -/// nk_widget(...); -/// nk_widget(...); -/// nk_widget(...); -/// nk_widget(...); -/// } -/// nk_end(...); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// 5. __nk_layout_row_template_xxx__

    -/// The most complex and second most flexible API is a simplified flexbox version without -/// line wrapping and weights for dynamic widgets. It is an immediate mode API but -/// unlike `nk_layout_row_xxx` it has auto repeat behavior and needs to be called -/// before calling the templated widgets. -/// The row template layout has three different per widget size specifier. The first -/// one is the `nk_layout_row_template_push_static` with fixed widget pixel width. -/// They do not grow if the row grows and will always stay the same. -/// The second size specifier is `nk_layout_row_template_push_variable` -/// which defines a minimum widget size but it also can grow if more space is available -/// not taken by other widgets. -/// Finally there are dynamic widgets with `nk_layout_row_template_push_dynamic` -/// which are completely flexible and unlike variable widgets can even shrink -/// to zero if not enough space is provided. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// if (nk_begin_xxx(...) { -/// // two rows with height: 30 composed of three widgets -/// nk_layout_row_template_begin(ctx, 30); -/// nk_layout_row_template_push_dynamic(ctx); -/// nk_layout_row_template_push_variable(ctx, 80); -/// nk_layout_row_template_push_static(ctx, 80); -/// nk_layout_row_template_end(ctx); -/// // -/// // first row -/// nk_widget(...); // dynamic widget can go to zero if not enough space -/// nk_widget(...); // variable widget with min 80 pixel but can grow bigger if enough space -/// nk_widget(...); // static widget with fixed 80 pixel width -/// // -/// // second row same layout -/// nk_widget(...); -/// nk_widget(...); -/// nk_widget(...); -/// } -/// nk_end(...); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// 6. __nk_layout_space_xxx__

    -/// Finally the most flexible API directly allows you to place widgets inside the -/// window. The space layout API is an immediate mode API which does not support -/// row auto repeat and directly sets position and size of a widget. Position -/// and size hereby can be either specified as ratio of allocated space or -/// allocated space local position and pixel size. Since this API is quite -/// powerful there are a number of utility functions to get the available space -/// and convert between local allocated space and screen space. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// if (nk_begin_xxx(...) { -/// // static row with height: 500 (you can set column count to INT_MAX if you don't want to be bothered) -/// nk_layout_space_begin(ctx, NK_STATIC, 500, INT_MAX); -/// nk_layout_space_push(ctx, nk_rect(0,0,150,200)); -/// nk_widget(...); -/// nk_layout_space_push(ctx, nk_rect(200,200,100,200)); -/// nk_widget(...); -/// nk_layout_space_end(ctx); -/// // -/// // dynamic row with height: 500 (you can set column count to INT_MAX if you don't want to be bothered) -/// nk_layout_space_begin(ctx, NK_DYNAMIC, 500, INT_MAX); -/// nk_layout_space_push(ctx, nk_rect(0.5,0.5,0.1,0.1)); -/// nk_widget(...); -/// nk_layout_space_push(ctx, nk_rect(0.7,0.6,0.1,0.1)); -/// nk_widget(...); -/// } -/// nk_end(...); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// #### Reference -/// Function | Description -/// ----------------------------------------|------------------------------------ -/// nk_layout_set_min_row_height | Set the currently used minimum row height to a specified value -/// nk_layout_reset_min_row_height | Resets the currently used minimum row height to font height -/// nk_layout_widget_bounds | Calculates current width a static layout row can fit inside a window -/// nk_layout_ratio_from_pixel | Utility functions to calculate window ratio from pixel size -// -/// nk_layout_row_dynamic | Current layout is divided into n same sized growing columns -/// nk_layout_row_static | Current layout is divided into n same fixed sized columns -/// nk_layout_row_begin | Starts a new row with given height and number of columns -/// nk_layout_row_push | Pushes another column with given size or window ratio -/// nk_layout_row_end | Finished previously started row -/// nk_layout_row | Specifies row columns in array as either window ratio or size -// -/// nk_layout_row_template_begin | Begins the row template declaration -/// nk_layout_row_template_push_dynamic | Adds a dynamic column that dynamically grows and can go to zero if not enough space -/// nk_layout_row_template_push_variable | Adds a variable column that dynamically grows but does not shrink below specified pixel width -/// nk_layout_row_template_push_static | Adds a static column that does not grow and will always have the same size -/// nk_layout_row_template_end | Marks the end of the row template -// -/// nk_layout_space_begin | Begins a new layouting space that allows to specify each widgets position and size -/// nk_layout_space_push | Pushes position and size of the next widget in own coordinate space either as pixel or ratio -/// nk_layout_space_end | Marks the end of the layouting space -// -/// nk_layout_space_bounds | Callable after nk_layout_space_begin and returns total space allocated -/// nk_layout_space_to_screen | Converts vector from nk_layout_space coordinate space into screen space -/// nk_layout_space_to_local | Converts vector from screen space into nk_layout_space coordinates -/// nk_layout_space_rect_to_screen | Converts rectangle from nk_layout_space coordinate space into screen space -/// nk_layout_space_rect_to_local | Converts rectangle from screen space into nk_layout_space coordinates -*/ -/*/// #### nk_layout_set_min_row_height -/// Sets the currently used minimum row height. -/// !!! WARNING -/// The passed height needs to include both your preferred row height -/// as well as padding. No internal padding is added. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_layout_set_min_row_height(struct nk_context*, float height); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after call `nk_begin_xxx` -/// __height__ | New minimum row height to be used for auto generating the row height -*/ -NK_API void nk_layout_set_min_row_height(struct nk_context*, float height); -/*/// #### nk_layout_reset_min_row_height -/// Reset the currently used minimum row height back to `font_height + text_padding + padding` -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_layout_reset_min_row_height(struct nk_context*); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after call `nk_begin_xxx` -*/ -NK_API void nk_layout_reset_min_row_height(struct nk_context*); -/*/// #### nk_layout_widget_bounds -/// Returns the width of the next row allocate by one of the layouting functions -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// struct nk_rect nk_layout_widget_bounds(struct nk_context*); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after call `nk_begin_xxx` -/// -/// Return `nk_rect` with both position and size of the next row -*/ -NK_API struct nk_rect nk_layout_widget_bounds(struct nk_context*); -/*/// #### nk_layout_ratio_from_pixel -/// Utility functions to calculate window ratio from pixel size -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// float nk_layout_ratio_from_pixel(struct nk_context*, float pixel_width); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after call `nk_begin_xxx` -/// __pixel__ | Pixel_width to convert to window ratio -/// -/// Returns `nk_rect` with both position and size of the next row -*/ -NK_API float nk_layout_ratio_from_pixel(struct nk_context*, float pixel_width); -/*/// #### nk_layout_row_dynamic -/// Sets current row layout to share horizontal space -/// between @cols number of widgets evenly. Once called all subsequent widget -/// calls greater than @cols will allocate a new row with same layout. -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_layout_row_dynamic(struct nk_context *ctx, float height, int cols); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after call `nk_begin_xxx` -/// __height__ | Holds height of each widget in row or zero for auto layouting -/// __columns__ | Number of widget inside row -*/ -NK_API void nk_layout_row_dynamic(struct nk_context *ctx, float height, int cols); -/*/// #### nk_layout_row_static -/// Sets current row layout to fill @cols number of widgets -/// in row with same @item_width horizontal size. Once called all subsequent widget -/// calls greater than @cols will allocate a new row with same layout. -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_layout_row_static(struct nk_context *ctx, float height, int item_width, int cols); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after call `nk_begin_xxx` -/// __height__ | Holds height of each widget in row or zero for auto layouting -/// __width__ | Holds pixel width of each widget in the row -/// __columns__ | Number of widget inside row -*/ -NK_API void nk_layout_row_static(struct nk_context *ctx, float height, int item_width, int cols); -/*/// #### nk_layout_row_begin -/// Starts a new dynamic or fixed row with given height and columns. -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_layout_row_begin(struct nk_context *ctx, enum nk_layout_format fmt, float row_height, int cols); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after call `nk_begin_xxx` -/// __fmt__ | either `NK_DYNAMIC` for window ratio or `NK_STATIC` for fixed size columns -/// __height__ | holds height of each widget in row or zero for auto layouting -/// __columns__ | Number of widget inside row -*/ -NK_API void nk_layout_row_begin(struct nk_context *ctx, enum nk_layout_format fmt, float row_height, int cols); -/*/// #### nk_layout_row_push -/// Specifies either window ratio or width of a single column -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_layout_row_push(struct nk_context*, float value); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after call `nk_begin_xxx` -/// __value__ | either a window ratio or fixed width depending on @fmt in previous `nk_layout_row_begin` call -*/ -NK_API void nk_layout_row_push(struct nk_context*, float value); -/*/// #### nk_layout_row_end -/// Finished previously started row -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_layout_row_end(struct nk_context*); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after call `nk_begin_xxx` -*/ -NK_API void nk_layout_row_end(struct nk_context*); -/*/// #### nk_layout_row -/// Specifies row columns in array as either window ratio or size -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_layout_row(struct nk_context*, enum nk_layout_format, float height, int cols, const float *ratio); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after call `nk_begin_xxx` -/// __fmt__ | Either `NK_DYNAMIC` for window ratio or `NK_STATIC` for fixed size columns -/// __height__ | Holds height of each widget in row or zero for auto layouting -/// __columns__ | Number of widget inside row -*/ -NK_API void nk_layout_row(struct nk_context*, enum nk_layout_format, float height, int cols, const float *ratio); -/*/// #### nk_layout_row_template_begin -/// Begins the row template declaration -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_layout_row_template_begin(struct nk_context*, float row_height); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after call `nk_begin_xxx` -/// __height__ | Holds height of each widget in row or zero for auto layouting -*/ -NK_API void nk_layout_row_template_begin(struct nk_context*, float row_height); -/*/// #### nk_layout_row_template_push_dynamic -/// Adds a dynamic column that dynamically grows and can go to zero if not enough space -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_layout_row_template_push_dynamic(struct nk_context*); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after call `nk_begin_xxx` -/// __height__ | Holds height of each widget in row or zero for auto layouting -*/ -NK_API void nk_layout_row_template_push_dynamic(struct nk_context*); -/*/// #### nk_layout_row_template_push_variable -/// Adds a variable column that dynamically grows but does not shrink below specified pixel width -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_layout_row_template_push_variable(struct nk_context*, float min_width); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after call `nk_begin_xxx` -/// __width__ | Holds the minimum pixel width the next column must always be -*/ -NK_API void nk_layout_row_template_push_variable(struct nk_context*, float min_width); -/*/// #### nk_layout_row_template_push_static -/// Adds a static column that does not grow and will always have the same size -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_layout_row_template_push_static(struct nk_context*, float width); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after call `nk_begin_xxx` -/// __width__ | Holds the absolute pixel width value the next column must be -*/ -NK_API void nk_layout_row_template_push_static(struct nk_context*, float width); -/*/// #### nk_layout_row_template_end -/// Marks the end of the row template -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_layout_row_template_end(struct nk_context*); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after call `nk_begin_xxx` -*/ -NK_API void nk_layout_row_template_end(struct nk_context*); -/*/// #### nk_layout_space_begin -/// Begins a new layouting space that allows to specify each widgets position and size. -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_layout_space_begin(struct nk_context*, enum nk_layout_format, float height, int widget_count); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after call `nk_begin_xxx` -/// __fmt__ | Either `NK_DYNAMIC` for window ratio or `NK_STATIC` for fixed size columns -/// __height__ | Holds height of each widget in row or zero for auto layouting -/// __columns__ | Number of widgets inside row -*/ -NK_API void nk_layout_space_begin(struct nk_context*, enum nk_layout_format, float height, int widget_count); -/*/// #### nk_layout_space_push -/// Pushes position and size of the next widget in own coordinate space either as pixel or ratio -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_layout_space_push(struct nk_context *ctx, struct nk_rect bounds); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after call `nk_layout_space_begin` -/// __bounds__ | Position and size in laoyut space local coordinates -*/ -NK_API void nk_layout_space_push(struct nk_context*, struct nk_rect bounds); -/*/// #### nk_layout_space_end -/// Marks the end of the layout space -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_layout_space_end(struct nk_context*); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after call `nk_layout_space_begin` -*/ -NK_API void nk_layout_space_end(struct nk_context*); -/*/// #### nk_layout_space_bounds -/// Utility function to calculate total space allocated for `nk_layout_space` -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// struct nk_rect nk_layout_space_bounds(struct nk_context*); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after call `nk_layout_space_begin` -/// -/// Returns `nk_rect` holding the total space allocated -*/ -NK_API struct nk_rect nk_layout_space_bounds(struct nk_context*); -/*/// #### nk_layout_space_to_screen -/// Converts vector from nk_layout_space coordinate space into screen space -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// struct nk_vec2 nk_layout_space_to_screen(struct nk_context*, struct nk_vec2); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after call `nk_layout_space_begin` -/// __vec__ | Position to convert from layout space into screen coordinate space -/// -/// Returns transformed `nk_vec2` in screen space coordinates -*/ -NK_API struct nk_vec2 nk_layout_space_to_screen(struct nk_context*, struct nk_vec2); -/*/// #### nk_layout_space_to_local -/// Converts vector from layout space into screen space -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// struct nk_vec2 nk_layout_space_to_local(struct nk_context*, struct nk_vec2); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after call `nk_layout_space_begin` -/// __vec__ | Position to convert from screen space into layout coordinate space -/// -/// Returns transformed `nk_vec2` in layout space coordinates -*/ -NK_API struct nk_vec2 nk_layout_space_to_local(struct nk_context*, struct nk_vec2); -/*/// #### nk_layout_space_rect_to_screen -/// Converts rectangle from screen space into layout space -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// struct nk_rect nk_layout_space_rect_to_screen(struct nk_context*, struct nk_rect); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after call `nk_layout_space_begin` -/// __bounds__ | Rectangle to convert from layout space into screen space -/// -/// Returns transformed `nk_rect` in screen space coordinates -*/ -NK_API struct nk_rect nk_layout_space_rect_to_screen(struct nk_context*, struct nk_rect); -/*/// #### nk_layout_space_rect_to_local -/// Converts rectangle from layout space into screen space -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// struct nk_rect nk_layout_space_rect_to_local(struct nk_context*, struct nk_rect); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after call `nk_layout_space_begin` -/// __bounds__ | Rectangle to convert from layout space into screen space -/// -/// Returns transformed `nk_rect` in layout space coordinates -*/ -NK_API struct nk_rect nk_layout_space_rect_to_local(struct nk_context*, struct nk_rect); - -/*/// #### nk_spacer -/// Spacer is a dummy widget that consumes space as usual but doesn't draw anything -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_spacer(struct nk_context* ); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after call `nk_layout_space_begin` -/// -*/ -NK_API void nk_spacer(struct nk_context* ); - - -/* ============================================================================= - * - * GROUP - * - * ============================================================================= -/// ### Groups -/// Groups are basically windows inside windows. They allow to subdivide space -/// in a window to layout widgets as a group. Almost all more complex widget -/// layouting requirements can be solved using groups and basic layouting -/// fuctionality. Groups just like windows are identified by an unique name and -/// internally keep track of scrollbar offsets by default. However additional -/// versions are provided to directly manage the scrollbar. -/// -/// #### Usage -/// To create a group you have to call one of the three `nk_group_begin_xxx` -/// functions to start group declarations and `nk_group_end` at the end. Furthermore it -/// is required to check the return value of `nk_group_begin_xxx` and only process -/// widgets inside the window if the value is not 0. -/// Nesting groups is possible and even encouraged since many layouting schemes -/// can only be achieved by nesting. Groups, unlike windows, need `nk_group_end` -/// to be only called if the corresponding `nk_group_begin_xxx` call does not return 0: -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// if (nk_group_begin_xxx(ctx, ...) { -/// // [... widgets ...] -/// nk_group_end(ctx); -/// } -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// In the grand concept groups can be called after starting a window -/// with `nk_begin_xxx` and before calling `nk_end`: -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// struct nk_context ctx; -/// nk_init_xxx(&ctx, ...); -/// while (1) { -/// // Input -/// Event evt; -/// nk_input_begin(&ctx); -/// while (GetEvent(&evt)) { -/// if (evt.type == MOUSE_MOVE) -/// nk_input_motion(&ctx, evt.motion.x, evt.motion.y); -/// else if (evt.type == [...]) { -/// nk_input_xxx(...); -/// } -/// } -/// nk_input_end(&ctx); -/// // -/// // Window -/// if (nk_begin_xxx(...) { -/// // [...widgets...] -/// nk_layout_row_dynamic(...); -/// if (nk_group_begin_xxx(ctx, ...) { -/// //[... widgets ...] -/// nk_group_end(ctx); -/// } -/// } -/// nk_end(ctx); -/// // -/// // Draw -/// const struct nk_command *cmd = 0; -/// nk_foreach(cmd, &ctx) { -/// switch (cmd->type) { -/// case NK_COMMAND_LINE: -/// your_draw_line_function(...) -/// break; -/// case NK_COMMAND_RECT -/// your_draw_rect_function(...) -/// break; -/// case ...: -/// // [...] -/// } -/// nk_clear(&ctx); -/// } -/// nk_free(&ctx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// #### Reference -/// Function | Description -/// --------------------------------|------------------------------------------- -/// nk_group_begin | Start a new group with internal scrollbar handling -/// nk_group_begin_titled | Start a new group with separated name and title and internal scrollbar handling -/// nk_group_end | Ends a group. Should only be called if nk_group_begin returned non-zero -/// nk_group_scrolled_offset_begin | Start a new group with manual separated handling of scrollbar x- and y-offset -/// nk_group_scrolled_begin | Start a new group with manual scrollbar handling -/// nk_group_scrolled_end | Ends a group with manual scrollbar handling. Should only be called if nk_group_begin returned non-zero -/// nk_group_get_scroll | Gets the scroll offset for the given group -/// nk_group_set_scroll | Sets the scroll offset for the given group -*/ -/*/// #### nk_group_begin -/// Starts a new widget group. Requires a previous layouting function to specify a pos/size. -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// nk_bool nk_group_begin(struct nk_context*, const char *title, nk_flags); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __title__ | Must be an unique identifier for this group that is also used for the group header -/// __flags__ | Window flags defined in the nk_panel_flags section with a number of different group behaviors -/// -/// Returns `true(1)` if visible and fillable with widgets or `false(0)` otherwise -*/ -NK_API nk_bool nk_group_begin(struct nk_context*, const char *title, nk_flags); -/*/// #### nk_group_begin_titled -/// Starts a new widget group. Requires a previous layouting function to specify a pos/size. -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// nk_bool nk_group_begin_titled(struct nk_context*, const char *name, const char *title, nk_flags); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __id__ | Must be an unique identifier for this group -/// __title__ | Group header title -/// __flags__ | Window flags defined in the nk_panel_flags section with a number of different group behaviors -/// -/// Returns `true(1)` if visible and fillable with widgets or `false(0)` otherwise -*/ -NK_API nk_bool nk_group_begin_titled(struct nk_context*, const char *name, const char *title, nk_flags); -/*/// #### nk_group_end -/// Ends a widget group -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_group_end(struct nk_context*); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -*/ -NK_API void nk_group_end(struct nk_context*); -/*/// #### nk_group_scrolled_offset_begin -/// starts a new widget group. requires a previous layouting function to specify -/// a size. Does not keep track of scrollbar. -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// nk_bool nk_group_scrolled_offset_begin(struct nk_context*, nk_uint *x_offset, nk_uint *y_offset, const char *title, nk_flags flags); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __x_offset__| Scrollbar x-offset to offset all widgets inside the group horizontally. -/// __y_offset__| Scrollbar y-offset to offset all widgets inside the group vertically -/// __title__ | Window unique group title used to both identify and display in the group header -/// __flags__ | Window flags from the nk_panel_flags section -/// -/// Returns `true(1)` if visible and fillable with widgets or `false(0)` otherwise -*/ -NK_API nk_bool nk_group_scrolled_offset_begin(struct nk_context*, nk_uint *x_offset, nk_uint *y_offset, const char *title, nk_flags flags); -/*/// #### nk_group_scrolled_begin -/// Starts a new widget group. requires a previous -/// layouting function to specify a size. Does not keep track of scrollbar. -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// nk_bool nk_group_scrolled_begin(struct nk_context*, struct nk_scroll *off, const char *title, nk_flags); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __off__ | Both x- and y- scroll offset. Allows for manual scrollbar control -/// __title__ | Window unique group title used to both identify and display in the group header -/// __flags__ | Window flags from nk_panel_flags section -/// -/// Returns `true(1)` if visible and fillable with widgets or `false(0)` otherwise -*/ -NK_API nk_bool nk_group_scrolled_begin(struct nk_context*, struct nk_scroll *off, const char *title, nk_flags); -/*/// #### nk_group_scrolled_end -/// Ends a widget group after calling nk_group_scrolled_offset_begin or nk_group_scrolled_begin. -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_group_scrolled_end(struct nk_context*); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -*/ -NK_API void nk_group_scrolled_end(struct nk_context*); -/*/// #### nk_group_get_scroll -/// Gets the scroll position of the given group. -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_group_get_scroll(struct nk_context*, const char *id, nk_uint *x_offset, nk_uint *y_offset); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// -------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __id__ | The id of the group to get the scroll position of -/// __x_offset__ | A pointer to the x offset output (or NULL to ignore) -/// __y_offset__ | A pointer to the y offset output (or NULL to ignore) -*/ -NK_API void nk_group_get_scroll(struct nk_context*, const char *id, nk_uint *x_offset, nk_uint *y_offset); -/*/// #### nk_group_set_scroll -/// Sets the scroll position of the given group. -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_group_set_scroll(struct nk_context*, const char *id, nk_uint x_offset, nk_uint y_offset); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// -------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __id__ | The id of the group to scroll -/// __x_offset__ | The x offset to scroll to -/// __y_offset__ | The y offset to scroll to -*/ -NK_API void nk_group_set_scroll(struct nk_context*, const char *id, nk_uint x_offset, nk_uint y_offset); -/* ============================================================================= - * - * TREE - * - * ============================================================================= -/// ### Tree -/// Trees represent two different concept. First the concept of a collapsible -/// UI section that can be either in a hidden or visible state. They allow the UI -/// user to selectively minimize the current set of visible UI to comprehend. -/// The second concept are tree widgets for visual UI representation of trees.

    -/// -/// Trees thereby can be nested for tree representations and multiple nested -/// collapsible UI sections. All trees are started by calling of the -/// `nk_tree_xxx_push_tree` functions and ended by calling one of the -/// `nk_tree_xxx_pop_xxx()` functions. Each starting functions takes a title label -/// and optionally an image to be displayed and the initial collapse state from -/// the nk_collapse_states section.

    -/// -/// The runtime state of the tree is either stored outside the library by the caller -/// or inside which requires a unique ID. The unique ID can either be generated -/// automatically from `__FILE__` and `__LINE__` with function `nk_tree_push`, -/// by `__FILE__` and a user provided ID generated for example by loop index with -/// function `nk_tree_push_id` or completely provided from outside by user with -/// function `nk_tree_push_hashed`. -/// -/// #### Usage -/// To create a tree you have to call one of the seven `nk_tree_xxx_push_xxx` -/// functions to start a collapsible UI section and `nk_tree_xxx_pop` to mark the -/// end. -/// Each starting function will either return `false(0)` if the tree is collapsed -/// or hidden and therefore does not need to be filled with content or `true(1)` -/// if visible and required to be filled. -/// -/// !!! Note -/// The tree header does not require and layouting function and instead -/// calculates a auto height based on the currently used font size -/// -/// The tree ending functions only need to be called if the tree content is -/// actually visible. So make sure the tree push function is guarded by `if` -/// and the pop call is only taken if the tree is visible. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// if (nk_tree_push(ctx, NK_TREE_TAB, "Tree", NK_MINIMIZED)) { -/// nk_layout_row_dynamic(...); -/// nk_widget(...); -/// nk_tree_pop(ctx); -/// } -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// #### Reference -/// Function | Description -/// ----------------------------|------------------------------------------- -/// nk_tree_push | Start a collapsible UI section with internal state management -/// nk_tree_push_id | Start a collapsible UI section with internal state management callable in a look -/// nk_tree_push_hashed | Start a collapsible UI section with internal state management with full control over internal unique ID use to store state -/// nk_tree_image_push | Start a collapsible UI section with image and label header -/// nk_tree_image_push_id | Start a collapsible UI section with image and label header and internal state management callable in a look -/// nk_tree_image_push_hashed | Start a collapsible UI section with image and label header and internal state management with full control over internal unique ID use to store state -/// nk_tree_pop | Ends a collapsible UI section -// -/// nk_tree_state_push | Start a collapsible UI section with external state management -/// nk_tree_state_image_push | Start a collapsible UI section with image and label header and external state management -/// nk_tree_state_pop | Ends a collapsabale UI section -/// -/// #### nk_tree_type -/// Flag | Description -/// ----------------|---------------------------------------- -/// NK_TREE_NODE | Highlighted tree header to mark a collapsible UI section -/// NK_TREE_TAB | Non-highlighted tree header closer to tree representations -*/ -/*/// #### nk_tree_push -/// Starts a collapsible UI section with internal state management -/// !!! WARNING -/// To keep track of the runtime tree collapsible state this function uses -/// defines `__FILE__` and `__LINE__` to generate a unique ID. If you want -/// to call this function in a loop please use `nk_tree_push_id` or -/// `nk_tree_push_hashed` instead. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// #define nk_tree_push(ctx, type, title, state) -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __type__ | Value from the nk_tree_type section to visually mark a tree node header as either a collapseable UI section or tree node -/// __title__ | Label printed in the tree header -/// __state__ | Initial tree state value out of nk_collapse_states -/// -/// Returns `true(1)` if visible and fillable with widgets or `false(0)` otherwise -*/ -#define nk_tree_push(ctx, type, title, state) nk_tree_push_hashed(ctx, type, title, state, NK_FILE_LINE,nk_strlen(NK_FILE_LINE),__LINE__) -/*/// #### nk_tree_push_id -/// Starts a collapsible UI section with internal state management callable in a look -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// #define nk_tree_push_id(ctx, type, title, state, id) -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __type__ | Value from the nk_tree_type section to visually mark a tree node header as either a collapseable UI section or tree node -/// __title__ | Label printed in the tree header -/// __state__ | Initial tree state value out of nk_collapse_states -/// __id__ | Loop counter index if this function is called in a loop -/// -/// Returns `true(1)` if visible and fillable with widgets or `false(0)` otherwise -*/ -#define nk_tree_push_id(ctx, type, title, state, id) nk_tree_push_hashed(ctx, type, title, state, NK_FILE_LINE,nk_strlen(NK_FILE_LINE),id) -/*/// #### nk_tree_push_hashed -/// Start a collapsible UI section with internal state management with full -/// control over internal unique ID used to store state -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// nk_bool nk_tree_push_hashed(struct nk_context*, enum nk_tree_type, const char *title, enum nk_collapse_states initial_state, const char *hash, int len,int seed); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __type__ | Value from the nk_tree_type section to visually mark a tree node header as either a collapseable UI section or tree node -/// __title__ | Label printed in the tree header -/// __state__ | Initial tree state value out of nk_collapse_states -/// __hash__ | Memory block or string to generate the ID from -/// __len__ | Size of passed memory block or string in __hash__ -/// __seed__ | Seeding value if this function is called in a loop or default to `0` -/// -/// Returns `true(1)` if visible and fillable with widgets or `false(0)` otherwise -*/ -NK_API nk_bool nk_tree_push_hashed(struct nk_context*, enum nk_tree_type, const char *title, enum nk_collapse_states initial_state, const char *hash, int len,int seed); -/*/// #### nk_tree_image_push -/// Start a collapsible UI section with image and label header -/// !!! WARNING -/// To keep track of the runtime tree collapsible state this function uses -/// defines `__FILE__` and `__LINE__` to generate a unique ID. If you want -/// to call this function in a loop please use `nk_tree_image_push_id` or -/// `nk_tree_image_push_hashed` instead. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// #define nk_tree_image_push(ctx, type, img, title, state) -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __type__ | Value from the nk_tree_type section to visually mark a tree node header as either a collapseable UI section or tree node -/// __img__ | Image to display inside the header on the left of the label -/// __title__ | Label printed in the tree header -/// __state__ | Initial tree state value out of nk_collapse_states -/// -/// Returns `true(1)` if visible and fillable with widgets or `false(0)` otherwise -*/ -#define nk_tree_image_push(ctx, type, img, title, state) nk_tree_image_push_hashed(ctx, type, img, title, state, NK_FILE_LINE,nk_strlen(NK_FILE_LINE),__LINE__) -/*/// #### nk_tree_image_push_id -/// Start a collapsible UI section with image and label header and internal state -/// management callable in a look -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// #define nk_tree_image_push_id(ctx, type, img, title, state, id) -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __type__ | Value from the nk_tree_type section to visually mark a tree node header as either a collapseable UI section or tree node -/// __img__ | Image to display inside the header on the left of the label -/// __title__ | Label printed in the tree header -/// __state__ | Initial tree state value out of nk_collapse_states -/// __id__ | Loop counter index if this function is called in a loop -/// -/// Returns `true(1)` if visible and fillable with widgets or `false(0)` otherwise -*/ -#define nk_tree_image_push_id(ctx, type, img, title, state, id) nk_tree_image_push_hashed(ctx, type, img, title, state, NK_FILE_LINE,nk_strlen(NK_FILE_LINE),id) -/*/// #### nk_tree_image_push_hashed -/// Start a collapsible UI section with internal state management with full -/// control over internal unique ID used to store state -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// nk_bool nk_tree_image_push_hashed(struct nk_context*, enum nk_tree_type, struct nk_image, const char *title, enum nk_collapse_states initial_state, const char *hash, int len,int seed); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct -/// __type__ | Value from the nk_tree_type section to visually mark a tree node header as either a collapseable UI section or tree node -/// __img__ | Image to display inside the header on the left of the label -/// __title__ | Label printed in the tree header -/// __state__ | Initial tree state value out of nk_collapse_states -/// __hash__ | Memory block or string to generate the ID from -/// __len__ | Size of passed memory block or string in __hash__ -/// __seed__ | Seeding value if this function is called in a loop or default to `0` -/// -/// Returns `true(1)` if visible and fillable with widgets or `false(0)` otherwise -*/ -NK_API nk_bool nk_tree_image_push_hashed(struct nk_context*, enum nk_tree_type, struct nk_image, const char *title, enum nk_collapse_states initial_state, const char *hash, int len,int seed); -/*/// #### nk_tree_pop -/// Ends a collapsabale UI section -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_tree_pop(struct nk_context*); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after calling `nk_tree_xxx_push_xxx` -*/ -NK_API void nk_tree_pop(struct nk_context*); -/*/// #### nk_tree_state_push -/// Start a collapsible UI section with external state management -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// nk_bool nk_tree_state_push(struct nk_context*, enum nk_tree_type, const char *title, enum nk_collapse_states *state); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after calling `nk_tree_xxx_push_xxx` -/// __type__ | Value from the nk_tree_type section to visually mark a tree node header as either a collapseable UI section or tree node -/// __title__ | Label printed in the tree header -/// __state__ | Persistent state to update -/// -/// Returns `true(1)` if visible and fillable with widgets or `false(0)` otherwise -*/ -NK_API nk_bool nk_tree_state_push(struct nk_context*, enum nk_tree_type, const char *title, enum nk_collapse_states *state); -/*/// #### nk_tree_state_image_push -/// Start a collapsible UI section with image and label header and external state management -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// nk_bool nk_tree_state_image_push(struct nk_context*, enum nk_tree_type, struct nk_image, const char *title, enum nk_collapse_states *state); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after calling `nk_tree_xxx_push_xxx` -/// __img__ | Image to display inside the header on the left of the label -/// __type__ | Value from the nk_tree_type section to visually mark a tree node header as either a collapseable UI section or tree node -/// __title__ | Label printed in the tree header -/// __state__ | Persistent state to update -/// -/// Returns `true(1)` if visible and fillable with widgets or `false(0)` otherwise -*/ -NK_API nk_bool nk_tree_state_image_push(struct nk_context*, enum nk_tree_type, struct nk_image, const char *title, enum nk_collapse_states *state); -/*/// #### nk_tree_state_pop -/// Ends a collapsabale UI section -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_tree_state_pop(struct nk_context*); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// ------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after calling `nk_tree_xxx_push_xxx` -*/ -NK_API void nk_tree_state_pop(struct nk_context*); - -#define nk_tree_element_push(ctx, type, title, state, sel) nk_tree_element_push_hashed(ctx, type, title, state, sel, NK_FILE_LINE,nk_strlen(NK_FILE_LINE),__LINE__) -#define nk_tree_element_push_id(ctx, type, title, state, sel, id) nk_tree_element_push_hashed(ctx, type, title, state, sel, NK_FILE_LINE,nk_strlen(NK_FILE_LINE),id) -NK_API nk_bool nk_tree_element_push_hashed(struct nk_context*, enum nk_tree_type, const char *title, enum nk_collapse_states initial_state, nk_bool *selected, const char *hash, int len, int seed); -NK_API nk_bool nk_tree_element_image_push_hashed(struct nk_context*, enum nk_tree_type, struct nk_image, const char *title, enum nk_collapse_states initial_state, nk_bool *selected, const char *hash, int len,int seed); -NK_API void nk_tree_element_pop(struct nk_context*); - -/* ============================================================================= - * - * LIST VIEW - * - * ============================================================================= */ -struct nk_list_view { -/* public: */ - int begin, end, count; -/* private: */ - int total_height; - struct nk_context *ctx; - nk_uint *scroll_pointer; - nk_uint scroll_value; -}; -NK_API nk_bool nk_list_view_begin(struct nk_context*, struct nk_list_view *out, const char *id, nk_flags, int row_height, int row_count); -NK_API void nk_list_view_end(struct nk_list_view*); -/* ============================================================================= - * - * WIDGET - * - * ============================================================================= */ -enum nk_widget_layout_states { - NK_WIDGET_INVALID, /* The widget cannot be seen and is completely out of view */ - NK_WIDGET_VALID, /* The widget is completely inside the window and can be updated and drawn */ - NK_WIDGET_ROM /* The widget is partially visible and cannot be updated */ -}; -enum nk_widget_states { - NK_WIDGET_STATE_MODIFIED = NK_FLAG(1), - NK_WIDGET_STATE_INACTIVE = NK_FLAG(2), /* widget is neither active nor hovered */ - NK_WIDGET_STATE_ENTERED = NK_FLAG(3), /* widget has been hovered on the current frame */ - NK_WIDGET_STATE_HOVER = NK_FLAG(4), /* widget is being hovered */ - NK_WIDGET_STATE_ACTIVED = NK_FLAG(5),/* widget is currently activated */ - NK_WIDGET_STATE_LEFT = NK_FLAG(6), /* widget is from this frame on not hovered anymore */ - NK_WIDGET_STATE_HOVERED = NK_WIDGET_STATE_HOVER|NK_WIDGET_STATE_MODIFIED, /* widget is being hovered */ - NK_WIDGET_STATE_ACTIVE = NK_WIDGET_STATE_ACTIVED|NK_WIDGET_STATE_MODIFIED /* widget is currently activated */ -}; -NK_API enum nk_widget_layout_states nk_widget(struct nk_rect*, const struct nk_context*); -NK_API enum nk_widget_layout_states nk_widget_fitting(struct nk_rect*, struct nk_context*, struct nk_vec2); -NK_API struct nk_rect nk_widget_bounds(struct nk_context*); -NK_API struct nk_vec2 nk_widget_position(struct nk_context*); -NK_API struct nk_vec2 nk_widget_size(struct nk_context*); -NK_API float nk_widget_width(struct nk_context*); -NK_API float nk_widget_height(struct nk_context*); -NK_API nk_bool nk_widget_is_hovered(struct nk_context*); -NK_API nk_bool nk_widget_is_mouse_clicked(struct nk_context*, enum nk_buttons); -NK_API nk_bool nk_widget_has_mouse_click_down(struct nk_context*, enum nk_buttons, nk_bool down); -NK_API void nk_spacing(struct nk_context*, int cols); -/* ============================================================================= - * - * TEXT - * - * ============================================================================= */ -enum nk_text_align { - NK_TEXT_ALIGN_LEFT = 0x01, - NK_TEXT_ALIGN_CENTERED = 0x02, - NK_TEXT_ALIGN_RIGHT = 0x04, - NK_TEXT_ALIGN_TOP = 0x08, - NK_TEXT_ALIGN_MIDDLE = 0x10, - NK_TEXT_ALIGN_BOTTOM = 0x20 -}; -enum nk_text_alignment { - NK_TEXT_LEFT = NK_TEXT_ALIGN_MIDDLE|NK_TEXT_ALIGN_LEFT, - NK_TEXT_CENTERED = NK_TEXT_ALIGN_MIDDLE|NK_TEXT_ALIGN_CENTERED, - NK_TEXT_RIGHT = NK_TEXT_ALIGN_MIDDLE|NK_TEXT_ALIGN_RIGHT -}; -NK_API void nk_text(struct nk_context*, const char*, int, nk_flags); -NK_API void nk_text_colored(struct nk_context*, const char*, int, nk_flags, struct nk_color); -NK_API void nk_text_wrap(struct nk_context*, const char*, int); -NK_API void nk_text_wrap_colored(struct nk_context*, const char*, int, struct nk_color); -NK_API void nk_label(struct nk_context*, const char*, nk_flags align); -NK_API void nk_label_colored(struct nk_context*, const char*, nk_flags align, struct nk_color); -NK_API void nk_label_wrap(struct nk_context*, const char*); -NK_API void nk_label_colored_wrap(struct nk_context*, const char*, struct nk_color); -NK_API void nk_image(struct nk_context*, struct nk_image); -NK_API void nk_image_color(struct nk_context*, struct nk_image, struct nk_color); -#ifdef NK_INCLUDE_STANDARD_VARARGS -NK_API void nk_labelf(struct nk_context*, nk_flags, NK_PRINTF_FORMAT_STRING const char*, ...) NK_PRINTF_VARARG_FUNC(3); -NK_API void nk_labelf_colored(struct nk_context*, nk_flags, struct nk_color, NK_PRINTF_FORMAT_STRING const char*,...) NK_PRINTF_VARARG_FUNC(4); -NK_API void nk_labelf_wrap(struct nk_context*, NK_PRINTF_FORMAT_STRING const char*,...) NK_PRINTF_VARARG_FUNC(2); -NK_API void nk_labelf_colored_wrap(struct nk_context*, struct nk_color, NK_PRINTF_FORMAT_STRING const char*,...) NK_PRINTF_VARARG_FUNC(3); -NK_API void nk_labelfv(struct nk_context*, nk_flags, NK_PRINTF_FORMAT_STRING const char*, va_list) NK_PRINTF_VALIST_FUNC(3); -NK_API void nk_labelfv_colored(struct nk_context*, nk_flags, struct nk_color, NK_PRINTF_FORMAT_STRING const char*, va_list) NK_PRINTF_VALIST_FUNC(4); -NK_API void nk_labelfv_wrap(struct nk_context*, NK_PRINTF_FORMAT_STRING const char*, va_list) NK_PRINTF_VALIST_FUNC(2); -NK_API void nk_labelfv_colored_wrap(struct nk_context*, struct nk_color, NK_PRINTF_FORMAT_STRING const char*, va_list) NK_PRINTF_VALIST_FUNC(3); -NK_API void nk_value_bool(struct nk_context*, const char *prefix, int); -NK_API void nk_value_int(struct nk_context*, const char *prefix, int); -NK_API void nk_value_uint(struct nk_context*, const char *prefix, unsigned int); -NK_API void nk_value_float(struct nk_context*, const char *prefix, float); -NK_API void nk_value_color_byte(struct nk_context*, const char *prefix, struct nk_color); -NK_API void nk_value_color_float(struct nk_context*, const char *prefix, struct nk_color); -NK_API void nk_value_color_hex(struct nk_context*, const char *prefix, struct nk_color); -#endif -/* ============================================================================= - * - * BUTTON - * - * ============================================================================= */ -NK_API nk_bool nk_button_text(struct nk_context*, const char *title, int len); -NK_API nk_bool nk_button_label(struct nk_context*, const char *title); -NK_API nk_bool nk_button_color(struct nk_context*, struct nk_color); -NK_API nk_bool nk_button_symbol(struct nk_context*, enum nk_symbol_type); -NK_API nk_bool nk_button_image(struct nk_context*, struct nk_image img); -NK_API nk_bool nk_button_symbol_label(struct nk_context*, enum nk_symbol_type, const char*, nk_flags text_alignment); -NK_API nk_bool nk_button_symbol_text(struct nk_context*, enum nk_symbol_type, const char*, int, nk_flags alignment); -NK_API nk_bool nk_button_image_label(struct nk_context*, struct nk_image img, const char*, nk_flags text_alignment); -NK_API nk_bool nk_button_image_text(struct nk_context*, struct nk_image img, const char*, int, nk_flags alignment); -NK_API nk_bool nk_button_text_styled(struct nk_context*, const struct nk_style_button*, const char *title, int len); -NK_API nk_bool nk_button_label_styled(struct nk_context*, const struct nk_style_button*, const char *title); -NK_API nk_bool nk_button_symbol_styled(struct nk_context*, const struct nk_style_button*, enum nk_symbol_type); -NK_API nk_bool nk_button_image_styled(struct nk_context*, const struct nk_style_button*, struct nk_image img); -NK_API nk_bool nk_button_symbol_text_styled(struct nk_context*,const struct nk_style_button*, enum nk_symbol_type, const char*, int, nk_flags alignment); -NK_API nk_bool nk_button_symbol_label_styled(struct nk_context *ctx, const struct nk_style_button *style, enum nk_symbol_type symbol, const char *title, nk_flags align); -NK_API nk_bool nk_button_image_label_styled(struct nk_context*,const struct nk_style_button*, struct nk_image img, const char*, nk_flags text_alignment); -NK_API nk_bool nk_button_image_text_styled(struct nk_context*,const struct nk_style_button*, struct nk_image img, const char*, int, nk_flags alignment); -NK_API void nk_button_set_behavior(struct nk_context*, enum nk_button_behavior); -NK_API nk_bool nk_button_push_behavior(struct nk_context*, enum nk_button_behavior); -NK_API nk_bool nk_button_pop_behavior(struct nk_context*); -/* ============================================================================= - * - * CHECKBOX - * - * ============================================================================= */ -NK_API nk_bool nk_check_label(struct nk_context*, const char*, nk_bool active); -NK_API nk_bool nk_check_text(struct nk_context*, const char*, int, nk_bool active); -NK_API unsigned nk_check_flags_label(struct nk_context*, const char*, unsigned int flags, unsigned int value); -NK_API unsigned nk_check_flags_text(struct nk_context*, const char*, int, unsigned int flags, unsigned int value); -NK_API nk_bool nk_checkbox_label(struct nk_context*, const char*, nk_bool *active); -NK_API nk_bool nk_checkbox_text(struct nk_context*, const char*, int, nk_bool *active); -NK_API nk_bool nk_checkbox_flags_label(struct nk_context*, const char*, unsigned int *flags, unsigned int value); -NK_API nk_bool nk_checkbox_flags_text(struct nk_context*, const char*, int, unsigned int *flags, unsigned int value); -/* ============================================================================= - * - * RADIO BUTTON - * - * ============================================================================= */ -NK_API nk_bool nk_radio_label(struct nk_context*, const char*, nk_bool *active); -NK_API nk_bool nk_radio_text(struct nk_context*, const char*, int, nk_bool *active); -NK_API nk_bool nk_option_label(struct nk_context*, const char*, nk_bool active); -NK_API nk_bool nk_option_text(struct nk_context*, const char*, int, nk_bool active); -/* ============================================================================= - * - * SELECTABLE - * - * ============================================================================= */ -NK_API nk_bool nk_selectable_label(struct nk_context*, const char*, nk_flags align, nk_bool *value); -NK_API nk_bool nk_selectable_text(struct nk_context*, const char*, int, nk_flags align, nk_bool *value); -NK_API nk_bool nk_selectable_image_label(struct nk_context*,struct nk_image, const char*, nk_flags align, nk_bool *value); -NK_API nk_bool nk_selectable_image_text(struct nk_context*,struct nk_image, const char*, int, nk_flags align, nk_bool *value); -NK_API nk_bool nk_selectable_symbol_label(struct nk_context*,enum nk_symbol_type, const char*, nk_flags align, nk_bool *value); -NK_API nk_bool nk_selectable_symbol_text(struct nk_context*,enum nk_symbol_type, const char*, int, nk_flags align, nk_bool *value); - -NK_API nk_bool nk_select_label(struct nk_context*, const char*, nk_flags align, nk_bool value); -NK_API nk_bool nk_select_text(struct nk_context*, const char*, int, nk_flags align, nk_bool value); -NK_API nk_bool nk_select_image_label(struct nk_context*, struct nk_image,const char*, nk_flags align, nk_bool value); -NK_API nk_bool nk_select_image_text(struct nk_context*, struct nk_image,const char*, int, nk_flags align, nk_bool value); -NK_API nk_bool nk_select_symbol_label(struct nk_context*,enum nk_symbol_type, const char*, nk_flags align, nk_bool value); -NK_API nk_bool nk_select_symbol_text(struct nk_context*,enum nk_symbol_type, const char*, int, nk_flags align, nk_bool value); - -/* ============================================================================= - * - * SLIDER - * - * ============================================================================= */ -NK_API float nk_slide_float(struct nk_context*, float min, float val, float max, float step); -NK_API int nk_slide_int(struct nk_context*, int min, int val, int max, int step); -NK_API nk_bool nk_slider_float(struct nk_context*, float min, float *val, float max, float step); -NK_API nk_bool nk_slider_int(struct nk_context*, int min, int *val, int max, int step); -/* ============================================================================= - * - * PROGRESSBAR - * - * ============================================================================= */ -NK_API nk_bool nk_progress(struct nk_context*, nk_size *cur, nk_size max, nk_bool modifyable); -NK_API nk_size nk_prog(struct nk_context*, nk_size cur, nk_size max, nk_bool modifyable); - -/* ============================================================================= - * - * COLOR PICKER - * - * ============================================================================= */ -NK_API struct nk_colorf nk_color_picker(struct nk_context*, struct nk_colorf, enum nk_color_format); -NK_API nk_bool nk_color_pick(struct nk_context*, struct nk_colorf*, enum nk_color_format); -/* ============================================================================= - * - * PROPERTIES - * - * ============================================================================= -/// ### Properties -/// Properties are the main value modification widgets in Nuklear. Changing a value -/// can be achieved by dragging, adding/removing incremental steps on button click -/// or by directly typing a number. -/// -/// #### Usage -/// Each property requires a unique name for identification that is also used for -/// displaying a label. If you want to use the same name multiple times make sure -/// add a '#' before your name. The '#' will not be shown but will generate a -/// unique ID. Each property also takes in a minimum and maximum value. If you want -/// to make use of the complete number range of a type just use the provided -/// type limits from `limits.h`. For example `INT_MIN` and `INT_MAX` for -/// `nk_property_int` and `nk_propertyi`. In additional each property takes in -/// a increment value that will be added or subtracted if either the increment -/// decrement button is clicked. Finally there is a value for increment per pixel -/// dragged that is added or subtracted from the value. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// int value = 0; -/// struct nk_context ctx; -/// nk_init_xxx(&ctx, ...); -/// while (1) { -/// // Input -/// Event evt; -/// nk_input_begin(&ctx); -/// while (GetEvent(&evt)) { -/// if (evt.type == MOUSE_MOVE) -/// nk_input_motion(&ctx, evt.motion.x, evt.motion.y); -/// else if (evt.type == [...]) { -/// nk_input_xxx(...); -/// } -/// } -/// nk_input_end(&ctx); -/// // -/// // Window -/// if (nk_begin_xxx(...) { -/// // Property -/// nk_layout_row_dynamic(...); -/// nk_property_int(ctx, "ID", INT_MIN, &value, INT_MAX, 1, 1); -/// } -/// nk_end(ctx); -/// // -/// // Draw -/// const struct nk_command *cmd = 0; -/// nk_foreach(cmd, &ctx) { -/// switch (cmd->type) { -/// case NK_COMMAND_LINE: -/// your_draw_line_function(...) -/// break; -/// case NK_COMMAND_RECT -/// your_draw_rect_function(...) -/// break; -/// case ...: -/// // [...] -/// } -/// nk_clear(&ctx); -/// } -/// nk_free(&ctx); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// #### Reference -/// Function | Description -/// --------------------|------------------------------------------- -/// nk_property_int | Integer property directly modifying a passed in value -/// nk_property_float | Float property directly modifying a passed in value -/// nk_property_double | Double property directly modifying a passed in value -/// nk_propertyi | Integer property returning the modified int value -/// nk_propertyf | Float property returning the modified float value -/// nk_propertyd | Double property returning the modified double value -/// -*/ -/*/// #### nk_property_int -/// Integer property directly modifying a passed in value -/// !!! WARNING -/// To generate a unique property ID using the same label make sure to insert -/// a `#` at the beginning. It will not be shown but guarantees correct behavior. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_property_int(struct nk_context *ctx, const char *name, int min, int *val, int max, int step, float inc_per_pixel); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// --------------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after calling a layouting function -/// __name__ | String used both as a label as well as a unique identifier -/// __min__ | Minimum value not allowed to be underflown -/// __val__ | Integer pointer to be modified -/// __max__ | Maximum value not allowed to be overflown -/// __step__ | Increment added and subtracted on increment and decrement button -/// __inc_per_pixel__ | Value per pixel added or subtracted on dragging -*/ -NK_API void nk_property_int(struct nk_context*, const char *name, int min, int *val, int max, int step, float inc_per_pixel); -/*/// #### nk_property_float -/// Float property directly modifying a passed in value -/// !!! WARNING -/// To generate a unique property ID using the same label make sure to insert -/// a `#` at the beginning. It will not be shown but guarantees correct behavior. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_property_float(struct nk_context *ctx, const char *name, float min, float *val, float max, float step, float inc_per_pixel); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// --------------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after calling a layouting function -/// __name__ | String used both as a label as well as a unique identifier -/// __min__ | Minimum value not allowed to be underflown -/// __val__ | Float pointer to be modified -/// __max__ | Maximum value not allowed to be overflown -/// __step__ | Increment added and subtracted on increment and decrement button -/// __inc_per_pixel__ | Value per pixel added or subtracted on dragging -*/ -NK_API void nk_property_float(struct nk_context*, const char *name, float min, float *val, float max, float step, float inc_per_pixel); -/*/// #### nk_property_double -/// Double property directly modifying a passed in value -/// !!! WARNING -/// To generate a unique property ID using the same label make sure to insert -/// a `#` at the beginning. It will not be shown but guarantees correct behavior. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// void nk_property_double(struct nk_context *ctx, const char *name, double min, double *val, double max, double step, double inc_per_pixel); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// --------------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after calling a layouting function -/// __name__ | String used both as a label as well as a unique identifier -/// __min__ | Minimum value not allowed to be underflown -/// __val__ | Double pointer to be modified -/// __max__ | Maximum value not allowed to be overflown -/// __step__ | Increment added and subtracted on increment and decrement button -/// __inc_per_pixel__ | Value per pixel added or subtracted on dragging -*/ -NK_API void nk_property_double(struct nk_context*, const char *name, double min, double *val, double max, double step, float inc_per_pixel); -/*/// #### nk_propertyi -/// Integer property modifying a passed in value and returning the new value -/// !!! WARNING -/// To generate a unique property ID using the same label make sure to insert -/// a `#` at the beginning. It will not be shown but guarantees correct behavior. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// int nk_propertyi(struct nk_context *ctx, const char *name, int min, int val, int max, int step, float inc_per_pixel); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// --------------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after calling a layouting function -/// __name__ | String used both as a label as well as a unique identifier -/// __min__ | Minimum value not allowed to be underflown -/// __val__ | Current integer value to be modified and returned -/// __max__ | Maximum value not allowed to be overflown -/// __step__ | Increment added and subtracted on increment and decrement button -/// __inc_per_pixel__ | Value per pixel added or subtracted on dragging -/// -/// Returns the new modified integer value -*/ -NK_API int nk_propertyi(struct nk_context*, const char *name, int min, int val, int max, int step, float inc_per_pixel); -/*/// #### nk_propertyf -/// Float property modifying a passed in value and returning the new value -/// !!! WARNING -/// To generate a unique property ID using the same label make sure to insert -/// a `#` at the beginning. It will not be shown but guarantees correct behavior. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// float nk_propertyf(struct nk_context *ctx, const char *name, float min, float val, float max, float step, float inc_per_pixel); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// --------------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after calling a layouting function -/// __name__ | String used both as a label as well as a unique identifier -/// __min__ | Minimum value not allowed to be underflown -/// __val__ | Current float value to be modified and returned -/// __max__ | Maximum value not allowed to be overflown -/// __step__ | Increment added and subtracted on increment and decrement button -/// __inc_per_pixel__ | Value per pixel added or subtracted on dragging -/// -/// Returns the new modified float value -*/ -NK_API float nk_propertyf(struct nk_context*, const char *name, float min, float val, float max, float step, float inc_per_pixel); -/*/// #### nk_propertyd -/// Float property modifying a passed in value and returning the new value -/// !!! WARNING -/// To generate a unique property ID using the same label make sure to insert -/// a `#` at the beginning. It will not be shown but guarantees correct behavior. -/// -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~c -/// float nk_propertyd(struct nk_context *ctx, const char *name, double min, double val, double max, double step, double inc_per_pixel); -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// -/// Parameter | Description -/// --------------------|----------------------------------------------------------- -/// __ctx__ | Must point to an previously initialized `nk_context` struct after calling a layouting function -/// __name__ | String used both as a label as well as a unique identifier -/// __min__ | Minimum value not allowed to be underflown -/// __val__ | Current double value to be modified and returned -/// __max__ | Maximum value not allowed to be overflown -/// __step__ | Increment added and subtracted on increment and decrement button -/// __inc_per_pixel__ | Value per pixel added or subtracted on dragging -/// -/// Returns the new modified double value -*/ -NK_API double nk_propertyd(struct nk_context*, const char *name, double min, double val, double max, double step, float inc_per_pixel); -/* ============================================================================= - * - * TEXT EDIT - * - * ============================================================================= */ -enum nk_edit_flags { - NK_EDIT_DEFAULT = 0, - NK_EDIT_READ_ONLY = NK_FLAG(0), - NK_EDIT_AUTO_SELECT = NK_FLAG(1), - NK_EDIT_SIG_ENTER = NK_FLAG(2), - NK_EDIT_ALLOW_TAB = NK_FLAG(3), - NK_EDIT_NO_CURSOR = NK_FLAG(4), - NK_EDIT_SELECTABLE = NK_FLAG(5), - NK_EDIT_CLIPBOARD = NK_FLAG(6), - NK_EDIT_CTRL_ENTER_NEWLINE = NK_FLAG(7), - NK_EDIT_NO_HORIZONTAL_SCROLL = NK_FLAG(8), - NK_EDIT_ALWAYS_INSERT_MODE = NK_FLAG(9), - NK_EDIT_MULTILINE = NK_FLAG(10), - NK_EDIT_GOTO_END_ON_ACTIVATE = NK_FLAG(11) -}; -enum nk_edit_types { - NK_EDIT_SIMPLE = NK_EDIT_ALWAYS_INSERT_MODE, - NK_EDIT_FIELD = NK_EDIT_SIMPLE|NK_EDIT_SELECTABLE|NK_EDIT_CLIPBOARD, - NK_EDIT_BOX = NK_EDIT_ALWAYS_INSERT_MODE| NK_EDIT_SELECTABLE| NK_EDIT_MULTILINE|NK_EDIT_ALLOW_TAB|NK_EDIT_CLIPBOARD, - NK_EDIT_EDITOR = NK_EDIT_SELECTABLE|NK_EDIT_MULTILINE|NK_EDIT_ALLOW_TAB| NK_EDIT_CLIPBOARD -}; -enum nk_edit_events { - NK_EDIT_ACTIVE = NK_FLAG(0), /* edit widget is currently being modified */ - NK_EDIT_INACTIVE = NK_FLAG(1), /* edit widget is not active and is not being modified */ - NK_EDIT_ACTIVATED = NK_FLAG(2), /* edit widget went from state inactive to state active */ - NK_EDIT_DEACTIVATED = NK_FLAG(3), /* edit widget went from state active to state inactive */ - NK_EDIT_COMMITED = NK_FLAG(4) /* edit widget has received an enter and lost focus */ -}; -NK_API nk_flags nk_edit_string(struct nk_context*, nk_flags, char *buffer, int *len, int max, nk_plugin_filter); -NK_API nk_flags nk_edit_string_zero_terminated(struct nk_context*, nk_flags, char *buffer, int max, nk_plugin_filter); -NK_API nk_flags nk_edit_buffer(struct nk_context*, nk_flags, struct nk_text_edit*, nk_plugin_filter); -NK_API void nk_edit_focus(struct nk_context*, nk_flags flags); -NK_API void nk_edit_unfocus(struct nk_context*); -/* ============================================================================= - * - * CHART - * - * ============================================================================= */ -NK_API nk_bool nk_chart_begin(struct nk_context*, enum nk_chart_type, int num, float min, float max); -NK_API nk_bool nk_chart_begin_colored(struct nk_context*, enum nk_chart_type, struct nk_color, struct nk_color active, int num, float min, float max); -NK_API void nk_chart_add_slot(struct nk_context *ctx, const enum nk_chart_type, int count, float min_value, float max_value); -NK_API void nk_chart_add_slot_colored(struct nk_context *ctx, const enum nk_chart_type, struct nk_color, struct nk_color active, int count, float min_value, float max_value); -NK_API nk_flags nk_chart_push(struct nk_context*, float); -NK_API nk_flags nk_chart_push_slot(struct nk_context*, float, int); -NK_API void nk_chart_end(struct nk_context*); -NK_API void nk_plot(struct nk_context*, enum nk_chart_type, const float *values, int count, int offset); -NK_API void nk_plot_function(struct nk_context*, enum nk_chart_type, void *userdata, float(*value_getter)(void* user, int index), int count, int offset); -/* ============================================================================= - * - * POPUP - * - * ============================================================================= */ -NK_API nk_bool nk_popup_begin(struct nk_context*, enum nk_popup_type, const char*, nk_flags, struct nk_rect bounds); -NK_API void nk_popup_close(struct nk_context*); -NK_API void nk_popup_end(struct nk_context*); -NK_API void nk_popup_get_scroll(struct nk_context*, nk_uint *offset_x, nk_uint *offset_y); -NK_API void nk_popup_set_scroll(struct nk_context*, nk_uint offset_x, nk_uint offset_y); -/* ============================================================================= - * - * COMBOBOX - * - * ============================================================================= */ -NK_API int nk_combo(struct nk_context*, const char **items, int count, int selected, int item_height, struct nk_vec2 size); -NK_API int nk_combo_separator(struct nk_context*, const char *items_separated_by_separator, int separator, int selected, int count, int item_height, struct nk_vec2 size); -NK_API int nk_combo_string(struct nk_context*, const char *items_separated_by_zeros, int selected, int count, int item_height, struct nk_vec2 size); -NK_API int nk_combo_callback(struct nk_context*, void(*item_getter)(void*, int, const char**), void *userdata, int selected, int count, int item_height, struct nk_vec2 size); -NK_API void nk_combobox(struct nk_context*, const char **items, int count, int *selected, int item_height, struct nk_vec2 size); -NK_API void nk_combobox_string(struct nk_context*, const char *items_separated_by_zeros, int *selected, int count, int item_height, struct nk_vec2 size); -NK_API void nk_combobox_separator(struct nk_context*, const char *items_separated_by_separator, int separator, int *selected, int count, int item_height, struct nk_vec2 size); -NK_API void nk_combobox_callback(struct nk_context*, void(*item_getter)(void*, int, const char**), void*, int *selected, int count, int item_height, struct nk_vec2 size); -/* ============================================================================= - * - * ABSTRACT COMBOBOX - * - * ============================================================================= */ -NK_API nk_bool nk_combo_begin_text(struct nk_context*, const char *selected, int, struct nk_vec2 size); -NK_API nk_bool nk_combo_begin_label(struct nk_context*, const char *selected, struct nk_vec2 size); -NK_API nk_bool nk_combo_begin_color(struct nk_context*, struct nk_color color, struct nk_vec2 size); -NK_API nk_bool nk_combo_begin_symbol(struct nk_context*, enum nk_symbol_type, struct nk_vec2 size); -NK_API nk_bool nk_combo_begin_symbol_label(struct nk_context*, const char *selected, enum nk_symbol_type, struct nk_vec2 size); -NK_API nk_bool nk_combo_begin_symbol_text(struct nk_context*, const char *selected, int, enum nk_symbol_type, struct nk_vec2 size); -NK_API nk_bool nk_combo_begin_image(struct nk_context*, struct nk_image img, struct nk_vec2 size); -NK_API nk_bool nk_combo_begin_image_label(struct nk_context*, const char *selected, struct nk_image, struct nk_vec2 size); -NK_API nk_bool nk_combo_begin_image_text(struct nk_context*, const char *selected, int, struct nk_image, struct nk_vec2 size); -NK_API nk_bool nk_combo_item_label(struct nk_context*, const char*, nk_flags alignment); -NK_API nk_bool nk_combo_item_text(struct nk_context*, const char*,int, nk_flags alignment); -NK_API nk_bool nk_combo_item_image_label(struct nk_context*, struct nk_image, const char*, nk_flags alignment); -NK_API nk_bool nk_combo_item_image_text(struct nk_context*, struct nk_image, const char*, int,nk_flags alignment); -NK_API nk_bool nk_combo_item_symbol_label(struct nk_context*, enum nk_symbol_type, const char*, nk_flags alignment); -NK_API nk_bool nk_combo_item_symbol_text(struct nk_context*, enum nk_symbol_type, const char*, int, nk_flags alignment); -NK_API void nk_combo_close(struct nk_context*); -NK_API void nk_combo_end(struct nk_context*); -/* ============================================================================= - * - * CONTEXTUAL - * - * ============================================================================= */ -NK_API nk_bool nk_contextual_begin(struct nk_context*, nk_flags, struct nk_vec2, struct nk_rect trigger_bounds); -NK_API nk_bool nk_contextual_item_text(struct nk_context*, const char*, int,nk_flags align); -NK_API nk_bool nk_contextual_item_label(struct nk_context*, const char*, nk_flags align); -NK_API nk_bool nk_contextual_item_image_label(struct nk_context*, struct nk_image, const char*, nk_flags alignment); -NK_API nk_bool nk_contextual_item_image_text(struct nk_context*, struct nk_image, const char*, int len, nk_flags alignment); -NK_API nk_bool nk_contextual_item_symbol_label(struct nk_context*, enum nk_symbol_type, const char*, nk_flags alignment); -NK_API nk_bool nk_contextual_item_symbol_text(struct nk_context*, enum nk_symbol_type, const char*, int, nk_flags alignment); -NK_API void nk_contextual_close(struct nk_context*); -NK_API void nk_contextual_end(struct nk_context*); -/* ============================================================================= - * - * TOOLTIP - * - * ============================================================================= */ -NK_API void nk_tooltip(struct nk_context*, const char*); -#ifdef NK_INCLUDE_STANDARD_VARARGS -NK_API void nk_tooltipf(struct nk_context*, NK_PRINTF_FORMAT_STRING const char*, ...) NK_PRINTF_VARARG_FUNC(2); -NK_API void nk_tooltipfv(struct nk_context*, NK_PRINTF_FORMAT_STRING const char*, va_list) NK_PRINTF_VALIST_FUNC(2); -#endif -NK_API nk_bool nk_tooltip_begin(struct nk_context*, float width); -NK_API void nk_tooltip_end(struct nk_context*); -/* ============================================================================= - * - * MENU - * - * ============================================================================= */ -NK_API void nk_menubar_begin(struct nk_context*); -NK_API void nk_menubar_end(struct nk_context*); -NK_API nk_bool nk_menu_begin_text(struct nk_context*, const char* title, int title_len, nk_flags align, struct nk_vec2 size); -NK_API nk_bool nk_menu_begin_label(struct nk_context*, const char*, nk_flags align, struct nk_vec2 size); -NK_API nk_bool nk_menu_begin_image(struct nk_context*, const char*, struct nk_image, struct nk_vec2 size); -NK_API nk_bool nk_menu_begin_image_text(struct nk_context*, const char*, int,nk_flags align,struct nk_image, struct nk_vec2 size); -NK_API nk_bool nk_menu_begin_image_label(struct nk_context*, const char*, nk_flags align,struct nk_image, struct nk_vec2 size); -NK_API nk_bool nk_menu_begin_symbol(struct nk_context*, const char*, enum nk_symbol_type, struct nk_vec2 size); -NK_API nk_bool nk_menu_begin_symbol_text(struct nk_context*, const char*, int,nk_flags align,enum nk_symbol_type, struct nk_vec2 size); -NK_API nk_bool nk_menu_begin_symbol_label(struct nk_context*, const char*, nk_flags align,enum nk_symbol_type, struct nk_vec2 size); -NK_API nk_bool nk_menu_item_text(struct nk_context*, const char*, int,nk_flags align); -NK_API nk_bool nk_menu_item_label(struct nk_context*, const char*, nk_flags alignment); -NK_API nk_bool nk_menu_item_image_label(struct nk_context*, struct nk_image, const char*, nk_flags alignment); -NK_API nk_bool nk_menu_item_image_text(struct nk_context*, struct nk_image, const char*, int len, nk_flags alignment); -NK_API nk_bool nk_menu_item_symbol_text(struct nk_context*, enum nk_symbol_type, const char*, int, nk_flags alignment); -NK_API nk_bool nk_menu_item_symbol_label(struct nk_context*, enum nk_symbol_type, const char*, nk_flags alignment); -NK_API void nk_menu_close(struct nk_context*); -NK_API void nk_menu_end(struct nk_context*); -/* ============================================================================= - * - * STYLE - * - * ============================================================================= */ -enum nk_style_colors { - NK_COLOR_TEXT, - NK_COLOR_WINDOW, - NK_COLOR_HEADER, - NK_COLOR_BORDER, - NK_COLOR_BUTTON, - NK_COLOR_BUTTON_HOVER, - NK_COLOR_BUTTON_ACTIVE, - NK_COLOR_TOGGLE, - NK_COLOR_TOGGLE_HOVER, - NK_COLOR_TOGGLE_CURSOR, - NK_COLOR_SELECT, - NK_COLOR_SELECT_ACTIVE, - NK_COLOR_SLIDER, - NK_COLOR_SLIDER_CURSOR, - NK_COLOR_SLIDER_CURSOR_HOVER, - NK_COLOR_SLIDER_CURSOR_ACTIVE, - NK_COLOR_PROPERTY, - NK_COLOR_EDIT, - NK_COLOR_EDIT_CURSOR, - NK_COLOR_COMBO, - NK_COLOR_CHART, - NK_COLOR_CHART_COLOR, - NK_COLOR_CHART_COLOR_HIGHLIGHT, - NK_COLOR_SCROLLBAR, - NK_COLOR_SCROLLBAR_CURSOR, - NK_COLOR_SCROLLBAR_CURSOR_HOVER, - NK_COLOR_SCROLLBAR_CURSOR_ACTIVE, - NK_COLOR_TAB_HEADER, - NK_COLOR_COUNT -}; -enum nk_style_cursor { - NK_CURSOR_ARROW, - NK_CURSOR_TEXT, - NK_CURSOR_MOVE, - NK_CURSOR_RESIZE_VERTICAL, - NK_CURSOR_RESIZE_HORIZONTAL, - NK_CURSOR_RESIZE_TOP_LEFT_DOWN_RIGHT, - NK_CURSOR_RESIZE_TOP_RIGHT_DOWN_LEFT, - NK_CURSOR_COUNT -}; -NK_API void nk_style_default(struct nk_context*); -NK_API void nk_style_from_table(struct nk_context*, const struct nk_color*); -NK_API void nk_style_load_cursor(struct nk_context*, enum nk_style_cursor, const struct nk_cursor*); -NK_API void nk_style_load_all_cursors(struct nk_context*, struct nk_cursor*); -NK_API const char* nk_style_get_color_by_name(enum nk_style_colors); -NK_API void nk_style_set_font(struct nk_context*, const struct nk_user_font*); -NK_API nk_bool nk_style_set_cursor(struct nk_context*, enum nk_style_cursor); -NK_API void nk_style_show_cursor(struct nk_context*); -NK_API void nk_style_hide_cursor(struct nk_context*); - -NK_API nk_bool nk_style_push_font(struct nk_context*, const struct nk_user_font*); -NK_API nk_bool nk_style_push_float(struct nk_context*, float*, float); -NK_API nk_bool nk_style_push_vec2(struct nk_context*, struct nk_vec2*, struct nk_vec2); -NK_API nk_bool nk_style_push_style_item(struct nk_context*, struct nk_style_item*, struct nk_style_item); -NK_API nk_bool nk_style_push_flags(struct nk_context*, nk_flags*, nk_flags); -NK_API nk_bool nk_style_push_color(struct nk_context*, struct nk_color*, struct nk_color); - -NK_API nk_bool nk_style_pop_font(struct nk_context*); -NK_API nk_bool nk_style_pop_float(struct nk_context*); -NK_API nk_bool nk_style_pop_vec2(struct nk_context*); -NK_API nk_bool nk_style_pop_style_item(struct nk_context*); -NK_API nk_bool nk_style_pop_flags(struct nk_context*); -NK_API nk_bool nk_style_pop_color(struct nk_context*); -/* ============================================================================= - * - * COLOR - * - * ============================================================================= */ -NK_API struct nk_color nk_rgb(int r, int g, int b); -NK_API struct nk_color nk_rgb_iv(const int *rgb); -NK_API struct nk_color nk_rgb_bv(const nk_byte* rgb); -NK_API struct nk_color nk_rgb_f(float r, float g, float b); -NK_API struct nk_color nk_rgb_fv(const float *rgb); -NK_API struct nk_color nk_rgb_cf(struct nk_colorf c); -NK_API struct nk_color nk_rgb_hex(const char *rgb); - -NK_API struct nk_color nk_rgba(int r, int g, int b, int a); -NK_API struct nk_color nk_rgba_u32(nk_uint); -NK_API struct nk_color nk_rgba_iv(const int *rgba); -NK_API struct nk_color nk_rgba_bv(const nk_byte *rgba); -NK_API struct nk_color nk_rgba_f(float r, float g, float b, float a); -NK_API struct nk_color nk_rgba_fv(const float *rgba); -NK_API struct nk_color nk_rgba_cf(struct nk_colorf c); -NK_API struct nk_color nk_rgba_hex(const char *rgb); - -NK_API struct nk_colorf nk_hsva_colorf(float h, float s, float v, float a); -NK_API struct nk_colorf nk_hsva_colorfv(float *c); -NK_API void nk_colorf_hsva_f(float *out_h, float *out_s, float *out_v, float *out_a, struct nk_colorf in); -NK_API void nk_colorf_hsva_fv(float *hsva, struct nk_colorf in); - -NK_API struct nk_color nk_hsv(int h, int s, int v); -NK_API struct nk_color nk_hsv_iv(const int *hsv); -NK_API struct nk_color nk_hsv_bv(const nk_byte *hsv); -NK_API struct nk_color nk_hsv_f(float h, float s, float v); -NK_API struct nk_color nk_hsv_fv(const float *hsv); - -NK_API struct nk_color nk_hsva(int h, int s, int v, int a); -NK_API struct nk_color nk_hsva_iv(const int *hsva); -NK_API struct nk_color nk_hsva_bv(const nk_byte *hsva); -NK_API struct nk_color nk_hsva_f(float h, float s, float v, float a); -NK_API struct nk_color nk_hsva_fv(const float *hsva); - -/* color (conversion nuklear --> user) */ -NK_API void nk_color_f(float *r, float *g, float *b, float *a, struct nk_color); -NK_API void nk_color_fv(float *rgba_out, struct nk_color); -NK_API struct nk_colorf nk_color_cf(struct nk_color); -NK_API void nk_color_d(double *r, double *g, double *b, double *a, struct nk_color); -NK_API void nk_color_dv(double *rgba_out, struct nk_color); - -NK_API nk_uint nk_color_u32(struct nk_color); -NK_API void nk_color_hex_rgba(char *output, struct nk_color); -NK_API void nk_color_hex_rgb(char *output, struct nk_color); - -NK_API void nk_color_hsv_i(int *out_h, int *out_s, int *out_v, struct nk_color); -NK_API void nk_color_hsv_b(nk_byte *out_h, nk_byte *out_s, nk_byte *out_v, struct nk_color); -NK_API void nk_color_hsv_iv(int *hsv_out, struct nk_color); -NK_API void nk_color_hsv_bv(nk_byte *hsv_out, struct nk_color); -NK_API void nk_color_hsv_f(float *out_h, float *out_s, float *out_v, struct nk_color); -NK_API void nk_color_hsv_fv(float *hsv_out, struct nk_color); - -NK_API void nk_color_hsva_i(int *h, int *s, int *v, int *a, struct nk_color); -NK_API void nk_color_hsva_b(nk_byte *h, nk_byte *s, nk_byte *v, nk_byte *a, struct nk_color); -NK_API void nk_color_hsva_iv(int *hsva_out, struct nk_color); -NK_API void nk_color_hsva_bv(nk_byte *hsva_out, struct nk_color); -NK_API void nk_color_hsva_f(float *out_h, float *out_s, float *out_v, float *out_a, struct nk_color); -NK_API void nk_color_hsva_fv(float *hsva_out, struct nk_color); -/* ============================================================================= - * - * IMAGE - * - * ============================================================================= */ -NK_API nk_handle nk_handle_ptr(void*); -NK_API nk_handle nk_handle_id(int); -NK_API struct nk_image nk_image_handle(nk_handle); -NK_API struct nk_image nk_image_ptr(void*); -NK_API struct nk_image nk_image_id(int); -NK_API nk_bool nk_image_is_subimage(const struct nk_image* img); -NK_API struct nk_image nk_subimage_ptr(void*, nk_ushort w, nk_ushort h, struct nk_rect sub_region); -NK_API struct nk_image nk_subimage_id(int, nk_ushort w, nk_ushort h, struct nk_rect sub_region); -NK_API struct nk_image nk_subimage_handle(nk_handle, nk_ushort w, nk_ushort h, struct nk_rect sub_region); -/* ============================================================================= - * - * 9-SLICE - * - * ============================================================================= */ -NK_API struct nk_nine_slice nk_nine_slice_handle(nk_handle, nk_ushort l, nk_ushort t, nk_ushort r, nk_ushort b); -NK_API struct nk_nine_slice nk_nine_slice_ptr(void*, nk_ushort l, nk_ushort t, nk_ushort r, nk_ushort b); -NK_API struct nk_nine_slice nk_nine_slice_id(int, nk_ushort l, nk_ushort t, nk_ushort r, nk_ushort b); -NK_API int nk_nine_slice_is_sub9slice(const struct nk_nine_slice* img); -NK_API struct nk_nine_slice nk_sub9slice_ptr(void*, nk_ushort w, nk_ushort h, struct nk_rect sub_region, nk_ushort l, nk_ushort t, nk_ushort r, nk_ushort b); -NK_API struct nk_nine_slice nk_sub9slice_id(int, nk_ushort w, nk_ushort h, struct nk_rect sub_region, nk_ushort l, nk_ushort t, nk_ushort r, nk_ushort b); -NK_API struct nk_nine_slice nk_sub9slice_handle(nk_handle, nk_ushort w, nk_ushort h, struct nk_rect sub_region, nk_ushort l, nk_ushort t, nk_ushort r, nk_ushort b); -/* ============================================================================= - * - * MATH - * - * ============================================================================= */ -NK_API nk_hash nk_murmur_hash(const void *key, int len, nk_hash seed); -NK_API void nk_triangle_from_direction(struct nk_vec2 *result, struct nk_rect r, float pad_x, float pad_y, enum nk_heading); - -NK_API struct nk_vec2 nk_vec2(float x, float y); -NK_API struct nk_vec2 nk_vec2i(int x, int y); -NK_API struct nk_vec2 nk_vec2v(const float *xy); -NK_API struct nk_vec2 nk_vec2iv(const int *xy); - -NK_API struct nk_rect nk_get_null_rect(void); -NK_API struct nk_rect nk_rect(float x, float y, float w, float h); -NK_API struct nk_rect nk_recti(int x, int y, int w, int h); -NK_API struct nk_rect nk_recta(struct nk_vec2 pos, struct nk_vec2 size); -NK_API struct nk_rect nk_rectv(const float *xywh); -NK_API struct nk_rect nk_rectiv(const int *xywh); -NK_API struct nk_vec2 nk_rect_pos(struct nk_rect); -NK_API struct nk_vec2 nk_rect_size(struct nk_rect); -/* ============================================================================= - * - * STRING - * - * ============================================================================= */ -NK_API int nk_strlen(const char *str); -NK_API int nk_stricmp(const char *s1, const char *s2); -NK_API int nk_stricmpn(const char *s1, const char *s2, int n); -NK_API int nk_strtoi(const char *str, const char **endptr); -NK_API float nk_strtof(const char *str, const char **endptr); -#ifndef NK_STRTOD -#define NK_STRTOD nk_strtod -NK_API double nk_strtod(const char *str, const char **endptr); -#endif -NK_API int nk_strfilter(const char *text, const char *regexp); -NK_API int nk_strmatch_fuzzy_string(char const *str, char const *pattern, int *out_score); -NK_API int nk_strmatch_fuzzy_text(const char *txt, int txt_len, const char *pattern, int *out_score); -/* ============================================================================= - * - * UTF-8 - * - * ============================================================================= */ -NK_API int nk_utf_decode(const char*, nk_rune*, int); -NK_API int nk_utf_encode(nk_rune, char*, int); -NK_API int nk_utf_len(const char*, int byte_len); -NK_API const char* nk_utf_at(const char *buffer, int length, int index, nk_rune *unicode, int *len); -/* =============================================================== - * - * FONT - * - * ===============================================================*/ -/*/// ### Font -/// Font handling in this library was designed to be quite customizable and lets -/// you decide what you want to use and what you want to provide. There are three -/// different ways to use the font atlas. The first two will use your font -/// handling scheme and only requires essential data to run nuklear. The next -/// slightly more advanced features is font handling with vertex buffer output. -/// Finally the most complex API wise is using nuklear's font baking API. -// -/// #### Using your own implementation without vertex buffer output -/// -/// So first up the easiest way to do font handling is by just providing a -/// `nk_user_font` struct which only requires the height in pixel of the used -/// font and a callback to calculate the width of a string. This way of handling -/// fonts is best fitted for using the normal draw shape command API where you -/// do all the text drawing yourself and the library does not require any kind -/// of deeper knowledge about which font handling mechanism you use. -/// IMPORTANT: the `nk_user_font` pointer provided to nuklear has to persist -/// over the complete life time! I know this sucks but it is currently the only -/// way to switch between fonts. -/// -/// ```c -/// float your_text_width_calculation(nk_handle handle, float height, const char *text, int len) -/// { -/// your_font_type *type = handle.ptr; -/// float text_width = ...; -/// return text_width; -/// } -/// -/// struct nk_user_font font; -/// font.userdata.ptr = &your_font_class_or_struct; -/// font.height = your_font_height; -/// font.width = your_text_width_calculation; -/// -/// struct nk_context ctx; -/// nk_init_default(&ctx, &font); -/// ``` -/// #### Using your own implementation with vertex buffer output -/// -/// While the first approach works fine if you don't want to use the optional -/// vertex buffer output it is not enough if you do. To get font handling working -/// for these cases you have to provide two additional parameters inside the -/// `nk_user_font`. First a texture atlas handle used to draw text as subimages -/// of a bigger font atlas texture and a callback to query a character's glyph -/// information (offset, size, ...). So it is still possible to provide your own -/// font and use the vertex buffer output. -/// -/// ```c -/// float your_text_width_calculation(nk_handle handle, float height, const char *text, int len) -/// { -/// your_font_type *type = handle.ptr; -/// float text_width = ...; -/// return text_width; -/// } -/// void query_your_font_glyph(nk_handle handle, float font_height, struct nk_user_font_glyph *glyph, nk_rune codepoint, nk_rune next_codepoint) -/// { -/// your_font_type *type = handle.ptr; -/// glyph.width = ...; -/// glyph.height = ...; -/// glyph.xadvance = ...; -/// glyph.uv[0].x = ...; -/// glyph.uv[0].y = ...; -/// glyph.uv[1].x = ...; -/// glyph.uv[1].y = ...; -/// glyph.offset.x = ...; -/// glyph.offset.y = ...; -/// } -/// -/// struct nk_user_font font; -/// font.userdata.ptr = &your_font_class_or_struct; -/// font.height = your_font_height; -/// font.width = your_text_width_calculation; -/// font.query = query_your_font_glyph; -/// font.texture.id = your_font_texture; -/// -/// struct nk_context ctx; -/// nk_init_default(&ctx, &font); -/// ``` -/// -/// #### Nuklear font baker -/// -/// The final approach if you do not have a font handling functionality or don't -/// want to use it in this library is by using the optional font baker. -/// The font baker APIs can be used to create a font plus font atlas texture -/// and can be used with or without the vertex buffer output. -/// -/// It still uses the `nk_user_font` struct and the two different approaches -/// previously stated still work. The font baker is not located inside -/// `nk_context` like all other systems since it can be understood as more of -/// an extension to nuklear and does not really depend on any `nk_context` state. -/// -/// Font baker need to be initialized first by one of the nk_font_atlas_init_xxx -/// functions. If you don't care about memory just call the default version -/// `nk_font_atlas_init_default` which will allocate all memory from the standard library. -/// If you want to control memory allocation but you don't care if the allocated -/// memory is temporary and therefore can be freed directly after the baking process -/// is over or permanent you can call `nk_font_atlas_init`. -/// -/// After successfully initializing the font baker you can add Truetype(.ttf) fonts from -/// different sources like memory or from file by calling one of the `nk_font_atlas_add_xxx`. -/// functions. Adding font will permanently store each font, font config and ttf memory block(!) -/// inside the font atlas and allows to reuse the font atlas. If you don't want to reuse -/// the font baker by for example adding additional fonts you can call -/// `nk_font_atlas_cleanup` after the baking process is over (after calling nk_font_atlas_end). -/// -/// As soon as you added all fonts you wanted you can now start the baking process -/// for every selected glyph to image by calling `nk_font_atlas_bake`. -/// The baking process returns image memory, width and height which can be used to -/// either create your own image object or upload it to any graphics library. -/// No matter which case you finally have to call `nk_font_atlas_end` which -/// will free all temporary memory including the font atlas image so make sure -/// you created our texture beforehand. `nk_font_atlas_end` requires a handle -/// to your font texture or object and optionally fills a `struct nk_draw_null_texture` -/// which can be used for the optional vertex output. If you don't want it just -/// set the argument to `NULL`. -/// -/// At this point you are done and if you don't want to reuse the font atlas you -/// can call `nk_font_atlas_cleanup` to free all truetype blobs and configuration -/// memory. Finally if you don't use the font atlas and any of it's fonts anymore -/// you need to call `nk_font_atlas_clear` to free all memory still being used. -/// -/// ```c -/// struct nk_font_atlas atlas; -/// nk_font_atlas_init_default(&atlas); -/// nk_font_atlas_begin(&atlas); -/// nk_font *font = nk_font_atlas_add_from_file(&atlas, "Path/To/Your/TTF_Font.ttf", 13, 0); -/// nk_font *font2 = nk_font_atlas_add_from_file(&atlas, "Path/To/Your/TTF_Font2.ttf", 16, 0); -/// const void* img = nk_font_atlas_bake(&atlas, &img_width, &img_height, NK_FONT_ATLAS_RGBA32); -/// nk_font_atlas_end(&atlas, nk_handle_id(texture), 0); -/// -/// struct nk_context ctx; -/// nk_init_default(&ctx, &font->handle); -/// while (1) { -/// -/// } -/// nk_font_atlas_clear(&atlas); -/// ``` -/// The font baker API is probably the most complex API inside this library and -/// I would suggest reading some of my examples `example/` to get a grip on how -/// to use the font atlas. There are a number of details I left out. For example -/// how to merge fonts, configure a font with `nk_font_config` to use other languages, -/// use another texture coordinate format and a lot more: -/// -/// ```c -/// struct nk_font_config cfg = nk_font_config(font_pixel_height); -/// cfg.merge_mode = nk_false or nk_true; -/// cfg.range = nk_font_korean_glyph_ranges(); -/// cfg.coord_type = NK_COORD_PIXEL; -/// nk_font *font = nk_font_atlas_add_from_file(&atlas, "Path/To/Your/TTF_Font.ttf", 13, &cfg); -/// ``` -*/ -struct nk_user_font_glyph; -typedef float(*nk_text_width_f)(nk_handle, float h, const char*, int len); -typedef void(*nk_query_font_glyph_f)(nk_handle handle, float font_height, - struct nk_user_font_glyph *glyph, - nk_rune codepoint, nk_rune next_codepoint); - -#if defined(NK_INCLUDE_VERTEX_BUFFER_OUTPUT) || defined(NK_INCLUDE_SOFTWARE_FONT) -struct nk_user_font_glyph { - struct nk_vec2 uv[2]; - /* texture coordinates */ - struct nk_vec2 offset; - /* offset between top left and glyph */ - float width, height; - /* size of the glyph */ - float xadvance; - /* offset to the next glyph */ -}; -#endif - -struct nk_user_font { - nk_handle userdata; - /* user provided font handle */ - float height; - /* max height of the font */ - nk_text_width_f width; - /* font string width in pixel callback */ -#ifdef NK_INCLUDE_VERTEX_BUFFER_OUTPUT - nk_query_font_glyph_f query; - /* font glyph callback to query drawing info */ - nk_handle texture; - /* texture handle to the used font atlas or texture */ -#endif -}; - -#ifdef NK_INCLUDE_FONT_BAKING -enum nk_font_coord_type { - NK_COORD_UV, /* texture coordinates inside font glyphs are clamped between 0-1 */ - NK_COORD_PIXEL /* texture coordinates inside font glyphs are in absolute pixel */ -}; - -struct nk_font; -struct nk_baked_font { - float height; - /* height of the font */ - float ascent, descent; - /* font glyphs ascent and descent */ - nk_rune glyph_offset; - /* glyph array offset inside the font glyph baking output array */ - nk_rune glyph_count; - /* number of glyphs of this font inside the glyph baking array output */ - const nk_rune *ranges; - /* font codepoint ranges as pairs of (from/to) and 0 as last element */ -}; - -struct nk_font_config { - struct nk_font_config *next; - /* NOTE: only used internally */ - void *ttf_blob; - /* pointer to loaded TTF file memory block. - * NOTE: not needed for nk_font_atlas_add_from_memory and nk_font_atlas_add_from_file. */ - nk_size ttf_size; - /* size of the loaded TTF file memory block - * NOTE: not needed for nk_font_atlas_add_from_memory and nk_font_atlas_add_from_file. */ - - unsigned char ttf_data_owned_by_atlas; - /* used inside font atlas: default to: 0*/ - unsigned char merge_mode; - /* merges this font into the last font */ - unsigned char pixel_snap; - /* align every character to pixel boundary (if true set oversample (1,1)) */ - unsigned char oversample_v, oversample_h; - /* rasterize at high quality for sub-pixel position */ - unsigned char padding[3]; - - float size; - /* baked pixel height of the font */ - enum nk_font_coord_type coord_type; - /* texture coordinate format with either pixel or UV coordinates */ - struct nk_vec2 spacing; - /* extra pixel spacing between glyphs */ - const nk_rune *range; - /* list of unicode ranges (2 values per range, zero terminated) */ - struct nk_baked_font *font; - /* font to setup in the baking process: NOTE: not needed for font atlas */ - nk_rune fallback_glyph; - /* fallback glyph to use if a given rune is not found */ - struct nk_font_config *n; - struct nk_font_config *p; -}; - -struct nk_font_glyph { - nk_rune codepoint; - float xadvance; - float x0, y0, x1, y1, w, h; - float u0, v0, u1, v1; -}; - -struct nk_font { - struct nk_font *next; - struct nk_user_font handle; - struct nk_baked_font info; - float scale; - struct nk_font_glyph *glyphs; - const struct nk_font_glyph *fallback; - nk_rune fallback_codepoint; - nk_handle texture; - struct nk_font_config *config; -}; - -enum nk_font_atlas_format { - NK_FONT_ATLAS_ALPHA8, - NK_FONT_ATLAS_RGBA32 -}; - -struct nk_font_atlas { - void *pixel; - int tex_width; - int tex_height; - - struct nk_allocator permanent; - struct nk_allocator temporary; - - struct nk_recti custom; - struct nk_cursor cursors[NK_CURSOR_COUNT]; - - int glyph_count; - struct nk_font_glyph *glyphs; - struct nk_font *default_font; - struct nk_font *fonts; - struct nk_font_config *config; - int font_num; -}; - -/* some language glyph codepoint ranges */ -NK_API const nk_rune *nk_font_default_glyph_ranges(void); -NK_API const nk_rune *nk_font_chinese_glyph_ranges(void); -NK_API const nk_rune *nk_font_cyrillic_glyph_ranges(void); -NK_API const nk_rune *nk_font_korean_glyph_ranges(void); - -#ifdef NK_INCLUDE_DEFAULT_ALLOCATOR -NK_API void nk_font_atlas_init_default(struct nk_font_atlas*); -#endif -NK_API void nk_font_atlas_init(struct nk_font_atlas*, struct nk_allocator*); -NK_API void nk_font_atlas_init_custom(struct nk_font_atlas*, struct nk_allocator *persistent, struct nk_allocator *transient); -NK_API void nk_font_atlas_begin(struct nk_font_atlas*); -NK_API struct nk_font_config nk_font_config(float pixel_height); -NK_API struct nk_font *nk_font_atlas_add(struct nk_font_atlas*, const struct nk_font_config*); -#ifdef NK_INCLUDE_DEFAULT_FONT -NK_API struct nk_font* nk_font_atlas_add_default(struct nk_font_atlas*, float height, const struct nk_font_config*); -#endif -NK_API struct nk_font* nk_font_atlas_add_from_memory(struct nk_font_atlas *atlas, void *memory, nk_size size, float height, const struct nk_font_config *config); -#ifdef NK_INCLUDE_STANDARD_IO -NK_API struct nk_font* nk_font_atlas_add_from_file(struct nk_font_atlas *atlas, const char *file_path, float height, const struct nk_font_config*); -#endif -NK_API struct nk_font *nk_font_atlas_add_compressed(struct nk_font_atlas*, void *memory, nk_size size, float height, const struct nk_font_config*); -NK_API struct nk_font* nk_font_atlas_add_compressed_base85(struct nk_font_atlas*, const char *data, float height, const struct nk_font_config *config); -NK_API const void* nk_font_atlas_bake(struct nk_font_atlas*, int *width, int *height, enum nk_font_atlas_format); -NK_API void nk_font_atlas_end(struct nk_font_atlas*, nk_handle tex, struct nk_draw_null_texture*); -NK_API const struct nk_font_glyph* nk_font_find_glyph(struct nk_font*, nk_rune unicode); -NK_API void nk_font_atlas_cleanup(struct nk_font_atlas *atlas); -NK_API void nk_font_atlas_clear(struct nk_font_atlas*); - -#endif - -/* ============================================================== - * - * MEMORY BUFFER - * - * ===============================================================*/ -/*/// ### Memory Buffer -/// A basic (double)-buffer with linear allocation and resetting as only -/// freeing policy. The buffer's main purpose is to control all memory management -/// inside the GUI toolkit and still leave memory control as much as possible in -/// the hand of the user while also making sure the library is easy to use if -/// not as much control is needed. -/// In general all memory inside this library can be provided from the user in -/// three different ways. -/// -/// The first way and the one providing most control is by just passing a fixed -/// size memory block. In this case all control lies in the hand of the user -/// since he can exactly control where the memory comes from and how much memory -/// the library should consume. Of course using the fixed size API removes the -/// ability to automatically resize a buffer if not enough memory is provided so -/// you have to take over the resizing. While being a fixed sized buffer sounds -/// quite limiting, it is very effective in this library since the actual memory -/// consumption is quite stable and has a fixed upper bound for a lot of cases. -/// -/// If you don't want to think about how much memory the library should allocate -/// at all time or have a very dynamic UI with unpredictable memory consumption -/// habits but still want control over memory allocation you can use the dynamic -/// allocator based API. The allocator consists of two callbacks for allocating -/// and freeing memory and optional userdata so you can plugin your own allocator. -/// -/// The final and easiest way can be used by defining -/// NK_INCLUDE_DEFAULT_ALLOCATOR which uses the standard library memory -/// allocation functions malloc and free and takes over complete control over -/// memory in this library. -*/ -struct nk_memory_status { - void *memory; - unsigned int type; - nk_size size; - nk_size allocated; - nk_size needed; - nk_size calls; -}; - -enum nk_allocation_type { - NK_BUFFER_FIXED, - NK_BUFFER_DYNAMIC -}; - -enum nk_buffer_allocation_type { - NK_BUFFER_FRONT, - NK_BUFFER_BACK, - NK_BUFFER_MAX -}; - -struct nk_buffer_marker { - nk_bool active; - nk_size offset; -}; - -struct nk_memory {void *ptr;nk_size size;}; -struct nk_buffer { - struct nk_buffer_marker marker[NK_BUFFER_MAX]; - /* buffer marker to free a buffer to a certain offset */ - struct nk_allocator pool; - /* allocator callback for dynamic buffers */ - enum nk_allocation_type type; - /* memory management type */ - struct nk_memory memory; - /* memory and size of the current memory block */ - float grow_factor; - /* growing factor for dynamic memory management */ - nk_size allocated; - /* total amount of memory allocated */ - nk_size needed; - /* totally consumed memory given that enough memory is present */ - nk_size calls; - /* number of allocation calls */ - nk_size size; - /* current size of the buffer */ -}; - -#ifdef NK_INCLUDE_DEFAULT_ALLOCATOR -NK_API void nk_buffer_init_default(struct nk_buffer*); -#endif -NK_API void nk_buffer_init(struct nk_buffer*, const struct nk_allocator*, nk_size size); -NK_API void nk_buffer_init_fixed(struct nk_buffer*, void *memory, nk_size size); -NK_API void nk_buffer_info(struct nk_memory_status*, struct nk_buffer*); -NK_API void nk_buffer_push(struct nk_buffer*, enum nk_buffer_allocation_type type, const void *memory, nk_size size, nk_size align); -NK_API void nk_buffer_mark(struct nk_buffer*, enum nk_buffer_allocation_type type); -NK_API void nk_buffer_reset(struct nk_buffer*, enum nk_buffer_allocation_type type); -NK_API void nk_buffer_clear(struct nk_buffer*); -NK_API void nk_buffer_free(struct nk_buffer*); -NK_API void *nk_buffer_memory(struct nk_buffer*); -NK_API const void *nk_buffer_memory_const(const struct nk_buffer*); -NK_API nk_size nk_buffer_total(struct nk_buffer*); - -/* ============================================================== - * - * STRING - * - * ===============================================================*/ -/* Basic string buffer which is only used in context with the text editor - * to manage and manipulate dynamic or fixed size string content. This is _NOT_ - * the default string handling method. The only instance you should have any contact - * with this API is if you interact with an `nk_text_edit` object inside one of the - * copy and paste functions and even there only for more advanced cases. */ -struct nk_str { - struct nk_buffer buffer; - int len; /* in codepoints/runes/glyphs */ -}; - -#ifdef NK_INCLUDE_DEFAULT_ALLOCATOR -NK_API void nk_str_init_default(struct nk_str*); -#endif -NK_API void nk_str_init(struct nk_str*, const struct nk_allocator*, nk_size size); -NK_API void nk_str_init_fixed(struct nk_str*, void *memory, nk_size size); -NK_API void nk_str_clear(struct nk_str*); -NK_API void nk_str_free(struct nk_str*); - -NK_API int nk_str_append_text_char(struct nk_str*, const char*, int); -NK_API int nk_str_append_str_char(struct nk_str*, const char*); -NK_API int nk_str_append_text_utf8(struct nk_str*, const char*, int); -NK_API int nk_str_append_str_utf8(struct nk_str*, const char*); -NK_API int nk_str_append_text_runes(struct nk_str*, const nk_rune*, int); -NK_API int nk_str_append_str_runes(struct nk_str*, const nk_rune*); - -NK_API int nk_str_insert_at_char(struct nk_str*, int pos, const char*, int); -NK_API int nk_str_insert_at_rune(struct nk_str*, int pos, const char*, int); - -NK_API int nk_str_insert_text_char(struct nk_str*, int pos, const char*, int); -NK_API int nk_str_insert_str_char(struct nk_str*, int pos, const char*); -NK_API int nk_str_insert_text_utf8(struct nk_str*, int pos, const char*, int); -NK_API int nk_str_insert_str_utf8(struct nk_str*, int pos, const char*); -NK_API int nk_str_insert_text_runes(struct nk_str*, int pos, const nk_rune*, int); -NK_API int nk_str_insert_str_runes(struct nk_str*, int pos, const nk_rune*); - -NK_API void nk_str_remove_chars(struct nk_str*, int len); -NK_API void nk_str_remove_runes(struct nk_str *str, int len); -NK_API void nk_str_delete_chars(struct nk_str*, int pos, int len); -NK_API void nk_str_delete_runes(struct nk_str*, int pos, int len); - -NK_API char *nk_str_at_char(struct nk_str*, int pos); -NK_API char *nk_str_at_rune(struct nk_str*, int pos, nk_rune *unicode, int *len); -NK_API nk_rune nk_str_rune_at(const struct nk_str*, int pos); -NK_API const char *nk_str_at_char_const(const struct nk_str*, int pos); -NK_API const char *nk_str_at_const(const struct nk_str*, int pos, nk_rune *unicode, int *len); - -NK_API char *nk_str_get(struct nk_str*); -NK_API const char *nk_str_get_const(const struct nk_str*); -NK_API int nk_str_len(struct nk_str*); -NK_API int nk_str_len_char(struct nk_str*); - -/*=============================================================== - * - * TEXT EDITOR - * - * ===============================================================*/ -/*/// ### Text Editor -/// Editing text in this library is handled by either `nk_edit_string` or -/// `nk_edit_buffer`. But like almost everything in this library there are multiple -/// ways of doing it and a balance between control and ease of use with memory -/// as well as functionality controlled by flags. -/// -/// This library generally allows three different levels of memory control: -/// First of is the most basic way of just providing a simple char array with -/// string length. This method is probably the easiest way of handling simple -/// user text input. Main upside is complete control over memory while the biggest -/// downside in comparison with the other two approaches is missing undo/redo. -/// -/// For UIs that require undo/redo the second way was created. It is based on -/// a fixed size nk_text_edit struct, which has an internal undo/redo stack. -/// This is mainly useful if you want something more like a text editor but don't want -/// to have a dynamically growing buffer. -/// -/// The final way is using a dynamically growing nk_text_edit struct, which -/// has both a default version if you don't care where memory comes from and an -/// allocator version if you do. While the text editor is quite powerful for its -/// complexity I would not recommend editing gigabytes of data with it. -/// It is rather designed for uses cases which make sense for a GUI library not for -/// an full blown text editor. - */ -#ifndef NK_TEXTEDIT_UNDOSTATECOUNT -#define NK_TEXTEDIT_UNDOSTATECOUNT 99 -#endif - -#ifndef NK_TEXTEDIT_UNDOCHARCOUNT -#define NK_TEXTEDIT_UNDOCHARCOUNT 999 -#endif - -struct nk_text_edit; -struct nk_clipboard { - nk_handle userdata; - nk_plugin_paste paste; - nk_plugin_copy copy; -}; - -struct nk_text_undo_record { - int where; - short insert_length; - short delete_length; - short char_storage; -}; - -struct nk_text_undo_state { - struct nk_text_undo_record undo_rec[NK_TEXTEDIT_UNDOSTATECOUNT]; - nk_rune undo_char[NK_TEXTEDIT_UNDOCHARCOUNT]; - short undo_point; - short redo_point; - short undo_char_point; - short redo_char_point; -}; - -enum nk_text_edit_type { - NK_TEXT_EDIT_SINGLE_LINE, - NK_TEXT_EDIT_MULTI_LINE -}; - -enum nk_text_edit_mode { - NK_TEXT_EDIT_MODE_VIEW, - NK_TEXT_EDIT_MODE_INSERT, - NK_TEXT_EDIT_MODE_REPLACE -}; - -struct nk_text_edit { - struct nk_clipboard clip; - struct nk_str string; - nk_plugin_filter filter; - struct nk_vec2 scrollbar; - - int cursor; - int select_start; - int select_end; - unsigned char mode; - unsigned char cursor_at_end_of_line; - unsigned char initialized; - unsigned char has_preferred_x; - unsigned char single_line; - unsigned char active; - unsigned char padding1; - float preferred_x; - struct nk_text_undo_state undo; -}; - -/* filter function */ -NK_API nk_bool nk_filter_default(const struct nk_text_edit*, nk_rune unicode); -NK_API nk_bool nk_filter_ascii(const struct nk_text_edit*, nk_rune unicode); -NK_API nk_bool nk_filter_float(const struct nk_text_edit*, nk_rune unicode); -NK_API nk_bool nk_filter_decimal(const struct nk_text_edit*, nk_rune unicode); -NK_API nk_bool nk_filter_hex(const struct nk_text_edit*, nk_rune unicode); -NK_API nk_bool nk_filter_oct(const struct nk_text_edit*, nk_rune unicode); -NK_API nk_bool nk_filter_binary(const struct nk_text_edit*, nk_rune unicode); - -/* text editor */ -#ifdef NK_INCLUDE_DEFAULT_ALLOCATOR -NK_API void nk_textedit_init_default(struct nk_text_edit*); -#endif -NK_API void nk_textedit_init(struct nk_text_edit*, struct nk_allocator*, nk_size size); -NK_API void nk_textedit_init_fixed(struct nk_text_edit*, void *memory, nk_size size); -NK_API void nk_textedit_free(struct nk_text_edit*); -NK_API void nk_textedit_text(struct nk_text_edit*, const char*, int total_len); -NK_API void nk_textedit_delete(struct nk_text_edit*, int where, int len); -NK_API void nk_textedit_delete_selection(struct nk_text_edit*); -NK_API void nk_textedit_select_all(struct nk_text_edit*); -NK_API nk_bool nk_textedit_cut(struct nk_text_edit*); -NK_API nk_bool nk_textedit_paste(struct nk_text_edit*, char const*, int len); -NK_API void nk_textedit_undo(struct nk_text_edit*); -NK_API void nk_textedit_redo(struct nk_text_edit*); - -/* =============================================================== - * - * DRAWING - * - * ===============================================================*/ -/*/// ### Drawing -/// This library was designed to be render backend agnostic so it does -/// not draw anything to screen. Instead all drawn shapes, widgets -/// are made of, are buffered into memory and make up a command queue. -/// Each frame therefore fills the command buffer with draw commands -/// that then need to be executed by the user and his own render backend. -/// After that the command buffer needs to be cleared and a new frame can be -/// started. It is probably important to note that the command buffer is the main -/// drawing API and the optional vertex buffer API only takes this format and -/// converts it into a hardware accessible format. -/// -/// To use the command queue to draw your own widgets you can access the -/// command buffer of each window by calling `nk_window_get_canvas` after -/// previously having called `nk_begin`: -/// -/// ```c -/// void draw_red_rectangle_widget(struct nk_context *ctx) -/// { -/// struct nk_command_buffer *canvas; -/// struct nk_input *input = &ctx->input; -/// canvas = nk_window_get_canvas(ctx); -/// -/// struct nk_rect space; -/// enum nk_widget_layout_states state; -/// state = nk_widget(&space, ctx); -/// if (!state) return; -/// -/// if (state != NK_WIDGET_ROM) -/// update_your_widget_by_user_input(...); -/// nk_fill_rect(canvas, space, 0, nk_rgb(255,0,0)); -/// } -/// -/// if (nk_begin(...)) { -/// nk_layout_row_dynamic(ctx, 25, 1); -/// draw_red_rectangle_widget(ctx); -/// } -/// nk_end(..) -/// -/// ``` -/// Important to know if you want to create your own widgets is the `nk_widget` -/// call. It allocates space on the panel reserved for this widget to be used, -/// but also returns the state of the widget space. If your widget is not seen and does -/// not have to be updated it is '0' and you can just return. If it only has -/// to be drawn the state will be `NK_WIDGET_ROM` otherwise you can do both -/// update and draw your widget. The reason for separating is to only draw and -/// update what is actually necessary which is crucial for performance. -*/ -enum nk_command_type { - NK_COMMAND_NOP, - NK_COMMAND_SCISSOR, - NK_COMMAND_LINE, - NK_COMMAND_CURVE, - NK_COMMAND_RECT, - NK_COMMAND_RECT_FILLED, - NK_COMMAND_RECT_MULTI_COLOR, - NK_COMMAND_CIRCLE, - NK_COMMAND_CIRCLE_FILLED, - NK_COMMAND_ARC, - NK_COMMAND_ARC_FILLED, - NK_COMMAND_TRIANGLE, - NK_COMMAND_TRIANGLE_FILLED, - NK_COMMAND_POLYGON, - NK_COMMAND_POLYGON_FILLED, - NK_COMMAND_POLYLINE, - NK_COMMAND_TEXT, - NK_COMMAND_IMAGE, - NK_COMMAND_CUSTOM -}; - -/* command base and header of every command inside the buffer */ -struct nk_command { - enum nk_command_type type; - nk_size next; -#ifdef NK_INCLUDE_COMMAND_USERDATA - nk_handle userdata; -#endif -}; - -struct nk_command_scissor { - struct nk_command header; - short x, y; - unsigned short w, h; -}; - -struct nk_command_line { - struct nk_command header; - unsigned short line_thickness; - struct nk_vec2i begin; - struct nk_vec2i end; - struct nk_color color; -}; - -struct nk_command_curve { - struct nk_command header; - unsigned short line_thickness; - struct nk_vec2i begin; - struct nk_vec2i end; - struct nk_vec2i ctrl[2]; - struct nk_color color; -}; - -struct nk_command_rect { - struct nk_command header; - unsigned short rounding; - unsigned short line_thickness; - short x, y; - unsigned short w, h; - struct nk_color color; -}; - -struct nk_command_rect_filled { - struct nk_command header; - unsigned short rounding; - short x, y; - unsigned short w, h; - struct nk_color color; -}; - -struct nk_command_rect_multi_color { - struct nk_command header; - short x, y; - unsigned short w, h; - struct nk_color left; - struct nk_color top; - struct nk_color bottom; - struct nk_color right; -}; - -struct nk_command_triangle { - struct nk_command header; - unsigned short line_thickness; - struct nk_vec2i a; - struct nk_vec2i b; - struct nk_vec2i c; - struct nk_color color; -}; - -struct nk_command_triangle_filled { - struct nk_command header; - struct nk_vec2i a; - struct nk_vec2i b; - struct nk_vec2i c; - struct nk_color color; -}; - -struct nk_command_circle { - struct nk_command header; - short x, y; - unsigned short line_thickness; - unsigned short w, h; - struct nk_color color; -}; - -struct nk_command_circle_filled { - struct nk_command header; - short x, y; - unsigned short w, h; - struct nk_color color; -}; - -struct nk_command_arc { - struct nk_command header; - short cx, cy; - unsigned short r; - unsigned short line_thickness; - float a[2]; - struct nk_color color; -}; - -struct nk_command_arc_filled { - struct nk_command header; - short cx, cy; - unsigned short r; - float a[2]; - struct nk_color color; -}; - -struct nk_command_polygon { - struct nk_command header; - struct nk_color color; - unsigned short line_thickness; - unsigned short point_count; - struct nk_vec2i points[1]; -}; - -struct nk_command_polygon_filled { - struct nk_command header; - struct nk_color color; - unsigned short point_count; - struct nk_vec2i points[1]; -}; - -struct nk_command_polyline { - struct nk_command header; - struct nk_color color; - unsigned short line_thickness; - unsigned short point_count; - struct nk_vec2i points[1]; -}; - -struct nk_command_image { - struct nk_command header; - short x, y; - unsigned short w, h; - struct nk_image img; - struct nk_color col; -}; - -typedef void (*nk_command_custom_callback)(void *canvas, short x,short y, - unsigned short w, unsigned short h, nk_handle callback_data); -struct nk_command_custom { - struct nk_command header; - short x, y; - unsigned short w, h; - nk_handle callback_data; - nk_command_custom_callback callback; -}; - -struct nk_command_text { - struct nk_command header; - const struct nk_user_font *font; - struct nk_color background; - struct nk_color foreground; - short x, y; - unsigned short w, h; - float height; - int length; - char string[1]; -}; - -enum nk_command_clipping { - NK_CLIPPING_OFF = nk_false, - NK_CLIPPING_ON = nk_true -}; - -struct nk_command_buffer { - struct nk_buffer *base; - struct nk_rect clip; - int use_clipping; - nk_handle userdata; - nk_size begin, end, last; -}; - -/* shape outlines */ -NK_API void nk_stroke_line(struct nk_command_buffer *b, float x0, float y0, float x1, float y1, float line_thickness, struct nk_color); -NK_API void nk_stroke_curve(struct nk_command_buffer*, float, float, float, float, float, float, float, float, float line_thickness, struct nk_color); -NK_API void nk_stroke_rect(struct nk_command_buffer*, struct nk_rect, float rounding, float line_thickness, struct nk_color); -NK_API void nk_stroke_circle(struct nk_command_buffer*, struct nk_rect, float line_thickness, struct nk_color); -NK_API void nk_stroke_arc(struct nk_command_buffer*, float cx, float cy, float radius, float a_min, float a_max, float line_thickness, struct nk_color); -NK_API void nk_stroke_triangle(struct nk_command_buffer*, float, float, float, float, float, float, float line_thichness, struct nk_color); -NK_API void nk_stroke_polyline(struct nk_command_buffer*, float *points, int point_count, float line_thickness, struct nk_color col); -NK_API void nk_stroke_polygon(struct nk_command_buffer*, float*, int point_count, float line_thickness, struct nk_color); - -/* filled shades */ -NK_API void nk_fill_rect(struct nk_command_buffer*, struct nk_rect, float rounding, struct nk_color); -NK_API void nk_fill_rect_multi_color(struct nk_command_buffer*, struct nk_rect, struct nk_color left, struct nk_color top, struct nk_color right, struct nk_color bottom); -NK_API void nk_fill_circle(struct nk_command_buffer*, struct nk_rect, struct nk_color); -NK_API void nk_fill_arc(struct nk_command_buffer*, float cx, float cy, float radius, float a_min, float a_max, struct nk_color); -NK_API void nk_fill_triangle(struct nk_command_buffer*, float x0, float y0, float x1, float y1, float x2, float y2, struct nk_color); -NK_API void nk_fill_polygon(struct nk_command_buffer*, float*, int point_count, struct nk_color); - -/* misc */ -NK_API void nk_draw_image(struct nk_command_buffer*, struct nk_rect, const struct nk_image*, struct nk_color); -NK_API void nk_draw_nine_slice(struct nk_command_buffer*, struct nk_rect, const struct nk_nine_slice*, struct nk_color); -NK_API void nk_draw_text(struct nk_command_buffer*, struct nk_rect, const char *text, int len, const struct nk_user_font*, struct nk_color, struct nk_color); -NK_API void nk_push_scissor(struct nk_command_buffer*, struct nk_rect); -NK_API void nk_push_custom(struct nk_command_buffer*, struct nk_rect, nk_command_custom_callback, nk_handle usr); - -/* =============================================================== - * - * INPUT - * - * ===============================================================*/ -struct nk_mouse_button { - nk_bool down; - unsigned int clicked; - struct nk_vec2 clicked_pos; -}; -struct nk_mouse { - struct nk_mouse_button buttons[NK_BUTTON_MAX]; - struct nk_vec2 pos; -#ifdef NK_BUTTON_TRIGGER_ON_RELEASE - struct nk_vec2 down_pos; -#endif - struct nk_vec2 prev; - struct nk_vec2 delta; - struct nk_vec2 scroll_delta; - unsigned char grab; - unsigned char grabbed; - unsigned char ungrab; -}; - -struct nk_key { - nk_bool down; - unsigned int clicked; -}; -struct nk_keyboard { - struct nk_key keys[NK_KEY_MAX]; - char text[NK_INPUT_MAX]; - int text_len; -}; - -struct nk_input { - struct nk_keyboard keyboard; - struct nk_mouse mouse; -}; - -NK_API nk_bool nk_input_has_mouse_click(const struct nk_input*, enum nk_buttons); -NK_API nk_bool nk_input_has_mouse_click_in_rect(const struct nk_input*, enum nk_buttons, struct nk_rect); -NK_API nk_bool nk_input_has_mouse_click_in_button_rect(const struct nk_input*, enum nk_buttons, struct nk_rect); -NK_API nk_bool nk_input_has_mouse_click_down_in_rect(const struct nk_input*, enum nk_buttons, struct nk_rect, nk_bool down); -NK_API nk_bool nk_input_is_mouse_click_in_rect(const struct nk_input*, enum nk_buttons, struct nk_rect); -NK_API nk_bool nk_input_is_mouse_click_down_in_rect(const struct nk_input *i, enum nk_buttons id, struct nk_rect b, nk_bool down); -NK_API nk_bool nk_input_any_mouse_click_in_rect(const struct nk_input*, struct nk_rect); -NK_API nk_bool nk_input_is_mouse_prev_hovering_rect(const struct nk_input*, struct nk_rect); -NK_API nk_bool nk_input_is_mouse_hovering_rect(const struct nk_input*, struct nk_rect); -NK_API nk_bool nk_input_mouse_clicked(const struct nk_input*, enum nk_buttons, struct nk_rect); -NK_API nk_bool nk_input_is_mouse_down(const struct nk_input*, enum nk_buttons); -NK_API nk_bool nk_input_is_mouse_pressed(const struct nk_input*, enum nk_buttons); -NK_API nk_bool nk_input_is_mouse_released(const struct nk_input*, enum nk_buttons); -NK_API nk_bool nk_input_is_key_pressed(const struct nk_input*, enum nk_keys); -NK_API nk_bool nk_input_is_key_released(const struct nk_input*, enum nk_keys); -NK_API nk_bool nk_input_is_key_down(const struct nk_input*, enum nk_keys); - -/* =============================================================== - * - * DRAW LIST - * - * ===============================================================*/ -#ifdef NK_INCLUDE_VERTEX_BUFFER_OUTPUT -/* ### Draw List -/// The optional vertex buffer draw list provides a 2D drawing context -/// with antialiasing functionality which takes basic filled or outlined shapes -/// or a path and outputs vertexes, elements and draw commands. -/// The actual draw list API is not required to be used directly while using this -/// library since converting the default library draw command output is done by -/// just calling `nk_convert` but I decided to still make this library accessible -/// since it can be useful. -/// -/// The draw list is based on a path buffering and polygon and polyline -/// rendering API which allows a lot of ways to draw 2D content to screen. -/// In fact it is probably more powerful than needed but allows even more crazy -/// things than this library provides by default. -*/ -#ifdef NK_UINT_DRAW_INDEX -typedef nk_uint nk_draw_index; -#else -typedef nk_ushort nk_draw_index; -#endif -enum nk_draw_list_stroke { - NK_STROKE_OPEN = nk_false, - /* build up path has no connection back to the beginning */ - NK_STROKE_CLOSED = nk_true - /* build up path has a connection back to the beginning */ -}; - -enum nk_draw_vertex_layout_attribute { - NK_VERTEX_POSITION, - NK_VERTEX_COLOR, - NK_VERTEX_TEXCOORD, - NK_VERTEX_ATTRIBUTE_COUNT -}; - -enum nk_draw_vertex_layout_format { - NK_FORMAT_SCHAR, - NK_FORMAT_SSHORT, - NK_FORMAT_SINT, - NK_FORMAT_UCHAR, - NK_FORMAT_USHORT, - NK_FORMAT_UINT, - NK_FORMAT_FLOAT, - NK_FORMAT_DOUBLE, - -NK_FORMAT_COLOR_BEGIN, - NK_FORMAT_R8G8B8 = NK_FORMAT_COLOR_BEGIN, - NK_FORMAT_R16G15B16, - NK_FORMAT_R32G32B32, - - NK_FORMAT_R8G8B8A8, - NK_FORMAT_B8G8R8A8, - NK_FORMAT_R16G15B16A16, - NK_FORMAT_R32G32B32A32, - NK_FORMAT_R32G32B32A32_FLOAT, - NK_FORMAT_R32G32B32A32_DOUBLE, - - NK_FORMAT_RGB32, - NK_FORMAT_RGBA32, -NK_FORMAT_COLOR_END = NK_FORMAT_RGBA32, - NK_FORMAT_COUNT -}; - -#define NK_VERTEX_LAYOUT_END NK_VERTEX_ATTRIBUTE_COUNT,NK_FORMAT_COUNT,0 -struct nk_draw_vertex_layout_element { - enum nk_draw_vertex_layout_attribute attribute; - enum nk_draw_vertex_layout_format format; - nk_size offset; -}; - -struct nk_draw_command { - unsigned int elem_count; - /* number of elements in the current draw batch */ - struct nk_rect clip_rect; - /* current screen clipping rectangle */ - nk_handle texture; - /* current texture to set */ -#ifdef NK_INCLUDE_COMMAND_USERDATA - nk_handle userdata; -#endif -}; - -struct nk_draw_list { - struct nk_rect clip_rect; - struct nk_vec2 circle_vtx[12]; - struct nk_convert_config config; - - struct nk_buffer *buffer; - struct nk_buffer *vertices; - struct nk_buffer *elements; - - unsigned int element_count; - unsigned int vertex_count; - unsigned int cmd_count; - nk_size cmd_offset; - - unsigned int path_count; - unsigned int path_offset; - - enum nk_anti_aliasing line_AA; - enum nk_anti_aliasing shape_AA; - -#ifdef NK_INCLUDE_COMMAND_USERDATA - nk_handle userdata; -#endif -}; - -/* draw list */ -NK_API void nk_draw_list_init(struct nk_draw_list*); -NK_API void nk_draw_list_setup(struct nk_draw_list*, const struct nk_convert_config*, struct nk_buffer *cmds, struct nk_buffer *vertices, struct nk_buffer *elements, enum nk_anti_aliasing line_aa,enum nk_anti_aliasing shape_aa); - -/* drawing */ -#define nk_draw_list_foreach(cmd, can, b) for((cmd)=nk__draw_list_begin(can, b); (cmd)!=0; (cmd)=nk__draw_list_next(cmd, b, can)) -NK_API const struct nk_draw_command* nk__draw_list_begin(const struct nk_draw_list*, const struct nk_buffer*); -NK_API const struct nk_draw_command* nk__draw_list_next(const struct nk_draw_command*, const struct nk_buffer*, const struct nk_draw_list*); -NK_API const struct nk_draw_command* nk__draw_list_end(const struct nk_draw_list*, const struct nk_buffer*); - -/* path */ -NK_API void nk_draw_list_path_clear(struct nk_draw_list*); -NK_API void nk_draw_list_path_line_to(struct nk_draw_list*, struct nk_vec2 pos); -NK_API void nk_draw_list_path_arc_to_fast(struct nk_draw_list*, struct nk_vec2 center, float radius, int a_min, int a_max); -NK_API void nk_draw_list_path_arc_to(struct nk_draw_list*, struct nk_vec2 center, float radius, float a_min, float a_max, unsigned int segments); -NK_API void nk_draw_list_path_rect_to(struct nk_draw_list*, struct nk_vec2 a, struct nk_vec2 b, float rounding); -NK_API void nk_draw_list_path_curve_to(struct nk_draw_list*, struct nk_vec2 p2, struct nk_vec2 p3, struct nk_vec2 p4, unsigned int num_segments); -NK_API void nk_draw_list_path_fill(struct nk_draw_list*, struct nk_color); -NK_API void nk_draw_list_path_stroke(struct nk_draw_list*, struct nk_color, enum nk_draw_list_stroke closed, float thickness); - -/* stroke */ -NK_API void nk_draw_list_stroke_line(struct nk_draw_list*, struct nk_vec2 a, struct nk_vec2 b, struct nk_color, float thickness); -NK_API void nk_draw_list_stroke_rect(struct nk_draw_list*, struct nk_rect rect, struct nk_color, float rounding, float thickness); -NK_API void nk_draw_list_stroke_triangle(struct nk_draw_list*, struct nk_vec2 a, struct nk_vec2 b, struct nk_vec2 c, struct nk_color, float thickness); -NK_API void nk_draw_list_stroke_circle(struct nk_draw_list*, struct nk_vec2 center, float radius, struct nk_color, unsigned int segs, float thickness); -NK_API void nk_draw_list_stroke_curve(struct nk_draw_list*, struct nk_vec2 p0, struct nk_vec2 cp0, struct nk_vec2 cp1, struct nk_vec2 p1, struct nk_color, unsigned int segments, float thickness); -NK_API void nk_draw_list_stroke_poly_line(struct nk_draw_list*, const struct nk_vec2 *pnts, const unsigned int cnt, struct nk_color, enum nk_draw_list_stroke, float thickness, enum nk_anti_aliasing); - -/* fill */ -NK_API void nk_draw_list_fill_rect(struct nk_draw_list*, struct nk_rect rect, struct nk_color, float rounding); -NK_API void nk_draw_list_fill_rect_multi_color(struct nk_draw_list*, struct nk_rect rect, struct nk_color left, struct nk_color top, struct nk_color right, struct nk_color bottom); -NK_API void nk_draw_list_fill_triangle(struct nk_draw_list*, struct nk_vec2 a, struct nk_vec2 b, struct nk_vec2 c, struct nk_color); -NK_API void nk_draw_list_fill_circle(struct nk_draw_list*, struct nk_vec2 center, float radius, struct nk_color col, unsigned int segs); -NK_API void nk_draw_list_fill_poly_convex(struct nk_draw_list*, const struct nk_vec2 *points, const unsigned int count, struct nk_color, enum nk_anti_aliasing); - -/* misc */ -NK_API void nk_draw_list_add_image(struct nk_draw_list*, struct nk_image texture, struct nk_rect rect, struct nk_color); -NK_API void nk_draw_list_add_text(struct nk_draw_list*, const struct nk_user_font*, struct nk_rect, const char *text, int len, float font_height, struct nk_color); -#ifdef NK_INCLUDE_COMMAND_USERDATA -NK_API void nk_draw_list_push_userdata(struct nk_draw_list*, nk_handle userdata); -#endif - -#endif - -/* =============================================================== - * - * GUI - * - * ===============================================================*/ -enum nk_style_item_type { - NK_STYLE_ITEM_COLOR, - NK_STYLE_ITEM_IMAGE, - NK_STYLE_ITEM_NINE_SLICE -}; - -union nk_style_item_data { - struct nk_color color; - struct nk_image image; - struct nk_nine_slice slice; -}; - -struct nk_style_item { - enum nk_style_item_type type; - union nk_style_item_data data; -}; - -struct nk_style_text { - struct nk_color color; - struct nk_vec2 padding; -}; - -struct nk_style_button { - /* background */ - struct nk_style_item normal; - struct nk_style_item hover; - struct nk_style_item active; - struct nk_color border_color; - - /* text */ - struct nk_color text_background; - struct nk_color text_normal; - struct nk_color text_hover; - struct nk_color text_active; - nk_flags text_alignment; - - /* properties */ - float border; - float rounding; - struct nk_vec2 padding; - struct nk_vec2 image_padding; - struct nk_vec2 touch_padding; - - /* optional user callbacks */ - nk_handle userdata; - void(*draw_begin)(struct nk_command_buffer*, nk_handle userdata); - void(*draw_end)(struct nk_command_buffer*, nk_handle userdata); -}; - -struct nk_style_toggle { - /* background */ - struct nk_style_item normal; - struct nk_style_item hover; - struct nk_style_item active; - struct nk_color border_color; - - /* cursor */ - struct nk_style_item cursor_normal; - struct nk_style_item cursor_hover; - - /* text */ - struct nk_color text_normal; - struct nk_color text_hover; - struct nk_color text_active; - struct nk_color text_background; - nk_flags text_alignment; - - /* properties */ - struct nk_vec2 padding; - struct nk_vec2 touch_padding; - float spacing; - float border; - - /* optional user callbacks */ - nk_handle userdata; - void(*draw_begin)(struct nk_command_buffer*, nk_handle); - void(*draw_end)(struct nk_command_buffer*, nk_handle); -}; - -struct nk_style_selectable { - /* background (inactive) */ - struct nk_style_item normal; - struct nk_style_item hover; - struct nk_style_item pressed; - - /* background (active) */ - struct nk_style_item normal_active; - struct nk_style_item hover_active; - struct nk_style_item pressed_active; - - /* text color (inactive) */ - struct nk_color text_normal; - struct nk_color text_hover; - struct nk_color text_pressed; - - /* text color (active) */ - struct nk_color text_normal_active; - struct nk_color text_hover_active; - struct nk_color text_pressed_active; - struct nk_color text_background; - nk_flags text_alignment; - - /* properties */ - float rounding; - struct nk_vec2 padding; - struct nk_vec2 touch_padding; - struct nk_vec2 image_padding; - - /* optional user callbacks */ - nk_handle userdata; - void(*draw_begin)(struct nk_command_buffer*, nk_handle); - void(*draw_end)(struct nk_command_buffer*, nk_handle); -}; - -struct nk_style_slider { - /* background */ - struct nk_style_item normal; - struct nk_style_item hover; - struct nk_style_item active; - struct nk_color border_color; - - /* background bar */ - struct nk_color bar_normal; - struct nk_color bar_hover; - struct nk_color bar_active; - struct nk_color bar_filled; - - /* cursor */ - struct nk_style_item cursor_normal; - struct nk_style_item cursor_hover; - struct nk_style_item cursor_active; - - /* properties */ - float border; - float rounding; - float bar_height; - struct nk_vec2 padding; - struct nk_vec2 spacing; - struct nk_vec2 cursor_size; - - /* optional buttons */ - int show_buttons; - struct nk_style_button inc_button; - struct nk_style_button dec_button; - enum nk_symbol_type inc_symbol; - enum nk_symbol_type dec_symbol; - - /* optional user callbacks */ - nk_handle userdata; - void(*draw_begin)(struct nk_command_buffer*, nk_handle); - void(*draw_end)(struct nk_command_buffer*, nk_handle); -}; - -struct nk_style_progress { - /* background */ - struct nk_style_item normal; - struct nk_style_item hover; - struct nk_style_item active; - struct nk_color border_color; - - /* cursor */ - struct nk_style_item cursor_normal; - struct nk_style_item cursor_hover; - struct nk_style_item cursor_active; - struct nk_color cursor_border_color; - - /* properties */ - float rounding; - float border; - float cursor_border; - float cursor_rounding; - struct nk_vec2 padding; - - /* optional user callbacks */ - nk_handle userdata; - void(*draw_begin)(struct nk_command_buffer*, nk_handle); - void(*draw_end)(struct nk_command_buffer*, nk_handle); -}; - -struct nk_style_scrollbar { - /* background */ - struct nk_style_item normal; - struct nk_style_item hover; - struct nk_style_item active; - struct nk_color border_color; - - /* cursor */ - struct nk_style_item cursor_normal; - struct nk_style_item cursor_hover; - struct nk_style_item cursor_active; - struct nk_color cursor_border_color; - - /* properties */ - float border; - float rounding; - float border_cursor; - float rounding_cursor; - struct nk_vec2 padding; - - /* optional buttons */ - int show_buttons; - struct nk_style_button inc_button; - struct nk_style_button dec_button; - enum nk_symbol_type inc_symbol; - enum nk_symbol_type dec_symbol; - - /* optional user callbacks */ - nk_handle userdata; - void(*draw_begin)(struct nk_command_buffer*, nk_handle); - void(*draw_end)(struct nk_command_buffer*, nk_handle); -}; - -struct nk_style_edit { - /* background */ - struct nk_style_item normal; - struct nk_style_item hover; - struct nk_style_item active; - struct nk_color border_color; - struct nk_style_scrollbar scrollbar; - - /* cursor */ - struct nk_color cursor_normal; - struct nk_color cursor_hover; - struct nk_color cursor_text_normal; - struct nk_color cursor_text_hover; - - /* text (unselected) */ - struct nk_color text_normal; - struct nk_color text_hover; - struct nk_color text_active; - - /* text (selected) */ - struct nk_color selected_normal; - struct nk_color selected_hover; - struct nk_color selected_text_normal; - struct nk_color selected_text_hover; - - /* properties */ - float border; - float rounding; - float cursor_size; - struct nk_vec2 scrollbar_size; - struct nk_vec2 padding; - float row_padding; -}; - -struct nk_style_property { - /* background */ - struct nk_style_item normal; - struct nk_style_item hover; - struct nk_style_item active; - struct nk_color border_color; - - /* text */ - struct nk_color label_normal; - struct nk_color label_hover; - struct nk_color label_active; - - /* symbols */ - enum nk_symbol_type sym_left; - enum nk_symbol_type sym_right; - - /* properties */ - float border; - float rounding; - struct nk_vec2 padding; - - struct nk_style_edit edit; - struct nk_style_button inc_button; - struct nk_style_button dec_button; - - /* optional user callbacks */ - nk_handle userdata; - void(*draw_begin)(struct nk_command_buffer*, nk_handle); - void(*draw_end)(struct nk_command_buffer*, nk_handle); -}; - -struct nk_style_chart { - /* colors */ - struct nk_style_item background; - struct nk_color border_color; - struct nk_color selected_color; - struct nk_color color; - - /* properties */ - float border; - float rounding; - struct nk_vec2 padding; -}; - -struct nk_style_combo { - /* background */ - struct nk_style_item normal; - struct nk_style_item hover; - struct nk_style_item active; - struct nk_color border_color; - - /* label */ - struct nk_color label_normal; - struct nk_color label_hover; - struct nk_color label_active; - - /* symbol */ - struct nk_color symbol_normal; - struct nk_color symbol_hover; - struct nk_color symbol_active; - - /* button */ - struct nk_style_button button; - enum nk_symbol_type sym_normal; - enum nk_symbol_type sym_hover; - enum nk_symbol_type sym_active; - - /* properties */ - float border; - float rounding; - struct nk_vec2 content_padding; - struct nk_vec2 button_padding; - struct nk_vec2 spacing; -}; - -struct nk_style_tab { - /* background */ - struct nk_style_item background; - struct nk_color border_color; - struct nk_color text; - - /* button */ - struct nk_style_button tab_maximize_button; - struct nk_style_button tab_minimize_button; - struct nk_style_button node_maximize_button; - struct nk_style_button node_minimize_button; - enum nk_symbol_type sym_minimize; - enum nk_symbol_type sym_maximize; - - /* properties */ - float border; - float rounding; - float indent; - struct nk_vec2 padding; - struct nk_vec2 spacing; -}; - -enum nk_style_header_align { - NK_HEADER_LEFT, - NK_HEADER_RIGHT -}; -struct nk_style_window_header { - /* background */ - struct nk_style_item normal; - struct nk_style_item hover; - struct nk_style_item active; - - /* button */ - struct nk_style_button close_button; - struct nk_style_button minimize_button; - enum nk_symbol_type close_symbol; - enum nk_symbol_type minimize_symbol; - enum nk_symbol_type maximize_symbol; - - /* title */ - struct nk_color label_normal; - struct nk_color label_hover; - struct nk_color label_active; - - /* properties */ - enum nk_style_header_align align; - struct nk_vec2 padding; - struct nk_vec2 label_padding; - struct nk_vec2 spacing; -}; - -struct nk_style_window { - struct nk_style_window_header header; - struct nk_style_item fixed_background; - struct nk_color background; - - struct nk_color border_color; - struct nk_color popup_border_color; - struct nk_color combo_border_color; - struct nk_color contextual_border_color; - struct nk_color menu_border_color; - struct nk_color group_border_color; - struct nk_color tooltip_border_color; - struct nk_style_item scaler; - - float border; - float combo_border; - float contextual_border; - float menu_border; - float group_border; - float tooltip_border; - float popup_border; - float min_row_height_padding; - - float rounding; - struct nk_vec2 spacing; - struct nk_vec2 scrollbar_size; - struct nk_vec2 min_size; - - struct nk_vec2 padding; - struct nk_vec2 group_padding; - struct nk_vec2 popup_padding; - struct nk_vec2 combo_padding; - struct nk_vec2 contextual_padding; - struct nk_vec2 menu_padding; - struct nk_vec2 tooltip_padding; -}; - -struct nk_style { - const struct nk_user_font *font; - const struct nk_cursor *cursors[NK_CURSOR_COUNT]; - const struct nk_cursor *cursor_active; - struct nk_cursor *cursor_last; - int cursor_visible; - - struct nk_style_text text; - struct nk_style_button button; - struct nk_style_button contextual_button; - struct nk_style_button menu_button; - struct nk_style_toggle option; - struct nk_style_toggle checkbox; - struct nk_style_selectable selectable; - struct nk_style_slider slider; - struct nk_style_progress progress; - struct nk_style_property property; - struct nk_style_edit edit; - struct nk_style_chart chart; - struct nk_style_scrollbar scrollh; - struct nk_style_scrollbar scrollv; - struct nk_style_tab tab; - struct nk_style_combo combo; - struct nk_style_window window; -}; - -NK_API struct nk_style_item nk_style_item_color(struct nk_color); -NK_API struct nk_style_item nk_style_item_image(struct nk_image img); -NK_API struct nk_style_item nk_style_item_nine_slice(struct nk_nine_slice slice); -NK_API struct nk_style_item nk_style_item_hide(void); - -/*============================================================== - * PANEL - * =============================================================*/ -#ifndef NK_MAX_LAYOUT_ROW_TEMPLATE_COLUMNS -#define NK_MAX_LAYOUT_ROW_TEMPLATE_COLUMNS 16 -#endif -#ifndef NK_CHART_MAX_SLOT -#define NK_CHART_MAX_SLOT 4 -#endif - -enum nk_panel_type { - NK_PANEL_NONE = 0, - NK_PANEL_WINDOW = NK_FLAG(0), - NK_PANEL_GROUP = NK_FLAG(1), - NK_PANEL_POPUP = NK_FLAG(2), - NK_PANEL_CONTEXTUAL = NK_FLAG(4), - NK_PANEL_COMBO = NK_FLAG(5), - NK_PANEL_MENU = NK_FLAG(6), - NK_PANEL_TOOLTIP = NK_FLAG(7) -}; -enum nk_panel_set { - NK_PANEL_SET_NONBLOCK = NK_PANEL_CONTEXTUAL|NK_PANEL_COMBO|NK_PANEL_MENU|NK_PANEL_TOOLTIP, - NK_PANEL_SET_POPUP = NK_PANEL_SET_NONBLOCK|NK_PANEL_POPUP, - NK_PANEL_SET_SUB = NK_PANEL_SET_POPUP|NK_PANEL_GROUP -}; - -struct nk_chart_slot { - enum nk_chart_type type; - struct nk_color color; - struct nk_color highlight; - float min, max, range; - int count; - struct nk_vec2 last; - int index; -}; - -struct nk_chart { - int slot; - float x, y, w, h; - struct nk_chart_slot slots[NK_CHART_MAX_SLOT]; -}; - -enum nk_panel_row_layout_type { - NK_LAYOUT_DYNAMIC_FIXED = 0, - NK_LAYOUT_DYNAMIC_ROW, - NK_LAYOUT_DYNAMIC_FREE, - NK_LAYOUT_DYNAMIC, - NK_LAYOUT_STATIC_FIXED, - NK_LAYOUT_STATIC_ROW, - NK_LAYOUT_STATIC_FREE, - NK_LAYOUT_STATIC, - NK_LAYOUT_TEMPLATE, - NK_LAYOUT_COUNT -}; -struct nk_row_layout { - enum nk_panel_row_layout_type type; - int index; - float height; - float min_height; - int columns; - const float *ratio; - float item_width; - float item_height; - float item_offset; - float filled; - struct nk_rect item; - int tree_depth; - float templates[NK_MAX_LAYOUT_ROW_TEMPLATE_COLUMNS]; -}; - -struct nk_popup_buffer { - nk_size begin; - nk_size parent; - nk_size last; - nk_size end; - nk_bool active; -}; - -struct nk_menu_state { - float x, y, w, h; - struct nk_scroll offset; -}; - -struct nk_panel { - enum nk_panel_type type; - nk_flags flags; - struct nk_rect bounds; - nk_uint *offset_x; - nk_uint *offset_y; - float at_x, at_y, max_x; - float footer_height; - float header_height; - float border; - unsigned int has_scrolling; - struct nk_rect clip; - struct nk_menu_state menu; - struct nk_row_layout row; - struct nk_chart chart; - struct nk_command_buffer *buffer; - struct nk_panel *parent; -}; - -/*============================================================== - * WINDOW - * =============================================================*/ -#ifndef NK_WINDOW_MAX_NAME -#define NK_WINDOW_MAX_NAME 64 -#endif - -struct nk_table; -enum nk_window_flags { - NK_WINDOW_PRIVATE = NK_FLAG(11), - NK_WINDOW_DYNAMIC = NK_WINDOW_PRIVATE, - /* special window type growing up in height while being filled to a certain maximum height */ - NK_WINDOW_ROM = NK_FLAG(12), - /* sets window widgets into a read only mode and does not allow input changes */ - NK_WINDOW_NOT_INTERACTIVE = NK_WINDOW_ROM|NK_WINDOW_NO_INPUT, - /* prevents all interaction caused by input to either window or widgets inside */ - NK_WINDOW_HIDDEN = NK_FLAG(13), - /* Hides window and stops any window interaction and drawing */ - NK_WINDOW_CLOSED = NK_FLAG(14), - /* Directly closes and frees the window at the end of the frame */ - NK_WINDOW_MINIMIZED = NK_FLAG(15), - /* marks the window as minimized */ - NK_WINDOW_REMOVE_ROM = NK_FLAG(16) - /* Removes read only mode at the end of the window */ -}; - -struct nk_popup_state { - struct nk_window *win; - enum nk_panel_type type; - struct nk_popup_buffer buf; - nk_hash name; - nk_bool active; - unsigned combo_count; - unsigned con_count, con_old; - unsigned active_con; - struct nk_rect header; -}; - -struct nk_edit_state { - nk_hash name; - unsigned int seq; - unsigned int old; - int active, prev; - int cursor; - int sel_start; - int sel_end; - struct nk_scroll scrollbar; - unsigned char mode; - unsigned char single_line; -}; - -struct nk_property_state { - int active, prev; - char buffer[NK_MAX_NUMBER_BUFFER]; - int length; - int cursor; - int select_start; - int select_end; - nk_hash name; - unsigned int seq; - unsigned int old; - int state; -}; - -struct nk_window { - unsigned int seq; - nk_hash name; - char name_string[NK_WINDOW_MAX_NAME]; - nk_flags flags; - - struct nk_rect bounds; - struct nk_scroll scrollbar; - struct nk_command_buffer buffer; - struct nk_panel *layout; - float scrollbar_hiding_timer; - - /* persistent widget state */ - struct nk_property_state property; - struct nk_popup_state popup; - struct nk_edit_state edit; - unsigned int scrolled; - - struct nk_table *tables; - unsigned int table_count; - - /* window list hooks */ - struct nk_window *next; - struct nk_window *prev; - struct nk_window *parent; -}; - -/*============================================================== - * STACK - * =============================================================*/ -/*/// ### Stack -/// The style modifier stack can be used to temporarily change a -/// property inside `nk_style`. For example if you want a special -/// red button you can temporarily push the old button color onto a stack -/// draw the button with a red color and then you just pop the old color -/// back from the stack: -/// -/// nk_style_push_style_item(ctx, &ctx->style.button.normal, nk_style_item_color(nk_rgb(255,0,0))); -/// nk_style_push_style_item(ctx, &ctx->style.button.hover, nk_style_item_color(nk_rgb(255,0,0))); -/// nk_style_push_style_item(ctx, &ctx->style.button.active, nk_style_item_color(nk_rgb(255,0,0))); -/// nk_style_push_vec2(ctx, &cx->style.button.padding, nk_vec2(2,2)); -/// -/// nk_button(...); -/// -/// nk_style_pop_style_item(ctx); -/// nk_style_pop_style_item(ctx); -/// nk_style_pop_style_item(ctx); -/// nk_style_pop_vec2(ctx); -/// -/// Nuklear has a stack for style_items, float properties, vector properties, -/// flags, colors, fonts and for button_behavior. Each has it's own fixed size stack -/// which can be changed at compile time. - */ -#ifndef NK_BUTTON_BEHAVIOR_STACK_SIZE -#define NK_BUTTON_BEHAVIOR_STACK_SIZE 8 -#endif - -#ifndef NK_FONT_STACK_SIZE -#define NK_FONT_STACK_SIZE 8 -#endif - -#ifndef NK_STYLE_ITEM_STACK_SIZE -#define NK_STYLE_ITEM_STACK_SIZE 16 -#endif - -#ifndef NK_FLOAT_STACK_SIZE -#define NK_FLOAT_STACK_SIZE 32 -#endif - -#ifndef NK_VECTOR_STACK_SIZE -#define NK_VECTOR_STACK_SIZE 16 -#endif - -#ifndef NK_FLAGS_STACK_SIZE -#define NK_FLAGS_STACK_SIZE 32 -#endif - -#ifndef NK_COLOR_STACK_SIZE -#define NK_COLOR_STACK_SIZE 32 -#endif - -#define NK_CONFIGURATION_STACK_TYPE(prefix, name, type)\ - struct nk_config_stack_##name##_element {\ - prefix##_##type *address;\ - prefix##_##type old_value;\ - } -#define NK_CONFIG_STACK(type,size)\ - struct nk_config_stack_##type {\ - int head;\ - struct nk_config_stack_##type##_element elements[size];\ - } - -#define nk_float float -NK_CONFIGURATION_STACK_TYPE(struct nk, style_item, style_item); -NK_CONFIGURATION_STACK_TYPE(nk ,float, float); -NK_CONFIGURATION_STACK_TYPE(struct nk, vec2, vec2); -NK_CONFIGURATION_STACK_TYPE(nk ,flags, flags); -NK_CONFIGURATION_STACK_TYPE(struct nk, color, color); -NK_CONFIGURATION_STACK_TYPE(const struct nk, user_font, user_font*); -NK_CONFIGURATION_STACK_TYPE(enum nk, button_behavior, button_behavior); - -NK_CONFIG_STACK(style_item, NK_STYLE_ITEM_STACK_SIZE); -NK_CONFIG_STACK(float, NK_FLOAT_STACK_SIZE); -NK_CONFIG_STACK(vec2, NK_VECTOR_STACK_SIZE); -NK_CONFIG_STACK(flags, NK_FLAGS_STACK_SIZE); -NK_CONFIG_STACK(color, NK_COLOR_STACK_SIZE); -NK_CONFIG_STACK(user_font, NK_FONT_STACK_SIZE); -NK_CONFIG_STACK(button_behavior, NK_BUTTON_BEHAVIOR_STACK_SIZE); - -struct nk_configuration_stacks { - struct nk_config_stack_style_item style_items; - struct nk_config_stack_float floats; - struct nk_config_stack_vec2 vectors; - struct nk_config_stack_flags flags; - struct nk_config_stack_color colors; - struct nk_config_stack_user_font fonts; - struct nk_config_stack_button_behavior button_behaviors; -}; - -/*============================================================== - * CONTEXT - * =============================================================*/ -#define NK_VALUE_PAGE_CAPACITY \ - (((NK_MAX(sizeof(struct nk_window),sizeof(struct nk_panel)) / sizeof(nk_uint))) / 2) - -struct nk_table { - unsigned int seq; - unsigned int size; - nk_hash keys[NK_VALUE_PAGE_CAPACITY]; - nk_uint values[NK_VALUE_PAGE_CAPACITY]; - struct nk_table *next, *prev; -}; - -union nk_page_data { - struct nk_table tbl; - struct nk_panel pan; - struct nk_window win; -}; - -struct nk_page_element { - union nk_page_data data; - struct nk_page_element *next; - struct nk_page_element *prev; -}; - -struct nk_page { - unsigned int size; - struct nk_page *next; - struct nk_page_element win[1]; -}; - -struct nk_pool { - struct nk_allocator alloc; - enum nk_allocation_type type; - unsigned int page_count; - struct nk_page *pages; - struct nk_page_element *freelist; - unsigned capacity; - nk_size size; - nk_size cap; -}; - -struct nk_context { -/* public: can be accessed freely */ - struct nk_input input; - struct nk_style style; - struct nk_buffer memory; - struct nk_clipboard clip; - nk_flags last_widget_state; - enum nk_button_behavior button_behavior; - struct nk_configuration_stacks stacks; - float delta_time_seconds; - -/* private: - should only be accessed if you - know what you are doing */ -#ifdef NK_INCLUDE_VERTEX_BUFFER_OUTPUT - struct nk_draw_list draw_list; -#endif -#ifdef NK_INCLUDE_COMMAND_USERDATA - nk_handle userdata; -#endif - /* text editor objects are quite big because of an internal - * undo/redo stack. Therefore it does not make sense to have one for - * each window for temporary use cases, so I only provide *one* instance - * for all windows. This works because the content is cleared anyway */ - struct nk_text_edit text_edit; - /* draw buffer used for overlay drawing operation like cursor */ - struct nk_command_buffer overlay; - - /* windows */ - int build; - int use_pool; - struct nk_pool pool; - struct nk_window *begin; - struct nk_window *end; - struct nk_window *active; - struct nk_window *current; - struct nk_page_element *freelist; - unsigned int count; - unsigned int seq; -}; - -/* ============================================================== - * MATH - * =============================================================== */ -#define NK_PI 3.141592654f -#define NK_UTF_INVALID 0xFFFD -#define NK_MAX_FLOAT_PRECISION 2 - -#define NK_UNUSED(x) ((void)(x)) -#define NK_SATURATE(x) (NK_MAX(0, NK_MIN(1.0f, x))) -#define NK_LEN(a) (sizeof(a)/sizeof(a)[0]) -#define NK_ABS(a) (((a) < 0) ? -(a) : (a)) -#define NK_BETWEEN(x, a, b) ((a) <= (x) && (x) < (b)) -#define NK_INBOX(px, py, x, y, w, h)\ - (NK_BETWEEN(px,x,x+w) && NK_BETWEEN(py,y,y+h)) -#define NK_INTERSECT(x0, y0, w0, h0, x1, y1, w1, h1) \ - ((x1 < (x0 + w0)) && (x0 < (x1 + w1)) && \ - (y1 < (y0 + h0)) && (y0 < (y1 + h1))) -#define NK_CONTAINS(x, y, w, h, bx, by, bw, bh)\ - (NK_INBOX(x,y, bx, by, bw, bh) && NK_INBOX(x+w,y+h, bx, by, bw, bh)) - -#define nk_vec2_sub(a, b) nk_vec2((a).x - (b).x, (a).y - (b).y) -#define nk_vec2_add(a, b) nk_vec2((a).x + (b).x, (a).y + (b).y) -#define nk_vec2_len_sqr(a) ((a).x*(a).x+(a).y*(a).y) -#define nk_vec2_muls(a, t) nk_vec2((a).x * (t), (a).y * (t)) - -#define nk_ptr_add(t, p, i) ((t*)((void*)((nk_byte*)(p) + (i)))) -#define nk_ptr_add_const(t, p, i) ((const t*)((const void*)((const nk_byte*)(p) + (i)))) -#define nk_zero_struct(s) nk_zero(&s, sizeof(s)) - -/* ============================================================== - * ALIGNMENT - * =============================================================== */ -/* Pointer to Integer type conversion for pointer alignment */ -#if defined(__PTRDIFF_TYPE__) /* This case should work for GCC*/ -# define NK_UINT_TO_PTR(x) ((void*)(__PTRDIFF_TYPE__)(x)) -# define NK_PTR_TO_UINT(x) ((nk_size)(__PTRDIFF_TYPE__)(x)) -#elif !defined(__GNUC__) /* works for compilers other than LLVM */ -# define NK_UINT_TO_PTR(x) ((void*)&((char*)0)[x]) -# define NK_PTR_TO_UINT(x) ((nk_size)(((char*)x)-(char*)0)) -#elif defined(NK_USE_FIXED_TYPES) /* used if we have */ -# define NK_UINT_TO_PTR(x) ((void*)(uintptr_t)(x)) -# define NK_PTR_TO_UINT(x) ((uintptr_t)(x)) -#else /* generates warning but works */ -# define NK_UINT_TO_PTR(x) ((void*)(x)) -# define NK_PTR_TO_UINT(x) ((nk_size)(x)) -#endif - -#define NK_ALIGN_PTR(x, mask)\ - (NK_UINT_TO_PTR((NK_PTR_TO_UINT((nk_byte*)(x) + (mask-1)) & ~(mask-1)))) -#define NK_ALIGN_PTR_BACK(x, mask)\ - (NK_UINT_TO_PTR((NK_PTR_TO_UINT((nk_byte*)(x)) & ~(mask-1)))) - -#if (defined(__GNUC__) && __GNUC__ >= 4) || defined(__clang__) -#define NK_OFFSETOF(st,m) (__builtin_offsetof(st,m)) -#else -#define NK_OFFSETOF(st,m) ((nk_ptr)&(((st*)0)->m)) -#endif - -#ifdef __cplusplus -} -#endif - -#ifdef __cplusplus -template struct nk_alignof; -template struct nk_helper{enum {value = size_diff};}; -template struct nk_helper{enum {value = nk_alignof::value};}; -template struct nk_alignof{struct Big {T x; char c;}; enum { - diff = sizeof(Big) - sizeof(T), value = nk_helper::value};}; -#define NK_ALIGNOF(t) (nk_alignof::value) -#else -#define NK_ALIGNOF(t) NK_OFFSETOF(struct {char c; t _h;}, _h) -#endif - -#define NK_CONTAINER_OF(ptr,type,member)\ - (type*)((void*)((char*)(1 ? (ptr): &((type*)0)->member) - NK_OFFSETOF(type, member))) - - - -#endif /* NK_NUKLEAR_H_ */ - -#ifdef NK_IMPLEMENTATION - -#ifndef NK_INTERNAL_H -#define NK_INTERNAL_H - -#ifndef NK_POOL_DEFAULT_CAPACITY -#define NK_POOL_DEFAULT_CAPACITY 16 -#endif - -#ifndef NK_DEFAULT_COMMAND_BUFFER_SIZE -#define NK_DEFAULT_COMMAND_BUFFER_SIZE (4*1024) -#endif - -#ifndef NK_BUFFER_DEFAULT_INITIAL_SIZE -#define NK_BUFFER_DEFAULT_INITIAL_SIZE (4*1024) -#endif - -/* standard library headers */ -#ifdef NK_INCLUDE_DEFAULT_ALLOCATOR -#include /* malloc, free */ -#endif -#ifdef NK_INCLUDE_STANDARD_IO -#include /* fopen, fclose,... */ -#endif -#ifdef NK_INCLUDE_STANDARD_VARARGS -#include /* valist, va_start, va_end, ... */ -#endif -#ifndef NK_ASSERT -#include -#define NK_ASSERT(expr) assert(expr) -#endif - -#define NK_DEFAULT (-1) - -#ifndef NK_VSNPRINTF -/* If your compiler does support `vsnprintf` I would highly recommend - * defining this to vsnprintf instead since `vsprintf` is basically - * unbelievable unsafe and should *NEVER* be used. But I have to support - * it since C89 only provides this unsafe version. */ - #if (defined(__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L)) ||\ - (defined(__cplusplus) && (__cplusplus >= 201103L)) || \ - (defined(_POSIX_C_SOURCE) && (_POSIX_C_SOURCE >= 200112L)) ||\ - (defined(_XOPEN_SOURCE) && (_XOPEN_SOURCE >= 500)) ||\ - defined(_ISOC99_SOURCE) || defined(_BSD_SOURCE) - #define NK_VSNPRINTF(s,n,f,a) vsnprintf(s,n,f,a) - #else - #define NK_VSNPRINTF(s,n,f,a) vsprintf(s,f,a) - #endif -#endif - -#define NK_SCHAR_MIN (-127) -#define NK_SCHAR_MAX 127 -#define NK_UCHAR_MIN 0 -#define NK_UCHAR_MAX 256 -#define NK_SSHORT_MIN (-32767) -#define NK_SSHORT_MAX 32767 -#define NK_USHORT_MIN 0 -#define NK_USHORT_MAX 65535 -#define NK_SINT_MIN (-2147483647) -#define NK_SINT_MAX 2147483647 -#define NK_UINT_MIN 0 -#define NK_UINT_MAX 4294967295u - -/* Make sure correct type size: - * This will fire with a negative subscript error if the type sizes - * are set incorrectly by the compiler, and compile out if not */ -NK_STATIC_ASSERT(sizeof(nk_size) >= sizeof(void*)); -NK_STATIC_ASSERT(sizeof(nk_ptr) == sizeof(void*)); -NK_STATIC_ASSERT(sizeof(nk_flags) >= 4); -NK_STATIC_ASSERT(sizeof(nk_rune) >= 4); -NK_STATIC_ASSERT(sizeof(nk_ushort) == 2); -NK_STATIC_ASSERT(sizeof(nk_short) == 2); -NK_STATIC_ASSERT(sizeof(nk_uint) == 4); -NK_STATIC_ASSERT(sizeof(nk_int) == 4); -NK_STATIC_ASSERT(sizeof(nk_byte) == 1); -#ifdef NK_INCLUDE_STANDARD_BOOL -NK_STATIC_ASSERT(sizeof(nk_bool) == sizeof(bool)); -#else -NK_STATIC_ASSERT(sizeof(nk_bool) == 4); -#endif - -NK_GLOBAL const struct nk_rect nk_null_rect = {-8192.0f, -8192.0f, 16384, 16384}; -#define NK_FLOAT_PRECISION 0.00000000000001 - -NK_GLOBAL const struct nk_color nk_red = {255,0,0,255}; -NK_GLOBAL const struct nk_color nk_green = {0,255,0,255}; -NK_GLOBAL const struct nk_color nk_blue = {0,0,255,255}; -NK_GLOBAL const struct nk_color nk_white = {255,255,255,255}; -NK_GLOBAL const struct nk_color nk_black = {0,0,0,255}; -NK_GLOBAL const struct nk_color nk_yellow = {255,255,0,255}; - -/* widget */ -#define nk_widget_state_reset(s)\ - if ((*(s)) & NK_WIDGET_STATE_MODIFIED)\ - (*(s)) = NK_WIDGET_STATE_INACTIVE|NK_WIDGET_STATE_MODIFIED;\ - else (*(s)) = NK_WIDGET_STATE_INACTIVE; - -/* math */ -#ifndef NK_INV_SQRT -NK_LIB float nk_inv_sqrt(float n); -#endif -#ifndef NK_SIN -NK_LIB float nk_sin(float x); -#endif -#ifndef NK_COS -NK_LIB float nk_cos(float x); -#endif -NK_LIB nk_uint nk_round_up_pow2(nk_uint v); -NK_LIB struct nk_rect nk_shrink_rect(struct nk_rect r, float amount); -NK_LIB struct nk_rect nk_pad_rect(struct nk_rect r, struct nk_vec2 pad); -NK_LIB void nk_unify(struct nk_rect *clip, const struct nk_rect *a, float x0, float y0, float x1, float y1); -NK_LIB double nk_pow(double x, int n); -NK_LIB int nk_ifloord(double x); -NK_LIB int nk_ifloorf(float x); -NK_LIB int nk_iceilf(float x); -NK_LIB int nk_log10(double n); - -/* util */ -enum {NK_DO_NOT_STOP_ON_NEW_LINE, NK_STOP_ON_NEW_LINE}; -NK_LIB nk_bool nk_is_lower(int c); -NK_LIB nk_bool nk_is_upper(int c); -NK_LIB int nk_to_upper(int c); -NK_LIB int nk_to_lower(int c); - -#ifndef NK_MEMCPY -NK_LIB void* nk_memcopy(void *dst, const void *src, nk_size n); -#endif -#ifndef NK_MEMSET -NK_LIB void nk_memset(void *ptr, int c0, nk_size size); -#endif -NK_LIB void nk_zero(void *ptr, nk_size size); -NK_LIB char *nk_itoa(char *s, long n); -NK_LIB int nk_string_float_limit(char *string, int prec); -#ifndef NK_DTOA -NK_LIB char *nk_dtoa(char *s, double n); -#endif -NK_LIB int nk_text_clamp(const struct nk_user_font *font, const char *text, int text_len, float space, int *glyphs, float *text_width, nk_rune *sep_list, int sep_count); -NK_LIB struct nk_vec2 nk_text_calculate_text_bounds(const struct nk_user_font *font, const char *begin, int byte_len, float row_height, const char **remaining, struct nk_vec2 *out_offset, int *glyphs, int op); -#ifdef NK_INCLUDE_STANDARD_VARARGS -NK_LIB int nk_strfmt(char *buf, int buf_size, const char *fmt, va_list args); -#endif -#ifdef NK_INCLUDE_STANDARD_IO -NK_LIB char *nk_file_load(const char* path, nk_size* siz, struct nk_allocator *alloc); -#endif - -/* buffer */ -#ifdef NK_INCLUDE_DEFAULT_ALLOCATOR -NK_LIB void* nk_malloc(nk_handle unused, void *old,nk_size size); -NK_LIB void nk_mfree(nk_handle unused, void *ptr); -#endif -NK_LIB void* nk_buffer_align(void *unaligned, nk_size align, nk_size *alignment, enum nk_buffer_allocation_type type); -NK_LIB void* nk_buffer_alloc(struct nk_buffer *b, enum nk_buffer_allocation_type type, nk_size size, nk_size align); -NK_LIB void* nk_buffer_realloc(struct nk_buffer *b, nk_size capacity, nk_size *size); - -/* draw */ -NK_LIB void nk_command_buffer_init(struct nk_command_buffer *cb, struct nk_buffer *b, enum nk_command_clipping clip); -NK_LIB void nk_command_buffer_reset(struct nk_command_buffer *b); -NK_LIB void* nk_command_buffer_push(struct nk_command_buffer* b, enum nk_command_type t, nk_size size); -NK_LIB void nk_draw_symbol(struct nk_command_buffer *out, enum nk_symbol_type type, struct nk_rect content, struct nk_color background, struct nk_color foreground, float border_width, const struct nk_user_font *font); - -/* buffering */ -NK_LIB void nk_start_buffer(struct nk_context *ctx, struct nk_command_buffer *b); -NK_LIB void nk_start(struct nk_context *ctx, struct nk_window *win); -NK_LIB void nk_start_popup(struct nk_context *ctx, struct nk_window *win); -NK_LIB void nk_finish_popup(struct nk_context *ctx, struct nk_window*); -NK_LIB void nk_finish_buffer(struct nk_context *ctx, struct nk_command_buffer *b); -NK_LIB void nk_finish(struct nk_context *ctx, struct nk_window *w); -NK_LIB void nk_build(struct nk_context *ctx); - -/* text editor */ -NK_LIB void nk_textedit_clear_state(struct nk_text_edit *state, enum nk_text_edit_type type, nk_plugin_filter filter); -NK_LIB void nk_textedit_click(struct nk_text_edit *state, float x, float y, const struct nk_user_font *font, float row_height); -NK_LIB void nk_textedit_drag(struct nk_text_edit *state, float x, float y, const struct nk_user_font *font, float row_height); -NK_LIB void nk_textedit_key(struct nk_text_edit *state, enum nk_keys key, int shift_mod, const struct nk_user_font *font, float row_height); - -/* window */ -enum nk_window_insert_location { - NK_INSERT_BACK, /* inserts window into the back of list (front of screen) */ - NK_INSERT_FRONT /* inserts window into the front of list (back of screen) */ -}; -NK_LIB void *nk_create_window(struct nk_context *ctx); -NK_LIB void nk_remove_window(struct nk_context*, struct nk_window*); -NK_LIB void nk_free_window(struct nk_context *ctx, struct nk_window *win); -NK_LIB struct nk_window *nk_find_window(struct nk_context *ctx, nk_hash hash, const char *name); -NK_LIB void nk_insert_window(struct nk_context *ctx, struct nk_window *win, enum nk_window_insert_location loc); - -/* pool */ -NK_LIB void nk_pool_init(struct nk_pool *pool, struct nk_allocator *alloc, unsigned int capacity); -NK_LIB void nk_pool_free(struct nk_pool *pool); -NK_LIB void nk_pool_init_fixed(struct nk_pool *pool, void *memory, nk_size size); -NK_LIB struct nk_page_element *nk_pool_alloc(struct nk_pool *pool); - -/* page-element */ -NK_LIB struct nk_page_element* nk_create_page_element(struct nk_context *ctx); -NK_LIB void nk_link_page_element_into_freelist(struct nk_context *ctx, struct nk_page_element *elem); -NK_LIB void nk_free_page_element(struct nk_context *ctx, struct nk_page_element *elem); - -/* table */ -NK_LIB struct nk_table* nk_create_table(struct nk_context *ctx); -NK_LIB void nk_remove_table(struct nk_window *win, struct nk_table *tbl); -NK_LIB void nk_free_table(struct nk_context *ctx, struct nk_table *tbl); -NK_LIB void nk_push_table(struct nk_window *win, struct nk_table *tbl); -NK_LIB nk_uint *nk_add_value(struct nk_context *ctx, struct nk_window *win, nk_hash name, nk_uint value); -NK_LIB nk_uint *nk_find_value(struct nk_window *win, nk_hash name); - -/* panel */ -NK_LIB void *nk_create_panel(struct nk_context *ctx); -NK_LIB void nk_free_panel(struct nk_context*, struct nk_panel *pan); -NK_LIB nk_bool nk_panel_has_header(nk_flags flags, const char *title); -NK_LIB struct nk_vec2 nk_panel_get_padding(const struct nk_style *style, enum nk_panel_type type); -NK_LIB float nk_panel_get_border(const struct nk_style *style, nk_flags flags, enum nk_panel_type type); -NK_LIB struct nk_color nk_panel_get_border_color(const struct nk_style *style, enum nk_panel_type type); -NK_LIB nk_bool nk_panel_is_sub(enum nk_panel_type type); -NK_LIB nk_bool nk_panel_is_nonblock(enum nk_panel_type type); -NK_LIB nk_bool nk_panel_begin(struct nk_context *ctx, const char *title, enum nk_panel_type panel_type); -NK_LIB void nk_panel_end(struct nk_context *ctx); - -/* layout */ -NK_LIB float nk_layout_row_calculate_usable_space(const struct nk_style *style, enum nk_panel_type type, float total_space, int columns); -NK_LIB void nk_panel_layout(const struct nk_context *ctx, struct nk_window *win, float height, int cols); -NK_LIB void nk_row_layout(struct nk_context *ctx, enum nk_layout_format fmt, float height, int cols, int width); -NK_LIB void nk_panel_alloc_row(const struct nk_context *ctx, struct nk_window *win); -NK_LIB void nk_layout_widget_space(struct nk_rect *bounds, const struct nk_context *ctx, struct nk_window *win, int modify); -NK_LIB void nk_panel_alloc_space(struct nk_rect *bounds, const struct nk_context *ctx); -NK_LIB void nk_layout_peek(struct nk_rect *bounds, struct nk_context *ctx); - -/* popup */ -NK_LIB nk_bool nk_nonblock_begin(struct nk_context *ctx, nk_flags flags, struct nk_rect body, struct nk_rect header, enum nk_panel_type panel_type); - -/* text */ -struct nk_text { - struct nk_vec2 padding; - struct nk_color background; - struct nk_color text; -}; -NK_LIB void nk_widget_text(struct nk_command_buffer *o, struct nk_rect b, const char *string, int len, const struct nk_text *t, nk_flags a, const struct nk_user_font *f); -NK_LIB void nk_widget_text_wrap(struct nk_command_buffer *o, struct nk_rect b, const char *string, int len, const struct nk_text *t, const struct nk_user_font *f); - -/* button */ -NK_LIB nk_bool nk_button_behavior(nk_flags *state, struct nk_rect r, const struct nk_input *i, enum nk_button_behavior behavior); -NK_LIB const struct nk_style_item* nk_draw_button(struct nk_command_buffer *out, const struct nk_rect *bounds, nk_flags state, const struct nk_style_button *style); -NK_LIB nk_bool nk_do_button(nk_flags *state, struct nk_command_buffer *out, struct nk_rect r, const struct nk_style_button *style, const struct nk_input *in, enum nk_button_behavior behavior, struct nk_rect *content); -NK_LIB void nk_draw_button_text(struct nk_command_buffer *out, const struct nk_rect *bounds, const struct nk_rect *content, nk_flags state, const struct nk_style_button *style, const char *txt, int len, nk_flags text_alignment, const struct nk_user_font *font); -NK_LIB nk_bool nk_do_button_text(nk_flags *state, struct nk_command_buffer *out, struct nk_rect bounds, const char *string, int len, nk_flags align, enum nk_button_behavior behavior, const struct nk_style_button *style, const struct nk_input *in, const struct nk_user_font *font); -NK_LIB void nk_draw_button_symbol(struct nk_command_buffer *out, const struct nk_rect *bounds, const struct nk_rect *content, nk_flags state, const struct nk_style_button *style, enum nk_symbol_type type, const struct nk_user_font *font); -NK_LIB nk_bool nk_do_button_symbol(nk_flags *state, struct nk_command_buffer *out, struct nk_rect bounds, enum nk_symbol_type symbol, enum nk_button_behavior behavior, const struct nk_style_button *style, const struct nk_input *in, const struct nk_user_font *font); -NK_LIB void nk_draw_button_image(struct nk_command_buffer *out, const struct nk_rect *bounds, const struct nk_rect *content, nk_flags state, const struct nk_style_button *style, const struct nk_image *img); -NK_LIB nk_bool nk_do_button_image(nk_flags *state, struct nk_command_buffer *out, struct nk_rect bounds, struct nk_image img, enum nk_button_behavior b, const struct nk_style_button *style, const struct nk_input *in); -NK_LIB void nk_draw_button_text_symbol(struct nk_command_buffer *out, const struct nk_rect *bounds, const struct nk_rect *label, const struct nk_rect *symbol, nk_flags state, const struct nk_style_button *style, const char *str, int len, enum nk_symbol_type type, const struct nk_user_font *font); -NK_LIB nk_bool nk_do_button_text_symbol(nk_flags *state, struct nk_command_buffer *out, struct nk_rect bounds, enum nk_symbol_type symbol, const char *str, int len, nk_flags align, enum nk_button_behavior behavior, const struct nk_style_button *style, const struct nk_user_font *font, const struct nk_input *in); -NK_LIB void nk_draw_button_text_image(struct nk_command_buffer *out, const struct nk_rect *bounds, const struct nk_rect *label, const struct nk_rect *image, nk_flags state, const struct nk_style_button *style, const char *str, int len, const struct nk_user_font *font, const struct nk_image *img); -NK_LIB nk_bool nk_do_button_text_image(nk_flags *state, struct nk_command_buffer *out, struct nk_rect bounds, struct nk_image img, const char* str, int len, nk_flags align, enum nk_button_behavior behavior, const struct nk_style_button *style, const struct nk_user_font *font, const struct nk_input *in); - -/* toggle */ -enum nk_toggle_type { - NK_TOGGLE_CHECK, - NK_TOGGLE_OPTION -}; -NK_LIB nk_bool nk_toggle_behavior(const struct nk_input *in, struct nk_rect select, nk_flags *state, nk_bool active); -NK_LIB void nk_draw_checkbox(struct nk_command_buffer *out, nk_flags state, const struct nk_style_toggle *style, nk_bool active, const struct nk_rect *label, const struct nk_rect *selector, const struct nk_rect *cursors, const char *string, int len, const struct nk_user_font *font); -NK_LIB void nk_draw_option(struct nk_command_buffer *out, nk_flags state, const struct nk_style_toggle *style, nk_bool active, const struct nk_rect *label, const struct nk_rect *selector, const struct nk_rect *cursors, const char *string, int len, const struct nk_user_font *font); -NK_LIB nk_bool nk_do_toggle(nk_flags *state, struct nk_command_buffer *out, struct nk_rect r, nk_bool *active, const char *str, int len, enum nk_toggle_type type, const struct nk_style_toggle *style, const struct nk_input *in, const struct nk_user_font *font); - -/* progress */ -NK_LIB nk_size nk_progress_behavior(nk_flags *state, struct nk_input *in, struct nk_rect r, struct nk_rect cursor, nk_size max, nk_size value, nk_bool modifiable); -NK_LIB void nk_draw_progress(struct nk_command_buffer *out, nk_flags state, const struct nk_style_progress *style, const struct nk_rect *bounds, const struct nk_rect *scursor, nk_size value, nk_size max); -NK_LIB nk_size nk_do_progress(nk_flags *state, struct nk_command_buffer *out, struct nk_rect bounds, nk_size value, nk_size max, nk_bool modifiable, const struct nk_style_progress *style, struct nk_input *in); - -/* slider */ -NK_LIB float nk_slider_behavior(nk_flags *state, struct nk_rect *logical_cursor, struct nk_rect *visual_cursor, struct nk_input *in, struct nk_rect bounds, float slider_min, float slider_max, float slider_value, float slider_step, float slider_steps); -NK_LIB void nk_draw_slider(struct nk_command_buffer *out, nk_flags state, const struct nk_style_slider *style, const struct nk_rect *bounds, const struct nk_rect *visual_cursor, float min, float value, float max); -NK_LIB float nk_do_slider(nk_flags *state, struct nk_command_buffer *out, struct nk_rect bounds, float min, float val, float max, float step, const struct nk_style_slider *style, struct nk_input *in, const struct nk_user_font *font); - -/* scrollbar */ -NK_LIB float nk_scrollbar_behavior(nk_flags *state, struct nk_input *in, int has_scrolling, const struct nk_rect *scroll, const struct nk_rect *cursor, const struct nk_rect *empty0, const struct nk_rect *empty1, float scroll_offset, float target, float scroll_step, enum nk_orientation o); -NK_LIB void nk_draw_scrollbar(struct nk_command_buffer *out, nk_flags state, const struct nk_style_scrollbar *style, const struct nk_rect *bounds, const struct nk_rect *scroll); -NK_LIB float nk_do_scrollbarv(nk_flags *state, struct nk_command_buffer *out, struct nk_rect scroll, int has_scrolling, float offset, float target, float step, float button_pixel_inc, const struct nk_style_scrollbar *style, struct nk_input *in, const struct nk_user_font *font); -NK_LIB float nk_do_scrollbarh(nk_flags *state, struct nk_command_buffer *out, struct nk_rect scroll, int has_scrolling, float offset, float target, float step, float button_pixel_inc, const struct nk_style_scrollbar *style, struct nk_input *in, const struct nk_user_font *font); - -/* selectable */ -NK_LIB void nk_draw_selectable(struct nk_command_buffer *out, nk_flags state, const struct nk_style_selectable *style, nk_bool active, const struct nk_rect *bounds, const struct nk_rect *icon, const struct nk_image *img, enum nk_symbol_type sym, const char *string, int len, nk_flags align, const struct nk_user_font *font); -NK_LIB nk_bool nk_do_selectable(nk_flags *state, struct nk_command_buffer *out, struct nk_rect bounds, const char *str, int len, nk_flags align, nk_bool *value, const struct nk_style_selectable *style, const struct nk_input *in, const struct nk_user_font *font); -NK_LIB nk_bool nk_do_selectable_image(nk_flags *state, struct nk_command_buffer *out, struct nk_rect bounds, const char *str, int len, nk_flags align, nk_bool *value, const struct nk_image *img, const struct nk_style_selectable *style, const struct nk_input *in, const struct nk_user_font *font); - -/* edit */ -NK_LIB void nk_edit_draw_text(struct nk_command_buffer *out, const struct nk_style_edit *style, float pos_x, float pos_y, float x_offset, const char *text, int byte_len, float row_height, const struct nk_user_font *font, struct nk_color background, struct nk_color foreground, nk_bool is_selected); -NK_LIB nk_flags nk_do_edit(nk_flags *state, struct nk_command_buffer *out, struct nk_rect bounds, nk_flags flags, nk_plugin_filter filter, struct nk_text_edit *edit, const struct nk_style_edit *style, struct nk_input *in, const struct nk_user_font *font); - -/* color-picker */ -NK_LIB nk_bool nk_color_picker_behavior(nk_flags *state, const struct nk_rect *bounds, const struct nk_rect *matrix, const struct nk_rect *hue_bar, const struct nk_rect *alpha_bar, struct nk_colorf *color, const struct nk_input *in); -NK_LIB void nk_draw_color_picker(struct nk_command_buffer *o, const struct nk_rect *matrix, const struct nk_rect *hue_bar, const struct nk_rect *alpha_bar, struct nk_colorf col); -NK_LIB nk_bool nk_do_color_picker(nk_flags *state, struct nk_command_buffer *out, struct nk_colorf *col, enum nk_color_format fmt, struct nk_rect bounds, struct nk_vec2 padding, const struct nk_input *in, const struct nk_user_font *font); - -/* property */ -enum nk_property_status { - NK_PROPERTY_DEFAULT, - NK_PROPERTY_EDIT, - NK_PROPERTY_DRAG -}; -enum nk_property_filter { - NK_FILTER_INT, - NK_FILTER_FLOAT -}; -enum nk_property_kind { - NK_PROPERTY_INT, - NK_PROPERTY_FLOAT, - NK_PROPERTY_DOUBLE -}; -union nk_property { - int i; - float f; - double d; -}; -struct nk_property_variant { - enum nk_property_kind kind; - union nk_property value; - union nk_property min_value; - union nk_property max_value; - union nk_property step; -}; -NK_LIB struct nk_property_variant nk_property_variant_int(int value, int min_value, int max_value, int step); -NK_LIB struct nk_property_variant nk_property_variant_float(float value, float min_value, float max_value, float step); -NK_LIB struct nk_property_variant nk_property_variant_double(double value, double min_value, double max_value, double step); - -NK_LIB void nk_drag_behavior(nk_flags *state, const struct nk_input *in, struct nk_rect drag, struct nk_property_variant *variant, float inc_per_pixel); -NK_LIB void nk_property_behavior(nk_flags *ws, const struct nk_input *in, struct nk_rect property, struct nk_rect label, struct nk_rect edit, struct nk_rect empty, int *state, struct nk_property_variant *variant, float inc_per_pixel); -NK_LIB void nk_draw_property(struct nk_command_buffer *out, const struct nk_style_property *style, const struct nk_rect *bounds, const struct nk_rect *label, nk_flags state, const char *name, int len, const struct nk_user_font *font); -NK_LIB void nk_do_property(nk_flags *ws, struct nk_command_buffer *out, struct nk_rect property, const char *name, struct nk_property_variant *variant, float inc_per_pixel, char *buffer, int *len, int *state, int *cursor, int *select_begin, int *select_end, const struct nk_style_property *style, enum nk_property_filter filter, struct nk_input *in, const struct nk_user_font *font, struct nk_text_edit *text_edit, enum nk_button_behavior behavior); -NK_LIB void nk_property(struct nk_context *ctx, const char *name, struct nk_property_variant *variant, float inc_per_pixel, const enum nk_property_filter filter); - -#ifdef NK_INCLUDE_FONT_BAKING - -#define STB_RECT_PACK_IMPLEMENTATION -#define STB_TRUETYPE_IMPLEMENTATION - -/* Allow consumer to define own STBTT_malloc/STBTT_free, and use the font atlas' allocator otherwise */ -#ifndef STBTT_malloc -static void* -nk_stbtt_malloc(nk_size size, void *user_data) { - struct nk_allocator *alloc = (struct nk_allocator *) user_data; - return alloc->alloc(alloc->userdata, 0, size); -} - -static void -nk_stbtt_free(void *ptr, void *user_data) { - struct nk_allocator *alloc = (struct nk_allocator *) user_data; - alloc->free(alloc->userdata, ptr); -} - -#define STBTT_malloc(x,u) nk_stbtt_malloc(x,u) -#define STBTT_free(x,u) nk_stbtt_free(x,u) - -#endif /* STBTT_malloc */ - -#endif /* NK_INCLUDE_FONT_BAKING */ - -#endif - - - - - -/* =============================================================== - * - * MATH - * - * ===============================================================*/ -/*/// ### Math -/// Since nuklear is supposed to work on all systems providing floating point -/// math without any dependencies I also had to implement my own math functions -/// for sqrt, sin and cos. Since the actual highly accurate implementations for -/// the standard library functions are quite complex and I do not need high -/// precision for my use cases I use approximations. -/// -/// Sqrt -/// ---- -/// For square root nuklear uses the famous fast inverse square root: -/// https://en.wikipedia.org/wiki/Fast_inverse_square_root with -/// slightly tweaked magic constant. While on today's hardware it is -/// probably not faster it is still fast and accurate enough for -/// nuklear's use cases. IMPORTANT: this requires float format IEEE 754 -/// -/// Sine/Cosine -/// ----------- -/// All constants inside both function are generated Remez's minimax -/// approximations for value range 0...2*PI. The reason why I decided to -/// approximate exactly that range is that nuklear only needs sine and -/// cosine to generate circles which only requires that exact range. -/// In addition I used Remez instead of Taylor for additional precision: -/// www.lolengine.net/blog/2011/12/21/better-function-approximations. -/// -/// The tool I used to generate constants for both sine and cosine -/// (it can actually approximate a lot more functions) can be -/// found here: www.lolengine.net/wiki/oss/lolremez -*/ -#ifndef NK_INV_SQRT -#define NK_INV_SQRT nk_inv_sqrt -NK_LIB float -nk_inv_sqrt(float n) -{ - float x2; - const float threehalfs = 1.5f; - union {nk_uint i; float f;} conv = {0}; - conv.f = n; - x2 = n * 0.5f; - conv.i = 0x5f375A84 - (conv.i >> 1); - conv.f = conv.f * (threehalfs - (x2 * conv.f * conv.f)); - return conv.f; -} -#endif -#ifndef NK_SIN -#define NK_SIN nk_sin -NK_LIB float -nk_sin(float x) -{ - NK_STORAGE const float a0 = +1.91059300966915117e-31f; - NK_STORAGE const float a1 = +1.00086760103908896f; - NK_STORAGE const float a2 = -1.21276126894734565e-2f; - NK_STORAGE const float a3 = -1.38078780785773762e-1f; - NK_STORAGE const float a4 = -2.67353392911981221e-2f; - NK_STORAGE const float a5 = +2.08026600266304389e-2f; - NK_STORAGE const float a6 = -3.03996055049204407e-3f; - NK_STORAGE const float a7 = +1.38235642404333740e-4f; - return a0 + x*(a1 + x*(a2 + x*(a3 + x*(a4 + x*(a5 + x*(a6 + x*a7)))))); -} -#endif -#ifndef NK_COS -#define NK_COS nk_cos -NK_LIB float -nk_cos(float x) -{ - /* New implementation. Also generated using lolremez. */ - /* Old version significantly deviated from expected results. */ - NK_STORAGE const float a0 = 9.9995999154986614e-1f; - NK_STORAGE const float a1 = 1.2548995793001028e-3f; - NK_STORAGE const float a2 = -5.0648546280678015e-1f; - NK_STORAGE const float a3 = 1.2942246466519995e-2f; - NK_STORAGE const float a4 = 2.8668384702547972e-2f; - NK_STORAGE const float a5 = 7.3726485210586547e-3f; - NK_STORAGE const float a6 = -3.8510875386947414e-3f; - NK_STORAGE const float a7 = 4.7196604604366623e-4f; - NK_STORAGE const float a8 = -1.8776444013090451e-5f; - return a0 + x*(a1 + x*(a2 + x*(a3 + x*(a4 + x*(a5 + x*(a6 + x*(a7 + x*a8))))))); -} -#endif -NK_LIB nk_uint -nk_round_up_pow2(nk_uint v) -{ - v--; - v |= v >> 1; - v |= v >> 2; - v |= v >> 4; - v |= v >> 8; - v |= v >> 16; - v++; - return v; -} -NK_LIB double -nk_pow(double x, int n) -{ - /* check the sign of n */ - double r = 1; - int plus = n >= 0; - n = (plus) ? n : -n; - while (n > 0) { - if ((n & 1) == 1) - r *= x; - n /= 2; - x *= x; - } - return plus ? r : 1.0 / r; -} -NK_LIB int -nk_ifloord(double x) -{ - x = (double)((int)x - ((x < 0.0) ? 1 : 0)); - return (int)x; -} -NK_LIB int -nk_ifloorf(float x) -{ - x = (float)((int)x - ((x < 0.0f) ? 1 : 0)); - return (int)x; -} -NK_LIB int -nk_iceilf(float x) -{ - if (x >= 0) { - int i = (int)x; - return (x > i) ? i+1: i; - } else { - int t = (int)x; - float r = x - (float)t; - return (r > 0.0f) ? t+1: t; - } -} -NK_LIB int -nk_log10(double n) -{ - int neg; - int ret; - int exp = 0; - - neg = (n < 0) ? 1 : 0; - ret = (neg) ? (int)-n : (int)n; - while ((ret / 10) > 0) { - ret /= 10; - exp++; - } - if (neg) exp = -exp; - return exp; -} -NK_API struct nk_rect -nk_get_null_rect(void) -{ - return nk_null_rect; -} -NK_API struct nk_rect -nk_rect(float x, float y, float w, float h) -{ - struct nk_rect r; - r.x = x; r.y = y; - r.w = w; r.h = h; - return r; -} -NK_API struct nk_rect -nk_recti(int x, int y, int w, int h) -{ - struct nk_rect r; - r.x = (float)x; - r.y = (float)y; - r.w = (float)w; - r.h = (float)h; - return r; -} -NK_API struct nk_rect -nk_recta(struct nk_vec2 pos, struct nk_vec2 size) -{ - return nk_rect(pos.x, pos.y, size.x, size.y); -} -NK_API struct nk_rect -nk_rectv(const float *r) -{ - return nk_rect(r[0], r[1], r[2], r[3]); -} -NK_API struct nk_rect -nk_rectiv(const int *r) -{ - return nk_recti(r[0], r[1], r[2], r[3]); -} -NK_API struct nk_vec2 -nk_rect_pos(struct nk_rect r) -{ - struct nk_vec2 ret; - ret.x = r.x; ret.y = r.y; - return ret; -} -NK_API struct nk_vec2 -nk_rect_size(struct nk_rect r) -{ - struct nk_vec2 ret; - ret.x = r.w; ret.y = r.h; - return ret; -} -NK_LIB struct nk_rect -nk_shrink_rect(struct nk_rect r, float amount) -{ - struct nk_rect res; - r.w = NK_MAX(r.w, 2 * amount); - r.h = NK_MAX(r.h, 2 * amount); - res.x = r.x + amount; - res.y = r.y + amount; - res.w = r.w - 2 * amount; - res.h = r.h - 2 * amount; - return res; -} -NK_LIB struct nk_rect -nk_pad_rect(struct nk_rect r, struct nk_vec2 pad) -{ - r.w = NK_MAX(r.w, 2 * pad.x); - r.h = NK_MAX(r.h, 2 * pad.y); - r.x += pad.x; r.y += pad.y; - r.w -= 2 * pad.x; - r.h -= 2 * pad.y; - return r; -} -NK_API struct nk_vec2 -nk_vec2(float x, float y) -{ - struct nk_vec2 ret; - ret.x = x; ret.y = y; - return ret; -} -NK_API struct nk_vec2 -nk_vec2i(int x, int y) -{ - struct nk_vec2 ret; - ret.x = (float)x; - ret.y = (float)y; - return ret; -} -NK_API struct nk_vec2 -nk_vec2v(const float *v) -{ - return nk_vec2(v[0], v[1]); -} -NK_API struct nk_vec2 -nk_vec2iv(const int *v) -{ - return nk_vec2i(v[0], v[1]); -} -NK_LIB void -nk_unify(struct nk_rect *clip, const struct nk_rect *a, float x0, float y0, - float x1, float y1) -{ - NK_ASSERT(a); - NK_ASSERT(clip); - clip->x = NK_MAX(a->x, x0); - clip->y = NK_MAX(a->y, y0); - clip->w = NK_MIN(a->x + a->w, x1) - clip->x; - clip->h = NK_MIN(a->y + a->h, y1) - clip->y; - clip->w = NK_MAX(0, clip->w); - clip->h = NK_MAX(0, clip->h); -} - -NK_API void -nk_triangle_from_direction(struct nk_vec2 *result, struct nk_rect r, - float pad_x, float pad_y, enum nk_heading direction) -{ - float w_half, h_half; - NK_ASSERT(result); - - r.w = NK_MAX(2 * pad_x, r.w); - r.h = NK_MAX(2 * pad_y, r.h); - r.w = r.w - 2 * pad_x; - r.h = r.h - 2 * pad_y; - - r.x = r.x + pad_x; - r.y = r.y + pad_y; - - w_half = r.w / 2.0f; - h_half = r.h / 2.0f; - - if (direction == NK_UP) { - result[0] = nk_vec2(r.x + w_half, r.y); - result[1] = nk_vec2(r.x + r.w, r.y + r.h); - result[2] = nk_vec2(r.x, r.y + r.h); - } else if (direction == NK_RIGHT) { - result[0] = nk_vec2(r.x, r.y); - result[1] = nk_vec2(r.x + r.w, r.y + h_half); - result[2] = nk_vec2(r.x, r.y + r.h); - } else if (direction == NK_DOWN) { - result[0] = nk_vec2(r.x, r.y); - result[1] = nk_vec2(r.x + r.w, r.y); - result[2] = nk_vec2(r.x + w_half, r.y + r.h); - } else { - result[0] = nk_vec2(r.x, r.y + h_half); - result[1] = nk_vec2(r.x + r.w, r.y); - result[2] = nk_vec2(r.x + r.w, r.y + r.h); - } -} - - - - - -/* =============================================================== - * - * UTIL - * - * ===============================================================*/ -NK_INTERN int nk_str_match_here(const char *regexp, const char *text); -NK_INTERN int nk_str_match_star(int c, const char *regexp, const char *text); -NK_LIB nk_bool nk_is_lower(int c) {return (c >= 'a' && c <= 'z') || (c >= 0xE0 && c <= 0xFF);} -NK_LIB nk_bool nk_is_upper(int c){return (c >= 'A' && c <= 'Z') || (c >= 0xC0 && c <= 0xDF);} -NK_LIB int nk_to_upper(int c) {return (c >= 'a' && c <= 'z') ? (c - ('a' - 'A')) : c;} -NK_LIB int nk_to_lower(int c) {return (c >= 'A' && c <= 'Z') ? (c - ('a' + 'A')) : c;} - -#ifndef NK_MEMCPY -#define NK_MEMCPY nk_memcopy -NK_LIB void* -nk_memcopy(void *dst0, const void *src0, nk_size length) -{ - nk_ptr t; - char *dst = (char*)dst0; - const char *src = (const char*)src0; - if (length == 0 || dst == src) - goto done; - - #define nk_word int - #define nk_wsize sizeof(nk_word) - #define nk_wmask (nk_wsize-1) - #define NK_TLOOP(s) if (t) NK_TLOOP1(s) - #define NK_TLOOP1(s) do { s; } while (--t) - - if (dst < src) { - t = (nk_ptr)src; /* only need low bits */ - if ((t | (nk_ptr)dst) & nk_wmask) { - if ((t ^ (nk_ptr)dst) & nk_wmask || length < nk_wsize) - t = length; - else - t = nk_wsize - (t & nk_wmask); - length -= t; - NK_TLOOP1(*dst++ = *src++); - } - t = length / nk_wsize; - NK_TLOOP(*(nk_word*)(void*)dst = *(const nk_word*)(const void*)src; - src += nk_wsize; dst += nk_wsize); - t = length & nk_wmask; - NK_TLOOP(*dst++ = *src++); - } else { - src += length; - dst += length; - t = (nk_ptr)src; - if ((t | (nk_ptr)dst) & nk_wmask) { - if ((t ^ (nk_ptr)dst) & nk_wmask || length <= nk_wsize) - t = length; - else - t &= nk_wmask; - length -= t; - NK_TLOOP1(*--dst = *--src); - } - t = length / nk_wsize; - NK_TLOOP(src -= nk_wsize; dst -= nk_wsize; - *(nk_word*)(void*)dst = *(const nk_word*)(const void*)src); - t = length & nk_wmask; - NK_TLOOP(*--dst = *--src); - } - #undef nk_word - #undef nk_wsize - #undef nk_wmask - #undef NK_TLOOP - #undef NK_TLOOP1 -done: - return (dst0); -} -#endif -#ifndef NK_MEMSET -#define NK_MEMSET nk_memset -NK_LIB void -nk_memset(void *ptr, int c0, nk_size size) -{ - #define nk_word unsigned - #define nk_wsize sizeof(nk_word) - #define nk_wmask (nk_wsize - 1) - nk_byte *dst = (nk_byte*)ptr; - unsigned c = 0; - nk_size t = 0; - - if ((c = (nk_byte)c0) != 0) { - c = (c << 8) | c; /* at least 16-bits */ - if (sizeof(unsigned int) > 2) - c = (c << 16) | c; /* at least 32-bits*/ - } - - /* too small of a word count */ - dst = (nk_byte*)ptr; - if (size < 3 * nk_wsize) { - while (size--) *dst++ = (nk_byte)c0; - return; - } - - /* align destination */ - if ((t = NK_PTR_TO_UINT(dst) & nk_wmask) != 0) { - t = nk_wsize -t; - size -= t; - do { - *dst++ = (nk_byte)c0; - } while (--t != 0); - } - - /* fill word */ - t = size / nk_wsize; - do { - *(nk_word*)((void*)dst) = c; - dst += nk_wsize; - } while (--t != 0); - - /* fill trailing bytes */ - t = (size & nk_wmask); - if (t != 0) { - do { - *dst++ = (nk_byte)c0; - } while (--t != 0); - } - - #undef nk_word - #undef nk_wsize - #undef nk_wmask -} -#endif -NK_LIB void -nk_zero(void *ptr, nk_size size) -{ - NK_ASSERT(ptr); - NK_MEMSET(ptr, 0, size); -} -NK_API int -nk_strlen(const char *str) -{ - int siz = 0; - NK_ASSERT(str); - while (str && *str++ != '\0') siz++; - return siz; -} -NK_API int -nk_strtoi(const char *str, const char **endptr) -{ - int neg = 1; - const char *p = str; - int value = 0; - - NK_ASSERT(str); - if (!str) return 0; - - /* skip whitespace */ - while (*p == ' ') p++; - if (*p == '-') { - neg = -1; - p++; - } - while (*p && *p >= '0' && *p <= '9') { - value = value * 10 + (int) (*p - '0'); - p++; - } - if (endptr) - *endptr = p; - return neg*value; -} -NK_API double -nk_strtod(const char *str, const char **endptr) -{ - double m; - double neg = 1.0; - const char *p = str; - double value = 0; - double number = 0; - - NK_ASSERT(str); - if (!str) return 0; - - /* skip whitespace */ - while (*p == ' ') p++; - if (*p == '-') { - neg = -1.0; - p++; - } - - while (*p && *p != '.' && *p != 'e') { - value = value * 10.0 + (double) (*p - '0'); - p++; - } - - if (*p == '.') { - p++; - for(m = 0.1; *p && *p != 'e'; p++ ) { - value = value + (double) (*p - '0') * m; - m *= 0.1; - } - } - if (*p == 'e') { - int i, pow, div; - p++; - if (*p == '-') { - div = nk_true; - p++; - } else if (*p == '+') { - div = nk_false; - p++; - } else div = nk_false; - - for (pow = 0; *p; p++) - pow = pow * 10 + (int) (*p - '0'); - - for (m = 1.0, i = 0; i < pow; i++) - m *= 10.0; - - if (div) - value /= m; - else value *= m; - } - number = value * neg; - if (endptr) - *endptr = p; - return number; -} -NK_API float -nk_strtof(const char *str, const char **endptr) -{ - float float_value; - double double_value; - double_value = NK_STRTOD(str, endptr); - float_value = (float)double_value; - return float_value; -} -NK_API int -nk_stricmp(const char *s1, const char *s2) -{ - nk_int c1,c2,d; - do { - c1 = *s1++; - c2 = *s2++; - d = c1 - c2; - while (d) { - if (c1 <= 'Z' && c1 >= 'A') { - d += ('a' - 'A'); - if (!d) break; - } - if (c2 <= 'Z' && c2 >= 'A') { - d -= ('a' - 'A'); - if (!d) break; - } - return ((d >= 0) << 1) - 1; - } - } while (c1); - return 0; -} -NK_API int -nk_stricmpn(const char *s1, const char *s2, int n) -{ - int c1,c2,d; - NK_ASSERT(n >= 0); - do { - c1 = *s1++; - c2 = *s2++; - if (!n--) return 0; - - d = c1 - c2; - while (d) { - if (c1 <= 'Z' && c1 >= 'A') { - d += ('a' - 'A'); - if (!d) break; - } - if (c2 <= 'Z' && c2 >= 'A') { - d -= ('a' - 'A'); - if (!d) break; - } - return ((d >= 0) << 1) - 1; - } - } while (c1); - return 0; -} -NK_INTERN int -nk_str_match_here(const char *regexp, const char *text) -{ - if (regexp[0] == '\0') - return 1; - if (regexp[1] == '*') - return nk_str_match_star(regexp[0], regexp+2, text); - if (regexp[0] == '$' && regexp[1] == '\0') - return *text == '\0'; - if (*text!='\0' && (regexp[0]=='.' || regexp[0]==*text)) - return nk_str_match_here(regexp+1, text+1); - return 0; -} -NK_INTERN int -nk_str_match_star(int c, const char *regexp, const char *text) -{ - do {/* a '* matches zero or more instances */ - if (nk_str_match_here(regexp, text)) - return 1; - } while (*text != '\0' && (*text++ == c || c == '.')); - return 0; -} -NK_API int -nk_strfilter(const char *text, const char *regexp) -{ - /* - c matches any literal character c - . matches any single character - ^ matches the beginning of the input string - $ matches the end of the input string - * matches zero or more occurrences of the previous character*/ - if (regexp[0] == '^') - return nk_str_match_here(regexp+1, text); - do { /* must look even if string is empty */ - if (nk_str_match_here(regexp, text)) - return 1; - } while (*text++ != '\0'); - return 0; -} -NK_API int -nk_strmatch_fuzzy_text(const char *str, int str_len, - const char *pattern, int *out_score) -{ - /* Returns true if each character in pattern is found sequentially within str - * if found then out_score is also set. Score value has no intrinsic meaning. - * Range varies with pattern. Can only compare scores with same search pattern. */ - - /* bonus for adjacent matches */ - #define NK_ADJACENCY_BONUS 5 - /* bonus if match occurs after a separator */ - #define NK_SEPARATOR_BONUS 10 - /* bonus if match is uppercase and prev is lower */ - #define NK_CAMEL_BONUS 10 - /* penalty applied for every letter in str before the first match */ - #define NK_LEADING_LETTER_PENALTY (-3) - /* maximum penalty for leading letters */ - #define NK_MAX_LEADING_LETTER_PENALTY (-9) - /* penalty for every letter that doesn't matter */ - #define NK_UNMATCHED_LETTER_PENALTY (-1) - - /* loop variables */ - int score = 0; - char const * pattern_iter = pattern; - int str_iter = 0; - int prev_matched = nk_false; - int prev_lower = nk_false; - /* true so if first letter match gets separator bonus*/ - int prev_separator = nk_true; - - /* use "best" matched letter if multiple string letters match the pattern */ - char const * best_letter = 0; - int best_letter_score = 0; - - /* loop over strings */ - NK_ASSERT(str); - NK_ASSERT(pattern); - if (!str || !str_len || !pattern) return 0; - while (str_iter < str_len) - { - const char pattern_letter = *pattern_iter; - const char str_letter = str[str_iter]; - - int next_match = *pattern_iter != '\0' && - nk_to_lower(pattern_letter) == nk_to_lower(str_letter); - int rematch = best_letter && nk_to_upper(*best_letter) == nk_to_upper(str_letter); - - int advanced = next_match && best_letter; - int pattern_repeat = best_letter && *pattern_iter != '\0'; - pattern_repeat = pattern_repeat && - nk_to_lower(*best_letter) == nk_to_lower(pattern_letter); - - if (advanced || pattern_repeat) { - score += best_letter_score; - best_letter = 0; - best_letter_score = 0; - } - - if (next_match || rematch) - { - int new_score = 0; - /* Apply penalty for each letter before the first pattern match */ - if (pattern_iter == pattern) { - int count = (int)(&str[str_iter] - str); - int penalty = NK_LEADING_LETTER_PENALTY * count; - if (penalty < NK_MAX_LEADING_LETTER_PENALTY) - penalty = NK_MAX_LEADING_LETTER_PENALTY; - - score += penalty; - } - - /* apply bonus for consecutive bonuses */ - if (prev_matched) - new_score += NK_ADJACENCY_BONUS; - - /* apply bonus for matches after a separator */ - if (prev_separator) - new_score += NK_SEPARATOR_BONUS; - - /* apply bonus across camel case boundaries */ - if (prev_lower && nk_is_upper(str_letter)) - new_score += NK_CAMEL_BONUS; - - /* update pattern iter IFF the next pattern letter was matched */ - if (next_match) - ++pattern_iter; - - /* update best letter in str which may be for a "next" letter or a rematch */ - if (new_score >= best_letter_score) { - /* apply penalty for now skipped letter */ - if (best_letter != 0) - score += NK_UNMATCHED_LETTER_PENALTY; - - best_letter = &str[str_iter]; - best_letter_score = new_score; - } - prev_matched = nk_true; - } else { - score += NK_UNMATCHED_LETTER_PENALTY; - prev_matched = nk_false; - } - - /* separators should be more easily defined */ - prev_lower = nk_is_lower(str_letter) != 0; - prev_separator = str_letter == '_' || str_letter == ' '; - - ++str_iter; - } - - /* apply score for last match */ - if (best_letter) - score += best_letter_score; - - /* did not match full pattern */ - if (*pattern_iter != '\0') - return nk_false; - - if (out_score) - *out_score = score; - return nk_true; -} -NK_API int -nk_strmatch_fuzzy_string(char const *str, char const *pattern, int *out_score) -{ - return nk_strmatch_fuzzy_text(str, nk_strlen(str), pattern, out_score); -} -NK_LIB int -nk_string_float_limit(char *string, int prec) -{ - int dot = 0; - char *c = string; - while (*c) { - if (*c == '.') { - dot = 1; - c++; - continue; - } - if (dot == (prec+1)) { - *c = 0; - break; - } - if (dot > 0) dot++; - c++; - } - return (int)(c - string); -} -NK_INTERN void -nk_strrev_ascii(char *s) -{ - int len = nk_strlen(s); - int end = len / 2; - int i = 0; - char t; - for (; i < end; ++i) { - t = s[i]; - s[i] = s[len - 1 - i]; - s[len -1 - i] = t; - } -} -NK_LIB char* -nk_itoa(char *s, long n) -{ - long i = 0; - if (n == 0) { - s[i++] = '0'; - s[i] = 0; - return s; - } - if (n < 0) { - s[i++] = '-'; - n = -n; - } - while (n > 0) { - s[i++] = (char)('0' + (n % 10)); - n /= 10; - } - s[i] = 0; - if (s[0] == '-') - ++s; - - nk_strrev_ascii(s); - return s; -} -#ifndef NK_DTOA -#define NK_DTOA nk_dtoa -NK_LIB char* -nk_dtoa(char *s, double n) -{ - int useExp = 0; - int digit = 0, m = 0, m1 = 0; - char *c = s; - int neg = 0; - - NK_ASSERT(s); - if (!s) return 0; - - if (n == 0.0) { - s[0] = '0'; s[1] = '\0'; - return s; - } - - neg = (n < 0); - if (neg) n = -n; - - /* calculate magnitude */ - m = nk_log10(n); - useExp = (m >= 14 || (neg && m >= 9) || m <= -9); - if (neg) *(c++) = '-'; - - /* set up for scientific notation */ - if (useExp) { - if (m < 0) - m -= 1; - n = n / (double)nk_pow(10.0, m); - m1 = m; - m = 0; - } - if (m < 1.0) { - m = 0; - } - - /* convert the number */ - while (n > NK_FLOAT_PRECISION || m >= 0) { - double weight = nk_pow(10.0, m); - if (weight > 0) { - double t = (double)n / weight; - digit = nk_ifloord(t); - n -= ((double)digit * weight); - *(c++) = (char)('0' + (char)digit); - } - if (m == 0 && n > 0) - *(c++) = '.'; - m--; - } - - if (useExp) { - /* convert the exponent */ - int i, j; - *(c++) = 'e'; - if (m1 > 0) { - *(c++) = '+'; - } else { - *(c++) = '-'; - m1 = -m1; - } - m = 0; - while (m1 > 0) { - *(c++) = (char)('0' + (char)(m1 % 10)); - m1 /= 10; - m++; - } - c -= m; - for (i = 0, j = m-1; i= buf_size) break; - iter++; - - /* flag arguments */ - while (*iter) { - if (*iter == '-') flag |= NK_ARG_FLAG_LEFT; - else if (*iter == '+') flag |= NK_ARG_FLAG_PLUS; - else if (*iter == ' ') flag |= NK_ARG_FLAG_SPACE; - else if (*iter == '#') flag |= NK_ARG_FLAG_NUM; - else if (*iter == '0') flag |= NK_ARG_FLAG_ZERO; - else break; - iter++; - } - - /* width argument */ - width = NK_DEFAULT; - if (*iter >= '1' && *iter <= '9') { - const char *end; - width = nk_strtoi(iter, &end); - if (end == iter) - width = -1; - else iter = end; - } else if (*iter == '*') { - width = va_arg(args, int); - iter++; - } - - /* precision argument */ - precision = NK_DEFAULT; - if (*iter == '.') { - iter++; - if (*iter == '*') { - precision = va_arg(args, int); - iter++; - } else { - const char *end; - precision = nk_strtoi(iter, &end); - if (end == iter) - precision = -1; - else iter = end; - } - } - - /* length modifier */ - if (*iter == 'h') { - if (*(iter+1) == 'h') { - arg_type = NK_ARG_TYPE_CHAR; - iter++; - } else arg_type = NK_ARG_TYPE_SHORT; - iter++; - } else if (*iter == 'l') { - arg_type = NK_ARG_TYPE_LONG; - iter++; - } else arg_type = NK_ARG_TYPE_DEFAULT; - - /* specifier */ - if (*iter == '%') { - NK_ASSERT(arg_type == NK_ARG_TYPE_DEFAULT); - NK_ASSERT(precision == NK_DEFAULT); - NK_ASSERT(width == NK_DEFAULT); - if (len < buf_size) - buf[len++] = '%'; - } else if (*iter == 's') { - /* string */ - const char *str = va_arg(args, const char*); - NK_ASSERT(str != buf && "buffer and argument are not allowed to overlap!"); - NK_ASSERT(arg_type == NK_ARG_TYPE_DEFAULT); - NK_ASSERT(precision == NK_DEFAULT); - NK_ASSERT(width == NK_DEFAULT); - if (str == buf) return -1; - while (str && *str && len < buf_size) - buf[len++] = *str++; - } else if (*iter == 'n') { - /* current length callback */ - signed int *n = va_arg(args, int*); - NK_ASSERT(arg_type == NK_ARG_TYPE_DEFAULT); - NK_ASSERT(precision == NK_DEFAULT); - NK_ASSERT(width == NK_DEFAULT); - if (n) *n = len; - } else if (*iter == 'c' || *iter == 'i' || *iter == 'd') { - /* signed integer */ - long value = 0; - const char *num_iter; - int num_len, num_print, padding; - int cur_precision = NK_MAX(precision, 1); - int cur_width = NK_MAX(width, 0); - - /* retrieve correct value type */ - if (arg_type == NK_ARG_TYPE_CHAR) - value = (signed char)va_arg(args, int); - else if (arg_type == NK_ARG_TYPE_SHORT) - value = (signed short)va_arg(args, int); - else if (arg_type == NK_ARG_TYPE_LONG) - value = va_arg(args, signed long); - else if (*iter == 'c') - value = (unsigned char)va_arg(args, int); - else value = va_arg(args, signed int); - - /* convert number to string */ - nk_itoa(number_buffer, value); - num_len = nk_strlen(number_buffer); - padding = NK_MAX(cur_width - NK_MAX(cur_precision, num_len), 0); - if ((flag & NK_ARG_FLAG_PLUS) || (flag & NK_ARG_FLAG_SPACE)) - padding = NK_MAX(padding-1, 0); - - /* fill left padding up to a total of `width` characters */ - if (!(flag & NK_ARG_FLAG_LEFT)) { - while (padding-- > 0 && (len < buf_size)) { - if ((flag & NK_ARG_FLAG_ZERO) && (precision == NK_DEFAULT)) - buf[len++] = '0'; - else buf[len++] = ' '; - } - } - - /* copy string value representation into buffer */ - if ((flag & NK_ARG_FLAG_PLUS) && value >= 0 && len < buf_size) - buf[len++] = '+'; - else if ((flag & NK_ARG_FLAG_SPACE) && value >= 0 && len < buf_size) - buf[len++] = ' '; - - /* fill up to precision number of digits with '0' */ - num_print = NK_MAX(cur_precision, num_len); - while (precision && (num_print > num_len) && (len < buf_size)) { - buf[len++] = '0'; - num_print--; - } - - /* copy string value representation into buffer */ - num_iter = number_buffer; - while (precision && *num_iter && len < buf_size) - buf[len++] = *num_iter++; - - /* fill right padding up to width characters */ - if (flag & NK_ARG_FLAG_LEFT) { - while ((padding-- > 0) && (len < buf_size)) - buf[len++] = ' '; - } - } else if (*iter == 'o' || *iter == 'x' || *iter == 'X' || *iter == 'u') { - /* unsigned integer */ - unsigned long value = 0; - int num_len = 0, num_print, padding = 0; - int cur_precision = NK_MAX(precision, 1); - int cur_width = NK_MAX(width, 0); - unsigned int base = (*iter == 'o') ? 8: (*iter == 'u')? 10: 16; - - /* print oct/hex/dec value */ - const char *upper_output_format = "0123456789ABCDEF"; - const char *lower_output_format = "0123456789abcdef"; - const char *output_format = (*iter == 'x') ? - lower_output_format: upper_output_format; - - /* retrieve correct value type */ - if (arg_type == NK_ARG_TYPE_CHAR) - value = (unsigned char)va_arg(args, int); - else if (arg_type == NK_ARG_TYPE_SHORT) - value = (unsigned short)va_arg(args, int); - else if (arg_type == NK_ARG_TYPE_LONG) - value = va_arg(args, unsigned long); - else value = va_arg(args, unsigned int); - - do { - /* convert decimal number into hex/oct number */ - int digit = output_format[value % base]; - if (num_len < NK_MAX_NUMBER_BUFFER) - number_buffer[num_len++] = (char)digit; - value /= base; - } while (value > 0); - - num_print = NK_MAX(cur_precision, num_len); - padding = NK_MAX(cur_width - NK_MAX(cur_precision, num_len), 0); - if (flag & NK_ARG_FLAG_NUM) - padding = NK_MAX(padding-1, 0); - - /* fill left padding up to a total of `width` characters */ - if (!(flag & NK_ARG_FLAG_LEFT)) { - while ((padding-- > 0) && (len < buf_size)) { - if ((flag & NK_ARG_FLAG_ZERO) && (precision == NK_DEFAULT)) - buf[len++] = '0'; - else buf[len++] = ' '; - } - } - - /* fill up to precision number of digits */ - if (num_print && (flag & NK_ARG_FLAG_NUM)) { - if ((*iter == 'o') && (len < buf_size)) { - buf[len++] = '0'; - } else if ((*iter == 'x') && ((len+1) < buf_size)) { - buf[len++] = '0'; - buf[len++] = 'x'; - } else if ((*iter == 'X') && ((len+1) < buf_size)) { - buf[len++] = '0'; - buf[len++] = 'X'; - } - } - while (precision && (num_print > num_len) && (len < buf_size)) { - buf[len++] = '0'; - num_print--; - } - - /* reverse number direction */ - while (num_len > 0) { - if (precision && (len < buf_size)) - buf[len++] = number_buffer[num_len-1]; - num_len--; - } - - /* fill right padding up to width characters */ - if (flag & NK_ARG_FLAG_LEFT) { - while ((padding-- > 0) && (len < buf_size)) - buf[len++] = ' '; - } - } else if (*iter == 'f') { - /* floating point */ - const char *num_iter; - int cur_precision = (precision < 0) ? 6: precision; - int prefix, cur_width = NK_MAX(width, 0); - double value = va_arg(args, double); - int num_len = 0, frac_len = 0, dot = 0; - int padding = 0; - - NK_ASSERT(arg_type == NK_ARG_TYPE_DEFAULT); - NK_DTOA(number_buffer, value); - num_len = nk_strlen(number_buffer); - - /* calculate padding */ - num_iter = number_buffer; - while (*num_iter && *num_iter != '.') - num_iter++; - - prefix = (*num_iter == '.')?(int)(num_iter - number_buffer)+1:0; - padding = NK_MAX(cur_width - (prefix + NK_MIN(cur_precision, num_len - prefix)) , 0); - if ((flag & NK_ARG_FLAG_PLUS) || (flag & NK_ARG_FLAG_SPACE)) - padding = NK_MAX(padding-1, 0); - - /* fill left padding up to a total of `width` characters */ - if (!(flag & NK_ARG_FLAG_LEFT)) { - while (padding-- > 0 && (len < buf_size)) { - if (flag & NK_ARG_FLAG_ZERO) - buf[len++] = '0'; - else buf[len++] = ' '; - } - } - - /* copy string value representation into buffer */ - num_iter = number_buffer; - if ((flag & NK_ARG_FLAG_PLUS) && (value >= 0) && (len < buf_size)) - buf[len++] = '+'; - else if ((flag & NK_ARG_FLAG_SPACE) && (value >= 0) && (len < buf_size)) - buf[len++] = ' '; - while (*num_iter) { - if (dot) frac_len++; - if (len < buf_size) - buf[len++] = *num_iter; - if (*num_iter == '.') dot = 1; - if (frac_len >= cur_precision) break; - num_iter++; - } - - /* fill number up to precision */ - while (frac_len < cur_precision) { - if (!dot && len < buf_size) { - buf[len++] = '.'; - dot = 1; - } - if (len < buf_size) - buf[len++] = '0'; - frac_len++; - } - - /* fill right padding up to width characters */ - if (flag & NK_ARG_FLAG_LEFT) { - while ((padding-- > 0) && (len < buf_size)) - buf[len++] = ' '; - } - } else { - /* Specifier not supported: g,G,e,E,p,z */ - NK_ASSERT(0 && "specifier is not supported!"); - return result; - } - } - buf[(len >= buf_size)?(buf_size-1):len] = 0; - result = (len >= buf_size)?-1:len; - return result; -} -#endif -NK_LIB int -nk_strfmt(char *buf, int buf_size, const char *fmt, va_list args) -{ - int result = -1; - NK_ASSERT(buf); - NK_ASSERT(buf_size); - if (!buf || !buf_size || !fmt) return 0; -#ifdef NK_INCLUDE_STANDARD_IO - result = NK_VSNPRINTF(buf, (nk_size)buf_size, fmt, args); - result = (result >= buf_size) ? -1: result; - buf[buf_size-1] = 0; -#else - result = nk_vsnprintf(buf, buf_size, fmt, args); -#endif - return result; -} -#endif -NK_API nk_hash -nk_murmur_hash(const void * key, int len, nk_hash seed) -{ - /* 32-Bit MurmurHash3: https://code.google.com/p/smhasher/wiki/MurmurHash3*/ - #define NK_ROTL(x,r) ((x) << (r) | ((x) >> (32 - r))) - - nk_uint h1 = seed; - nk_uint k1; - const nk_byte *data = (const nk_byte*)key; - const nk_byte *keyptr = data; - nk_byte *k1ptr; - const int bsize = sizeof(k1); - const int nblocks = len/4; - - const nk_uint c1 = 0xcc9e2d51; - const nk_uint c2 = 0x1b873593; - const nk_byte *tail; - int i; - - /* body */ - if (!key) return 0; - for (i = 0; i < nblocks; ++i, keyptr += bsize) { - k1ptr = (nk_byte*)&k1; - k1ptr[0] = keyptr[0]; - k1ptr[1] = keyptr[1]; - k1ptr[2] = keyptr[2]; - k1ptr[3] = keyptr[3]; - - k1 *= c1; - k1 = NK_ROTL(k1,15); - k1 *= c2; - - h1 ^= k1; - h1 = NK_ROTL(h1,13); - h1 = h1*5+0xe6546b64; - } - - /* tail */ - tail = (const nk_byte*)(data + nblocks*4); - k1 = 0; - switch (len & 3) { - case 3: k1 ^= (nk_uint)(tail[2] << 16); /* fallthrough */ - case 2: k1 ^= (nk_uint)(tail[1] << 8u); /* fallthrough */ - case 1: k1 ^= tail[0]; - k1 *= c1; - k1 = NK_ROTL(k1,15); - k1 *= c2; - h1 ^= k1; - break; - default: break; - } - - /* finalization */ - h1 ^= (nk_uint)len; - /* fmix32 */ - h1 ^= h1 >> 16; - h1 *= 0x85ebca6b; - h1 ^= h1 >> 13; - h1 *= 0xc2b2ae35; - h1 ^= h1 >> 16; - - #undef NK_ROTL - return h1; -} -#ifdef NK_INCLUDE_STANDARD_IO -NK_LIB char* -nk_file_load(const char* path, nk_size* siz, struct nk_allocator *alloc) -{ - char *buf; - FILE *fd; - long ret; - - NK_ASSERT(path); - NK_ASSERT(siz); - NK_ASSERT(alloc); - if (!path || !siz || !alloc) - return 0; - - fd = fopen(path, "rb"); - if (!fd) return 0; - fseek(fd, 0, SEEK_END); - ret = ftell(fd); - if (ret < 0) { - fclose(fd); - return 0; - } - *siz = (nk_size)ret; - fseek(fd, 0, SEEK_SET); - buf = (char*)alloc->alloc(alloc->userdata,0, *siz); - NK_ASSERT(buf); - if (!buf) { - fclose(fd); - return 0; - } - *siz = (nk_size)fread(buf, 1,*siz, fd); - fclose(fd); - return buf; -} -#endif -NK_LIB int -nk_text_clamp(const struct nk_user_font *font, const char *text, - int text_len, float space, int *glyphs, float *text_width, - nk_rune *sep_list, int sep_count) -{ - int i = 0; - int glyph_len = 0; - float last_width = 0; - nk_rune unicode = 0; - float width = 0; - int len = 0; - int g = 0; - float s; - - int sep_len = 0; - int sep_g = 0; - float sep_width = 0; - sep_count = NK_MAX(sep_count,0); - - glyph_len = nk_utf_decode(text, &unicode, text_len); - while (glyph_len && (width < space) && (len < text_len)) { - len += glyph_len; - s = font->width(font->userdata, font->height, text, len); - for (i = 0; i < sep_count; ++i) { - if (unicode != sep_list[i]) continue; - sep_width = last_width = width; - sep_g = g+1; - sep_len = len; - break; - } - if (i == sep_count){ - last_width = sep_width = width; - sep_g = g+1; - } - width = s; - glyph_len = nk_utf_decode(&text[len], &unicode, text_len - len); - g++; - } - if (len >= text_len) { - *glyphs = g; - *text_width = last_width; - return len; - } else { - *glyphs = sep_g; - *text_width = sep_width; - return (!sep_len) ? len: sep_len; - } -} -NK_LIB struct nk_vec2 -nk_text_calculate_text_bounds(const struct nk_user_font *font, - const char *begin, int byte_len, float row_height, const char **remaining, - struct nk_vec2 *out_offset, int *glyphs, int op) -{ - float line_height = row_height; - struct nk_vec2 text_size = nk_vec2(0,0); - float line_width = 0.0f; - - float glyph_width; - int glyph_len = 0; - nk_rune unicode = 0; - int text_len = 0; - if (!begin || byte_len <= 0 || !font) - return nk_vec2(0,row_height); - - glyph_len = nk_utf_decode(begin, &unicode, byte_len); - if (!glyph_len) return text_size; - glyph_width = font->width(font->userdata, font->height, begin, glyph_len); - - *glyphs = 0; - while ((text_len < byte_len) && glyph_len) { - if (unicode == '\n') { - text_size.x = NK_MAX(text_size.x, line_width); - text_size.y += line_height; - line_width = 0; - *glyphs+=1; - if (op == NK_STOP_ON_NEW_LINE) - break; - - text_len++; - glyph_len = nk_utf_decode(begin + text_len, &unicode, byte_len-text_len); - continue; - } - - if (unicode == '\r') { - text_len++; - *glyphs+=1; - glyph_len = nk_utf_decode(begin + text_len, &unicode, byte_len-text_len); - continue; - } - - *glyphs = *glyphs + 1; - text_len += glyph_len; - line_width += (float)glyph_width; - glyph_len = nk_utf_decode(begin + text_len, &unicode, byte_len-text_len); - glyph_width = font->width(font->userdata, font->height, begin+text_len, glyph_len); - continue; - } - - if (text_size.x < line_width) - text_size.x = line_width; - if (out_offset) - *out_offset = nk_vec2(line_width, text_size.y + line_height); - if (line_width > 0 || text_size.y == 0.0f) - text_size.y += line_height; - if (remaining) - *remaining = begin+text_len; - return text_size; -} - - - - - -/* ============================================================== - * - * COLOR - * - * ===============================================================*/ -NK_INTERN int -nk_parse_hex(const char *p, int length) -{ - int i = 0; - int len = 0; - while (len < length) { - i <<= 4; - if (p[len] >= 'a' && p[len] <= 'f') - i += ((p[len] - 'a') + 10); - else if (p[len] >= 'A' && p[len] <= 'F') - i += ((p[len] - 'A') + 10); - else i += (p[len] - '0'); - len++; - } - return i; -} -NK_API struct nk_color -nk_rgba(int r, int g, int b, int a) -{ - struct nk_color ret; - ret.r = (nk_byte)NK_CLAMP(0, r, 255); - ret.g = (nk_byte)NK_CLAMP(0, g, 255); - ret.b = (nk_byte)NK_CLAMP(0, b, 255); - ret.a = (nk_byte)NK_CLAMP(0, a, 255); - return ret; -} -NK_API struct nk_color -nk_rgb_hex(const char *rgb) -{ - struct nk_color col; - const char *c = rgb; - if (*c == '#') c++; - col.r = (nk_byte)nk_parse_hex(c, 2); - col.g = (nk_byte)nk_parse_hex(c+2, 2); - col.b = (nk_byte)nk_parse_hex(c+4, 2); - col.a = 255; - return col; -} -NK_API struct nk_color -nk_rgba_hex(const char *rgb) -{ - struct nk_color col; - const char *c = rgb; - if (*c == '#') c++; - col.r = (nk_byte)nk_parse_hex(c, 2); - col.g = (nk_byte)nk_parse_hex(c+2, 2); - col.b = (nk_byte)nk_parse_hex(c+4, 2); - col.a = (nk_byte)nk_parse_hex(c+6, 2); - return col; -} -NK_API void -nk_color_hex_rgba(char *output, struct nk_color col) -{ - #define NK_TO_HEX(i) ((i) <= 9 ? '0' + (i): 'A' - 10 + (i)) - output[0] = (char)NK_TO_HEX((col.r & 0xF0) >> 4); - output[1] = (char)NK_TO_HEX((col.r & 0x0F)); - output[2] = (char)NK_TO_HEX((col.g & 0xF0) >> 4); - output[3] = (char)NK_TO_HEX((col.g & 0x0F)); - output[4] = (char)NK_TO_HEX((col.b & 0xF0) >> 4); - output[5] = (char)NK_TO_HEX((col.b & 0x0F)); - output[6] = (char)NK_TO_HEX((col.a & 0xF0) >> 4); - output[7] = (char)NK_TO_HEX((col.a & 0x0F)); - output[8] = '\0'; - #undef NK_TO_HEX -} -NK_API void -nk_color_hex_rgb(char *output, struct nk_color col) -{ - #define NK_TO_HEX(i) ((i) <= 9 ? '0' + (i): 'A' - 10 + (i)) - output[0] = (char)NK_TO_HEX((col.r & 0xF0) >> 4); - output[1] = (char)NK_TO_HEX((col.r & 0x0F)); - output[2] = (char)NK_TO_HEX((col.g & 0xF0) >> 4); - output[3] = (char)NK_TO_HEX((col.g & 0x0F)); - output[4] = (char)NK_TO_HEX((col.b & 0xF0) >> 4); - output[5] = (char)NK_TO_HEX((col.b & 0x0F)); - output[6] = '\0'; - #undef NK_TO_HEX -} -NK_API struct nk_color -nk_rgba_iv(const int *c) -{ - return nk_rgba(c[0], c[1], c[2], c[3]); -} -NK_API struct nk_color -nk_rgba_bv(const nk_byte *c) -{ - return nk_rgba(c[0], c[1], c[2], c[3]); -} -NK_API struct nk_color -nk_rgb(int r, int g, int b) -{ - struct nk_color ret; - ret.r = (nk_byte)NK_CLAMP(0, r, 255); - ret.g = (nk_byte)NK_CLAMP(0, g, 255); - ret.b = (nk_byte)NK_CLAMP(0, b, 255); - ret.a = (nk_byte)255; - return ret; -} -NK_API struct nk_color -nk_rgb_iv(const int *c) -{ - return nk_rgb(c[0], c[1], c[2]); -} -NK_API struct nk_color -nk_rgb_bv(const nk_byte* c) -{ - return nk_rgb(c[0], c[1], c[2]); -} -NK_API struct nk_color -nk_rgba_u32(nk_uint in) -{ - struct nk_color ret; - ret.r = (in & 0xFF); - ret.g = ((in >> 8) & 0xFF); - ret.b = ((in >> 16) & 0xFF); - ret.a = (nk_byte)((in >> 24) & 0xFF); - return ret; -} -NK_API struct nk_color -nk_rgba_f(float r, float g, float b, float a) -{ - struct nk_color ret; - ret.r = (nk_byte)(NK_SATURATE(r) * 255.0f); - ret.g = (nk_byte)(NK_SATURATE(g) * 255.0f); - ret.b = (nk_byte)(NK_SATURATE(b) * 255.0f); - ret.a = (nk_byte)(NK_SATURATE(a) * 255.0f); - return ret; -} -NK_API struct nk_color -nk_rgba_fv(const float *c) -{ - return nk_rgba_f(c[0], c[1], c[2], c[3]); -} -NK_API struct nk_color -nk_rgba_cf(struct nk_colorf c) -{ - return nk_rgba_f(c.r, c.g, c.b, c.a); -} -NK_API struct nk_color -nk_rgb_f(float r, float g, float b) -{ - struct nk_color ret; - ret.r = (nk_byte)(NK_SATURATE(r) * 255.0f); - ret.g = (nk_byte)(NK_SATURATE(g) * 255.0f); - ret.b = (nk_byte)(NK_SATURATE(b) * 255.0f); - ret.a = 255; - return ret; -} -NK_API struct nk_color -nk_rgb_fv(const float *c) -{ - return nk_rgb_f(c[0], c[1], c[2]); -} -NK_API struct nk_color -nk_rgb_cf(struct nk_colorf c) -{ - return nk_rgb_f(c.r, c.g, c.b); -} -NK_API struct nk_color -nk_hsv(int h, int s, int v) -{ - return nk_hsva(h, s, v, 255); -} -NK_API struct nk_color -nk_hsv_iv(const int *c) -{ - return nk_hsv(c[0], c[1], c[2]); -} -NK_API struct nk_color -nk_hsv_bv(const nk_byte *c) -{ - return nk_hsv(c[0], c[1], c[2]); -} -NK_API struct nk_color -nk_hsv_f(float h, float s, float v) -{ - return nk_hsva_f(h, s, v, 1.0f); -} -NK_API struct nk_color -nk_hsv_fv(const float *c) -{ - return nk_hsv_f(c[0], c[1], c[2]); -} -NK_API struct nk_color -nk_hsva(int h, int s, int v, int a) -{ - float hf = ((float)NK_CLAMP(0, h, 255)) / 255.0f; - float sf = ((float)NK_CLAMP(0, s, 255)) / 255.0f; - float vf = ((float)NK_CLAMP(0, v, 255)) / 255.0f; - float af = ((float)NK_CLAMP(0, a, 255)) / 255.0f; - return nk_hsva_f(hf, sf, vf, af); -} -NK_API struct nk_color -nk_hsva_iv(const int *c) -{ - return nk_hsva(c[0], c[1], c[2], c[3]); -} -NK_API struct nk_color -nk_hsva_bv(const nk_byte *c) -{ - return nk_hsva(c[0], c[1], c[2], c[3]); -} -NK_API struct nk_colorf -nk_hsva_colorf(float h, float s, float v, float a) -{ - int i; - float p, q, t, f; - struct nk_colorf out = {0,0,0,0}; - if (s <= 0.0f) { - out.r = v; out.g = v; out.b = v; out.a = a; - return out; - } - h = h / (60.0f/360.0f); - i = (int)h; - f = h - (float)i; - p = v * (1.0f - s); - q = v * (1.0f - (s * f)); - t = v * (1.0f - s * (1.0f - f)); - - switch (i) { - case 0: default: out.r = v; out.g = t; out.b = p; break; - case 1: out.r = q; out.g = v; out.b = p; break; - case 2: out.r = p; out.g = v; out.b = t; break; - case 3: out.r = p; out.g = q; out.b = v; break; - case 4: out.r = t; out.g = p; out.b = v; break; - case 5: out.r = v; out.g = p; out.b = q; break;} - out.a = a; - return out; -} -NK_API struct nk_colorf -nk_hsva_colorfv(float *c) -{ - return nk_hsva_colorf(c[0], c[1], c[2], c[3]); -} -NK_API struct nk_color -nk_hsva_f(float h, float s, float v, float a) -{ - struct nk_colorf c = nk_hsva_colorf(h, s, v, a); - return nk_rgba_f(c.r, c.g, c.b, c.a); -} -NK_API struct nk_color -nk_hsva_fv(const float *c) -{ - return nk_hsva_f(c[0], c[1], c[2], c[3]); -} -NK_API nk_uint -nk_color_u32(struct nk_color in) -{ - nk_uint out = (nk_uint)in.r; - out |= ((nk_uint)in.g << 8); - out |= ((nk_uint)in.b << 16); - out |= ((nk_uint)in.a << 24); - return out; -} -NK_API void -nk_color_f(float *r, float *g, float *b, float *a, struct nk_color in) -{ - NK_STORAGE const float s = 1.0f/255.0f; - *r = (float)in.r * s; - *g = (float)in.g * s; - *b = (float)in.b * s; - *a = (float)in.a * s; -} -NK_API void -nk_color_fv(float *c, struct nk_color in) -{ - nk_color_f(&c[0], &c[1], &c[2], &c[3], in); -} -NK_API struct nk_colorf -nk_color_cf(struct nk_color in) -{ - struct nk_colorf o; - nk_color_f(&o.r, &o.g, &o.b, &o.a, in); - return o; -} -NK_API void -nk_color_d(double *r, double *g, double *b, double *a, struct nk_color in) -{ - NK_STORAGE const double s = 1.0/255.0; - *r = (double)in.r * s; - *g = (double)in.g * s; - *b = (double)in.b * s; - *a = (double)in.a * s; -} -NK_API void -nk_color_dv(double *c, struct nk_color in) -{ - nk_color_d(&c[0], &c[1], &c[2], &c[3], in); -} -NK_API void -nk_color_hsv_f(float *out_h, float *out_s, float *out_v, struct nk_color in) -{ - float a; - nk_color_hsva_f(out_h, out_s, out_v, &a, in); -} -NK_API void -nk_color_hsv_fv(float *out, struct nk_color in) -{ - float a; - nk_color_hsva_f(&out[0], &out[1], &out[2], &a, in); -} -NK_API void -nk_colorf_hsva_f(float *out_h, float *out_s, - float *out_v, float *out_a, struct nk_colorf in) -{ - float chroma; - float K = 0.0f; - if (in.g < in.b) { - const float t = in.g; in.g = in.b; in.b = t; - K = -1.f; - } - if (in.r < in.g) { - const float t = in.r; in.r = in.g; in.g = t; - K = -2.f/6.0f - K; - } - chroma = in.r - ((in.g < in.b) ? in.g: in.b); - *out_h = NK_ABS(K + (in.g - in.b)/(6.0f * chroma + 1e-20f)); - *out_s = chroma / (in.r + 1e-20f); - *out_v = in.r; - *out_a = in.a; - -} -NK_API void -nk_colorf_hsva_fv(float *hsva, struct nk_colorf in) -{ - nk_colorf_hsva_f(&hsva[0], &hsva[1], &hsva[2], &hsva[3], in); -} -NK_API void -nk_color_hsva_f(float *out_h, float *out_s, - float *out_v, float *out_a, struct nk_color in) -{ - struct nk_colorf col; - nk_color_f(&col.r,&col.g,&col.b,&col.a, in); - nk_colorf_hsva_f(out_h, out_s, out_v, out_a, col); -} -NK_API void -nk_color_hsva_fv(float *out, struct nk_color in) -{ - nk_color_hsva_f(&out[0], &out[1], &out[2], &out[3], in); -} -NK_API void -nk_color_hsva_i(int *out_h, int *out_s, int *out_v, - int *out_a, struct nk_color in) -{ - float h,s,v,a; - nk_color_hsva_f(&h, &s, &v, &a, in); - *out_h = (nk_byte)(h * 255.0f); - *out_s = (nk_byte)(s * 255.0f); - *out_v = (nk_byte)(v * 255.0f); - *out_a = (nk_byte)(a * 255.0f); -} -NK_API void -nk_color_hsva_iv(int *out, struct nk_color in) -{ - nk_color_hsva_i(&out[0], &out[1], &out[2], &out[3], in); -} -NK_API void -nk_color_hsva_bv(nk_byte *out, struct nk_color in) -{ - int tmp[4]; - nk_color_hsva_i(&tmp[0], &tmp[1], &tmp[2], &tmp[3], in); - out[0] = (nk_byte)tmp[0]; - out[1] = (nk_byte)tmp[1]; - out[2] = (nk_byte)tmp[2]; - out[3] = (nk_byte)tmp[3]; -} -NK_API void -nk_color_hsva_b(nk_byte *h, nk_byte *s, nk_byte *v, nk_byte *a, struct nk_color in) -{ - int tmp[4]; - nk_color_hsva_i(&tmp[0], &tmp[1], &tmp[2], &tmp[3], in); - *h = (nk_byte)tmp[0]; - *s = (nk_byte)tmp[1]; - *v = (nk_byte)tmp[2]; - *a = (nk_byte)tmp[3]; -} -NK_API void -nk_color_hsv_i(int *out_h, int *out_s, int *out_v, struct nk_color in) -{ - int a; - nk_color_hsva_i(out_h, out_s, out_v, &a, in); -} -NK_API void -nk_color_hsv_b(nk_byte *out_h, nk_byte *out_s, nk_byte *out_v, struct nk_color in) -{ - int tmp[4]; - nk_color_hsva_i(&tmp[0], &tmp[1], &tmp[2], &tmp[3], in); - *out_h = (nk_byte)tmp[0]; - *out_s = (nk_byte)tmp[1]; - *out_v = (nk_byte)tmp[2]; -} -NK_API void -nk_color_hsv_iv(int *out, struct nk_color in) -{ - nk_color_hsv_i(&out[0], &out[1], &out[2], in); -} -NK_API void -nk_color_hsv_bv(nk_byte *out, struct nk_color in) -{ - int tmp[4]; - nk_color_hsv_i(&tmp[0], &tmp[1], &tmp[2], in); - out[0] = (nk_byte)tmp[0]; - out[1] = (nk_byte)tmp[1]; - out[2] = (nk_byte)tmp[2]; -} - - - - - -/* =============================================================== - * - * UTF-8 - * - * ===============================================================*/ -NK_GLOBAL const nk_byte nk_utfbyte[NK_UTF_SIZE+1] = {0x80, 0, 0xC0, 0xE0, 0xF0}; -NK_GLOBAL const nk_byte nk_utfmask[NK_UTF_SIZE+1] = {0xC0, 0x80, 0xE0, 0xF0, 0xF8}; -NK_GLOBAL const nk_uint nk_utfmin[NK_UTF_SIZE+1] = {0, 0, 0x80, 0x800, 0x10000}; -NK_GLOBAL const nk_uint nk_utfmax[NK_UTF_SIZE+1] = {0x10FFFF, 0x7F, 0x7FF, 0xFFFF, 0x10FFFF}; - -NK_INTERN int -nk_utf_validate(nk_rune *u, int i) -{ - NK_ASSERT(u); - if (!u) return 0; - if (!NK_BETWEEN(*u, nk_utfmin[i], nk_utfmax[i]) || - NK_BETWEEN(*u, 0xD800, 0xDFFF)) - *u = NK_UTF_INVALID; - for (i = 1; *u > nk_utfmax[i]; ++i); - return i; -} -NK_INTERN nk_rune -nk_utf_decode_byte(char c, int *i) -{ - NK_ASSERT(i); - if (!i) return 0; - for(*i = 0; *i < (int)NK_LEN(nk_utfmask); ++(*i)) { - if (((nk_byte)c & nk_utfmask[*i]) == nk_utfbyte[*i]) - return (nk_byte)(c & ~nk_utfmask[*i]); - } - return 0; -} -NK_API int -nk_utf_decode(const char *c, nk_rune *u, int clen) -{ - int i, j, len, type=0; - nk_rune udecoded; - - NK_ASSERT(c); - NK_ASSERT(u); - - if (!c || !u) return 0; - if (!clen) return 0; - *u = NK_UTF_INVALID; - - udecoded = nk_utf_decode_byte(c[0], &len); - if (!NK_BETWEEN(len, 1, NK_UTF_SIZE)) - return 1; - - for (i = 1, j = 1; i < clen && j < len; ++i, ++j) { - udecoded = (udecoded << 6) | nk_utf_decode_byte(c[i], &type); - if (type != 0) - return j; - } - if (j < len) - return 0; - *u = udecoded; - nk_utf_validate(u, len); - return len; -} -NK_INTERN char -nk_utf_encode_byte(nk_rune u, int i) -{ - return (char)((nk_utfbyte[i]) | ((nk_byte)u & ~nk_utfmask[i])); -} -NK_API int -nk_utf_encode(nk_rune u, char *c, int clen) -{ - int len, i; - len = nk_utf_validate(&u, 0); - if (clen < len || !len || len > NK_UTF_SIZE) - return 0; - - for (i = len - 1; i != 0; --i) { - c[i] = nk_utf_encode_byte(u, 0); - u >>= 6; - } - c[0] = nk_utf_encode_byte(u, len); - return len; -} -NK_API int -nk_utf_len(const char *str, int len) -{ - const char *text; - int glyphs = 0; - int text_len; - int glyph_len; - int src_len = 0; - nk_rune unicode; - - NK_ASSERT(str); - if (!str || !len) return 0; - - text = str; - text_len = len; - glyph_len = nk_utf_decode(text, &unicode, text_len); - while (glyph_len && src_len < len) { - glyphs++; - src_len = src_len + glyph_len; - glyph_len = nk_utf_decode(text + src_len, &unicode, text_len - src_len); - } - return glyphs; -} -NK_API const char* -nk_utf_at(const char *buffer, int length, int index, - nk_rune *unicode, int *len) -{ - int i = 0; - int src_len = 0; - int glyph_len = 0; - const char *text; - int text_len; - - NK_ASSERT(buffer); - NK_ASSERT(unicode); - NK_ASSERT(len); - - if (!buffer || !unicode || !len) return 0; - if (index < 0) { - *unicode = NK_UTF_INVALID; - *len = 0; - return 0; - } - - text = buffer; - text_len = length; - glyph_len = nk_utf_decode(text, unicode, text_len); - while (glyph_len) { - if (i == index) { - *len = glyph_len; - break; - } - - i++; - src_len = src_len + glyph_len; - glyph_len = nk_utf_decode(text + src_len, unicode, text_len - src_len); - } - if (i != index) return 0; - return buffer + src_len; -} - - - - - -/* ============================================================== - * - * BUFFER - * - * ===============================================================*/ -#ifdef NK_INCLUDE_DEFAULT_ALLOCATOR -NK_LIB void* -nk_malloc(nk_handle unused, void *old,nk_size size) -{ - NK_UNUSED(unused); - NK_UNUSED(old); - return malloc(size); -} -NK_LIB void -nk_mfree(nk_handle unused, void *ptr) -{ - NK_UNUSED(unused); - free(ptr); -} -NK_API void -nk_buffer_init_default(struct nk_buffer *buffer) -{ - struct nk_allocator alloc; - alloc.userdata.ptr = 0; - alloc.alloc = nk_malloc; - alloc.free = nk_mfree; - nk_buffer_init(buffer, &alloc, NK_BUFFER_DEFAULT_INITIAL_SIZE); -} -#endif - -NK_API void -nk_buffer_init(struct nk_buffer *b, const struct nk_allocator *a, - nk_size initial_size) -{ - NK_ASSERT(b); - NK_ASSERT(a); - NK_ASSERT(initial_size); - if (!b || !a || !initial_size) return; - - nk_zero(b, sizeof(*b)); - b->type = NK_BUFFER_DYNAMIC; - b->memory.ptr = a->alloc(a->userdata,0, initial_size); - b->memory.size = initial_size; - b->size = initial_size; - b->grow_factor = 2.0f; - b->pool = *a; -} -NK_API void -nk_buffer_init_fixed(struct nk_buffer *b, void *m, nk_size size) -{ - NK_ASSERT(b); - NK_ASSERT(m); - NK_ASSERT(size); - if (!b || !m || !size) return; - - nk_zero(b, sizeof(*b)); - b->type = NK_BUFFER_FIXED; - b->memory.ptr = m; - b->memory.size = size; - b->size = size; -} -NK_LIB void* -nk_buffer_align(void *unaligned, - nk_size align, nk_size *alignment, - enum nk_buffer_allocation_type type) -{ - void *memory = 0; - switch (type) { - default: - case NK_BUFFER_MAX: - case NK_BUFFER_FRONT: - if (align) { - memory = NK_ALIGN_PTR(unaligned, align); - *alignment = (nk_size)((nk_byte*)memory - (nk_byte*)unaligned); - } else { - memory = unaligned; - *alignment = 0; - } - break; - case NK_BUFFER_BACK: - if (align) { - memory = NK_ALIGN_PTR_BACK(unaligned, align); - *alignment = (nk_size)((nk_byte*)unaligned - (nk_byte*)memory); - } else { - memory = unaligned; - *alignment = 0; - } - break; - } - return memory; -} -NK_LIB void* -nk_buffer_realloc(struct nk_buffer *b, nk_size capacity, nk_size *size) -{ - void *temp; - nk_size buffer_size; - - NK_ASSERT(b); - NK_ASSERT(size); - if (!b || !size || !b->pool.alloc || !b->pool.free) - return 0; - - buffer_size = b->memory.size; - temp = b->pool.alloc(b->pool.userdata, b->memory.ptr, capacity); - NK_ASSERT(temp); - if (!temp) return 0; - - *size = capacity; - if (temp != b->memory.ptr) { - NK_MEMCPY(temp, b->memory.ptr, buffer_size); - b->pool.free(b->pool.userdata, b->memory.ptr); - } - - if (b->size == buffer_size) { - /* no back buffer so just set correct size */ - b->size = capacity; - return temp; - } else { - /* copy back buffer to the end of the new buffer */ - void *dst, *src; - nk_size back_size; - back_size = buffer_size - b->size; - dst = nk_ptr_add(void, temp, capacity - back_size); - src = nk_ptr_add(void, temp, b->size); - NK_MEMCPY(dst, src, back_size); - b->size = capacity - back_size; - } - return temp; -} -NK_LIB void* -nk_buffer_alloc(struct nk_buffer *b, enum nk_buffer_allocation_type type, - nk_size size, nk_size align) -{ - int full; - nk_size alignment; - void *unaligned; - void *memory; - - NK_ASSERT(b); - NK_ASSERT(size); - if (!b || !size) return 0; - b->needed += size; - - /* calculate total size with needed alignment + size */ - if (type == NK_BUFFER_FRONT) - unaligned = nk_ptr_add(void, b->memory.ptr, b->allocated); - else unaligned = nk_ptr_add(void, b->memory.ptr, b->size - size); - memory = nk_buffer_align(unaligned, align, &alignment, type); - - /* check if buffer has enough memory*/ - if (type == NK_BUFFER_FRONT) - full = ((b->allocated + size + alignment) > b->size); - else full = ((b->size - NK_MIN(b->size,(size + alignment))) <= b->allocated); - - if (full) { - nk_size capacity; - if (b->type != NK_BUFFER_DYNAMIC) - return 0; - NK_ASSERT(b->pool.alloc && b->pool.free); - if (b->type != NK_BUFFER_DYNAMIC || !b->pool.alloc || !b->pool.free) - return 0; - - /* buffer is full so allocate bigger buffer if dynamic */ - capacity = (nk_size)((float)b->memory.size * b->grow_factor); - capacity = NK_MAX(capacity, nk_round_up_pow2((nk_uint)(b->allocated + size))); - b->memory.ptr = nk_buffer_realloc(b, capacity, &b->memory.size); - if (!b->memory.ptr) return 0; - - /* align newly allocated pointer */ - if (type == NK_BUFFER_FRONT) - unaligned = nk_ptr_add(void, b->memory.ptr, b->allocated); - else unaligned = nk_ptr_add(void, b->memory.ptr, b->size - size); - memory = nk_buffer_align(unaligned, align, &alignment, type); - } - if (type == NK_BUFFER_FRONT) - b->allocated += size + alignment; - else b->size -= (size + alignment); - b->needed += alignment; - b->calls++; - return memory; -} -NK_API void -nk_buffer_push(struct nk_buffer *b, enum nk_buffer_allocation_type type, - const void *memory, nk_size size, nk_size align) -{ - void *mem = nk_buffer_alloc(b, type, size, align); - if (!mem) return; - NK_MEMCPY(mem, memory, size); -} -NK_API void -nk_buffer_mark(struct nk_buffer *buffer, enum nk_buffer_allocation_type type) -{ - NK_ASSERT(buffer); - if (!buffer) return; - buffer->marker[type].active = nk_true; - if (type == NK_BUFFER_BACK) - buffer->marker[type].offset = buffer->size; - else buffer->marker[type].offset = buffer->allocated; -} -NK_API void -nk_buffer_reset(struct nk_buffer *buffer, enum nk_buffer_allocation_type type) -{ - NK_ASSERT(buffer); - if (!buffer) return; - if (type == NK_BUFFER_BACK) { - /* reset back buffer either back to marker or empty */ - buffer->needed -= (buffer->memory.size - buffer->marker[type].offset); - if (buffer->marker[type].active) - buffer->size = buffer->marker[type].offset; - else buffer->size = buffer->memory.size; - buffer->marker[type].active = nk_false; - } else { - /* reset front buffer either back to back marker or empty */ - buffer->needed -= (buffer->allocated - buffer->marker[type].offset); - if (buffer->marker[type].active) - buffer->allocated = buffer->marker[type].offset; - else buffer->allocated = 0; - buffer->marker[type].active = nk_false; - } -} -NK_API void -nk_buffer_clear(struct nk_buffer *b) -{ - NK_ASSERT(b); - if (!b) return; - b->allocated = 0; - b->size = b->memory.size; - b->calls = 0; - b->needed = 0; -} -NK_API void -nk_buffer_free(struct nk_buffer *b) -{ - NK_ASSERT(b); - if (!b || !b->memory.ptr) return; - if (b->type == NK_BUFFER_FIXED) return; - if (!b->pool.free) return; - NK_ASSERT(b->pool.free); - b->pool.free(b->pool.userdata, b->memory.ptr); -} -NK_API void -nk_buffer_info(struct nk_memory_status *s, struct nk_buffer *b) -{ - NK_ASSERT(b); - NK_ASSERT(s); - if (!s || !b) return; - s->allocated = b->allocated; - s->size = b->memory.size; - s->needed = b->needed; - s->memory = b->memory.ptr; - s->calls = b->calls; -} -NK_API void* -nk_buffer_memory(struct nk_buffer *buffer) -{ - NK_ASSERT(buffer); - if (!buffer) return 0; - return buffer->memory.ptr; -} -NK_API const void* -nk_buffer_memory_const(const struct nk_buffer *buffer) -{ - NK_ASSERT(buffer); - if (!buffer) return 0; - return buffer->memory.ptr; -} -NK_API nk_size -nk_buffer_total(struct nk_buffer *buffer) -{ - NK_ASSERT(buffer); - if (!buffer) return 0; - return buffer->memory.size; -} - - - - - -/* =============================================================== - * - * STRING - * - * ===============================================================*/ -#ifdef NK_INCLUDE_DEFAULT_ALLOCATOR -NK_API void -nk_str_init_default(struct nk_str *str) -{ - struct nk_allocator alloc; - alloc.userdata.ptr = 0; - alloc.alloc = nk_malloc; - alloc.free = nk_mfree; - nk_buffer_init(&str->buffer, &alloc, 32); - str->len = 0; -} -#endif - -NK_API void -nk_str_init(struct nk_str *str, const struct nk_allocator *alloc, nk_size size) -{ - nk_buffer_init(&str->buffer, alloc, size); - str->len = 0; -} -NK_API void -nk_str_init_fixed(struct nk_str *str, void *memory, nk_size size) -{ - nk_buffer_init_fixed(&str->buffer, memory, size); - str->len = 0; -} -NK_API int -nk_str_append_text_char(struct nk_str *s, const char *str, int len) -{ - char *mem; - NK_ASSERT(s); - NK_ASSERT(str); - if (!s || !str || !len) return 0; - mem = (char*)nk_buffer_alloc(&s->buffer, NK_BUFFER_FRONT, (nk_size)len * sizeof(char), 0); - if (!mem) return 0; - NK_MEMCPY(mem, str, (nk_size)len * sizeof(char)); - s->len += nk_utf_len(str, len); - return len; -} -NK_API int -nk_str_append_str_char(struct nk_str *s, const char *str) -{ - return nk_str_append_text_char(s, str, nk_strlen(str)); -} -NK_API int -nk_str_append_text_utf8(struct nk_str *str, const char *text, int len) -{ - int i = 0; - int byte_len = 0; - nk_rune unicode; - if (!str || !text || !len) return 0; - for (i = 0; i < len; ++i) - byte_len += nk_utf_decode(text+byte_len, &unicode, 4); - nk_str_append_text_char(str, text, byte_len); - return len; -} -NK_API int -nk_str_append_str_utf8(struct nk_str *str, const char *text) -{ - int byte_len = 0; - int num_runes = 0; - int glyph_len = 0; - nk_rune unicode; - if (!str || !text) return 0; - - glyph_len = byte_len = nk_utf_decode(text+byte_len, &unicode, 4); - while (unicode != '\0' && glyph_len) { - glyph_len = nk_utf_decode(text+byte_len, &unicode, 4); - byte_len += glyph_len; - num_runes++; - } - nk_str_append_text_char(str, text, byte_len); - return num_runes; -} -NK_API int -nk_str_append_text_runes(struct nk_str *str, const nk_rune *text, int len) -{ - int i = 0; - int byte_len = 0; - nk_glyph glyph; - - NK_ASSERT(str); - if (!str || !text || !len) return 0; - for (i = 0; i < len; ++i) { - byte_len = nk_utf_encode(text[i], glyph, NK_UTF_SIZE); - if (!byte_len) break; - nk_str_append_text_char(str, glyph, byte_len); - } - return len; -} -NK_API int -nk_str_append_str_runes(struct nk_str *str, const nk_rune *runes) -{ - int i = 0; - nk_glyph glyph; - int byte_len; - NK_ASSERT(str); - if (!str || !runes) return 0; - while (runes[i] != '\0') { - byte_len = nk_utf_encode(runes[i], glyph, NK_UTF_SIZE); - nk_str_append_text_char(str, glyph, byte_len); - i++; - } - return i; -} -NK_API int -nk_str_insert_at_char(struct nk_str *s, int pos, const char *str, int len) -{ - int i; - void *mem; - char *src; - char *dst; - - int copylen; - NK_ASSERT(s); - NK_ASSERT(str); - NK_ASSERT(len >= 0); - if (!s || !str || !len || (nk_size)pos > s->buffer.allocated) return 0; - if ((s->buffer.allocated + (nk_size)len >= s->buffer.memory.size) && - (s->buffer.type == NK_BUFFER_FIXED)) return 0; - - copylen = (int)s->buffer.allocated - pos; - if (!copylen) { - nk_str_append_text_char(s, str, len); - return 1; - } - mem = nk_buffer_alloc(&s->buffer, NK_BUFFER_FRONT, (nk_size)len * sizeof(char), 0); - if (!mem) return 0; - - /* memmove */ - NK_ASSERT(((int)pos + (int)len + ((int)copylen - 1)) >= 0); - NK_ASSERT(((int)pos + ((int)copylen - 1)) >= 0); - dst = nk_ptr_add(char, s->buffer.memory.ptr, pos + len + (copylen - 1)); - src = nk_ptr_add(char, s->buffer.memory.ptr, pos + (copylen-1)); - for (i = 0; i < copylen; ++i) *dst-- = *src--; - mem = nk_ptr_add(void, s->buffer.memory.ptr, pos); - NK_MEMCPY(mem, str, (nk_size)len * sizeof(char)); - s->len = nk_utf_len((char *)s->buffer.memory.ptr, (int)s->buffer.allocated); - return 1; -} -NK_API int -nk_str_insert_at_rune(struct nk_str *str, int pos, const char *cstr, int len) -{ - int glyph_len; - nk_rune unicode; - const char *begin; - const char *buffer; - - NK_ASSERT(str); - NK_ASSERT(cstr); - NK_ASSERT(len); - if (!str || !cstr || !len) return 0; - begin = nk_str_at_rune(str, pos, &unicode, &glyph_len); - if (!str->len) - return nk_str_append_text_char(str, cstr, len); - buffer = nk_str_get_const(str); - if (!begin) return 0; - return nk_str_insert_at_char(str, (int)(begin - buffer), cstr, len); -} -NK_API int -nk_str_insert_text_char(struct nk_str *str, int pos, const char *text, int len) -{ - return nk_str_insert_text_utf8(str, pos, text, len); -} -NK_API int -nk_str_insert_str_char(struct nk_str *str, int pos, const char *text) -{ - return nk_str_insert_text_utf8(str, pos, text, nk_strlen(text)); -} -NK_API int -nk_str_insert_text_utf8(struct nk_str *str, int pos, const char *text, int len) -{ - int i = 0; - int byte_len = 0; - nk_rune unicode; - - NK_ASSERT(str); - NK_ASSERT(text); - if (!str || !text || !len) return 0; - for (i = 0; i < len; ++i) - byte_len += nk_utf_decode(text+byte_len, &unicode, 4); - nk_str_insert_at_rune(str, pos, text, byte_len); - return len; -} -NK_API int -nk_str_insert_str_utf8(struct nk_str *str, int pos, const char *text) -{ - int byte_len = 0; - int num_runes = 0; - int glyph_len = 0; - nk_rune unicode; - if (!str || !text) return 0; - - glyph_len = byte_len = nk_utf_decode(text+byte_len, &unicode, 4); - while (unicode != '\0' && glyph_len) { - glyph_len = nk_utf_decode(text+byte_len, &unicode, 4); - byte_len += glyph_len; - num_runes++; - } - nk_str_insert_at_rune(str, pos, text, byte_len); - return num_runes; -} -NK_API int -nk_str_insert_text_runes(struct nk_str *str, int pos, const nk_rune *runes, int len) -{ - int i = 0; - int byte_len = 0; - nk_glyph glyph; - - NK_ASSERT(str); - if (!str || !runes || !len) return 0; - for (i = 0; i < len; ++i) { - byte_len = nk_utf_encode(runes[i], glyph, NK_UTF_SIZE); - if (!byte_len) break; - nk_str_insert_at_rune(str, pos+i, glyph, byte_len); - } - return len; -} -NK_API int -nk_str_insert_str_runes(struct nk_str *str, int pos, const nk_rune *runes) -{ - int i = 0; - nk_glyph glyph; - int byte_len; - NK_ASSERT(str); - if (!str || !runes) return 0; - while (runes[i] != '\0') { - byte_len = nk_utf_encode(runes[i], glyph, NK_UTF_SIZE); - nk_str_insert_at_rune(str, pos+i, glyph, byte_len); - i++; - } - return i; -} -NK_API void -nk_str_remove_chars(struct nk_str *s, int len) -{ - NK_ASSERT(s); - NK_ASSERT(len >= 0); - if (!s || len < 0 || (nk_size)len > s->buffer.allocated) return; - NK_ASSERT(((int)s->buffer.allocated - (int)len) >= 0); - s->buffer.allocated -= (nk_size)len; - s->len = nk_utf_len((char *)s->buffer.memory.ptr, (int)s->buffer.allocated); -} -NK_API void -nk_str_remove_runes(struct nk_str *str, int len) -{ - int index; - const char *begin; - const char *end; - nk_rune unicode; - - NK_ASSERT(str); - NK_ASSERT(len >= 0); - if (!str || len < 0) return; - if (len >= str->len) { - str->len = 0; - return; - } - - index = str->len - len; - begin = nk_str_at_rune(str, index, &unicode, &len); - end = (const char*)str->buffer.memory.ptr + str->buffer.allocated; - nk_str_remove_chars(str, (int)(end-begin)+1); -} -NK_API void -nk_str_delete_chars(struct nk_str *s, int pos, int len) -{ - NK_ASSERT(s); - if (!s || !len || (nk_size)pos > s->buffer.allocated || - (nk_size)(pos + len) > s->buffer.allocated) return; - - if ((nk_size)(pos + len) < s->buffer.allocated) { - /* memmove */ - char *dst = nk_ptr_add(char, s->buffer.memory.ptr, pos); - char *src = nk_ptr_add(char, s->buffer.memory.ptr, pos + len); - NK_MEMCPY(dst, src, s->buffer.allocated - (nk_size)(pos + len)); - NK_ASSERT(((int)s->buffer.allocated - (int)len) >= 0); - s->buffer.allocated -= (nk_size)len; - } else nk_str_remove_chars(s, len); - s->len = nk_utf_len((char *)s->buffer.memory.ptr, (int)s->buffer.allocated); -} -NK_API void -nk_str_delete_runes(struct nk_str *s, int pos, int len) -{ - char *temp; - nk_rune unicode; - char *begin; - char *end; - int unused; - - NK_ASSERT(s); - NK_ASSERT(s->len >= pos + len); - if (s->len < pos + len) - len = NK_CLAMP(0, (s->len - pos), s->len); - if (!len) return; - - temp = (char *)s->buffer.memory.ptr; - begin = nk_str_at_rune(s, pos, &unicode, &unused); - if (!begin) return; - s->buffer.memory.ptr = begin; - end = nk_str_at_rune(s, len, &unicode, &unused); - s->buffer.memory.ptr = temp; - if (!end) return; - nk_str_delete_chars(s, (int)(begin - temp), (int)(end - begin)); -} -NK_API char* -nk_str_at_char(struct nk_str *s, int pos) -{ - NK_ASSERT(s); - if (!s || pos > (int)s->buffer.allocated) return 0; - return nk_ptr_add(char, s->buffer.memory.ptr, pos); -} -NK_API char* -nk_str_at_rune(struct nk_str *str, int pos, nk_rune *unicode, int *len) -{ - int i = 0; - int src_len = 0; - int glyph_len = 0; - char *text; - int text_len; - - NK_ASSERT(str); - NK_ASSERT(unicode); - NK_ASSERT(len); - - if (!str || !unicode || !len) return 0; - if (pos < 0) { - *unicode = 0; - *len = 0; - return 0; - } - - text = (char*)str->buffer.memory.ptr; - text_len = (int)str->buffer.allocated; - glyph_len = nk_utf_decode(text, unicode, text_len); - while (glyph_len) { - if (i == pos) { - *len = glyph_len; - break; - } - - i++; - src_len = src_len + glyph_len; - glyph_len = nk_utf_decode(text + src_len, unicode, text_len - src_len); - } - if (i != pos) return 0; - return text + src_len; -} -NK_API const char* -nk_str_at_char_const(const struct nk_str *s, int pos) -{ - NK_ASSERT(s); - if (!s || pos > (int)s->buffer.allocated) return 0; - return nk_ptr_add(char, s->buffer.memory.ptr, pos); -} -NK_API const char* -nk_str_at_const(const struct nk_str *str, int pos, nk_rune *unicode, int *len) -{ - int i = 0; - int src_len = 0; - int glyph_len = 0; - char *text; - int text_len; - - NK_ASSERT(str); - NK_ASSERT(unicode); - NK_ASSERT(len); - - if (!str || !unicode || !len) return 0; - if (pos < 0) { - *unicode = 0; - *len = 0; - return 0; - } - - text = (char*)str->buffer.memory.ptr; - text_len = (int)str->buffer.allocated; - glyph_len = nk_utf_decode(text, unicode, text_len); - while (glyph_len) { - if (i == pos) { - *len = glyph_len; - break; - } - - i++; - src_len = src_len + glyph_len; - glyph_len = nk_utf_decode(text + src_len, unicode, text_len - src_len); - } - if (i != pos) return 0; - return text + src_len; -} -NK_API nk_rune -nk_str_rune_at(const struct nk_str *str, int pos) -{ - int len; - nk_rune unicode = 0; - nk_str_at_const(str, pos, &unicode, &len); - return unicode; -} -NK_API char* -nk_str_get(struct nk_str *s) -{ - NK_ASSERT(s); - if (!s || !s->len || !s->buffer.allocated) return 0; - return (char*)s->buffer.memory.ptr; -} -NK_API const char* -nk_str_get_const(const struct nk_str *s) -{ - NK_ASSERT(s); - if (!s || !s->len || !s->buffer.allocated) return 0; - return (const char*)s->buffer.memory.ptr; -} -NK_API int -nk_str_len(struct nk_str *s) -{ - NK_ASSERT(s); - if (!s || !s->len || !s->buffer.allocated) return 0; - return s->len; -} -NK_API int -nk_str_len_char(struct nk_str *s) -{ - NK_ASSERT(s); - if (!s || !s->len || !s->buffer.allocated) return 0; - return (int)s->buffer.allocated; -} -NK_API void -nk_str_clear(struct nk_str *str) -{ - NK_ASSERT(str); - nk_buffer_clear(&str->buffer); - str->len = 0; -} -NK_API void -nk_str_free(struct nk_str *str) -{ - NK_ASSERT(str); - nk_buffer_free(&str->buffer); - str->len = 0; -} - - - - - -/* ============================================================== - * - * DRAW - * - * ===============================================================*/ -NK_LIB void -nk_command_buffer_init(struct nk_command_buffer *cb, - struct nk_buffer *b, enum nk_command_clipping clip) -{ - NK_ASSERT(cb); - NK_ASSERT(b); - if (!cb || !b) return; - cb->base = b; - cb->use_clipping = (int)clip; - cb->begin = b->allocated; - cb->end = b->allocated; - cb->last = b->allocated; -} -NK_LIB void -nk_command_buffer_reset(struct nk_command_buffer *b) -{ - NK_ASSERT(b); - if (!b) return; - b->begin = 0; - b->end = 0; - b->last = 0; - b->clip = nk_null_rect; -#ifdef NK_INCLUDE_COMMAND_USERDATA - b->userdata.ptr = 0; -#endif -} -NK_LIB void* -nk_command_buffer_push(struct nk_command_buffer* b, - enum nk_command_type t, nk_size size) -{ - NK_STORAGE const nk_size align = NK_ALIGNOF(struct nk_command); - struct nk_command *cmd; - nk_size alignment; - void *unaligned; - void *memory; - - NK_ASSERT(b); - NK_ASSERT(b->base); - if (!b) return 0; - cmd = (struct nk_command*)nk_buffer_alloc(b->base,NK_BUFFER_FRONT,size,align); - if (!cmd) return 0; - - /* make sure the offset to the next command is aligned */ - b->last = (nk_size)((nk_byte*)cmd - (nk_byte*)b->base->memory.ptr); - unaligned = (nk_byte*)cmd + size; - memory = NK_ALIGN_PTR(unaligned, align); - alignment = (nk_size)((nk_byte*)memory - (nk_byte*)unaligned); -#ifdef NK_ZERO_COMMAND_MEMORY - NK_MEMSET(cmd, 0, size + alignment); -#endif - - cmd->type = t; - cmd->next = b->base->allocated + alignment; -#ifdef NK_INCLUDE_COMMAND_USERDATA - cmd->userdata = b->userdata; -#endif - b->end = cmd->next; - return cmd; -} -NK_API void -nk_push_scissor(struct nk_command_buffer *b, struct nk_rect r) -{ - struct nk_command_scissor *cmd; - NK_ASSERT(b); - if (!b) return; - - b->clip.x = r.x; - b->clip.y = r.y; - b->clip.w = r.w; - b->clip.h = r.h; - cmd = (struct nk_command_scissor*) - nk_command_buffer_push(b, NK_COMMAND_SCISSOR, sizeof(*cmd)); - - if (!cmd) return; - cmd->x = (short)r.x; - cmd->y = (short)r.y; - cmd->w = (unsigned short)NK_MAX(0, r.w); - cmd->h = (unsigned short)NK_MAX(0, r.h); -} -NK_API void -nk_stroke_line(struct nk_command_buffer *b, float x0, float y0, - float x1, float y1, float line_thickness, struct nk_color c) -{ - struct nk_command_line *cmd; - NK_ASSERT(b); - if (!b || line_thickness <= 0) return; - cmd = (struct nk_command_line*) - nk_command_buffer_push(b, NK_COMMAND_LINE, sizeof(*cmd)); - if (!cmd) return; - cmd->line_thickness = (unsigned short)line_thickness; - cmd->begin.x = (short)x0; - cmd->begin.y = (short)y0; - cmd->end.x = (short)x1; - cmd->end.y = (short)y1; - cmd->color = c; -} -NK_API void -nk_stroke_curve(struct nk_command_buffer *b, float ax, float ay, - float ctrl0x, float ctrl0y, float ctrl1x, float ctrl1y, - float bx, float by, float line_thickness, struct nk_color col) -{ - struct nk_command_curve *cmd; - NK_ASSERT(b); - if (!b || col.a == 0 || line_thickness <= 0) return; - - cmd = (struct nk_command_curve*) - nk_command_buffer_push(b, NK_COMMAND_CURVE, sizeof(*cmd)); - if (!cmd) return; - cmd->line_thickness = (unsigned short)line_thickness; - cmd->begin.x = (short)ax; - cmd->begin.y = (short)ay; - cmd->ctrl[0].x = (short)ctrl0x; - cmd->ctrl[0].y = (short)ctrl0y; - cmd->ctrl[1].x = (short)ctrl1x; - cmd->ctrl[1].y = (short)ctrl1y; - cmd->end.x = (short)bx; - cmd->end.y = (short)by; - cmd->color = col; -} -NK_API void -nk_stroke_rect(struct nk_command_buffer *b, struct nk_rect rect, - float rounding, float line_thickness, struct nk_color c) -{ - struct nk_command_rect *cmd; - NK_ASSERT(b); - if (!b || c.a == 0 || rect.w == 0 || rect.h == 0 || line_thickness <= 0) return; - if (b->use_clipping) { - const struct nk_rect *clip = &b->clip; - if (!NK_INTERSECT(rect.x, rect.y, rect.w, rect.h, - clip->x, clip->y, clip->w, clip->h)) return; - } - cmd = (struct nk_command_rect*) - nk_command_buffer_push(b, NK_COMMAND_RECT, sizeof(*cmd)); - if (!cmd) return; - cmd->rounding = (unsigned short)rounding; - cmd->line_thickness = (unsigned short)line_thickness; - cmd->x = (short)rect.x; - cmd->y = (short)rect.y; - cmd->w = (unsigned short)NK_MAX(0, rect.w); - cmd->h = (unsigned short)NK_MAX(0, rect.h); - cmd->color = c; -} -NK_API void -nk_fill_rect(struct nk_command_buffer *b, struct nk_rect rect, - float rounding, struct nk_color c) -{ - struct nk_command_rect_filled *cmd; - NK_ASSERT(b); - if (!b || c.a == 0 || rect.w == 0 || rect.h == 0) return; - if (b->use_clipping) { - const struct nk_rect *clip = &b->clip; - if (!NK_INTERSECT(rect.x, rect.y, rect.w, rect.h, - clip->x, clip->y, clip->w, clip->h)) return; - } - - cmd = (struct nk_command_rect_filled*) - nk_command_buffer_push(b, NK_COMMAND_RECT_FILLED, sizeof(*cmd)); - if (!cmd) return; - cmd->rounding = (unsigned short)rounding; - cmd->x = (short)rect.x; - cmd->y = (short)rect.y; - cmd->w = (unsigned short)NK_MAX(0, rect.w); - cmd->h = (unsigned short)NK_MAX(0, rect.h); - cmd->color = c; -} -NK_API void -nk_fill_rect_multi_color(struct nk_command_buffer *b, struct nk_rect rect, - struct nk_color left, struct nk_color top, struct nk_color right, - struct nk_color bottom) -{ - struct nk_command_rect_multi_color *cmd; - NK_ASSERT(b); - if (!b || rect.w == 0 || rect.h == 0) return; - if (b->use_clipping) { - const struct nk_rect *clip = &b->clip; - if (!NK_INTERSECT(rect.x, rect.y, rect.w, rect.h, - clip->x, clip->y, clip->w, clip->h)) return; - } - - cmd = (struct nk_command_rect_multi_color*) - nk_command_buffer_push(b, NK_COMMAND_RECT_MULTI_COLOR, sizeof(*cmd)); - if (!cmd) return; - cmd->x = (short)rect.x; - cmd->y = (short)rect.y; - cmd->w = (unsigned short)NK_MAX(0, rect.w); - cmd->h = (unsigned short)NK_MAX(0, rect.h); - cmd->left = left; - cmd->top = top; - cmd->right = right; - cmd->bottom = bottom; -} -NK_API void -nk_stroke_circle(struct nk_command_buffer *b, struct nk_rect r, - float line_thickness, struct nk_color c) -{ - struct nk_command_circle *cmd; - if (!b || r.w == 0 || r.h == 0 || line_thickness <= 0) return; - if (b->use_clipping) { - const struct nk_rect *clip = &b->clip; - if (!NK_INTERSECT(r.x, r.y, r.w, r.h, clip->x, clip->y, clip->w, clip->h)) - return; - } - - cmd = (struct nk_command_circle*) - nk_command_buffer_push(b, NK_COMMAND_CIRCLE, sizeof(*cmd)); - if (!cmd) return; - cmd->line_thickness = (unsigned short)line_thickness; - cmd->x = (short)r.x; - cmd->y = (short)r.y; - cmd->w = (unsigned short)NK_MAX(r.w, 0); - cmd->h = (unsigned short)NK_MAX(r.h, 0); - cmd->color = c; -} -NK_API void -nk_fill_circle(struct nk_command_buffer *b, struct nk_rect r, struct nk_color c) -{ - struct nk_command_circle_filled *cmd; - NK_ASSERT(b); - if (!b || c.a == 0 || r.w == 0 || r.h == 0) return; - if (b->use_clipping) { - const struct nk_rect *clip = &b->clip; - if (!NK_INTERSECT(r.x, r.y, r.w, r.h, clip->x, clip->y, clip->w, clip->h)) - return; - } - - cmd = (struct nk_command_circle_filled*) - nk_command_buffer_push(b, NK_COMMAND_CIRCLE_FILLED, sizeof(*cmd)); - if (!cmd) return; - cmd->x = (short)r.x; - cmd->y = (short)r.y; - cmd->w = (unsigned short)NK_MAX(r.w, 0); - cmd->h = (unsigned short)NK_MAX(r.h, 0); - cmd->color = c; -} -NK_API void -nk_stroke_arc(struct nk_command_buffer *b, float cx, float cy, float radius, - float a_min, float a_max, float line_thickness, struct nk_color c) -{ - struct nk_command_arc *cmd; - if (!b || c.a == 0 || line_thickness <= 0) return; - cmd = (struct nk_command_arc*) - nk_command_buffer_push(b, NK_COMMAND_ARC, sizeof(*cmd)); - if (!cmd) return; - cmd->line_thickness = (unsigned short)line_thickness; - cmd->cx = (short)cx; - cmd->cy = (short)cy; - cmd->r = (unsigned short)radius; - cmd->a[0] = a_min; - cmd->a[1] = a_max; - cmd->color = c; -} -NK_API void -nk_fill_arc(struct nk_command_buffer *b, float cx, float cy, float radius, - float a_min, float a_max, struct nk_color c) -{ - struct nk_command_arc_filled *cmd; - NK_ASSERT(b); - if (!b || c.a == 0) return; - cmd = (struct nk_command_arc_filled*) - nk_command_buffer_push(b, NK_COMMAND_ARC_FILLED, sizeof(*cmd)); - if (!cmd) return; - cmd->cx = (short)cx; - cmd->cy = (short)cy; - cmd->r = (unsigned short)radius; - cmd->a[0] = a_min; - cmd->a[1] = a_max; - cmd->color = c; -} -NK_API void -nk_stroke_triangle(struct nk_command_buffer *b, float x0, float y0, float x1, - float y1, float x2, float y2, float line_thickness, struct nk_color c) -{ - struct nk_command_triangle *cmd; - NK_ASSERT(b); - if (!b || c.a == 0 || line_thickness <= 0) return; - if (b->use_clipping) { - const struct nk_rect *clip = &b->clip; - if (!NK_INBOX(x0, y0, clip->x, clip->y, clip->w, clip->h) && - !NK_INBOX(x1, y1, clip->x, clip->y, clip->w, clip->h) && - !NK_INBOX(x2, y2, clip->x, clip->y, clip->w, clip->h)) - return; - } - - cmd = (struct nk_command_triangle*) - nk_command_buffer_push(b, NK_COMMAND_TRIANGLE, sizeof(*cmd)); - if (!cmd) return; - cmd->line_thickness = (unsigned short)line_thickness; - cmd->a.x = (short)x0; - cmd->a.y = (short)y0; - cmd->b.x = (short)x1; - cmd->b.y = (short)y1; - cmd->c.x = (short)x2; - cmd->c.y = (short)y2; - cmd->color = c; -} -NK_API void -nk_fill_triangle(struct nk_command_buffer *b, float x0, float y0, float x1, - float y1, float x2, float y2, struct nk_color c) -{ - struct nk_command_triangle_filled *cmd; - NK_ASSERT(b); - if (!b || c.a == 0) return; - if (!b) return; - if (b->use_clipping) { - const struct nk_rect *clip = &b->clip; - if (!NK_INBOX(x0, y0, clip->x, clip->y, clip->w, clip->h) && - !NK_INBOX(x1, y1, clip->x, clip->y, clip->w, clip->h) && - !NK_INBOX(x2, y2, clip->x, clip->y, clip->w, clip->h)) - return; - } - - cmd = (struct nk_command_triangle_filled*) - nk_command_buffer_push(b, NK_COMMAND_TRIANGLE_FILLED, sizeof(*cmd)); - if (!cmd) return; - cmd->a.x = (short)x0; - cmd->a.y = (short)y0; - cmd->b.x = (short)x1; - cmd->b.y = (short)y1; - cmd->c.x = (short)x2; - cmd->c.y = (short)y2; - cmd->color = c; -} -NK_API void -nk_stroke_polygon(struct nk_command_buffer *b, float *points, int point_count, - float line_thickness, struct nk_color col) -{ - int i; - nk_size size = 0; - struct nk_command_polygon *cmd; - - NK_ASSERT(b); - if (!b || col.a == 0 || line_thickness <= 0) return; - size = sizeof(*cmd) + sizeof(short) * 2 * (nk_size)point_count; - cmd = (struct nk_command_polygon*) nk_command_buffer_push(b, NK_COMMAND_POLYGON, size); - if (!cmd) return; - cmd->color = col; - cmd->line_thickness = (unsigned short)line_thickness; - cmd->point_count = (unsigned short)point_count; - for (i = 0; i < point_count; ++i) { - cmd->points[i].x = (short)points[i*2]; - cmd->points[i].y = (short)points[i*2+1]; - } -} -NK_API void -nk_fill_polygon(struct nk_command_buffer *b, float *points, int point_count, - struct nk_color col) -{ - int i; - nk_size size = 0; - struct nk_command_polygon_filled *cmd; - - NK_ASSERT(b); - if (!b || col.a == 0) return; - size = sizeof(*cmd) + sizeof(short) * 2 * (nk_size)point_count; - cmd = (struct nk_command_polygon_filled*) - nk_command_buffer_push(b, NK_COMMAND_POLYGON_FILLED, size); - if (!cmd) return; - cmd->color = col; - cmd->point_count = (unsigned short)point_count; - for (i = 0; i < point_count; ++i) { - cmd->points[i].x = (short)points[i*2+0]; - cmd->points[i].y = (short)points[i*2+1]; - } -} -NK_API void -nk_stroke_polyline(struct nk_command_buffer *b, float *points, int point_count, - float line_thickness, struct nk_color col) -{ - int i; - nk_size size = 0; - struct nk_command_polyline *cmd; - - NK_ASSERT(b); - if (!b || col.a == 0 || line_thickness <= 0) return; - size = sizeof(*cmd) + sizeof(short) * 2 * (nk_size)point_count; - cmd = (struct nk_command_polyline*) nk_command_buffer_push(b, NK_COMMAND_POLYLINE, size); - if (!cmd) return; - cmd->color = col; - cmd->point_count = (unsigned short)point_count; - cmd->line_thickness = (unsigned short)line_thickness; - for (i = 0; i < point_count; ++i) { - cmd->points[i].x = (short)points[i*2]; - cmd->points[i].y = (short)points[i*2+1]; - } -} -NK_API void -nk_draw_image(struct nk_command_buffer *b, struct nk_rect r, - const struct nk_image *img, struct nk_color col) -{ - struct nk_command_image *cmd; - NK_ASSERT(b); - if (!b) return; - if (b->use_clipping) { - const struct nk_rect *c = &b->clip; - if (c->w == 0 || c->h == 0 || !NK_INTERSECT(r.x, r.y, r.w, r.h, c->x, c->y, c->w, c->h)) - return; - } - - cmd = (struct nk_command_image*) - nk_command_buffer_push(b, NK_COMMAND_IMAGE, sizeof(*cmd)); - if (!cmd) return; - cmd->x = (short)r.x; - cmd->y = (short)r.y; - cmd->w = (unsigned short)NK_MAX(0, r.w); - cmd->h = (unsigned short)NK_MAX(0, r.h); - cmd->img = *img; - cmd->col = col; -} -NK_API void -nk_draw_nine_slice(struct nk_command_buffer *b, struct nk_rect r, - const struct nk_nine_slice *slc, struct nk_color col) -{ - struct nk_image img; - const struct nk_image *slcimg = (const struct nk_image*)slc; - nk_ushort rgnX, rgnY, rgnW, rgnH; - rgnX = slcimg->region[0]; - rgnY = slcimg->region[1]; - rgnW = slcimg->region[2]; - rgnH = slcimg->region[3]; - - /* top-left */ - img.handle = slcimg->handle; - img.w = slcimg->w; - img.h = slcimg->h; - img.region[0] = rgnX; - img.region[1] = rgnY; - img.region[2] = slc->l; - img.region[3] = slc->t; - - nk_draw_image(b, - nk_rect(r.x, r.y, (float)slc->l, (float)slc->t), - &img, col); - -#define IMG_RGN(x, y, w, h) img.region[0] = (nk_ushort)(x); img.region[1] = (nk_ushort)(y); img.region[2] = (nk_ushort)(w); img.region[3] = (nk_ushort)(h); - - /* top-center */ - IMG_RGN(rgnX + slc->l, rgnY, rgnW - slc->l - slc->r, slc->t); - nk_draw_image(b, - nk_rect(r.x + (float)slc->l, r.y, (float)(r.w - slc->l - slc->r), (float)slc->t), - &img, col); - - /* top-right */ - IMG_RGN(rgnX + rgnW - slc->r, rgnY, slc->r, slc->t); - nk_draw_image(b, - nk_rect(r.x + r.w - (float)slc->r, r.y, (float)slc->r, (float)slc->t), - &img, col); - - /* center-left */ - IMG_RGN(rgnX, rgnY + slc->t, slc->l, rgnH - slc->t - slc->b); - nk_draw_image(b, - nk_rect(r.x, r.y + (float)slc->t, (float)slc->l, (float)(r.h - slc->t - slc->b)), - &img, col); - - /* center */ - IMG_RGN(rgnX + slc->l, rgnY + slc->t, rgnW - slc->l - slc->r, rgnH - slc->t - slc->b); - nk_draw_image(b, - nk_rect(r.x + (float)slc->l, r.y + (float)slc->t, (float)(r.w - slc->l - slc->r), (float)(r.h - slc->t - slc->b)), - &img, col); - - /* center-right */ - IMG_RGN(rgnX + rgnW - slc->r, rgnY + slc->t, slc->r, rgnH - slc->t - slc->b); - nk_draw_image(b, - nk_rect(r.x + r.w - (float)slc->r, r.y + (float)slc->t, (float)slc->r, (float)(r.h - slc->t - slc->b)), - &img, col); - - /* bottom-left */ - IMG_RGN(rgnX, rgnY + rgnH - slc->b, slc->l, slc->b); - nk_draw_image(b, - nk_rect(r.x, r.y + r.h - (float)slc->b, (float)slc->l, (float)slc->b), - &img, col); - - /* bottom-center */ - IMG_RGN(rgnX + slc->l, rgnY + rgnH - slc->b, rgnW - slc->l - slc->r, slc->b); - nk_draw_image(b, - nk_rect(r.x + (float)slc->l, r.y + r.h - (float)slc->b, (float)(r.w - slc->l - slc->r), (float)slc->b), - &img, col); - - /* bottom-right */ - IMG_RGN(rgnX + rgnW - slc->r, rgnY + rgnH - slc->b, slc->r, slc->b); - nk_draw_image(b, - nk_rect(r.x + r.w - (float)slc->r, r.y + r.h - (float)slc->b, (float)slc->r, (float)slc->b), - &img, col); - -#undef IMG_RGN -} -NK_API void -nk_push_custom(struct nk_command_buffer *b, struct nk_rect r, - nk_command_custom_callback cb, nk_handle usr) -{ - struct nk_command_custom *cmd; - NK_ASSERT(b); - if (!b) return; - if (b->use_clipping) { - const struct nk_rect *c = &b->clip; - if (c->w == 0 || c->h == 0 || !NK_INTERSECT(r.x, r.y, r.w, r.h, c->x, c->y, c->w, c->h)) - return; - } - - cmd = (struct nk_command_custom*) - nk_command_buffer_push(b, NK_COMMAND_CUSTOM, sizeof(*cmd)); - if (!cmd) return; - cmd->x = (short)r.x; - cmd->y = (short)r.y; - cmd->w = (unsigned short)NK_MAX(0, r.w); - cmd->h = (unsigned short)NK_MAX(0, r.h); - cmd->callback_data = usr; - cmd->callback = cb; -} -NK_API void -nk_draw_text(struct nk_command_buffer *b, struct nk_rect r, - const char *string, int length, const struct nk_user_font *font, - struct nk_color bg, struct nk_color fg) -{ - float text_width = 0; - struct nk_command_text *cmd; - - NK_ASSERT(b); - NK_ASSERT(font); - if (!b || !string || !length || (bg.a == 0 && fg.a == 0)) return; - if (b->use_clipping) { - const struct nk_rect *c = &b->clip; - if (c->w == 0 || c->h == 0 || !NK_INTERSECT(r.x, r.y, r.w, r.h, c->x, c->y, c->w, c->h)) - return; - } - - /* make sure text fits inside bounds */ - text_width = font->width(font->userdata, font->height, string, length); - if (text_width > r.w){ - int glyphs = 0; - float txt_width = (float)text_width; - length = nk_text_clamp(font, string, length, r.w, &glyphs, &txt_width, 0,0); - } - - if (!length) return; - cmd = (struct nk_command_text*) - nk_command_buffer_push(b, NK_COMMAND_TEXT, sizeof(*cmd) + (nk_size)(length + 1)); - if (!cmd) return; - cmd->x = (short)r.x; - cmd->y = (short)r.y; - cmd->w = (unsigned short)r.w; - cmd->h = (unsigned short)r.h; - cmd->background = bg; - cmd->foreground = fg; - cmd->font = font; - cmd->length = length; - cmd->height = font->height; - NK_MEMCPY(cmd->string, string, (nk_size)length); - cmd->string[length] = '\0'; -} - - - - - -/* =============================================================== - * - * VERTEX - * - * ===============================================================*/ -#ifdef NK_INCLUDE_VERTEX_BUFFER_OUTPUT -NK_API void -nk_draw_list_init(struct nk_draw_list *list) -{ - nk_size i = 0; - NK_ASSERT(list); - if (!list) return; - nk_zero(list, sizeof(*list)); - for (i = 0; i < NK_LEN(list->circle_vtx); ++i) { - const float a = ((float)i / (float)NK_LEN(list->circle_vtx)) * 2 * NK_PI; - list->circle_vtx[i].x = (float)NK_COS(a); - list->circle_vtx[i].y = (float)NK_SIN(a); - } -} -NK_API void -nk_draw_list_setup(struct nk_draw_list *canvas, const struct nk_convert_config *config, - struct nk_buffer *cmds, struct nk_buffer *vertices, struct nk_buffer *elements, - enum nk_anti_aliasing line_aa, enum nk_anti_aliasing shape_aa) -{ - NK_ASSERT(canvas); - NK_ASSERT(config); - NK_ASSERT(cmds); - NK_ASSERT(vertices); - NK_ASSERT(elements); - if (!canvas || !config || !cmds || !vertices || !elements) - return; - - canvas->buffer = cmds; - canvas->config = *config; - canvas->elements = elements; - canvas->vertices = vertices; - canvas->line_AA = line_aa; - canvas->shape_AA = shape_aa; - canvas->clip_rect = nk_null_rect; - - canvas->cmd_offset = 0; - canvas->element_count = 0; - canvas->vertex_count = 0; - canvas->cmd_offset = 0; - canvas->cmd_count = 0; - canvas->path_count = 0; -} -NK_API const struct nk_draw_command* -nk__draw_list_begin(const struct nk_draw_list *canvas, const struct nk_buffer *buffer) -{ - nk_byte *memory; - nk_size offset; - const struct nk_draw_command *cmd; - - NK_ASSERT(buffer); - if (!buffer || !buffer->size || !canvas->cmd_count) - return 0; - - memory = (nk_byte*)buffer->memory.ptr; - offset = buffer->memory.size - canvas->cmd_offset; - cmd = nk_ptr_add(const struct nk_draw_command, memory, offset); - return cmd; -} -NK_API const struct nk_draw_command* -nk__draw_list_end(const struct nk_draw_list *canvas, const struct nk_buffer *buffer) -{ - nk_size size; - nk_size offset; - nk_byte *memory; - const struct nk_draw_command *end; - - NK_ASSERT(buffer); - NK_ASSERT(canvas); - if (!buffer || !canvas) - return 0; - - memory = (nk_byte*)buffer->memory.ptr; - size = buffer->memory.size; - offset = size - canvas->cmd_offset; - end = nk_ptr_add(const struct nk_draw_command, memory, offset); - end -= (canvas->cmd_count-1); - return end; -} -NK_API const struct nk_draw_command* -nk__draw_list_next(const struct nk_draw_command *cmd, - const struct nk_buffer *buffer, const struct nk_draw_list *canvas) -{ - const struct nk_draw_command *end; - NK_ASSERT(buffer); - NK_ASSERT(canvas); - if (!cmd || !buffer || !canvas) - return 0; - - end = nk__draw_list_end(canvas, buffer); - if (cmd <= end) return 0; - return (cmd-1); -} -NK_INTERN struct nk_vec2* -nk_draw_list_alloc_path(struct nk_draw_list *list, int count) -{ - struct nk_vec2 *points; - NK_STORAGE const nk_size point_align = NK_ALIGNOF(struct nk_vec2); - NK_STORAGE const nk_size point_size = sizeof(struct nk_vec2); - points = (struct nk_vec2*) - nk_buffer_alloc(list->buffer, NK_BUFFER_FRONT, - point_size * (nk_size)count, point_align); - - if (!points) return 0; - if (!list->path_offset) { - void *memory = nk_buffer_memory(list->buffer); - list->path_offset = (unsigned int)((nk_byte*)points - (nk_byte*)memory); - } - list->path_count += (unsigned int)count; - return points; -} -NK_INTERN struct nk_vec2 -nk_draw_list_path_last(struct nk_draw_list *list) -{ - void *memory; - struct nk_vec2 *point; - NK_ASSERT(list->path_count); - memory = nk_buffer_memory(list->buffer); - point = nk_ptr_add(struct nk_vec2, memory, list->path_offset); - point += (list->path_count-1); - return *point; -} -NK_INTERN struct nk_draw_command* -nk_draw_list_push_command(struct nk_draw_list *list, struct nk_rect clip, - nk_handle texture) -{ - NK_STORAGE const nk_size cmd_align = NK_ALIGNOF(struct nk_draw_command); - NK_STORAGE const nk_size cmd_size = sizeof(struct nk_draw_command); - struct nk_draw_command *cmd; - - NK_ASSERT(list); - cmd = (struct nk_draw_command*) - nk_buffer_alloc(list->buffer, NK_BUFFER_BACK, cmd_size, cmd_align); - - if (!cmd) return 0; - if (!list->cmd_count) { - nk_byte *memory = (nk_byte*)nk_buffer_memory(list->buffer); - nk_size total = nk_buffer_total(list->buffer); - memory = nk_ptr_add(nk_byte, memory, total); - list->cmd_offset = (nk_size)(memory - (nk_byte*)cmd); - } - - cmd->elem_count = 0; - cmd->clip_rect = clip; - cmd->texture = texture; -#ifdef NK_INCLUDE_COMMAND_USERDATA - cmd->userdata = list->userdata; -#endif - - list->cmd_count++; - list->clip_rect = clip; - return cmd; -} -NK_INTERN struct nk_draw_command* -nk_draw_list_command_last(struct nk_draw_list *list) -{ - void *memory; - nk_size size; - struct nk_draw_command *cmd; - NK_ASSERT(list->cmd_count); - - memory = nk_buffer_memory(list->buffer); - size = nk_buffer_total(list->buffer); - cmd = nk_ptr_add(struct nk_draw_command, memory, size - list->cmd_offset); - return (cmd - (list->cmd_count-1)); -} -NK_INTERN void -nk_draw_list_add_clip(struct nk_draw_list *list, struct nk_rect rect) -{ - NK_ASSERT(list); - if (!list) return; - if (!list->cmd_count) { - nk_draw_list_push_command(list, rect, list->config.tex_null.texture); - } else { - struct nk_draw_command *prev = nk_draw_list_command_last(list); - if (prev->elem_count == 0) - prev->clip_rect = rect; - nk_draw_list_push_command(list, rect, prev->texture); - } -} -NK_INTERN void -nk_draw_list_push_image(struct nk_draw_list *list, nk_handle texture) -{ - NK_ASSERT(list); - if (!list) return; - if (!list->cmd_count) { - nk_draw_list_push_command(list, nk_null_rect, texture); - } else { - struct nk_draw_command *prev = nk_draw_list_command_last(list); - if (prev->elem_count == 0) { - prev->texture = texture; - #ifdef NK_INCLUDE_COMMAND_USERDATA - prev->userdata = list->userdata; - #endif - } else if (prev->texture.id != texture.id - #ifdef NK_INCLUDE_COMMAND_USERDATA - || prev->userdata.id != list->userdata.id - #endif - ) nk_draw_list_push_command(list, prev->clip_rect, texture); - } -} -#ifdef NK_INCLUDE_COMMAND_USERDATA -NK_API void -nk_draw_list_push_userdata(struct nk_draw_list *list, nk_handle userdata) -{ - list->userdata = userdata; -} -#endif -NK_INTERN void* -nk_draw_list_alloc_vertices(struct nk_draw_list *list, nk_size count) -{ - void *vtx; - NK_ASSERT(list); - if (!list) return 0; - vtx = nk_buffer_alloc(list->vertices, NK_BUFFER_FRONT, - list->config.vertex_size*count, list->config.vertex_alignment); - if (!vtx) return 0; - list->vertex_count += (unsigned int)count; - - /* This assert triggers because your are drawing a lot of stuff and nuklear - * defined `nk_draw_index` as `nk_ushort` to safe space be default. - * - * So you reached the maximum number of indices or rather vertexes. - * To solve this issue please change typedef `nk_draw_index` to `nk_uint` - * and don't forget to specify the new element size in your drawing - * backend (OpenGL, DirectX, ...). For example in OpenGL for `glDrawElements` - * instead of specifying `GL_UNSIGNED_SHORT` you have to define `GL_UNSIGNED_INT`. - * Sorry for the inconvenience. */ - if(sizeof(nk_draw_index)==2) NK_ASSERT((list->vertex_count < NK_USHORT_MAX && - "To many vertices for 16-bit vertex indices. Please read comment above on how to solve this problem")); - return vtx; -} -NK_INTERN nk_draw_index* -nk_draw_list_alloc_elements(struct nk_draw_list *list, nk_size count) -{ - nk_draw_index *ids; - struct nk_draw_command *cmd; - NK_STORAGE const nk_size elem_align = NK_ALIGNOF(nk_draw_index); - NK_STORAGE const nk_size elem_size = sizeof(nk_draw_index); - NK_ASSERT(list); - if (!list) return 0; - - ids = (nk_draw_index*) - nk_buffer_alloc(list->elements, NK_BUFFER_FRONT, elem_size*count, elem_align); - if (!ids) return 0; - cmd = nk_draw_list_command_last(list); - list->element_count += (unsigned int)count; - cmd->elem_count += (unsigned int)count; - return ids; -} -NK_INTERN int -nk_draw_vertex_layout_element_is_end_of_layout( - const struct nk_draw_vertex_layout_element *element) -{ - return (element->attribute == NK_VERTEX_ATTRIBUTE_COUNT || - element->format == NK_FORMAT_COUNT); -} -NK_INTERN void -nk_draw_vertex_color(void *attr, const float *vals, - enum nk_draw_vertex_layout_format format) -{ - /* if this triggers you tried to provide a value format for a color */ - float val[4]; - NK_ASSERT(format >= NK_FORMAT_COLOR_BEGIN); - NK_ASSERT(format <= NK_FORMAT_COLOR_END); - if (format < NK_FORMAT_COLOR_BEGIN || format > NK_FORMAT_COLOR_END) return; - - val[0] = NK_SATURATE(vals[0]); - val[1] = NK_SATURATE(vals[1]); - val[2] = NK_SATURATE(vals[2]); - val[3] = NK_SATURATE(vals[3]); - - switch (format) { - default: NK_ASSERT(0 && "Invalid vertex layout color format"); break; - case NK_FORMAT_R8G8B8A8: - case NK_FORMAT_R8G8B8: { - struct nk_color col = nk_rgba_fv(val); - NK_MEMCPY(attr, &col.r, sizeof(col)); - } break; - case NK_FORMAT_B8G8R8A8: { - struct nk_color col = nk_rgba_fv(val); - struct nk_color bgra = nk_rgba(col.b, col.g, col.r, col.a); - NK_MEMCPY(attr, &bgra, sizeof(bgra)); - } break; - case NK_FORMAT_R16G15B16: { - nk_ushort col[3]; - col[0] = (nk_ushort)(val[0]*(float)NK_USHORT_MAX); - col[1] = (nk_ushort)(val[1]*(float)NK_USHORT_MAX); - col[2] = (nk_ushort)(val[2]*(float)NK_USHORT_MAX); - NK_MEMCPY(attr, col, sizeof(col)); - } break; - case NK_FORMAT_R16G15B16A16: { - nk_ushort col[4]; - col[0] = (nk_ushort)(val[0]*(float)NK_USHORT_MAX); - col[1] = (nk_ushort)(val[1]*(float)NK_USHORT_MAX); - col[2] = (nk_ushort)(val[2]*(float)NK_USHORT_MAX); - col[3] = (nk_ushort)(val[3]*(float)NK_USHORT_MAX); - NK_MEMCPY(attr, col, sizeof(col)); - } break; - case NK_FORMAT_R32G32B32: { - nk_uint col[3]; - col[0] = (nk_uint)(val[0]*(float)NK_UINT_MAX); - col[1] = (nk_uint)(val[1]*(float)NK_UINT_MAX); - col[2] = (nk_uint)(val[2]*(float)NK_UINT_MAX); - NK_MEMCPY(attr, col, sizeof(col)); - } break; - case NK_FORMAT_R32G32B32A32: { - nk_uint col[4]; - col[0] = (nk_uint)(val[0]*(float)NK_UINT_MAX); - col[1] = (nk_uint)(val[1]*(float)NK_UINT_MAX); - col[2] = (nk_uint)(val[2]*(float)NK_UINT_MAX); - col[3] = (nk_uint)(val[3]*(float)NK_UINT_MAX); - NK_MEMCPY(attr, col, sizeof(col)); - } break; - case NK_FORMAT_R32G32B32A32_FLOAT: - NK_MEMCPY(attr, val, sizeof(float)*4); - break; - case NK_FORMAT_R32G32B32A32_DOUBLE: { - double col[4]; - col[0] = (double)val[0]; - col[1] = (double)val[1]; - col[2] = (double)val[2]; - col[3] = (double)val[3]; - NK_MEMCPY(attr, col, sizeof(col)); - } break; - case NK_FORMAT_RGB32: - case NK_FORMAT_RGBA32: { - struct nk_color col = nk_rgba_fv(val); - nk_uint color = nk_color_u32(col); - NK_MEMCPY(attr, &color, sizeof(color)); - } break; } -} -NK_INTERN void -nk_draw_vertex_element(void *dst, const float *values, int value_count, - enum nk_draw_vertex_layout_format format) -{ - int value_index; - void *attribute = dst; - /* if this triggers you tried to provide a color format for a value */ - NK_ASSERT(format < NK_FORMAT_COLOR_BEGIN); - if (format >= NK_FORMAT_COLOR_BEGIN && format <= NK_FORMAT_COLOR_END) return; - for (value_index = 0; value_index < value_count; ++value_index) { - switch (format) { - default: NK_ASSERT(0 && "invalid vertex layout format"); break; - case NK_FORMAT_SCHAR: { - char value = (char)NK_CLAMP((float)NK_SCHAR_MIN, values[value_index], (float)NK_SCHAR_MAX); - NK_MEMCPY(attribute, &value, sizeof(value)); - attribute = (void*)((char*)attribute + sizeof(char)); - } break; - case NK_FORMAT_SSHORT: { - nk_short value = (nk_short)NK_CLAMP((float)NK_SSHORT_MIN, values[value_index], (float)NK_SSHORT_MAX); - NK_MEMCPY(attribute, &value, sizeof(value)); - attribute = (void*)((char*)attribute + sizeof(value)); - } break; - case NK_FORMAT_SINT: { - nk_int value = (nk_int)NK_CLAMP((float)NK_SINT_MIN, values[value_index], (float)NK_SINT_MAX); - NK_MEMCPY(attribute, &value, sizeof(value)); - attribute = (void*)((char*)attribute + sizeof(nk_int)); - } break; - case NK_FORMAT_UCHAR: { - unsigned char value = (unsigned char)NK_CLAMP((float)NK_UCHAR_MIN, values[value_index], (float)NK_UCHAR_MAX); - NK_MEMCPY(attribute, &value, sizeof(value)); - attribute = (void*)((char*)attribute + sizeof(unsigned char)); - } break; - case NK_FORMAT_USHORT: { - nk_ushort value = (nk_ushort)NK_CLAMP((float)NK_USHORT_MIN, values[value_index], (float)NK_USHORT_MAX); - NK_MEMCPY(attribute, &value, sizeof(value)); - attribute = (void*)((char*)attribute + sizeof(value)); - } break; - case NK_FORMAT_UINT: { - nk_uint value = (nk_uint)NK_CLAMP((float)NK_UINT_MIN, values[value_index], (float)NK_UINT_MAX); - NK_MEMCPY(attribute, &value, sizeof(value)); - attribute = (void*)((char*)attribute + sizeof(nk_uint)); - } break; - case NK_FORMAT_FLOAT: - NK_MEMCPY(attribute, &values[value_index], sizeof(values[value_index])); - attribute = (void*)((char*)attribute + sizeof(float)); - break; - case NK_FORMAT_DOUBLE: { - double value = (double)values[value_index]; - NK_MEMCPY(attribute, &value, sizeof(value)); - attribute = (void*)((char*)attribute + sizeof(double)); - } break; - } - } -} -NK_INTERN void* -nk_draw_vertex(void *dst, const struct nk_convert_config *config, - struct nk_vec2 pos, struct nk_vec2 uv, struct nk_colorf color) -{ - void *result = (void*)((char*)dst + config->vertex_size); - const struct nk_draw_vertex_layout_element *elem_iter = config->vertex_layout; - while (!nk_draw_vertex_layout_element_is_end_of_layout(elem_iter)) { - void *address = (void*)((char*)dst + elem_iter->offset); - switch (elem_iter->attribute) { - case NK_VERTEX_ATTRIBUTE_COUNT: - default: NK_ASSERT(0 && "wrong element attribute"); break; - case NK_VERTEX_POSITION: nk_draw_vertex_element(address, &pos.x, 2, elem_iter->format); break; - case NK_VERTEX_TEXCOORD: nk_draw_vertex_element(address, &uv.x, 2, elem_iter->format); break; - case NK_VERTEX_COLOR: nk_draw_vertex_color(address, &color.r, elem_iter->format); break; - } - elem_iter++; - } - return result; -} -NK_API void -nk_draw_list_stroke_poly_line(struct nk_draw_list *list, const struct nk_vec2 *points, - const unsigned int points_count, struct nk_color color, enum nk_draw_list_stroke closed, - float thickness, enum nk_anti_aliasing aliasing) -{ - nk_size count; - int thick_line; - struct nk_colorf col; - struct nk_colorf col_trans; - NK_ASSERT(list); - if (!list || points_count < 2) return; - - color.a = (nk_byte)((float)color.a * list->config.global_alpha); - count = points_count; - if (!closed) count = points_count-1; - thick_line = thickness > 1.0f; - -#ifdef NK_INCLUDE_COMMAND_USERDATA - nk_draw_list_push_userdata(list, list->userdata); -#endif - - color.a = (nk_byte)((float)color.a * list->config.global_alpha); - nk_color_fv(&col.r, color); - col_trans = col; - col_trans.a = 0; - - if (aliasing == NK_ANTI_ALIASING_ON) { - /* ANTI-ALIASED STROKE */ - const float AA_SIZE = 1.0f; - NK_STORAGE const nk_size pnt_align = NK_ALIGNOF(struct nk_vec2); - NK_STORAGE const nk_size pnt_size = sizeof(struct nk_vec2); - - /* allocate vertices and elements */ - nk_size i1 = 0; - nk_size vertex_offset; - nk_size index = list->vertex_count; - - const nk_size idx_count = (thick_line) ? (count * 18) : (count * 12); - const nk_size vtx_count = (thick_line) ? (points_count * 4): (points_count *3); - - void *vtx = nk_draw_list_alloc_vertices(list, vtx_count); - nk_draw_index *ids = nk_draw_list_alloc_elements(list, idx_count); - - nk_size size; - struct nk_vec2 *normals, *temp; - if (!vtx || !ids) return; - - /* temporary allocate normals + points */ - vertex_offset = (nk_size)((nk_byte*)vtx - (nk_byte*)list->vertices->memory.ptr); - nk_buffer_mark(list->vertices, NK_BUFFER_FRONT); - size = pnt_size * ((thick_line) ? 5 : 3) * points_count; - normals = (struct nk_vec2*) nk_buffer_alloc(list->vertices, NK_BUFFER_FRONT, size, pnt_align); - if (!normals) return; - temp = normals + points_count; - - /* make sure vertex pointer is still correct */ - vtx = (void*)((nk_byte*)list->vertices->memory.ptr + vertex_offset); - - /* calculate normals */ - for (i1 = 0; i1 < count; ++i1) { - const nk_size i2 = ((i1 + 1) == points_count) ? 0 : (i1 + 1); - struct nk_vec2 diff = nk_vec2_sub(points[i2], points[i1]); - float len; - - /* vec2 inverted length */ - len = nk_vec2_len_sqr(diff); - if (len != 0.0f) - len = NK_INV_SQRT(len); - else len = 1.0f; - - diff = nk_vec2_muls(diff, len); - normals[i1].x = diff.y; - normals[i1].y = -diff.x; - } - - if (!closed) - normals[points_count-1] = normals[points_count-2]; - - if (!thick_line) { - nk_size idx1, i; - if (!closed) { - struct nk_vec2 d; - temp[0] = nk_vec2_add(points[0], nk_vec2_muls(normals[0], AA_SIZE)); - temp[1] = nk_vec2_sub(points[0], nk_vec2_muls(normals[0], AA_SIZE)); - d = nk_vec2_muls(normals[points_count-1], AA_SIZE); - temp[(points_count-1) * 2 + 0] = nk_vec2_add(points[points_count-1], d); - temp[(points_count-1) * 2 + 1] = nk_vec2_sub(points[points_count-1], d); - } - - /* fill elements */ - idx1 = index; - for (i1 = 0; i1 < count; i1++) { - struct nk_vec2 dm; - float dmr2; - nk_size i2 = ((i1 + 1) == points_count) ? 0 : (i1 + 1); - nk_size idx2 = ((i1+1) == points_count) ? index: (idx1 + 3); - - /* average normals */ - dm = nk_vec2_muls(nk_vec2_add(normals[i1], normals[i2]), 0.5f); - dmr2 = dm.x * dm.x + dm.y* dm.y; - if (dmr2 > 0.000001f) { - float scale = 1.0f/dmr2; - scale = NK_MIN(100.0f, scale); - dm = nk_vec2_muls(dm, scale); - } - - dm = nk_vec2_muls(dm, AA_SIZE); - temp[i2*2+0] = nk_vec2_add(points[i2], dm); - temp[i2*2+1] = nk_vec2_sub(points[i2], dm); - - ids[0] = (nk_draw_index)(idx2 + 0); ids[1] = (nk_draw_index)(idx1+0); - ids[2] = (nk_draw_index)(idx1 + 2); ids[3] = (nk_draw_index)(idx1+2); - ids[4] = (nk_draw_index)(idx2 + 2); ids[5] = (nk_draw_index)(idx2+0); - ids[6] = (nk_draw_index)(idx2 + 1); ids[7] = (nk_draw_index)(idx1+1); - ids[8] = (nk_draw_index)(idx1 + 0); ids[9] = (nk_draw_index)(idx1+0); - ids[10]= (nk_draw_index)(idx2 + 0); ids[11]= (nk_draw_index)(idx2+1); - ids += 12; - idx1 = idx2; - } - - /* fill vertices */ - for (i = 0; i < points_count; ++i) { - const struct nk_vec2 uv = list->config.tex_null.uv; - vtx = nk_draw_vertex(vtx, &list->config, points[i], uv, col); - vtx = nk_draw_vertex(vtx, &list->config, temp[i*2+0], uv, col_trans); - vtx = nk_draw_vertex(vtx, &list->config, temp[i*2+1], uv, col_trans); - } - } else { - nk_size idx1, i; - const float half_inner_thickness = (thickness - AA_SIZE) * 0.5f; - if (!closed) { - struct nk_vec2 d1 = nk_vec2_muls(normals[0], half_inner_thickness + AA_SIZE); - struct nk_vec2 d2 = nk_vec2_muls(normals[0], half_inner_thickness); - - temp[0] = nk_vec2_add(points[0], d1); - temp[1] = nk_vec2_add(points[0], d2); - temp[2] = nk_vec2_sub(points[0], d2); - temp[3] = nk_vec2_sub(points[0], d1); - - d1 = nk_vec2_muls(normals[points_count-1], half_inner_thickness + AA_SIZE); - d2 = nk_vec2_muls(normals[points_count-1], half_inner_thickness); - - temp[(points_count-1)*4+0] = nk_vec2_add(points[points_count-1], d1); - temp[(points_count-1)*4+1] = nk_vec2_add(points[points_count-1], d2); - temp[(points_count-1)*4+2] = nk_vec2_sub(points[points_count-1], d2); - temp[(points_count-1)*4+3] = nk_vec2_sub(points[points_count-1], d1); - } - - /* add all elements */ - idx1 = index; - for (i1 = 0; i1 < count; ++i1) { - struct nk_vec2 dm_out, dm_in; - const nk_size i2 = ((i1+1) == points_count) ? 0: (i1 + 1); - nk_size idx2 = ((i1+1) == points_count) ? index: (idx1 + 4); - - /* average normals */ - struct nk_vec2 dm = nk_vec2_muls(nk_vec2_add(normals[i1], normals[i2]), 0.5f); - float dmr2 = dm.x * dm.x + dm.y* dm.y; - if (dmr2 > 0.000001f) { - float scale = 1.0f/dmr2; - scale = NK_MIN(100.0f, scale); - dm = nk_vec2_muls(dm, scale); - } - - dm_out = nk_vec2_muls(dm, ((half_inner_thickness) + AA_SIZE)); - dm_in = nk_vec2_muls(dm, half_inner_thickness); - temp[i2*4+0] = nk_vec2_add(points[i2], dm_out); - temp[i2*4+1] = nk_vec2_add(points[i2], dm_in); - temp[i2*4+2] = nk_vec2_sub(points[i2], dm_in); - temp[i2*4+3] = nk_vec2_sub(points[i2], dm_out); - - /* add indexes */ - ids[0] = (nk_draw_index)(idx2 + 1); ids[1] = (nk_draw_index)(idx1+1); - ids[2] = (nk_draw_index)(idx1 + 2); ids[3] = (nk_draw_index)(idx1+2); - ids[4] = (nk_draw_index)(idx2 + 2); ids[5] = (nk_draw_index)(idx2+1); - ids[6] = (nk_draw_index)(idx2 + 1); ids[7] = (nk_draw_index)(idx1+1); - ids[8] = (nk_draw_index)(idx1 + 0); ids[9] = (nk_draw_index)(idx1+0); - ids[10]= (nk_draw_index)(idx2 + 0); ids[11] = (nk_draw_index)(idx2+1); - ids[12]= (nk_draw_index)(idx2 + 2); ids[13] = (nk_draw_index)(idx1+2); - ids[14]= (nk_draw_index)(idx1 + 3); ids[15] = (nk_draw_index)(idx1+3); - ids[16]= (nk_draw_index)(idx2 + 3); ids[17] = (nk_draw_index)(idx2+2); - ids += 18; - idx1 = idx2; - } - - /* add vertices */ - for (i = 0; i < points_count; ++i) { - const struct nk_vec2 uv = list->config.tex_null.uv; - vtx = nk_draw_vertex(vtx, &list->config, temp[i*4+0], uv, col_trans); - vtx = nk_draw_vertex(vtx, &list->config, temp[i*4+1], uv, col); - vtx = nk_draw_vertex(vtx, &list->config, temp[i*4+2], uv, col); - vtx = nk_draw_vertex(vtx, &list->config, temp[i*4+3], uv, col_trans); - } - } - /* free temporary normals + points */ - nk_buffer_reset(list->vertices, NK_BUFFER_FRONT); - } else { - /* NON ANTI-ALIASED STROKE */ - nk_size i1 = 0; - nk_size idx = list->vertex_count; - const nk_size idx_count = count * 6; - const nk_size vtx_count = count * 4; - void *vtx = nk_draw_list_alloc_vertices(list, vtx_count); - nk_draw_index *ids = nk_draw_list_alloc_elements(list, idx_count); - if (!vtx || !ids) return; - - for (i1 = 0; i1 < count; ++i1) { - float dx, dy; - const struct nk_vec2 uv = list->config.tex_null.uv; - const nk_size i2 = ((i1+1) == points_count) ? 0 : i1 + 1; - const struct nk_vec2 p1 = points[i1]; - const struct nk_vec2 p2 = points[i2]; - struct nk_vec2 diff = nk_vec2_sub(p2, p1); - float len; - - /* vec2 inverted length */ - len = nk_vec2_len_sqr(diff); - if (len != 0.0f) - len = NK_INV_SQRT(len); - else len = 1.0f; - diff = nk_vec2_muls(diff, len); - - /* add vertices */ - dx = diff.x * (thickness * 0.5f); - dy = diff.y * (thickness * 0.5f); - - vtx = nk_draw_vertex(vtx, &list->config, nk_vec2(p1.x + dy, p1.y - dx), uv, col); - vtx = nk_draw_vertex(vtx, &list->config, nk_vec2(p2.x + dy, p2.y - dx), uv, col); - vtx = nk_draw_vertex(vtx, &list->config, nk_vec2(p2.x - dy, p2.y + dx), uv, col); - vtx = nk_draw_vertex(vtx, &list->config, nk_vec2(p1.x - dy, p1.y + dx), uv, col); - - ids[0] = (nk_draw_index)(idx+0); ids[1] = (nk_draw_index)(idx+1); - ids[2] = (nk_draw_index)(idx+2); ids[3] = (nk_draw_index)(idx+0); - ids[4] = (nk_draw_index)(idx+2); ids[5] = (nk_draw_index)(idx+3); - - ids += 6; - idx += 4; - } - } -} -NK_API void -nk_draw_list_fill_poly_convex(struct nk_draw_list *list, - const struct nk_vec2 *points, const unsigned int points_count, - struct nk_color color, enum nk_anti_aliasing aliasing) -{ - struct nk_colorf col; - struct nk_colorf col_trans; - - NK_STORAGE const nk_size pnt_align = NK_ALIGNOF(struct nk_vec2); - NK_STORAGE const nk_size pnt_size = sizeof(struct nk_vec2); - NK_ASSERT(list); - if (!list || points_count < 3) return; - -#ifdef NK_INCLUDE_COMMAND_USERDATA - nk_draw_list_push_userdata(list, list->userdata); -#endif - - color.a = (nk_byte)((float)color.a * list->config.global_alpha); - nk_color_fv(&col.r, color); - col_trans = col; - col_trans.a = 0; - - if (aliasing == NK_ANTI_ALIASING_ON) { - nk_size i = 0; - nk_size i0 = 0; - nk_size i1 = 0; - - const float AA_SIZE = 1.0f; - nk_size vertex_offset = 0; - nk_size index = list->vertex_count; - - const nk_size idx_count = (points_count-2)*3 + points_count*6; - const nk_size vtx_count = (points_count*2); - - void *vtx = nk_draw_list_alloc_vertices(list, vtx_count); - nk_draw_index *ids = nk_draw_list_alloc_elements(list, idx_count); - - nk_size size = 0; - struct nk_vec2 *normals = 0; - unsigned int vtx_inner_idx = (unsigned int)(index + 0); - unsigned int vtx_outer_idx = (unsigned int)(index + 1); - if (!vtx || !ids) return; - - /* temporary allocate normals */ - vertex_offset = (nk_size)((nk_byte*)vtx - (nk_byte*)list->vertices->memory.ptr); - nk_buffer_mark(list->vertices, NK_BUFFER_FRONT); - size = pnt_size * points_count; - normals = (struct nk_vec2*) nk_buffer_alloc(list->vertices, NK_BUFFER_FRONT, size, pnt_align); - if (!normals) return; - vtx = (void*)((nk_byte*)list->vertices->memory.ptr + vertex_offset); - - /* add elements */ - for (i = 2; i < points_count; i++) { - ids[0] = (nk_draw_index)(vtx_inner_idx); - ids[1] = (nk_draw_index)(vtx_inner_idx + ((i-1) << 1)); - ids[2] = (nk_draw_index)(vtx_inner_idx + (i << 1)); - ids += 3; - } - - /* compute normals */ - for (i0 = points_count-1, i1 = 0; i1 < points_count; i0 = i1++) { - struct nk_vec2 p0 = points[i0]; - struct nk_vec2 p1 = points[i1]; - struct nk_vec2 diff = nk_vec2_sub(p1, p0); - - /* vec2 inverted length */ - float len = nk_vec2_len_sqr(diff); - if (len != 0.0f) - len = NK_INV_SQRT(len); - else len = 1.0f; - diff = nk_vec2_muls(diff, len); - - normals[i0].x = diff.y; - normals[i0].y = -diff.x; - } - - /* add vertices + indexes */ - for (i0 = points_count-1, i1 = 0; i1 < points_count; i0 = i1++) { - const struct nk_vec2 uv = list->config.tex_null.uv; - struct nk_vec2 n0 = normals[i0]; - struct nk_vec2 n1 = normals[i1]; - struct nk_vec2 dm = nk_vec2_muls(nk_vec2_add(n0, n1), 0.5f); - float dmr2 = dm.x*dm.x + dm.y*dm.y; - if (dmr2 > 0.000001f) { - float scale = 1.0f / dmr2; - scale = NK_MIN(scale, 100.0f); - dm = nk_vec2_muls(dm, scale); - } - dm = nk_vec2_muls(dm, AA_SIZE * 0.5f); - - /* add vertices */ - vtx = nk_draw_vertex(vtx, &list->config, nk_vec2_sub(points[i1], dm), uv, col); - vtx = nk_draw_vertex(vtx, &list->config, nk_vec2_add(points[i1], dm), uv, col_trans); - - /* add indexes */ - ids[0] = (nk_draw_index)(vtx_inner_idx+(i1<<1)); - ids[1] = (nk_draw_index)(vtx_inner_idx+(i0<<1)); - ids[2] = (nk_draw_index)(vtx_outer_idx+(i0<<1)); - ids[3] = (nk_draw_index)(vtx_outer_idx+(i0<<1)); - ids[4] = (nk_draw_index)(vtx_outer_idx+(i1<<1)); - ids[5] = (nk_draw_index)(vtx_inner_idx+(i1<<1)); - ids += 6; - } - /* free temporary normals + points */ - nk_buffer_reset(list->vertices, NK_BUFFER_FRONT); - } else { - nk_size i = 0; - nk_size index = list->vertex_count; - const nk_size idx_count = (points_count-2)*3; - const nk_size vtx_count = points_count; - void *vtx = nk_draw_list_alloc_vertices(list, vtx_count); - nk_draw_index *ids = nk_draw_list_alloc_elements(list, idx_count); - - if (!vtx || !ids) return; - for (i = 0; i < vtx_count; ++i) - vtx = nk_draw_vertex(vtx, &list->config, points[i], list->config.tex_null.uv, col); - for (i = 2; i < points_count; ++i) { - ids[0] = (nk_draw_index)index; - ids[1] = (nk_draw_index)(index+ i - 1); - ids[2] = (nk_draw_index)(index+i); - ids += 3; - } - } -} -NK_API void -nk_draw_list_path_clear(struct nk_draw_list *list) -{ - NK_ASSERT(list); - if (!list) return; - nk_buffer_reset(list->buffer, NK_BUFFER_FRONT); - list->path_count = 0; - list->path_offset = 0; -} -NK_API void -nk_draw_list_path_line_to(struct nk_draw_list *list, struct nk_vec2 pos) -{ - struct nk_vec2 *points = 0; - struct nk_draw_command *cmd = 0; - NK_ASSERT(list); - if (!list) return; - if (!list->cmd_count) - nk_draw_list_add_clip(list, nk_null_rect); - - cmd = nk_draw_list_command_last(list); - if (cmd && cmd->texture.ptr != list->config.tex_null.texture.ptr) - nk_draw_list_push_image(list, list->config.tex_null.texture); - - points = nk_draw_list_alloc_path(list, 1); - if (!points) return; - points[0] = pos; -} -NK_API void -nk_draw_list_path_arc_to_fast(struct nk_draw_list *list, struct nk_vec2 center, - float radius, int a_min, int a_max) -{ - int a = 0; - NK_ASSERT(list); - if (!list) return; - if (a_min <= a_max) { - for (a = a_min; a <= a_max; a++) { - const struct nk_vec2 c = list->circle_vtx[(nk_size)a % NK_LEN(list->circle_vtx)]; - const float x = center.x + c.x * radius; - const float y = center.y + c.y * radius; - nk_draw_list_path_line_to(list, nk_vec2(x, y)); - } - } -} -NK_API void -nk_draw_list_path_arc_to(struct nk_draw_list *list, struct nk_vec2 center, - float radius, float a_min, float a_max, unsigned int segments) -{ - unsigned int i = 0; - NK_ASSERT(list); - if (!list) return; - if (radius == 0.0f) return; - - /* This algorithm for arc drawing relies on these two trigonometric identities[1]: - sin(a + b) = sin(a) * cos(b) + cos(a) * sin(b) - cos(a + b) = cos(a) * cos(b) - sin(a) * sin(b) - - Two coordinates (x, y) of a point on a circle centered on - the origin can be written in polar form as: - x = r * cos(a) - y = r * sin(a) - where r is the radius of the circle, - a is the angle between (x, y) and the origin. - - This allows us to rotate the coordinates around the - origin by an angle b using the following transformation: - x' = r * cos(a + b) = x * cos(b) - y * sin(b) - y' = r * sin(a + b) = y * cos(b) + x * sin(b) - - [1] https://en.wikipedia.org/wiki/List_of_trigonometric_identities#Angle_sum_and_difference_identities - */ - {const float d_angle = (a_max - a_min) / (float)segments; - const float sin_d = (float)NK_SIN(d_angle); - const float cos_d = (float)NK_COS(d_angle); - - float cx = (float)NK_COS(a_min) * radius; - float cy = (float)NK_SIN(a_min) * radius; - for(i = 0; i <= segments; ++i) { - float new_cx, new_cy; - const float x = center.x + cx; - const float y = center.y + cy; - nk_draw_list_path_line_to(list, nk_vec2(x, y)); - - new_cx = cx * cos_d - cy * sin_d; - new_cy = cy * cos_d + cx * sin_d; - cx = new_cx; - cy = new_cy; - }} -} -NK_API void -nk_draw_list_path_rect_to(struct nk_draw_list *list, struct nk_vec2 a, - struct nk_vec2 b, float rounding) -{ - float r; - NK_ASSERT(list); - if (!list) return; - r = rounding; - r = NK_MIN(r, ((b.x-a.x) < 0) ? -(b.x-a.x): (b.x-a.x)); - r = NK_MIN(r, ((b.y-a.y) < 0) ? -(b.y-a.y): (b.y-a.y)); - - if (r == 0.0f) { - nk_draw_list_path_line_to(list, a); - nk_draw_list_path_line_to(list, nk_vec2(b.x,a.y)); - nk_draw_list_path_line_to(list, b); - nk_draw_list_path_line_to(list, nk_vec2(a.x,b.y)); - } else { - nk_draw_list_path_arc_to_fast(list, nk_vec2(a.x + r, a.y + r), r, 6, 9); - nk_draw_list_path_arc_to_fast(list, nk_vec2(b.x - r, a.y + r), r, 9, 12); - nk_draw_list_path_arc_to_fast(list, nk_vec2(b.x - r, b.y - r), r, 0, 3); - nk_draw_list_path_arc_to_fast(list, nk_vec2(a.x + r, b.y - r), r, 3, 6); - } -} -NK_API void -nk_draw_list_path_curve_to(struct nk_draw_list *list, struct nk_vec2 p2, - struct nk_vec2 p3, struct nk_vec2 p4, unsigned int num_segments) -{ - float t_step; - unsigned int i_step; - struct nk_vec2 p1; - - NK_ASSERT(list); - NK_ASSERT(list->path_count); - if (!list || !list->path_count) return; - num_segments = NK_MAX(num_segments, 1); - - p1 = nk_draw_list_path_last(list); - t_step = 1.0f/(float)num_segments; - for (i_step = 1; i_step <= num_segments; ++i_step) { - float t = t_step * (float)i_step; - float u = 1.0f - t; - float w1 = u*u*u; - float w2 = 3*u*u*t; - float w3 = 3*u*t*t; - float w4 = t * t *t; - float x = w1 * p1.x + w2 * p2.x + w3 * p3.x + w4 * p4.x; - float y = w1 * p1.y + w2 * p2.y + w3 * p3.y + w4 * p4.y; - nk_draw_list_path_line_to(list, nk_vec2(x,y)); - } -} -NK_API void -nk_draw_list_path_fill(struct nk_draw_list *list, struct nk_color color) -{ - struct nk_vec2 *points; - NK_ASSERT(list); - if (!list) return; - points = (struct nk_vec2*)nk_buffer_memory(list->buffer); - nk_draw_list_fill_poly_convex(list, points, list->path_count, color, list->config.shape_AA); - nk_draw_list_path_clear(list); -} -NK_API void -nk_draw_list_path_stroke(struct nk_draw_list *list, struct nk_color color, - enum nk_draw_list_stroke closed, float thickness) -{ - struct nk_vec2 *points; - NK_ASSERT(list); - if (!list) return; - points = (struct nk_vec2*)nk_buffer_memory(list->buffer); - nk_draw_list_stroke_poly_line(list, points, list->path_count, color, - closed, thickness, list->config.line_AA); - nk_draw_list_path_clear(list); -} -NK_API void -nk_draw_list_stroke_line(struct nk_draw_list *list, struct nk_vec2 a, - struct nk_vec2 b, struct nk_color col, float thickness) -{ - NK_ASSERT(list); - if (!list || !col.a) return; - if (list->line_AA == NK_ANTI_ALIASING_ON) { - nk_draw_list_path_line_to(list, a); - nk_draw_list_path_line_to(list, b); - } else { - nk_draw_list_path_line_to(list, nk_vec2_sub(a,nk_vec2(0.5f,0.5f))); - nk_draw_list_path_line_to(list, nk_vec2_sub(b,nk_vec2(0.5f,0.5f))); - } - nk_draw_list_path_stroke(list, col, NK_STROKE_OPEN, thickness); -} -NK_API void -nk_draw_list_fill_rect(struct nk_draw_list *list, struct nk_rect rect, - struct nk_color col, float rounding) -{ - NK_ASSERT(list); - if (!list || !col.a) return; - - if (list->line_AA == NK_ANTI_ALIASING_ON) { - nk_draw_list_path_rect_to(list, nk_vec2(rect.x, rect.y), - nk_vec2(rect.x + rect.w, rect.y + rect.h), rounding); - } else { - nk_draw_list_path_rect_to(list, nk_vec2(rect.x-0.5f, rect.y-0.5f), - nk_vec2(rect.x + rect.w, rect.y + rect.h), rounding); - } nk_draw_list_path_fill(list, col); -} -NK_API void -nk_draw_list_stroke_rect(struct nk_draw_list *list, struct nk_rect rect, - struct nk_color col, float rounding, float thickness) -{ - NK_ASSERT(list); - if (!list || !col.a) return; - if (list->line_AA == NK_ANTI_ALIASING_ON) { - nk_draw_list_path_rect_to(list, nk_vec2(rect.x, rect.y), - nk_vec2(rect.x + rect.w, rect.y + rect.h), rounding); - } else { - nk_draw_list_path_rect_to(list, nk_vec2(rect.x-0.5f, rect.y-0.5f), - nk_vec2(rect.x + rect.w, rect.y + rect.h), rounding); - } nk_draw_list_path_stroke(list, col, NK_STROKE_CLOSED, thickness); -} -NK_API void -nk_draw_list_fill_rect_multi_color(struct nk_draw_list *list, struct nk_rect rect, - struct nk_color left, struct nk_color top, struct nk_color right, - struct nk_color bottom) -{ - void *vtx; - struct nk_colorf col_left, col_top; - struct nk_colorf col_right, col_bottom; - nk_draw_index *idx; - nk_draw_index index; - - nk_color_fv(&col_left.r, left); - nk_color_fv(&col_right.r, right); - nk_color_fv(&col_top.r, top); - nk_color_fv(&col_bottom.r, bottom); - - NK_ASSERT(list); - if (!list) return; - - nk_draw_list_push_image(list, list->config.tex_null.texture); - index = (nk_draw_index)list->vertex_count; - vtx = nk_draw_list_alloc_vertices(list, 4); - idx = nk_draw_list_alloc_elements(list, 6); - if (!vtx || !idx) return; - - idx[0] = (nk_draw_index)(index+0); idx[1] = (nk_draw_index)(index+1); - idx[2] = (nk_draw_index)(index+2); idx[3] = (nk_draw_index)(index+0); - idx[4] = (nk_draw_index)(index+2); idx[5] = (nk_draw_index)(index+3); - - vtx = nk_draw_vertex(vtx, &list->config, nk_vec2(rect.x, rect.y), list->config.tex_null.uv, col_left); - vtx = nk_draw_vertex(vtx, &list->config, nk_vec2(rect.x + rect.w, rect.y), list->config.tex_null.uv, col_top); - vtx = nk_draw_vertex(vtx, &list->config, nk_vec2(rect.x + rect.w, rect.y + rect.h), list->config.tex_null.uv, col_right); - vtx = nk_draw_vertex(vtx, &list->config, nk_vec2(rect.x, rect.y + rect.h), list->config.tex_null.uv, col_bottom); -} -NK_API void -nk_draw_list_fill_triangle(struct nk_draw_list *list, struct nk_vec2 a, - struct nk_vec2 b, struct nk_vec2 c, struct nk_color col) -{ - NK_ASSERT(list); - if (!list || !col.a) return; - nk_draw_list_path_line_to(list, a); - nk_draw_list_path_line_to(list, b); - nk_draw_list_path_line_to(list, c); - nk_draw_list_path_fill(list, col); -} -NK_API void -nk_draw_list_stroke_triangle(struct nk_draw_list *list, struct nk_vec2 a, - struct nk_vec2 b, struct nk_vec2 c, struct nk_color col, float thickness) -{ - NK_ASSERT(list); - if (!list || !col.a) return; - nk_draw_list_path_line_to(list, a); - nk_draw_list_path_line_to(list, b); - nk_draw_list_path_line_to(list, c); - nk_draw_list_path_stroke(list, col, NK_STROKE_CLOSED, thickness); -} -NK_API void -nk_draw_list_fill_circle(struct nk_draw_list *list, struct nk_vec2 center, - float radius, struct nk_color col, unsigned int segs) -{ - float a_max; - NK_ASSERT(list); - if (!list || !col.a) return; - a_max = NK_PI * 2.0f * ((float)segs - 1.0f) / (float)segs; - nk_draw_list_path_arc_to(list, center, radius, 0.0f, a_max, segs); - nk_draw_list_path_fill(list, col); -} -NK_API void -nk_draw_list_stroke_circle(struct nk_draw_list *list, struct nk_vec2 center, - float radius, struct nk_color col, unsigned int segs, float thickness) -{ - float a_max; - NK_ASSERT(list); - if (!list || !col.a) return; - a_max = NK_PI * 2.0f * ((float)segs - 1.0f) / (float)segs; - nk_draw_list_path_arc_to(list, center, radius, 0.0f, a_max, segs); - nk_draw_list_path_stroke(list, col, NK_STROKE_CLOSED, thickness); -} -NK_API void -nk_draw_list_stroke_curve(struct nk_draw_list *list, struct nk_vec2 p0, - struct nk_vec2 cp0, struct nk_vec2 cp1, struct nk_vec2 p1, - struct nk_color col, unsigned int segments, float thickness) -{ - NK_ASSERT(list); - if (!list || !col.a) return; - nk_draw_list_path_line_to(list, p0); - nk_draw_list_path_curve_to(list, cp0, cp1, p1, segments); - nk_draw_list_path_stroke(list, col, NK_STROKE_OPEN, thickness); -} -NK_INTERN void -nk_draw_list_push_rect_uv(struct nk_draw_list *list, struct nk_vec2 a, - struct nk_vec2 c, struct nk_vec2 uva, struct nk_vec2 uvc, - struct nk_color color) -{ - void *vtx; - struct nk_vec2 uvb; - struct nk_vec2 uvd; - struct nk_vec2 b; - struct nk_vec2 d; - - struct nk_colorf col; - nk_draw_index *idx; - nk_draw_index index; - NK_ASSERT(list); - if (!list) return; - - nk_color_fv(&col.r, color); - uvb = nk_vec2(uvc.x, uva.y); - uvd = nk_vec2(uva.x, uvc.y); - b = nk_vec2(c.x, a.y); - d = nk_vec2(a.x, c.y); - - index = (nk_draw_index)list->vertex_count; - vtx = nk_draw_list_alloc_vertices(list, 4); - idx = nk_draw_list_alloc_elements(list, 6); - if (!vtx || !idx) return; - - idx[0] = (nk_draw_index)(index+0); idx[1] = (nk_draw_index)(index+1); - idx[2] = (nk_draw_index)(index+2); idx[3] = (nk_draw_index)(index+0); - idx[4] = (nk_draw_index)(index+2); idx[5] = (nk_draw_index)(index+3); - - vtx = nk_draw_vertex(vtx, &list->config, a, uva, col); - vtx = nk_draw_vertex(vtx, &list->config, b, uvb, col); - vtx = nk_draw_vertex(vtx, &list->config, c, uvc, col); - vtx = nk_draw_vertex(vtx, &list->config, d, uvd, col); -} -NK_API void -nk_draw_list_add_image(struct nk_draw_list *list, struct nk_image texture, - struct nk_rect rect, struct nk_color color) -{ - NK_ASSERT(list); - if (!list) return; - /* push new command with given texture */ - nk_draw_list_push_image(list, texture.handle); - if (nk_image_is_subimage(&texture)) { - /* add region inside of the texture */ - struct nk_vec2 uv[2]; - uv[0].x = (float)texture.region[0]/(float)texture.w; - uv[0].y = (float)texture.region[1]/(float)texture.h; - uv[1].x = (float)(texture.region[0] + texture.region[2])/(float)texture.w; - uv[1].y = (float)(texture.region[1] + texture.region[3])/(float)texture.h; - nk_draw_list_push_rect_uv(list, nk_vec2(rect.x, rect.y), - nk_vec2(rect.x + rect.w, rect.y + rect.h), uv[0], uv[1], color); - } else nk_draw_list_push_rect_uv(list, nk_vec2(rect.x, rect.y), - nk_vec2(rect.x + rect.w, rect.y + rect.h), - nk_vec2(0.0f, 0.0f), nk_vec2(1.0f, 1.0f),color); -} -NK_API void -nk_draw_list_add_text(struct nk_draw_list *list, const struct nk_user_font *font, - struct nk_rect rect, const char *text, int len, float font_height, - struct nk_color fg) -{ - float x = 0; - int text_len = 0; - nk_rune unicode = 0; - nk_rune next = 0; - int glyph_len = 0; - int next_glyph_len = 0; - struct nk_user_font_glyph g; - - NK_ASSERT(list); - if (!list || !len || !text) return; - if (!NK_INTERSECT(rect.x, rect.y, rect.w, rect.h, - list->clip_rect.x, list->clip_rect.y, list->clip_rect.w, list->clip_rect.h)) return; - - nk_draw_list_push_image(list, font->texture); - x = rect.x; - glyph_len = nk_utf_decode(text, &unicode, len); - if (!glyph_len) return; - - /* draw every glyph image */ - fg.a = (nk_byte)((float)fg.a * list->config.global_alpha); - while (text_len < len && glyph_len) { - float gx, gy, gh, gw; - float char_width = 0; - if (unicode == NK_UTF_INVALID) break; - - /* query currently drawn glyph information */ - next_glyph_len = nk_utf_decode(text + text_len + glyph_len, &next, (int)len - text_len); - font->query(font->userdata, font_height, &g, unicode, - (next == NK_UTF_INVALID) ? '\0' : next); - - /* calculate and draw glyph drawing rectangle and image */ - gx = x + g.offset.x; - gy = rect.y + g.offset.y; - gw = g.width; gh = g.height; - char_width = g.xadvance; - nk_draw_list_push_rect_uv(list, nk_vec2(gx,gy), nk_vec2(gx + gw, gy+ gh), - g.uv[0], g.uv[1], fg); - - /* offset next glyph */ - text_len += glyph_len; - x += char_width; - glyph_len = next_glyph_len; - unicode = next; - } -} -NK_API nk_flags -nk_convert(struct nk_context *ctx, struct nk_buffer *cmds, - struct nk_buffer *vertices, struct nk_buffer *elements, - const struct nk_convert_config *config) -{ - nk_flags res = NK_CONVERT_SUCCESS; - const struct nk_command *cmd; - NK_ASSERT(ctx); - NK_ASSERT(cmds); - NK_ASSERT(vertices); - NK_ASSERT(elements); - NK_ASSERT(config); - NK_ASSERT(config->vertex_layout); - NK_ASSERT(config->vertex_size); - if (!ctx || !cmds || !vertices || !elements || !config || !config->vertex_layout) - return NK_CONVERT_INVALID_PARAM; - - nk_draw_list_setup(&ctx->draw_list, config, cmds, vertices, elements, - config->line_AA, config->shape_AA); - nk_foreach(cmd, ctx) - { -#ifdef NK_INCLUDE_COMMAND_USERDATA - ctx->draw_list.userdata = cmd->userdata; -#endif - switch (cmd->type) { - case NK_COMMAND_NOP: break; - case NK_COMMAND_SCISSOR: { - const struct nk_command_scissor *s = (const struct nk_command_scissor*)cmd; - nk_draw_list_add_clip(&ctx->draw_list, nk_rect(s->x, s->y, s->w, s->h)); - } break; - case NK_COMMAND_LINE: { - const struct nk_command_line *l = (const struct nk_command_line*)cmd; - nk_draw_list_stroke_line(&ctx->draw_list, nk_vec2(l->begin.x, l->begin.y), - nk_vec2(l->end.x, l->end.y), l->color, l->line_thickness); - } break; - case NK_COMMAND_CURVE: { - const struct nk_command_curve *q = (const struct nk_command_curve*)cmd; - nk_draw_list_stroke_curve(&ctx->draw_list, nk_vec2(q->begin.x, q->begin.y), - nk_vec2(q->ctrl[0].x, q->ctrl[0].y), nk_vec2(q->ctrl[1].x, - q->ctrl[1].y), nk_vec2(q->end.x, q->end.y), q->color, - config->curve_segment_count, q->line_thickness); - } break; - case NK_COMMAND_RECT: { - const struct nk_command_rect *r = (const struct nk_command_rect*)cmd; - nk_draw_list_stroke_rect(&ctx->draw_list, nk_rect(r->x, r->y, r->w, r->h), - r->color, (float)r->rounding, r->line_thickness); - } break; - case NK_COMMAND_RECT_FILLED: { - const struct nk_command_rect_filled *r = (const struct nk_command_rect_filled*)cmd; - nk_draw_list_fill_rect(&ctx->draw_list, nk_rect(r->x, r->y, r->w, r->h), - r->color, (float)r->rounding); - } break; - case NK_COMMAND_RECT_MULTI_COLOR: { - const struct nk_command_rect_multi_color *r = (const struct nk_command_rect_multi_color*)cmd; - nk_draw_list_fill_rect_multi_color(&ctx->draw_list, nk_rect(r->x, r->y, r->w, r->h), - r->left, r->top, r->right, r->bottom); - } break; - case NK_COMMAND_CIRCLE: { - const struct nk_command_circle *c = (const struct nk_command_circle*)cmd; - nk_draw_list_stroke_circle(&ctx->draw_list, nk_vec2((float)c->x + (float)c->w/2, - (float)c->y + (float)c->h/2), (float)c->w/2, c->color, - config->circle_segment_count, c->line_thickness); - } break; - case NK_COMMAND_CIRCLE_FILLED: { - const struct nk_command_circle_filled *c = (const struct nk_command_circle_filled *)cmd; - nk_draw_list_fill_circle(&ctx->draw_list, nk_vec2((float)c->x + (float)c->w/2, - (float)c->y + (float)c->h/2), (float)c->w/2, c->color, - config->circle_segment_count); - } break; - case NK_COMMAND_ARC: { - const struct nk_command_arc *c = (const struct nk_command_arc*)cmd; - nk_draw_list_path_line_to(&ctx->draw_list, nk_vec2(c->cx, c->cy)); - nk_draw_list_path_arc_to(&ctx->draw_list, nk_vec2(c->cx, c->cy), c->r, - c->a[0], c->a[1], config->arc_segment_count); - nk_draw_list_path_stroke(&ctx->draw_list, c->color, NK_STROKE_CLOSED, c->line_thickness); - } break; - case NK_COMMAND_ARC_FILLED: { - const struct nk_command_arc_filled *c = (const struct nk_command_arc_filled*)cmd; - nk_draw_list_path_line_to(&ctx->draw_list, nk_vec2(c->cx, c->cy)); - nk_draw_list_path_arc_to(&ctx->draw_list, nk_vec2(c->cx, c->cy), c->r, - c->a[0], c->a[1], config->arc_segment_count); - nk_draw_list_path_fill(&ctx->draw_list, c->color); - } break; - case NK_COMMAND_TRIANGLE: { - const struct nk_command_triangle *t = (const struct nk_command_triangle*)cmd; - nk_draw_list_stroke_triangle(&ctx->draw_list, nk_vec2(t->a.x, t->a.y), - nk_vec2(t->b.x, t->b.y), nk_vec2(t->c.x, t->c.y), t->color, - t->line_thickness); - } break; - case NK_COMMAND_TRIANGLE_FILLED: { - const struct nk_command_triangle_filled *t = (const struct nk_command_triangle_filled*)cmd; - nk_draw_list_fill_triangle(&ctx->draw_list, nk_vec2(t->a.x, t->a.y), - nk_vec2(t->b.x, t->b.y), nk_vec2(t->c.x, t->c.y), t->color); - } break; - case NK_COMMAND_POLYGON: { - int i; - const struct nk_command_polygon*p = (const struct nk_command_polygon*)cmd; - for (i = 0; i < p->point_count; ++i) { - struct nk_vec2 pnt = nk_vec2((float)p->points[i].x, (float)p->points[i].y); - nk_draw_list_path_line_to(&ctx->draw_list, pnt); - } - nk_draw_list_path_stroke(&ctx->draw_list, p->color, NK_STROKE_CLOSED, p->line_thickness); - } break; - case NK_COMMAND_POLYGON_FILLED: { - int i; - const struct nk_command_polygon_filled *p = (const struct nk_command_polygon_filled*)cmd; - for (i = 0; i < p->point_count; ++i) { - struct nk_vec2 pnt = nk_vec2((float)p->points[i].x, (float)p->points[i].y); - nk_draw_list_path_line_to(&ctx->draw_list, pnt); - } - nk_draw_list_path_fill(&ctx->draw_list, p->color); - } break; - case NK_COMMAND_POLYLINE: { - int i; - const struct nk_command_polyline *p = (const struct nk_command_polyline*)cmd; - for (i = 0; i < p->point_count; ++i) { - struct nk_vec2 pnt = nk_vec2((float)p->points[i].x, (float)p->points[i].y); - nk_draw_list_path_line_to(&ctx->draw_list, pnt); - } - nk_draw_list_path_stroke(&ctx->draw_list, p->color, NK_STROKE_OPEN, p->line_thickness); - } break; - case NK_COMMAND_TEXT: { - const struct nk_command_text *t = (const struct nk_command_text*)cmd; - nk_draw_list_add_text(&ctx->draw_list, t->font, nk_rect(t->x, t->y, t->w, t->h), - t->string, t->length, t->height, t->foreground); - } break; - case NK_COMMAND_IMAGE: { - const struct nk_command_image *i = (const struct nk_command_image*)cmd; - nk_draw_list_add_image(&ctx->draw_list, i->img, nk_rect(i->x, i->y, i->w, i->h), i->col); - } break; - case NK_COMMAND_CUSTOM: { - const struct nk_command_custom *c = (const struct nk_command_custom*)cmd; - c->callback(&ctx->draw_list, c->x, c->y, c->w, c->h, c->callback_data); - } break; - default: break; - } - } - res |= (cmds->needed > cmds->allocated + (cmds->memory.size - cmds->size)) ? NK_CONVERT_COMMAND_BUFFER_FULL: 0; - res |= (vertices->needed > vertices->allocated) ? NK_CONVERT_VERTEX_BUFFER_FULL: 0; - res |= (elements->needed > elements->allocated) ? NK_CONVERT_ELEMENT_BUFFER_FULL: 0; - return res; -} -NK_API const struct nk_draw_command* -nk__draw_begin(const struct nk_context *ctx, - const struct nk_buffer *buffer) -{ - return nk__draw_list_begin(&ctx->draw_list, buffer); -} -NK_API const struct nk_draw_command* -nk__draw_end(const struct nk_context *ctx, const struct nk_buffer *buffer) -{ - return nk__draw_list_end(&ctx->draw_list, buffer); -} -NK_API const struct nk_draw_command* -nk__draw_next(const struct nk_draw_command *cmd, - const struct nk_buffer *buffer, const struct nk_context *ctx) -{ - return nk__draw_list_next(cmd, buffer, &ctx->draw_list); -} -#endif - - -/* stb_rect_pack.h - v1.01 - public domain - rectangle packing */ -/* Sean Barrett 2014 */ -/* */ -/* Useful for e.g. packing rectangular textures into an atlas. */ -/* Does not do rotation. */ -/* */ -/* Before #including, */ -/* */ -/* #define STB_RECT_PACK_IMPLEMENTATION */ -/* */ -/* in the file that you want to have the implementation. */ -/* */ -/* Not necessarily the awesomest packing method, but better than */ -/* the totally naive one in stb_truetype (which is primarily what */ -/* this is meant to replace). */ -/* */ -/* Has only had a few tests run, may have issues. */ -/* */ -/* More docs to come. */ -/* */ -/* No memory allocations; uses qsort() and assert() from stdlib. */ -/* Can override those by defining STBRP_SORT and STBRP_ASSERT. */ -/* */ -/* This library currently uses the Skyline Bottom-Left algorithm. */ -/* */ -/* Please note: better rectangle packers are welcome! Please */ -/* implement them to the same API, but with a different init */ -/* function. */ -/* */ -/* Credits */ -/* */ -/* Library */ -/* Sean Barrett */ -/* Minor features */ -/* Martins Mozeiko */ -/* github:IntellectualKitty */ -/* */ -/* Bugfixes / warning fixes */ -/* Jeremy Jaussaud */ -/* Fabian Giesen */ -/* */ -/* Version history: */ -/* */ -/* 1.01 (2021-07-11) always use large rect mode, expose STBRP__MAXVAL in public section */ -/* 1.00 (2019-02-25) avoid small space waste; gracefully fail too-wide rectangles */ -/* 0.99 (2019-02-07) warning fixes */ -/* 0.11 (2017-03-03) return packing success/fail result */ -/* 0.10 (2016-10-25) remove cast-away-const to avoid warnings */ -/* 0.09 (2016-08-27) fix compiler warnings */ -/* 0.08 (2015-09-13) really fix bug with empty rects (w=0 or h=0) */ -/* 0.07 (2015-09-13) fix bug with empty rects (w=0 or h=0) */ -/* 0.06 (2015-04-15) added STBRP_SORT to allow replacing qsort */ -/* 0.05: added STBRP_ASSERT to allow replacing assert */ -/* 0.04: fixed minor bug in STBRP_LARGE_RECTS support */ -/* 0.01: initial release */ -/* */ -/* LICENSE */ -/* */ -/* See end of file for license information. */ - -/* //////////////////////////////////////////////////////////////////////////// */ -/* */ -/* INCLUDE SECTION */ -/* */ - -#ifndef STB_INCLUDE_STB_RECT_PACK_H -#define STB_INCLUDE_STB_RECT_PACK_H - -#define STB_RECT_PACK_VERSION 1 - -#ifdef STBRP_STATIC -#define STBRP_DEF static -#else -#define STBRP_DEF extern -#endif - -#ifdef __cplusplus -extern "C" { -#endif - -typedef struct stbrp_context stbrp_context; -typedef struct stbrp_node stbrp_node; -typedef struct stbrp_rect stbrp_rect; - -typedef int stbrp_coord; - -#define STBRP__MAXVAL 0x7fffffff -/* Mostly for internal use, but this is the maximum supported coordinate value. */ - -STBRP_DEF int stbrp_pack_rects (stbrp_context *context, stbrp_rect *rects, int num_rects); -/* Assign packed locations to rectangles. The rectangles are of type */ -/* 'stbrp_rect' defined below, stored in the array 'rects', and there */ -/* are 'num_rects' many of them. */ -/* */ -/* Rectangles which are successfully packed have the 'was_packed' flag */ -/* set to a non-zero value and 'x' and 'y' store the minimum location */ -/* on each axis (i.e. bottom-left in cartesian coordinates, top-left */ -/* if you imagine y increasing downwards). Rectangles which do not fit */ -/* have the 'was_packed' flag set to 0. */ -/* */ -/* You should not try to access the 'rects' array from another thread */ -/* while this function is running, as the function temporarily reorders */ -/* the array while it executes. */ -/* */ -/* To pack into another rectangle, you need to call stbrp_init_target */ -/* again. To continue packing into the same rectangle, you can call */ -/* this function again. Calling this multiple times with multiple rect */ -/* arrays will probably produce worse packing results than calling it */ -/* a single time with the full rectangle array, but the option is */ -/* available. */ -/* */ -/* The function returns 1 if all of the rectangles were successfully */ -/* packed and 0 otherwise. */ - -struct stbrp_rect -{ - /* reserved for your use: */ - int id; - - /* input: */ - stbrp_coord w, h; - - /* output: */ - stbrp_coord x, y; - int was_packed; /* non-zero if valid packing */ - -}; /* 16 bytes, nominally */ - - -STBRP_DEF void stbrp_init_target (stbrp_context *context, int width, int height, stbrp_node *nodes, int num_nodes); -/* Initialize a rectangle packer to: */ -/* pack a rectangle that is 'width' by 'height' in dimensions */ -/* using temporary storage provided by the array 'nodes', which is 'num_nodes' long */ -/* */ -/* You must call this function every time you start packing into a new target. */ -/* */ -/* There is no "shutdown" function. The 'nodes' memory must stay valid for */ -/* the following stbrp_pack_rects() call (or calls), but can be freed after */ -/* the call (or calls) finish. */ -/* */ -/* Note: to guarantee best results, either: */ -/* 1. make sure 'num_nodes' >= 'width' */ -/* or 2. call stbrp_allow_out_of_mem() defined below with 'allow_out_of_mem = 1' */ -/* */ -/* If you don't do either of the above things, widths will be quantized to multiples */ -/* of small integers to guarantee the algorithm doesn't run out of temporary storage. */ -/* */ -/* If you do #2, then the non-quantized algorithm will be used, but the algorithm */ -/* may run out of temporary storage and be unable to pack some rectangles. */ - -STBRP_DEF void stbrp_setup_allow_out_of_mem (stbrp_context *context, int allow_out_of_mem); -/* Optionally call this function after init but before doing any packing to */ -/* change the handling of the out-of-temp-memory scenario, described above. */ -/* If you call init again, this will be reset to the default (false). */ - - -STBRP_DEF void stbrp_setup_heuristic (stbrp_context *context, int heuristic); -/* Optionally select which packing heuristic the library should use. Different */ -/* heuristics will produce better/worse results for different data sets. */ -/* If you call init again, this will be reset to the default. */ - -enum -{ - STBRP_HEURISTIC_Skyline_default=0, - STBRP_HEURISTIC_Skyline_BL_sortHeight = STBRP_HEURISTIC_Skyline_default, - STBRP_HEURISTIC_Skyline_BF_sortHeight -}; - - -/* //////////////////////////////////////////////////////////////////////////// */ -/* */ -/* the details of the following structures don't matter to you, but they must */ -/* be visible so you can handle the memory allocations for them */ - -struct stbrp_node -{ - stbrp_coord x,y; - stbrp_node *next; -}; - -struct stbrp_context -{ - int width; - int height; - int align; - int init_mode; - int heuristic; - int num_nodes; - stbrp_node *active_head; - stbrp_node *free_head; - stbrp_node extra[2]; /* we allocate two extra nodes so optimal user-node-count is 'width' not 'width+2' */ -}; - -#ifdef __cplusplus -} -#endif - -#endif - -/* //////////////////////////////////////////////////////////////////////////// */ -/* */ -/* IMPLEMENTATION SECTION */ -/* */ - -#ifdef STB_RECT_PACK_IMPLEMENTATION -#ifndef STBRP_SORT -#include -#define STBRP_SORT qsort -#endif - -#ifndef STBRP_ASSERT -#include -#define STBRP_ASSERT assert -#endif - -#ifdef _MSC_VER -#define STBRP__NOTUSED(v) (void)(v) -#define STBRP__CDECL __cdecl -#else -#define STBRP__NOTUSED(v) (void)sizeof(v) -#define STBRP__CDECL -#endif - -enum -{ - STBRP__INIT_skyline = 1 -}; - -STBRP_DEF void stbrp_setup_heuristic(stbrp_context *context, int heuristic) -{ - switch (context->init_mode) { - case STBRP__INIT_skyline: - STBRP_ASSERT(heuristic == STBRP_HEURISTIC_Skyline_BL_sortHeight || heuristic == STBRP_HEURISTIC_Skyline_BF_sortHeight); - context->heuristic = heuristic; - break; - default: - STBRP_ASSERT(0); - } -} - -STBRP_DEF void stbrp_setup_allow_out_of_mem(stbrp_context *context, int allow_out_of_mem) -{ - if (allow_out_of_mem) - /* if it's ok to run out of memory, then don't bother aligning them; */ - /* this gives better packing, but may fail due to OOM (even though */ - /* the rectangles easily fit). @TODO a smarter approach would be to only */ - /* quantize once we've hit OOM, then we could get rid of this parameter. */ - context->align = 1; - else { - /* if it's not ok to run out of memory, then quantize the widths */ - /* so that num_nodes is always enough nodes. */ - /* */ - /* I.e. num_nodes * align >= width */ - /* align >= width / num_nodes */ - /* align = ceil(width/num_nodes) */ - - context->align = (context->width + context->num_nodes-1) / context->num_nodes; - } -} - -STBRP_DEF void stbrp_init_target(stbrp_context *context, int width, int height, stbrp_node *nodes, int num_nodes) -{ - int i; - - for (i=0; i < num_nodes-1; ++i) - nodes[i].next = &nodes[i+1]; - nodes[i].next = NULL; - context->init_mode = STBRP__INIT_skyline; - context->heuristic = STBRP_HEURISTIC_Skyline_default; - context->free_head = &nodes[0]; - context->active_head = &context->extra[0]; - context->width = width; - context->height = height; - context->num_nodes = num_nodes; - stbrp_setup_allow_out_of_mem(context, 0); - - /* node 0 is the full width, node 1 is the sentinel (lets us not store width explicitly) */ - context->extra[0].x = 0; - context->extra[0].y = 0; - context->extra[0].next = &context->extra[1]; - context->extra[1].x = (stbrp_coord) width; - context->extra[1].y = (1<<30); - context->extra[1].next = NULL; -} - -/* find minimum y position if it starts at x1 */ -static int stbrp__skyline_find_min_y(stbrp_context *c, stbrp_node *first, int x0, int width, int *pwaste) -{ - stbrp_node *node = first; - int x1 = x0 + width; - int min_y, visited_width, waste_area; - - STBRP__NOTUSED(c); - - STBRP_ASSERT(first->x <= x0); - - #if 0 - /* skip in case we're past the node */ - while (node->next->x <= x0) - ++node; - #else - STBRP_ASSERT(node->next->x > x0); /* we ended up handling this in the caller for efficiency */ - #endif - - STBRP_ASSERT(node->x <= x0); - - min_y = 0; - waste_area = 0; - visited_width = 0; - while (node->x < x1) { - if (node->y > min_y) { - /* raise min_y higher. */ - /* we've accounted for all waste up to min_y, */ - /* but we'll now add more waste for everything we've visted */ - waste_area += visited_width * (node->y - min_y); - min_y = node->y; - /* the first time through, visited_width might be reduced */ - if (node->x < x0) - visited_width += node->next->x - x0; - else - visited_width += node->next->x - node->x; - } else { - /* add waste area */ - int under_width = node->next->x - node->x; - if (under_width + visited_width > width) - under_width = width - visited_width; - waste_area += under_width * (min_y - node->y); - visited_width += under_width; - } - node = node->next; - } - - *pwaste = waste_area; - return min_y; -} - -typedef struct -{ - int x,y; - stbrp_node **prev_link; -} stbrp__findresult; - -static stbrp__findresult stbrp__skyline_find_best_pos(stbrp_context *c, int width, int height) -{ - int best_waste = (1<<30), best_x, best_y = (1 << 30); - stbrp__findresult fr; - stbrp_node **prev, *node, *tail, **best = NULL; - - /* align to multiple of c->align */ - width = (width + c->align - 1); - width -= width % c->align; - STBRP_ASSERT(width % c->align == 0); - - /* if it can't possibly fit, bail immediately */ - if (width > c->width || height > c->height) { - fr.prev_link = NULL; - fr.x = fr.y = 0; - return fr; - } - - node = c->active_head; - prev = &c->active_head; - while (node->x + width <= c->width) { - int y,waste; - y = stbrp__skyline_find_min_y(c, node, node->x, width, &waste); - if (c->heuristic == STBRP_HEURISTIC_Skyline_BL_sortHeight) { /* actually just want to test BL */ - /* bottom left */ - if (y < best_y) { - best_y = y; - best = prev; - } - } else { - /* best-fit */ - if (y + height <= c->height) { - /* can only use it if it first vertically */ - if (y < best_y || (y == best_y && waste < best_waste)) { - best_y = y; - best_waste = waste; - best = prev; - } - } - } - prev = &node->next; - node = node->next; - } - - best_x = (best == NULL) ? 0 : (*best)->x; - - /* if doing best-fit (BF), we also have to try aligning right edge to each node position */ - /* */ - /* e.g, if fitting */ - /* */ - /* ____________________ */ - /* |____________________| */ - /* */ - /* into */ - /* */ - /* | | */ - /* | ____________| */ - /* |____________| */ - /* */ - /* then right-aligned reduces waste, but bottom-left BL is always chooses left-aligned */ - /* */ - /* This makes BF take about 2x the time */ - - if (c->heuristic == STBRP_HEURISTIC_Skyline_BF_sortHeight) { - tail = c->active_head; - node = c->active_head; - prev = &c->active_head; - /* find first node that's admissible */ - while (tail->x < width) - tail = tail->next; - while (tail) { - int xpos = tail->x - width; - int y,waste; - STBRP_ASSERT(xpos >= 0); - /* find the left position that matches this */ - while (node->next->x <= xpos) { - prev = &node->next; - node = node->next; - } - STBRP_ASSERT(node->next->x > xpos && node->x <= xpos); - y = stbrp__skyline_find_min_y(c, node, xpos, width, &waste); - if (y + height <= c->height) { - if (y <= best_y) { - if (y < best_y || waste < best_waste || (waste==best_waste && xpos < best_x)) { - best_x = xpos; - STBRP_ASSERT(y <= best_y); - best_y = y; - best_waste = waste; - best = prev; - } - } - } - tail = tail->next; - } - } - - fr.prev_link = best; - fr.x = best_x; - fr.y = best_y; - return fr; -} - -static stbrp__findresult stbrp__skyline_pack_rectangle(stbrp_context *context, int width, int height) -{ - /* find best position according to heuristic */ - stbrp__findresult res = stbrp__skyline_find_best_pos(context, width, height); - stbrp_node *node, *cur; - - /* bail if: */ - /* 1. it failed */ - /* 2. the best node doesn't fit (we don't always check this) */ - /* 3. we're out of memory */ - if (res.prev_link == NULL || res.y + height > context->height || context->free_head == NULL) { - res.prev_link = NULL; - return res; - } - - /* on success, create new node */ - node = context->free_head; - node->x = (stbrp_coord) res.x; - node->y = (stbrp_coord) (res.y + height); - - context->free_head = node->next; - - /* insert the new node into the right starting point, and */ - /* let 'cur' point to the remaining nodes needing to be */ - /* stiched back in */ - - cur = *res.prev_link; - if (cur->x < res.x) { - /* preserve the existing one, so start testing with the next one */ - stbrp_node *next = cur->next; - cur->next = node; - cur = next; - } else { - *res.prev_link = node; - } - - /* from here, traverse cur and free the nodes, until we get to one */ - /* that shouldn't be freed */ - while (cur->next && cur->next->x <= res.x + width) { - stbrp_node *next = cur->next; - /* move the current node to the free list */ - cur->next = context->free_head; - context->free_head = cur; - cur = next; - } - - /* stitch the list back in */ - node->next = cur; - - if (cur->x < res.x + width) - cur->x = (stbrp_coord) (res.x + width); - -#ifdef _DEBUG - cur = context->active_head; - while (cur->x < context->width) { - STBRP_ASSERT(cur->x < cur->next->x); - cur = cur->next; - } - STBRP_ASSERT(cur->next == NULL); - - { - int count=0; - cur = context->active_head; - while (cur) { - cur = cur->next; - ++count; - } - cur = context->free_head; - while (cur) { - cur = cur->next; - ++count; - } - STBRP_ASSERT(count == context->num_nodes+2); - } -#endif - - return res; -} - -static int STBRP__CDECL rect_height_compare(const void *a, const void *b) -{ - const stbrp_rect *p = (const stbrp_rect *) a; - const stbrp_rect *q = (const stbrp_rect *) b; - if (p->h > q->h) - return -1; - if (p->h < q->h) - return 1; - return (p->w > q->w) ? -1 : (p->w < q->w); -} - -static int STBRP__CDECL rect_original_order(const void *a, const void *b) -{ - const stbrp_rect *p = (const stbrp_rect *) a; - const stbrp_rect *q = (const stbrp_rect *) b; - return (p->was_packed < q->was_packed) ? -1 : (p->was_packed > q->was_packed); -} - -STBRP_DEF int stbrp_pack_rects(stbrp_context *context, stbrp_rect *rects, int num_rects) -{ - int i, all_rects_packed = 1; - - /* we use the 'was_packed' field internally to allow sorting/unsorting */ - for (i=0; i < num_rects; ++i) { - rects[i].was_packed = i; - } - - /* sort according to heuristic */ - STBRP_SORT(rects, num_rects, sizeof(rects[0]), rect_height_compare); - - for (i=0; i < num_rects; ++i) { - if (rects[i].w == 0 || rects[i].h == 0) { - rects[i].x = rects[i].y = 0; /* empty rect needs no space */ - } else { - stbrp__findresult fr = stbrp__skyline_pack_rectangle(context, rects[i].w, rects[i].h); - if (fr.prev_link) { - rects[i].x = (stbrp_coord) fr.x; - rects[i].y = (stbrp_coord) fr.y; - } else { - rects[i].x = rects[i].y = STBRP__MAXVAL; - } - } - } - - /* unsort */ - STBRP_SORT(rects, num_rects, sizeof(rects[0]), rect_original_order); - - /* set was_packed flags and all_rects_packed status */ - for (i=0; i < num_rects; ++i) { - rects[i].was_packed = !(rects[i].x == STBRP__MAXVAL && rects[i].y == STBRP__MAXVAL); - if (!rects[i].was_packed) - all_rects_packed = 0; - } - - /* return the all_rects_packed status */ - return all_rects_packed; -} -#endif - -/* ------------------------------------------------------------------------------- -This software is available under 2 licenses -- choose whichever you prefer. ------------------------------------------------------------------------------- -ALTERNATIVE A - MIT License -Copyright (c) 2017 Sean Barrett -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. ------------------------------------------------------------------------------- -ALTERNATIVE B - Public Domain (www.unlicense.org) -This is free and unencumbered software released into the public domain. -Anyone is free to copy, modify, publish, use, compile, sell, or distribute this -software, either in source code form or as a compiled binary, for any purpose, -commercial or non-commercial, and by any means. -In jurisdictions that recognize copyright laws, the author or authors of this -software dedicate any and all copyright interest in the software to the public -domain. We make this dedication for the benefit of the public at large and to -the detriment of our heirs and successors. We intend this dedication to be an -overt act of relinquishment in perpetuity of all present and future rights to -this software under copyright law. -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 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. ------------------------------------------------------------------------------- -*/ - -/* stb_truetype.h - v1.26 - public domain */ -/* authored from 2009-2021 by Sean Barrett / RAD Game Tools */ -/* */ -/* ======================================================================= */ -/* */ -/* NO SECURITY GUARANTEE -- DO NOT USE THIS ON UNTRUSTED FONT FILES */ -/* */ -/* This library does no range checking of the offsets found in the file, */ -/* meaning an attacker can use it to read arbitrary memory. */ -/* */ -/* ======================================================================= */ -/* */ -/* This library processes TrueType files: */ -/* parse files */ -/* extract glyph metrics */ -/* extract glyph shapes */ -/* render glyphs to one-channel bitmaps with antialiasing (box filter) */ -/* render glyphs to one-channel SDF bitmaps (signed-distance field/function) */ -/* */ -/* Todo: */ -/* non-MS cmaps */ -/* crashproof on bad data */ -/* hinting? (no longer patented) */ -/* cleartype-style AA? */ -/* optimize: use simple memory allocator for intermediates */ -/* optimize: build edge-list directly from curves */ -/* optimize: rasterize directly from curves? */ -/* */ -/* ADDITIONAL CONTRIBUTORS */ -/* */ -/* Mikko Mononen: compound shape support, more cmap formats */ -/* Tor Andersson: kerning, subpixel rendering */ -/* Dougall Johnson: OpenType / Type 2 font handling */ -/* Daniel Ribeiro Maciel: basic GPOS-based kerning */ -/* */ -/* Misc other: */ -/* Ryan Gordon */ -/* Simon Glass */ -/* github:IntellectualKitty */ -/* Imanol Celaya */ -/* Daniel Ribeiro Maciel */ -/* */ -/* Bug/warning reports/fixes: */ -/* "Zer" on mollyrocket Fabian "ryg" Giesen github:NiLuJe */ -/* Cass Everitt Martins Mozeiko github:aloucks */ -/* stoiko (Haemimont Games) Cap Petschulat github:oyvindjam */ -/* Brian Hook Omar Cornut github:vassvik */ -/* Walter van Niftrik Ryan Griege */ -/* David Gow Peter LaValle */ -/* David Given Sergey Popov */ -/* Ivan-Assen Ivanov Giumo X. Clanjor */ -/* Anthony Pesch Higor Euripedes */ -/* Johan Duparc Thomas Fields */ -/* Hou Qiming Derek Vinyard */ -/* Rob Loach Cort Stratton */ -/* Kenney Phillis Jr. Brian Costabile */ -/* Ken Voskuil (kaesve) */ -/* */ -/* VERSION HISTORY */ -/* */ -/* 1.26 (2021-08-28) fix broken rasterizer */ -/* 1.25 (2021-07-11) many fixes */ -/* 1.24 (2020-02-05) fix warning */ -/* 1.23 (2020-02-02) query SVG data for glyphs; query whole kerning table (but only kern not GPOS) */ -/* 1.22 (2019-08-11) minimize missing-glyph duplication; fix kerning if both 'GPOS' and 'kern' are defined */ -/* 1.21 (2019-02-25) fix warning */ -/* 1.20 (2019-02-07) PackFontRange skips missing codepoints; GetScaleFontVMetrics() */ -/* 1.19 (2018-02-11) GPOS kerning, STBTT_fmod */ -/* 1.18 (2018-01-29) add missing function */ -/* 1.17 (2017-07-23) make more arguments const; doc fix */ -/* 1.16 (2017-07-12) SDF support */ -/* 1.15 (2017-03-03) make more arguments const */ -/* 1.14 (2017-01-16) num-fonts-in-TTC function */ -/* 1.13 (2017-01-02) support OpenType fonts, certain Apple fonts */ -/* 1.12 (2016-10-25) suppress warnings about casting away const with -Wcast-qual */ -/* 1.11 (2016-04-02) fix unused-variable warning */ -/* 1.10 (2016-04-02) user-defined fabs(); rare memory leak; remove duplicate typedef */ -/* 1.09 (2016-01-16) warning fix; avoid crash on outofmem; use allocation userdata properly */ -/* 1.08 (2015-09-13) document stbtt_Rasterize(); fixes for vertical & horizontal edges */ -/* 1.07 (2015-08-01) allow PackFontRanges to accept arrays of sparse codepoints; */ -/* variant PackFontRanges to pack and render in separate phases; */ -/* fix stbtt_GetFontOFfsetForIndex (never worked for non-0 input?); */ -/* fixed an assert() bug in the new rasterizer */ -/* replace assert() with STBTT_assert() in new rasterizer */ -/* */ -/* Full history can be found at the end of this file. */ -/* */ -/* LICENSE */ -/* */ -/* See end of file for license information. */ -/* */ -/* USAGE */ -/* */ -/* Include this file in whatever places need to refer to it. In ONE C/C++ */ -/* file, write: */ -/* #define STB_TRUETYPE_IMPLEMENTATION */ -/* before the #include of this file. This expands out the actual */ -/* implementation into that C/C++ file. */ -/* */ -/* To make the implementation private to the file that generates the implementation, */ -/* #define STBTT_STATIC */ -/* */ -/* Simple 3D API (don't ship this, but it's fine for tools and quick start) */ -/* stbtt_BakeFontBitmap() -- bake a font to a bitmap for use as texture */ -/* stbtt_GetBakedQuad() -- compute quad to draw for a given char */ -/* */ -/* Improved 3D API (more shippable): */ -/* #include "stb_rect_pack.h" -- optional, but you really want it */ -/* stbtt_PackBegin() */ -/* stbtt_PackSetOversampling() -- for improved quality on small fonts */ -/* stbtt_PackFontRanges() -- pack and renders */ -/* stbtt_PackEnd() */ -/* stbtt_GetPackedQuad() */ -/* */ -/* "Load" a font file from a memory buffer (you have to keep the buffer loaded) */ -/* stbtt_InitFont() */ -/* stbtt_GetFontOffsetForIndex() -- indexing for TTC font collections */ -/* stbtt_GetNumberOfFonts() -- number of fonts for TTC font collections */ -/* */ -/* Render a unicode codepoint to a bitmap */ -/* stbtt_GetCodepointBitmap() -- allocates and returns a bitmap */ -/* stbtt_MakeCodepointBitmap() -- renders into bitmap you provide */ -/* stbtt_GetCodepointBitmapBox() -- how big the bitmap must be */ -/* */ -/* Character advance/positioning */ -/* stbtt_GetCodepointHMetrics() */ -/* stbtt_GetFontVMetrics() */ -/* stbtt_GetFontVMetricsOS2() */ -/* stbtt_GetCodepointKernAdvance() */ -/* */ -/* Starting with version 1.06, the rasterizer was replaced with a new, */ -/* faster and generally-more-precise rasterizer. The new rasterizer more */ -/* accurately measures pixel coverage for anti-aliasing, except in the case */ -/* where multiple shapes overlap, in which case it overestimates the AA pixel */ -/* coverage. Thus, anti-aliasing of intersecting shapes may look wrong. If */ -/* this turns out to be a problem, you can re-enable the old rasterizer with */ -/* #define STBTT_RASTERIZER_VERSION 1 */ -/* which will incur about a 15% speed hit. */ -/* */ -/* ADDITIONAL DOCUMENTATION */ -/* */ -/* Immediately after this block comment are a series of sample programs. */ -/* */ -/* After the sample programs is the "header file" section. This section */ -/* includes documentation for each API function. */ -/* */ -/* Some important concepts to understand to use this library: */ -/* */ -/* Codepoint */ -/* Characters are defined by unicode codepoints, e.g. 65 is */ -/* uppercase A, 231 is lowercase c with a cedilla, 0x7e30 is */ -/* the hiragana for "ma". */ -/* */ -/* Glyph */ -/* A visual character shape (every codepoint is rendered as */ -/* some glyph) */ -/* */ -/* Glyph index */ -/* A font-specific integer ID representing a glyph */ -/* */ -/* Baseline */ -/* Glyph shapes are defined relative to a baseline, which is the */ -/* bottom of uppercase characters. Characters extend both above */ -/* and below the baseline. */ -/* */ -/* Current Point */ -/* As you draw text to the screen, you keep track of a "current point" */ -/* which is the origin of each character. The current point's vertical */ -/* position is the baseline. Even "baked fonts" use this model. */ -/* */ -/* Vertical Font Metrics */ -/* The vertical qualities of the font, used to vertically position */ -/* and space the characters. See docs for stbtt_GetFontVMetrics. */ -/* */ -/* Font Size in Pixels or Points */ -/* The preferred interface for specifying font sizes in stb_truetype */ -/* is to specify how tall the font's vertical extent should be in pixels. */ -/* If that sounds good enough, skip the next paragraph. */ -/* */ -/* Most font APIs instead use "points", which are a common typographic */ -/* measurement for describing font size, defined as 72 points per inch. */ -/* stb_truetype provides a point API for compatibility. However, true */ -/* "per inch" conventions don't make much sense on computer displays */ -/* since different monitors have different number of pixels per */ -/* inch. For example, Windows traditionally uses a convention that */ -/* there are 96 pixels per inch, thus making 'inch' measurements have */ -/* nothing to do with inches, and thus effectively defining a point to */ -/* be 1.333 pixels. Additionally, the TrueType font data provides */ -/* an explicit scale factor to scale a given font's glyphs to points, */ -/* but the author has observed that this scale factor is often wrong */ -/* for non-commercial fonts, thus making fonts scaled in points */ -/* according to the TrueType spec incoherently sized in practice. */ -/* */ -/* DETAILED USAGE: */ -/* */ -/* Scale: */ -/* Select how high you want the font to be, in points or pixels. */ -/* Call ScaleForPixelHeight or ScaleForMappingEmToPixels to compute */ -/* a scale factor SF that will be used by all other functions. */ -/* */ -/* Baseline: */ -/* You need to select a y-coordinate that is the baseline of where */ -/* your text will appear. Call GetFontBoundingBox to get the baseline-relative */ -/* bounding box for all characters. SF*-y0 will be the distance in pixels */ -/* that the worst-case character could extend above the baseline, so if */ -/* you want the top edge of characters to appear at the top of the */ -/* screen where y=0, then you would set the baseline to SF*-y0. */ -/* */ -/* Current point: */ -/* Set the current point where the first character will appear. The */ -/* first character could extend left of the current point; this is font */ -/* dependent. You can either choose a current point that is the leftmost */ -/* point and hope, or add some padding, or check the bounding box or */ -/* left-side-bearing of the first character to be displayed and set */ -/* the current point based on that. */ -/* */ -/* Displaying a character: */ -/* Compute the bounding box of the character. It will contain signed values */ -/* relative to . I.e. if it returns x0,y0,x1,y1, */ -/* then the character should be displayed in the rectangle from */ -/* to = 32 && *text < 128) { - stbtt_aligned_quad q; - stbtt_GetBakedQuad(cdata, 512,512, *text-32, &x,&y,&q,1);/* 1=opengl & d3d10+,0=d3d9 */ - glTexCoord2f(q.s0,q.t0); glVertex2f(q.x0,q.y0); - glTexCoord2f(q.s1,q.t0); glVertex2f(q.x1,q.y0); - glTexCoord2f(q.s1,q.t1); glVertex2f(q.x1,q.y1); - glTexCoord2f(q.s0,q.t1); glVertex2f(q.x0,q.y1); - } - ++text; - } - glEnd(); -} -#endif -/* */ -/* */ -/* //////////////////////////////////////////////////////////////////////////// */ -/* */ -/* Complete program (this compiles): get a single bitmap, print as ASCII art */ -/* */ -#if 0 -#include -#define STB_TRUETYPE_IMPLEMENTATION /* force following include to generate implementation */ -#include "stb_truetype.h" - -char ttf_buffer[1<<25]; - -int main(int argc, char **argv) -{ - stbtt_fontinfo font; - unsigned char *bitmap; - int w,h,i,j,c = (argc > 1 ? atoi(argv[1]) : 'a'), s = (argc > 2 ? atoi(argv[2]) : 20); - - fread(ttf_buffer, 1, 1<<25, fopen(argc > 3 ? argv[3] : "c:/windows/fonts/arialbd.ttf", "rb")); - - stbtt_InitFont(&font, ttf_buffer, stbtt_GetFontOffsetForIndex(ttf_buffer,0)); - bitmap = stbtt_GetCodepointBitmap(&font, 0,stbtt_ScaleForPixelHeight(&font, s), c, &w, &h, 0,0); - - for (j=0; j < h; ++j) { - for (i=0; i < w; ++i) - putchar(" .:ioVM@"[bitmap[j*w+i]>>5]); - putchar('\n'); - } - return 0; -} -#endif -/* */ -/* Output: */ -/* */ -/* .ii. */ -/* @@@@@@. */ -/* V@Mio@@o */ -/* :i. V@V */ -/* :oM@@M */ -/* :@@@MM@M */ -/* @@o o@M */ -/* :@@. M@M */ -/* @@@o@@@@ */ -/* :M@@V:@@. */ -/* */ -/* //////////////////////////////////////////////////////////////////////////// */ -/* */ -/* Complete program: print "Hello World!" banner, with bugs */ -/* */ -#if 0 -char buffer[24<<20]; -unsigned char screen[20][79]; - -int main(int arg, char **argv) -{ - stbtt_fontinfo font; - int i,j,ascent,baseline,ch=0; - float scale, xpos=2; /* leave a little padding in case the character extends left */ - char *text = "Heljo World!"; /* intentionally misspelled to show 'lj' brokenness */ - - fread(buffer, 1, 1000000, fopen("c:/windows/fonts/arialbd.ttf", "rb")); - stbtt_InitFont(&font, buffer, 0); - - scale = stbtt_ScaleForPixelHeight(&font, 15); - stbtt_GetFontVMetrics(&font, &ascent,0,0); - baseline = (int) (ascent*scale); - - while (text[ch]) { - int advance,lsb,x0,y0,x1,y1; - float x_shift = xpos - (float) floor(xpos); - stbtt_GetCodepointHMetrics(&font, text[ch], &advance, &lsb); - stbtt_GetCodepointBitmapBoxSubpixel(&font, text[ch], scale,scale,x_shift,0, &x0,&y0,&x1,&y1); - stbtt_MakeCodepointBitmapSubpixel(&font, &screen[baseline + y0][(int) xpos + x0], x1-x0,y1-y0, 79, scale,scale,x_shift,0, text[ch]); - /* note that this stomps the old data, so where character boxes overlap (e.g. 'lj') it's wrong */ - /* because this API is really for baking character bitmaps into textures. if you want to render */ - /* a sequence of characters, you really need to render each bitmap to a temp buffer, then */ - /* "alpha blend" that into the working buffer */ - xpos += (advance * scale); - if (text[ch+1]) - xpos += scale*stbtt_GetCodepointKernAdvance(&font, text[ch],text[ch+1]); - ++ch; - } - - for (j=0; j < 20; ++j) { - for (i=0; i < 78; ++i) - putchar(" .:ioVM@"[screen[j][i]>>5]); - putchar('\n'); - } - - return 0; -} -#endif - - -/* //////////////////////////////////////////////////////////////////////////// */ -/* //////////////////////////////////////////////////////////////////////////// */ -/* // */ -/* // INTEGRATION WITH YOUR CODEBASE */ -/* // */ -/* // The following sections allow you to supply alternate definitions */ -/* // of C library functions used by stb_truetype, e.g. if you don't */ -/* // link with the C runtime library. */ - -#ifdef STB_TRUETYPE_IMPLEMENTATION - /* #define your own (u)stbtt_int8/16/32 before including to override this */ - #ifndef stbtt_uint8 - typedef unsigned char stbtt_uint8; - typedef signed char stbtt_int8; - typedef unsigned short stbtt_uint16; - typedef signed short stbtt_int16; - typedef unsigned int stbtt_uint32; - typedef signed int stbtt_int32; - #endif - - typedef char stbtt__check_size32[sizeof(stbtt_int32)==4 ? 1 : -1]; - typedef char stbtt__check_size16[sizeof(stbtt_int16)==2 ? 1 : -1]; - - /* e.g. #define your own STBTT_ifloor/STBTT_iceil() to avoid math.h */ - #ifndef STBTT_ifloor - #include - #define STBTT_ifloor(x) ((int) floor(x)) - #define STBTT_iceil(x) ((int) ceil(x)) - #endif - - #ifndef STBTT_sqrt - #include - #define STBTT_sqrt(x) sqrt(x) - #define STBTT_pow(x,y) pow(x,y) - #endif - - #ifndef STBTT_fmod - #include - #define STBTT_fmod(x,y) fmod(x,y) - #endif - - #ifndef STBTT_cos - #include - #define STBTT_cos(x) cos(x) - #define STBTT_acos(x) acos(x) - #endif - - #ifndef STBTT_fabs - #include - #define STBTT_fabs(x) fabs(x) - #endif - - /* #define your own functions "STBTT_malloc" / "STBTT_free" to avoid malloc.h */ - #ifndef STBTT_malloc - #include - #define STBTT_malloc(x,u) ((void)(u),malloc(x)) - #define STBTT_free(x,u) ((void)(u),free(x)) - #endif - - #ifndef STBTT_assert - #include - #define STBTT_assert(x) assert(x) - #endif - - #ifndef STBTT_strlen - #include - #define STBTT_strlen(x) strlen(x) - #endif - - #ifndef STBTT_memcpy - #include - #define STBTT_memcpy memcpy - #define STBTT_memset memset - #endif -#endif - -/* ///////////////////////////////////////////////////////////////////////////// */ -/* ///////////////////////////////////////////////////////////////////////////// */ -/* // */ -/* // INTERFACE */ -/* // */ -/* // */ - -#ifndef __STB_INCLUDE_STB_TRUETYPE_H__ -#define __STB_INCLUDE_STB_TRUETYPE_H__ - -#ifdef STBTT_STATIC -#define STBTT_DEF static -#else -#define STBTT_DEF extern -#endif - -#ifdef __cplusplus -extern "C" { -#endif - -/* private structure */ -typedef struct -{ - unsigned char *data; - int cursor; - int size; -} stbtt__buf; - -/* //////////////////////////////////////////////////////////////////////////// */ -/* */ -/* TEXTURE BAKING API */ -/* */ -/* If you use this API, you only have to call two functions ever. */ -/* */ - -typedef struct -{ - unsigned short x0,y0,x1,y1; /* coordinates of bbox in bitmap */ - float xoff,yoff,xadvance; -} stbtt_bakedchar; - -STBTT_DEF int stbtt_BakeFontBitmap(const unsigned char *data, int offset, /* font location (use offset=0 for plain .ttf) */ - float pixel_height, /* height of font in pixels */ - unsigned char *pixels, int pw, int ph, /* bitmap to be filled in */ - int first_char, int num_chars, /* characters to bake */ - stbtt_bakedchar *chardata); /* you allocate this, it's num_chars long */ -/* if return is positive, the first unused row of the bitmap */ -/* if return is negative, returns the negative of the number of characters that fit */ -/* if return is 0, no characters fit and no rows were used */ -/* This uses a very crappy packing. */ - -typedef struct -{ - float x0,y0,s0,t0; /* top-left */ - float x1,y1,s1,t1; /* bottom-right */ -} stbtt_aligned_quad; - -STBTT_DEF void stbtt_GetBakedQuad(const stbtt_bakedchar *chardata, int pw, int ph, /* same data as above */ - int char_index, /* character to display */ - float *xpos, float *ypos, /* pointers to current position in screen pixel space */ - stbtt_aligned_quad *q, /* output: quad to draw */ - int opengl_fillrule); /* true if opengl fill rule; false if DX9 or earlier */ -/* Call GetBakedQuad with char_index = 'character - first_char', and it */ -/* creates the quad you need to draw and advances the current position. */ -/* */ -/* The coordinate system used assumes y increases downwards. */ -/* */ -/* Characters will extend both above and below the current position; */ -/* see discussion of "BASELINE" above. */ -/* */ -/* It's inefficient; you might want to c&p it and optimize it. */ - -STBTT_DEF void stbtt_GetScaledFontVMetrics(const unsigned char *fontdata, int index, float size, float *ascent, float *descent, float *lineGap); -/* Query the font vertical metrics without having to create a font first. */ - - -/* //////////////////////////////////////////////////////////////////////////// */ -/* */ -/* NEW TEXTURE BAKING API */ -/* */ -/* This provides options for packing multiple fonts into one atlas, not */ -/* perfectly but better than nothing. */ - -typedef struct -{ - unsigned short x0,y0,x1,y1; /* coordinates of bbox in bitmap */ - float xoff,yoff,xadvance; - float xoff2,yoff2; -} stbtt_packedchar; - -typedef struct stbtt_pack_context stbtt_pack_context; -typedef struct stbtt_fontinfo stbtt_fontinfo; -#ifndef STB_RECT_PACK_VERSION -typedef struct stbrp_rect stbrp_rect; -#endif - -STBTT_DEF int stbtt_PackBegin(stbtt_pack_context *spc, unsigned char *pixels, int width, int height, int stride_in_bytes, int padding, void *alloc_context); -/* Initializes a packing context stored in the passed-in stbtt_pack_context. */ -/* Future calls using this context will pack characters into the bitmap passed */ -/* in here: a 1-channel bitmap that is width * height. stride_in_bytes is */ -/* the distance from one row to the next (or 0 to mean they are packed tightly */ -/* together). "padding" is the amount of padding to leave between each */ -/* character (normally you want '1' for bitmaps you'll use as textures with */ -/* bilinear filtering). */ -/* */ -/* Returns 0 on failure, 1 on success. */ - -STBTT_DEF void stbtt_PackEnd (stbtt_pack_context *spc); -/* Cleans up the packing context and frees all memory. */ - -#define STBTT_POINT_SIZE(x) (-(x)) - -STBTT_DEF int stbtt_PackFontRange(stbtt_pack_context *spc, const unsigned char *fontdata, int font_index, float font_size, - int first_unicode_char_in_range, int num_chars_in_range, stbtt_packedchar *chardata_for_range); -/* Creates character bitmaps from the font_index'th font found in fontdata (use */ -/* font_index=0 if you don't know what that is). It creates num_chars_in_range */ -/* bitmaps for characters with unicode values starting at first_unicode_char_in_range */ -/* and increasing. Data for how to render them is stored in chardata_for_range; */ -/* pass these to stbtt_GetPackedQuad to get back renderable quads. */ -/* */ -/* font_size is the full height of the character from ascender to descender, */ -/* as computed by stbtt_ScaleForPixelHeight. To use a point size as computed */ -/* by stbtt_ScaleForMappingEmToPixels, wrap the point size in STBTT_POINT_SIZE() */ -/* and pass that result as 'font_size': */ -/* ..., 20 , ... // font max minus min y is 20 pixels tall */ -/* ..., STBTT_POINT_SIZE(20), ... // 'M' is 20 pixels tall */ - -typedef struct -{ - float font_size; - int first_unicode_codepoint_in_range; /* if non-zero, then the chars are continuous, and this is the first codepoint */ - int *array_of_unicode_codepoints; /* if non-zero, then this is an array of unicode codepoints */ - int num_chars; - stbtt_packedchar *chardata_for_range; /* output */ - unsigned char h_oversample, v_oversample; /* don't set these, they're used internally */ -} stbtt_pack_range; - -STBTT_DEF int stbtt_PackFontRanges(stbtt_pack_context *spc, const unsigned char *fontdata, int font_index, stbtt_pack_range *ranges, int num_ranges); -/* Creates character bitmaps from multiple ranges of characters stored in */ -/* ranges. This will usually create a better-packed bitmap than multiple */ -/* calls to stbtt_PackFontRange. Note that you can call this multiple */ -/* times within a single PackBegin/PackEnd. */ - -STBTT_DEF void stbtt_PackSetOversampling(stbtt_pack_context *spc, unsigned int h_oversample, unsigned int v_oversample); -/* Oversampling a font increases the quality by allowing higher-quality subpixel */ -/* positioning, and is especially valuable at smaller text sizes. */ -/* */ -/* This function sets the amount of oversampling for all following calls to */ -/* stbtt_PackFontRange(s) or stbtt_PackFontRangesGatherRects for a given */ -/* pack context. The default (no oversampling) is achieved by h_oversample=1 */ -/* and v_oversample=1. The total number of pixels required is */ -/* h_oversample*v_oversample larger than the default; for example, 2x2 */ -/* oversampling requires 4x the storage of 1x1. For best results, render */ -/* oversampled textures with bilinear filtering. Look at the readme in */ -/* stb/tests/oversample for information about oversampled fonts */ -/* */ -/* To use with PackFontRangesGather etc., you must set it before calls */ -/* call to PackFontRangesGatherRects. */ - -STBTT_DEF void stbtt_PackSetSkipMissingCodepoints(stbtt_pack_context *spc, int skip); -/* If skip != 0, this tells stb_truetype to skip any codepoints for which */ -/* there is no corresponding glyph. If skip=0, which is the default, then */ -/* codepoints without a glyph recived the font's "missing character" glyph, */ -/* typically an empty box by convention. */ - -STBTT_DEF void stbtt_GetPackedQuad(const stbtt_packedchar *chardata, int pw, int ph, /* same data as above */ - int char_index, /* character to display */ - float *xpos, float *ypos, /* pointers to current position in screen pixel space */ - stbtt_aligned_quad *q, /* output: quad to draw */ - int align_to_integer); - -STBTT_DEF int stbtt_PackFontRangesGatherRects(stbtt_pack_context *spc, const stbtt_fontinfo *info, stbtt_pack_range *ranges, int num_ranges, stbrp_rect *rects); -STBTT_DEF void stbtt_PackFontRangesPackRects(stbtt_pack_context *spc, stbrp_rect *rects, int num_rects); -STBTT_DEF int stbtt_PackFontRangesRenderIntoRects(stbtt_pack_context *spc, const stbtt_fontinfo *info, stbtt_pack_range *ranges, int num_ranges, stbrp_rect *rects); -/* Calling these functions in sequence is roughly equivalent to calling */ -/* stbtt_PackFontRanges(). If you more control over the packing of multiple */ -/* fonts, or if you want to pack custom data into a font texture, take a look */ -/* at the source to of stbtt_PackFontRanges() and create a custom version */ -/* using these functions, e.g. call GatherRects multiple times, */ -/* building up a single array of rects, then call PackRects once, */ -/* then call RenderIntoRects repeatedly. This may result in a */ -/* better packing than calling PackFontRanges multiple times */ -/* (or it may not). */ - -/* this is an opaque structure that you shouldn't mess with which holds */ -/* all the context needed from PackBegin to PackEnd. */ -struct stbtt_pack_context { - void *user_allocator_context; - void *pack_info; - int width; - int height; - int stride_in_bytes; - int padding; - int skip_missing; - unsigned int h_oversample, v_oversample; - unsigned char *pixels; - void *nodes; -}; - -/* //////////////////////////////////////////////////////////////////////////// */ -/* */ -/* FONT LOADING */ -/* */ -/* */ - -STBTT_DEF int stbtt_GetNumberOfFonts(const unsigned char *data); -/* This function will determine the number of fonts in a font file. TrueType */ -/* collection (.ttc) files may contain multiple fonts, while TrueType font */ -/* (.ttf) files only contain one font. The number of fonts can be used for */ -/* indexing with the previous function where the index is between zero and one */ -/* less than the total fonts. If an error occurs, -1 is returned. */ - -STBTT_DEF int stbtt_GetFontOffsetForIndex(const unsigned char *data, int index); -/* Each .ttf/.ttc file may have more than one font. Each font has a sequential */ -/* index number starting from 0. Call this function to get the font offset for */ -/* a given index; it returns -1 if the index is out of range. A regular .ttf */ -/* file will only define one font and it always be at offset 0, so it will */ -/* return '0' for index 0, and -1 for all other indices. */ - -/* The following structure is defined publicly so you can declare one on */ -/* the stack or as a global or etc, but you should treat it as opaque. */ -struct stbtt_fontinfo -{ - void * userdata; - unsigned char * data; /* pointer to .ttf file */ - int fontstart; /* offset of start of font */ - - int numGlyphs; /* number of glyphs, needed for range checking */ - - int loca,head,glyf,hhea,hmtx,kern,gpos,svg; /* table locations as offset from start of .ttf */ - int index_map; /* a cmap mapping for our chosen character encoding */ - int indexToLocFormat; /* format needed to map from glyph index to glyph */ - - stbtt__buf cff; /* cff font data */ - stbtt__buf charstrings; /* the charstring index */ - stbtt__buf gsubrs; /* global charstring subroutines index */ - stbtt__buf subrs; /* private charstring subroutines index */ - stbtt__buf fontdicts; /* array of font dicts */ - stbtt__buf fdselect; /* map from glyph to fontdict */ -}; - -STBTT_DEF int stbtt_InitFont(stbtt_fontinfo *info, const unsigned char *data, int offset); -/* Given an offset into the file that defines a font, this function builds */ -/* the necessary cached info for the rest of the system. You must allocate */ -/* the stbtt_fontinfo yourself, and stbtt_InitFont will fill it out. You don't */ -/* need to do anything special to free it, because the contents are pure */ -/* value data with no additional data structures. Returns 0 on failure. */ - - -/* //////////////////////////////////////////////////////////////////////////// */ -/* */ -/* CHARACTER TO GLYPH-INDEX CONVERSIOn */ - -STBTT_DEF int stbtt_FindGlyphIndex(const stbtt_fontinfo *info, int unicode_codepoint); -/* If you're going to perform multiple operations on the same character */ -/* and you want a speed-up, call this function with the character you're */ -/* going to process, then use glyph-based functions instead of the */ -/* codepoint-based functions. */ -/* Returns 0 if the character codepoint is not defined in the font. */ - - -/* //////////////////////////////////////////////////////////////////////////// */ -/* */ -/* CHARACTER PROPERTIES */ -/* */ - -STBTT_DEF float stbtt_ScaleForPixelHeight(const stbtt_fontinfo *info, float pixels); -/* computes a scale factor to produce a font whose "height" is 'pixels' tall. */ -/* Height is measured as the distance from the highest ascender to the lowest */ -/* descender; in other words, it's equivalent to calling stbtt_GetFontVMetrics */ -/* and computing: */ -/* scale = pixels / (ascent - descent) */ -/* so if you prefer to measure height by the ascent only, use a similar calculation. */ - -STBTT_DEF float stbtt_ScaleForMappingEmToPixels(const stbtt_fontinfo *info, float pixels); -/* computes a scale factor to produce a font whose EM size is mapped to */ -/* 'pixels' tall. This is probably what traditional APIs compute, but */ -/* I'm not positive. */ - -STBTT_DEF void stbtt_GetFontVMetrics(const stbtt_fontinfo *info, int *ascent, int *descent, int *lineGap); -/* ascent is the coordinate above the baseline the font extends; descent */ -/* is the coordinate below the baseline the font extends (i.e. it is typically negative) */ -/* lineGap is the spacing between one row's descent and the next row's ascent... */ -/* so you should advance the vertical position by "*ascent - *descent + *lineGap" */ -/* these are expressed in unscaled coordinates, so you must multiply by */ -/* the scale factor for a given size */ - -STBTT_DEF int stbtt_GetFontVMetricsOS2(const stbtt_fontinfo *info, int *typoAscent, int *typoDescent, int *typoLineGap); -/* analogous to GetFontVMetrics, but returns the "typographic" values from the OS/2 */ -/* table (specific to MS/Windows TTF files). */ -/* */ -/* Returns 1 on success (table present), 0 on failure. */ - -STBTT_DEF void stbtt_GetFontBoundingBox(const stbtt_fontinfo *info, int *x0, int *y0, int *x1, int *y1); -/* the bounding box around all possible characters */ - -STBTT_DEF void stbtt_GetCodepointHMetrics(const stbtt_fontinfo *info, int codepoint, int *advanceWidth, int *leftSideBearing); -/* leftSideBearing is the offset from the current horizontal position to the left edge of the character */ -/* advanceWidth is the offset from the current horizontal position to the next horizontal position */ -/* these are expressed in unscaled coordinates */ - -STBTT_DEF int stbtt_GetCodepointKernAdvance(const stbtt_fontinfo *info, int ch1, int ch2); -/* an additional amount to add to the 'advance' value between ch1 and ch2 */ - -STBTT_DEF int stbtt_GetCodepointBox(const stbtt_fontinfo *info, int codepoint, int *x0, int *y0, int *x1, int *y1); -/* Gets the bounding box of the visible part of the glyph, in unscaled coordinates */ - -STBTT_DEF void stbtt_GetGlyphHMetrics(const stbtt_fontinfo *info, int glyph_index, int *advanceWidth, int *leftSideBearing); -STBTT_DEF int stbtt_GetGlyphKernAdvance(const stbtt_fontinfo *info, int glyph1, int glyph2); -STBTT_DEF int stbtt_GetGlyphBox(const stbtt_fontinfo *info, int glyph_index, int *x0, int *y0, int *x1, int *y1); -/* as above, but takes one or more glyph indices for greater efficiency */ - -typedef struct stbtt_kerningentry -{ - int glyph1; /* use stbtt_FindGlyphIndex */ - int glyph2; - int advance; -} stbtt_kerningentry; - -STBTT_DEF int stbtt_GetKerningTableLength(const stbtt_fontinfo *info); -STBTT_DEF int stbtt_GetKerningTable(const stbtt_fontinfo *info, stbtt_kerningentry* table, int table_length); -/* Retrieves a complete list of all of the kerning pairs provided by the font */ -/* stbtt_GetKerningTable never writes more than table_length entries and returns how many entries it did write. */ -/* The table will be sorted by (a.glyph1 == b.glyph1)?(a.glyph2 < b.glyph2):(a.glyph1 < b.glyph1) */ - -/* //////////////////////////////////////////////////////////////////////////// */ -/* */ -/* GLYPH SHAPES (you probably don't need these, but they have to go before */ -/* the bitmaps for C declaration-order reasons) */ -/* */ - -#ifndef STBTT_vmove /* you can predefine these to use different values (but why?) */ - enum { - STBTT_vmove=1, - STBTT_vline, - STBTT_vcurve, - STBTT_vcubic - }; -#endif - -#ifndef stbtt_vertex /* you can predefine this to use different values */ - /* (we share this with other code at RAD) */ - #define stbtt_vertex_type short /* can't use stbtt_int16 because that's not visible in the header file */ - typedef struct - { - stbtt_vertex_type x,y,cx,cy,cx1,cy1; - unsigned char type,padding; - } stbtt_vertex; -#endif - -STBTT_DEF int stbtt_IsGlyphEmpty(const stbtt_fontinfo *info, int glyph_index); -/* returns non-zero if nothing is drawn for this glyph */ - -STBTT_DEF int stbtt_GetCodepointShape(const stbtt_fontinfo *info, int unicode_codepoint, stbtt_vertex **vertices); -STBTT_DEF int stbtt_GetGlyphShape(const stbtt_fontinfo *info, int glyph_index, stbtt_vertex **vertices); -/* returns # of vertices and fills *vertices with the pointer to them */ -/* these are expressed in "unscaled" coordinates */ -/* */ -/* The shape is a series of contours. Each one starts with */ -/* a STBTT_moveto, then consists of a series of mixed */ -/* STBTT_lineto and STBTT_curveto segments. A lineto */ -/* draws a line from previous endpoint to its x,y; a curveto */ -/* draws a quadratic bezier from previous endpoint to */ -/* its x,y, using cx,cy as the bezier control point. */ - -STBTT_DEF void stbtt_FreeShape(const stbtt_fontinfo *info, stbtt_vertex *vertices); -/* frees the data allocated above */ - -STBTT_DEF unsigned char *stbtt_FindSVGDoc(const stbtt_fontinfo *info, int gl); -STBTT_DEF int stbtt_GetCodepointSVG(const stbtt_fontinfo *info, int unicode_codepoint, const char **svg); -STBTT_DEF int stbtt_GetGlyphSVG(const stbtt_fontinfo *info, int gl, const char **svg); -/* fills svg with the character's SVG data. */ -/* returns data size or 0 if SVG not found. */ - -/* //////////////////////////////////////////////////////////////////////////// */ -/* */ -/* BITMAP RENDERING */ -/* */ - -STBTT_DEF void stbtt_FreeBitmap(unsigned char *bitmap, void *userdata); -/* frees the bitmap allocated below */ - -STBTT_DEF unsigned char *stbtt_GetCodepointBitmap(const stbtt_fontinfo *info, float scale_x, float scale_y, int codepoint, int *width, int *height, int *xoff, int *yoff); -/* allocates a large-enough single-channel 8bpp bitmap and renders the */ -/* specified character/glyph at the specified scale into it, with */ -/* antialiasing. 0 is no coverage (transparent), 255 is fully covered (opaque). */ -/* *width & *height are filled out with the width & height of the bitmap, */ -/* which is stored left-to-right, top-to-bottom. */ -/* */ -/* xoff/yoff are the offset it pixel space from the glyph origin to the top-left of the bitmap */ - -STBTT_DEF unsigned char *stbtt_GetCodepointBitmapSubpixel(const stbtt_fontinfo *info, float scale_x, float scale_y, float shift_x, float shift_y, int codepoint, int *width, int *height, int *xoff, int *yoff); -/* the same as stbtt_GetCodepoitnBitmap, but you can specify a subpixel */ -/* shift for the character */ - -STBTT_DEF void stbtt_MakeCodepointBitmap(const stbtt_fontinfo *info, unsigned char *output, int out_w, int out_h, int out_stride, float scale_x, float scale_y, int codepoint); -/* the same as stbtt_GetCodepointBitmap, but you pass in storage for the bitmap */ -/* in the form of 'output', with row spacing of 'out_stride' bytes. the bitmap */ -/* is clipped to out_w/out_h bytes. Call stbtt_GetCodepointBitmapBox to get the */ -/* width and height and positioning info for it first. */ - -STBTT_DEF void stbtt_MakeCodepointBitmapSubpixel(const stbtt_fontinfo *info, unsigned char *output, int out_w, int out_h, int out_stride, float scale_x, float scale_y, float shift_x, float shift_y, int codepoint); -/* same as stbtt_MakeCodepointBitmap, but you can specify a subpixel */ -/* shift for the character */ - -STBTT_DEF void stbtt_MakeCodepointBitmapSubpixelPrefilter(const stbtt_fontinfo *info, unsigned char *output, int out_w, int out_h, int out_stride, float scale_x, float scale_y, float shift_x, float shift_y, int oversample_x, int oversample_y, float *sub_x, float *sub_y, int codepoint); -/* same as stbtt_MakeCodepointBitmapSubpixel, but prefiltering */ -/* is performed (see stbtt_PackSetOversampling) */ - -STBTT_DEF void stbtt_GetCodepointBitmapBox(const stbtt_fontinfo *font, int codepoint, float scale_x, float scale_y, int *ix0, int *iy0, int *ix1, int *iy1); -/* get the bbox of the bitmap centered around the glyph origin; so the */ -/* bitmap width is ix1-ix0, height is iy1-iy0, and location to place */ -/* the bitmap top left is (leftSideBearing*scale,iy0). */ -/* (Note that the bitmap uses y-increases-down, but the shape uses */ -/* y-increases-up, so CodepointBitmapBox and CodepointBox are inverted.) */ - -STBTT_DEF void stbtt_GetCodepointBitmapBoxSubpixel(const stbtt_fontinfo *font, int codepoint, float scale_x, float scale_y, float shift_x, float shift_y, int *ix0, int *iy0, int *ix1, int *iy1); -/* same as stbtt_GetCodepointBitmapBox, but you can specify a subpixel */ -/* shift for the character */ - -/* the following functions are equivalent to the above functions, but operate */ -/* on glyph indices instead of Unicode codepoints (for efficiency) */ -STBTT_DEF unsigned char *stbtt_GetGlyphBitmap(const stbtt_fontinfo *info, float scale_x, float scale_y, int glyph, int *width, int *height, int *xoff, int *yoff); -STBTT_DEF unsigned char *stbtt_GetGlyphBitmapSubpixel(const stbtt_fontinfo *info, float scale_x, float scale_y, float shift_x, float shift_y, int glyph, int *width, int *height, int *xoff, int *yoff); -STBTT_DEF void stbtt_MakeGlyphBitmap(const stbtt_fontinfo *info, unsigned char *output, int out_w, int out_h, int out_stride, float scale_x, float scale_y, int glyph); -STBTT_DEF void stbtt_MakeGlyphBitmapSubpixel(const stbtt_fontinfo *info, unsigned char *output, int out_w, int out_h, int out_stride, float scale_x, float scale_y, float shift_x, float shift_y, int glyph); -STBTT_DEF void stbtt_MakeGlyphBitmapSubpixelPrefilter(const stbtt_fontinfo *info, unsigned char *output, int out_w, int out_h, int out_stride, float scale_x, float scale_y, float shift_x, float shift_y, int oversample_x, int oversample_y, float *sub_x, float *sub_y, int glyph); -STBTT_DEF void stbtt_GetGlyphBitmapBox(const stbtt_fontinfo *font, int glyph, float scale_x, float scale_y, int *ix0, int *iy0, int *ix1, int *iy1); -STBTT_DEF void stbtt_GetGlyphBitmapBoxSubpixel(const stbtt_fontinfo *font, int glyph, float scale_x, float scale_y,float shift_x, float shift_y, int *ix0, int *iy0, int *ix1, int *iy1); - - -/* @TODO: don't expose this structure */ -typedef struct -{ - int w,h,stride; - unsigned char *pixels; -} stbtt__bitmap; - -/* rasterize a shape with quadratic beziers into a bitmap */ -STBTT_DEF void stbtt_Rasterize(stbtt__bitmap *result, /* 1-channel bitmap to draw into */ - float flatness_in_pixels, /* allowable error of curve in pixels */ - stbtt_vertex *vertices, /* array of vertices defining shape */ - int num_verts, /* number of vertices in above array */ - float scale_x, float scale_y, /* scale applied to input vertices */ - float shift_x, float shift_y, /* translation applied to input vertices */ - int x_off, int y_off, /* another translation applied to input */ - int invert, /* if non-zero, vertically flip shape */ - void *userdata); /* context for to STBTT_MALLOC */ - -/* //////////////////////////////////////////////////////////////////////////// */ -/* */ -/* Signed Distance Function (or Field) rendering */ - -STBTT_DEF void stbtt_FreeSDF(unsigned char *bitmap, void *userdata); -/* frees the SDF bitmap allocated below */ - -STBTT_DEF unsigned char * stbtt_GetGlyphSDF(const stbtt_fontinfo *info, float scale, int glyph, int padding, unsigned char onedge_value, float pixel_dist_scale, int *width, int *height, int *xoff, int *yoff); -STBTT_DEF unsigned char * stbtt_GetCodepointSDF(const stbtt_fontinfo *info, float scale, int codepoint, int padding, unsigned char onedge_value, float pixel_dist_scale, int *width, int *height, int *xoff, int *yoff); -/* These functions compute a discretized SDF field for a single character, suitable for storing */ -/* in a single-channel texture, sampling with bilinear filtering, and testing against */ -/* larger than some threshold to produce scalable fonts. */ -/* info -- the font */ -/* scale -- controls the size of the resulting SDF bitmap, same as it would be creating a regular bitmap */ -/* glyph/codepoint -- the character to generate the SDF for */ -/* padding -- extra "pixels" around the character which are filled with the distance to the character (not 0), */ -/* which allows effects like bit outlines */ -/* onedge_value -- value 0-255 to test the SDF against to reconstruct the character (i.e. the isocontour of the character) */ -/* pixel_dist_scale -- what value the SDF should increase by when moving one SDF "pixel" away from the edge (on the 0..255 scale) */ -/* if positive, > onedge_value is inside; if negative, < onedge_value is inside */ -/* width,height -- output height & width of the SDF bitmap (including padding) */ -/* xoff,yoff -- output origin of the character */ -/* return value -- a 2D array of bytes 0..255, width*height in size */ -/* */ -/* pixel_dist_scale & onedge_value are a scale & bias that allows you to make */ -/* optimal use of the limited 0..255 for your application, trading off precision */ -/* and special effects. SDF values outside the range 0..255 are clamped to 0..255. */ -/* */ -/* Example: */ -/* scale = stbtt_ScaleForPixelHeight(22) */ -/* padding = 5 */ -/* onedge_value = 180 */ -/* pixel_dist_scale = 180/5.0 = 36.0 */ -/* */ -/* This will create an SDF bitmap in which the character is about 22 pixels */ -/* high but the whole bitmap is about 22+5+5=32 pixels high. To produce a filled */ -/* shape, sample the SDF at each pixel and fill the pixel if the SDF value */ -/* is greater than or equal to 180/255. (You'll actually want to antialias, */ -/* which is beyond the scope of this example.) Additionally, you can compute */ -/* offset outlines (e.g. to stroke the character border inside & outside, */ -/* or only outside). For example, to fill outside the character up to 3 SDF */ -/* pixels, you would compare against (180-36.0*3)/255 = 72/255. The above */ -/* choice of variables maps a range from 5 pixels outside the shape to */ -/* 2 pixels inside the shape to 0..255; this is intended primarily for apply */ -/* outside effects only (the interior range is needed to allow proper */ -/* antialiasing of the font at *smaller* sizes) */ -/* */ -/* The function computes the SDF analytically at each SDF pixel, not by e.g. */ -/* building a higher-res bitmap and approximating it. In theory the quality */ -/* should be as high as possible for an SDF of this size & representation, but */ -/* unclear if this is true in practice (perhaps building a higher-res bitmap */ -/* and computing from that can allow drop-out prevention). */ -/* */ -/* The algorithm has not been optimized at all, so expect it to be slow */ -/* if computing lots of characters or very large sizes. */ - - - -/* //////////////////////////////////////////////////////////////////////////// */ -/* */ -/* Finding the right font... */ -/* */ -/* You should really just solve this offline, keep your own tables */ -/* of what font is what, and don't try to get it out of the .ttf file. */ -/* That's because getting it out of the .ttf file is really hard, because */ -/* the names in the file can appear in many possible encodings, in many */ -/* possible languages, and e.g. if you need a case-insensitive comparison, */ -/* the details of that depend on the encoding & language in a complex way */ -/* (actually underspecified in truetype, but also gigantic). */ -/* */ -/* But you can use the provided functions in two possible ways: */ -/* stbtt_FindMatchingFont() will use *case-sensitive* comparisons on */ -/* unicode-encoded names to try to find the font you want; */ -/* you can run this before calling stbtt_InitFont() */ -/* */ -/* stbtt_GetFontNameString() lets you get any of the various strings */ -/* from the file yourself and do your own comparisons on them. */ -/* You have to have called stbtt_InitFont() first. */ - - -STBTT_DEF int stbtt_FindMatchingFont(const unsigned char *fontdata, const char *name, int flags); -/* returns the offset (not index) of the font that matches, or -1 if none */ -/* if you use STBTT_MACSTYLE_DONTCARE, use a font name like "Arial Bold". */ -/* if you use any other flag, use a font name like "Arial"; this checks */ -/* the 'macStyle' header field; i don't know if fonts set this consistently */ -#define STBTT_MACSTYLE_DONTCARE 0 -#define STBTT_MACSTYLE_BOLD 1 -#define STBTT_MACSTYLE_ITALIC 2 -#define STBTT_MACSTYLE_UNDERSCORE 4 -#define STBTT_MACSTYLE_NONE 8 /* <= not same as 0, this makes us check the bitfield is 0 */ - -STBTT_DEF int stbtt_CompareUTF8toUTF16_bigendian(const char *s1, int len1, const char *s2, int len2); -/* returns 1/0 whether the first string interpreted as utf8 is identical to */ -/* the second string interpreted as big-endian utf16... useful for strings from next func */ - -STBTT_DEF const char *stbtt_GetFontNameString(const stbtt_fontinfo *font, int *length, int platformID, int encodingID, int languageID, int nameID); -/* returns the string (which may be big-endian double byte, e.g. for unicode) */ -/* and puts the length in bytes in *length. */ -/* */ -/* some of the values for the IDs are below; for more see the truetype spec: */ -/* http://developer.apple.com/textfonts/TTRefMan/RM06/Chap6name.html */ -/* http://www.microsoft.com/typography/otspec/name.htm */ - -enum { /* platformID */ - STBTT_PLATFORM_ID_UNICODE =0, - STBTT_PLATFORM_ID_MAC =1, - STBTT_PLATFORM_ID_ISO =2, - STBTT_PLATFORM_ID_MICROSOFT =3 -}; - -enum { /* encodingID for STBTT_PLATFORM_ID_UNICODE */ - STBTT_UNICODE_EID_UNICODE_1_0 =0, - STBTT_UNICODE_EID_UNICODE_1_1 =1, - STBTT_UNICODE_EID_ISO_10646 =2, - STBTT_UNICODE_EID_UNICODE_2_0_BMP=3, - STBTT_UNICODE_EID_UNICODE_2_0_FULL=4 -}; - -enum { /* encodingID for STBTT_PLATFORM_ID_MICROSOFT */ - STBTT_MS_EID_SYMBOL =0, - STBTT_MS_EID_UNICODE_BMP =1, - STBTT_MS_EID_SHIFTJIS =2, - STBTT_MS_EID_UNICODE_FULL =10 -}; - -enum { /* encodingID for STBTT_PLATFORM_ID_MAC; same as Script Manager codes */ - STBTT_MAC_EID_ROMAN =0, STBTT_MAC_EID_ARABIC =4, - STBTT_MAC_EID_JAPANESE =1, STBTT_MAC_EID_HEBREW =5, - STBTT_MAC_EID_CHINESE_TRAD =2, STBTT_MAC_EID_GREEK =6, - STBTT_MAC_EID_KOREAN =3, STBTT_MAC_EID_RUSSIAN =7 -}; - -enum { /* languageID for STBTT_PLATFORM_ID_MICROSOFT; same as LCID... */ - /* problematic because there are e.g. 16 english LCIDs and 16 arabic LCIDs */ - STBTT_MS_LANG_ENGLISH =0x0409, STBTT_MS_LANG_ITALIAN =0x0410, - STBTT_MS_LANG_CHINESE =0x0804, STBTT_MS_LANG_JAPANESE =0x0411, - STBTT_MS_LANG_DUTCH =0x0413, STBTT_MS_LANG_KOREAN =0x0412, - STBTT_MS_LANG_FRENCH =0x040c, STBTT_MS_LANG_RUSSIAN =0x0419, - STBTT_MS_LANG_GERMAN =0x0407, STBTT_MS_LANG_SPANISH =0x0409, - STBTT_MS_LANG_HEBREW =0x040d, STBTT_MS_LANG_SWEDISH =0x041D -}; - -enum { /* languageID for STBTT_PLATFORM_ID_MAC */ - STBTT_MAC_LANG_ENGLISH =0 , STBTT_MAC_LANG_JAPANESE =11, - STBTT_MAC_LANG_ARABIC =12, STBTT_MAC_LANG_KOREAN =23, - STBTT_MAC_LANG_DUTCH =4 , STBTT_MAC_LANG_RUSSIAN =32, - STBTT_MAC_LANG_FRENCH =1 , STBTT_MAC_LANG_SPANISH =6 , - STBTT_MAC_LANG_GERMAN =2 , STBTT_MAC_LANG_SWEDISH =5 , - STBTT_MAC_LANG_HEBREW =10, STBTT_MAC_LANG_CHINESE_SIMPLIFIED =33, - STBTT_MAC_LANG_ITALIAN =3 , STBTT_MAC_LANG_CHINESE_TRAD =19 -}; - -#ifdef __cplusplus -} -#endif - -#endif /* __STB_INCLUDE_STB_TRUETYPE_H__ */ - -/* ///////////////////////////////////////////////////////////////////////////// */ -/* ///////////////////////////////////////////////////////////////////////////// */ -/* // */ -/* // IMPLEMENTATION */ -/* // */ -/* // */ - -#ifdef STB_TRUETYPE_IMPLEMENTATION - -#ifndef STBTT_MAX_OVERSAMPLE -#define STBTT_MAX_OVERSAMPLE 8 -#endif - -#if STBTT_MAX_OVERSAMPLE > 255 -#error "STBTT_MAX_OVERSAMPLE cannot be > 255" -#endif - -typedef int stbtt__test_oversample_pow2[(STBTT_MAX_OVERSAMPLE & (STBTT_MAX_OVERSAMPLE-1)) == 0 ? 1 : -1]; - -#ifndef STBTT_RASTERIZER_VERSION -#define STBTT_RASTERIZER_VERSION 2 -#endif - -#ifdef _MSC_VER -#define STBTT__NOTUSED(v) (void)(v) -#else -#define STBTT__NOTUSED(v) (void)sizeof(v) -#endif - -/* //////////////////////////////////////////////////////////////////////// */ -/* */ -/* stbtt__buf helpers to parse data from file */ -/* */ - -static stbtt_uint8 stbtt__buf_get8(stbtt__buf *b) -{ - if (b->cursor >= b->size) - return 0; - return b->data[b->cursor++]; -} - -static stbtt_uint8 stbtt__buf_peek8(stbtt__buf *b) -{ - if (b->cursor >= b->size) - return 0; - return b->data[b->cursor]; -} - -static void stbtt__buf_seek(stbtt__buf *b, int o) -{ - STBTT_assert(!(o > b->size || o < 0)); - b->cursor = (o > b->size || o < 0) ? b->size : o; -} - -static void stbtt__buf_skip(stbtt__buf *b, int o) -{ - stbtt__buf_seek(b, b->cursor + o); -} - -static stbtt_uint32 stbtt__buf_get(stbtt__buf *b, int n) -{ - stbtt_uint32 v = 0; - int i; - STBTT_assert(n >= 1 && n <= 4); - for (i = 0; i < n; i++) - v = (v << 8) | stbtt__buf_get8(b); - return v; -} - -static stbtt__buf stbtt__new_buf(const void *p, size_t size) -{ - stbtt__buf r; - STBTT_assert(size < 0x40000000); - r.data = (stbtt_uint8*) p; - r.size = (int) size; - r.cursor = 0; - return r; -} - -#define stbtt__buf_get16(b) stbtt__buf_get((b), 2) -#define stbtt__buf_get32(b) stbtt__buf_get((b), 4) - -static stbtt__buf stbtt__buf_range(const stbtt__buf *b, int o, int s) -{ - stbtt__buf r = stbtt__new_buf(NULL, 0); - if (o < 0 || s < 0 || o > b->size || s > b->size - o) return r; - r.data = b->data + o; - r.size = s; - return r; -} - -static stbtt__buf stbtt__cff_get_index(stbtt__buf *b) -{ - int count, start, offsize; - start = b->cursor; - count = stbtt__buf_get16(b); - if (count) { - offsize = stbtt__buf_get8(b); - STBTT_assert(offsize >= 1 && offsize <= 4); - stbtt__buf_skip(b, offsize * count); - stbtt__buf_skip(b, stbtt__buf_get(b, offsize) - 1); - } - return stbtt__buf_range(b, start, b->cursor - start); -} - -static stbtt_uint32 stbtt__cff_int(stbtt__buf *b) -{ - int b0 = stbtt__buf_get8(b); - if (b0 >= 32 && b0 <= 246) return b0 - 139; - else if (b0 >= 247 && b0 <= 250) return (b0 - 247)*256 + stbtt__buf_get8(b) + 108; - else if (b0 >= 251 && b0 <= 254) return -(b0 - 251)*256 - stbtt__buf_get8(b) - 108; - else if (b0 == 28) return stbtt__buf_get16(b); - else if (b0 == 29) return stbtt__buf_get32(b); - STBTT_assert(0); - return 0; -} - -static void stbtt__cff_skip_operand(stbtt__buf *b) { - int v, b0 = stbtt__buf_peek8(b); - STBTT_assert(b0 >= 28); - if (b0 == 30) { - stbtt__buf_skip(b, 1); - while (b->cursor < b->size) { - v = stbtt__buf_get8(b); - if ((v & 0xF) == 0xF || (v >> 4) == 0xF) - break; - } - } else { - stbtt__cff_int(b); - } -} - -static stbtt__buf stbtt__dict_get(stbtt__buf *b, int key) -{ - stbtt__buf_seek(b, 0); - while (b->cursor < b->size) { - int start = b->cursor, end, op; - while (stbtt__buf_peek8(b) >= 28) - stbtt__cff_skip_operand(b); - end = b->cursor; - op = stbtt__buf_get8(b); - if (op == 12) op = stbtt__buf_get8(b) | 0x100; - if (op == key) return stbtt__buf_range(b, start, end-start); - } - return stbtt__buf_range(b, 0, 0); -} - -static void stbtt__dict_get_ints(stbtt__buf *b, int key, int outcount, stbtt_uint32 *out) -{ - int i; - stbtt__buf operands = stbtt__dict_get(b, key); - for (i = 0; i < outcount && operands.cursor < operands.size; i++) - out[i] = stbtt__cff_int(&operands); -} - -static int stbtt__cff_index_count(stbtt__buf *b) -{ - stbtt__buf_seek(b, 0); - return stbtt__buf_get16(b); -} - -static stbtt__buf stbtt__cff_index_get(stbtt__buf b, int i) -{ - int count, offsize, start, end; - stbtt__buf_seek(&b, 0); - count = stbtt__buf_get16(&b); - offsize = stbtt__buf_get8(&b); - STBTT_assert(i >= 0 && i < count); - STBTT_assert(offsize >= 1 && offsize <= 4); - stbtt__buf_skip(&b, i*offsize); - start = stbtt__buf_get(&b, offsize); - end = stbtt__buf_get(&b, offsize); - return stbtt__buf_range(&b, 2+(count+1)*offsize+start, end - start); -} - -/* //////////////////////////////////////////////////////////////////////// */ -/* */ -/* accessors to parse data from file */ -/* */ - -/* on platforms that don't allow misaligned reads, if we want to allow */ -/* truetype fonts that aren't padded to alignment, define ALLOW_UNALIGNED_TRUETYPE */ - -#define ttBYTE(p) (* (stbtt_uint8 *) (p)) -#define ttCHAR(p) (* (stbtt_int8 *) (p)) -#define ttFixed(p) ttLONG(p) - -static stbtt_uint16 ttUSHORT(stbtt_uint8 *p) { return p[0]*256 + p[1]; } -static stbtt_int16 ttSHORT(stbtt_uint8 *p) { return p[0]*256 + p[1]; } -static stbtt_uint32 ttULONG(stbtt_uint8 *p) { return (p[0]<<24) + (p[1]<<16) + (p[2]<<8) + p[3]; } -static stbtt_int32 ttLONG(stbtt_uint8 *p) { return (p[0]<<24) + (p[1]<<16) + (p[2]<<8) + p[3]; } - -#define stbtt_tag4(p,c0,c1,c2,c3) ((p)[0] == (c0) && (p)[1] == (c1) && (p)[2] == (c2) && (p)[3] == (c3)) -#define stbtt_tag(p,str) stbtt_tag4(p,str[0],str[1],str[2],str[3]) - -static int stbtt__isfont(stbtt_uint8 *font) -{ - /* check the version number */ - if (stbtt_tag4(font, '1',0,0,0)) return 1; /* TrueType 1 */ - if (stbtt_tag(font, "typ1")) return 1; /* TrueType with type 1 font -- we don't support this! */ - if (stbtt_tag(font, "OTTO")) return 1; /* OpenType with CFF */ - if (stbtt_tag4(font, 0,1,0,0)) return 1; /* OpenType 1.0 */ - if (stbtt_tag(font, "true")) return 1; /* Apple specification for TrueType fonts */ - return 0; -} - -/* @OPTIMIZE: binary search */ -static stbtt_uint32 stbtt__find_table(stbtt_uint8 *data, stbtt_uint32 fontstart, const char *tag) -{ - stbtt_int32 num_tables = ttUSHORT(data+fontstart+4); - stbtt_uint32 tabledir = fontstart + 12; - stbtt_int32 i; - for (i=0; i < num_tables; ++i) { - stbtt_uint32 loc = tabledir + 16*i; - if (stbtt_tag(data+loc+0, tag)) - return ttULONG(data+loc+8); - } - return 0; -} - -static int stbtt_GetFontOffsetForIndex_internal(unsigned char *font_collection, int index) -{ - /* if it's just a font, there's only one valid index */ - if (stbtt__isfont(font_collection)) - return index == 0 ? 0 : -1; - - /* check if it's a TTC */ - if (stbtt_tag(font_collection, "ttcf")) { - /* version 1? */ - if (ttULONG(font_collection+4) == 0x00010000 || ttULONG(font_collection+4) == 0x00020000) { - stbtt_int32 n = ttLONG(font_collection+8); - if (index >= n) - return -1; - return ttULONG(font_collection+12+index*4); - } - } - return -1; -} - -static int stbtt_GetNumberOfFonts_internal(unsigned char *font_collection) -{ - /* if it's just a font, there's only one valid font */ - if (stbtt__isfont(font_collection)) - return 1; - - /* check if it's a TTC */ - if (stbtt_tag(font_collection, "ttcf")) { - /* version 1? */ - if (ttULONG(font_collection+4) == 0x00010000 || ttULONG(font_collection+4) == 0x00020000) { - return ttLONG(font_collection+8); - } - } - return 0; -} - -static stbtt__buf stbtt__get_subrs(stbtt__buf cff, stbtt__buf fontdict) -{ - stbtt_uint32 subrsoff = 0, private_loc[2] = { 0, 0 }; - stbtt__buf pdict; - stbtt__dict_get_ints(&fontdict, 18, 2, private_loc); - if (!private_loc[1] || !private_loc[0]) return stbtt__new_buf(NULL, 0); - pdict = stbtt__buf_range(&cff, private_loc[1], private_loc[0]); - stbtt__dict_get_ints(&pdict, 19, 1, &subrsoff); - if (!subrsoff) return stbtt__new_buf(NULL, 0); - stbtt__buf_seek(&cff, private_loc[1]+subrsoff); - return stbtt__cff_get_index(&cff); -} - -/* since most people won't use this, find this table the first time it's needed */ -static int stbtt__get_svg(stbtt_fontinfo *info) -{ - stbtt_uint32 t; - if (info->svg < 0) { - t = stbtt__find_table(info->data, info->fontstart, "SVG "); - if (t) { - stbtt_uint32 offset = ttULONG(info->data + t + 2); - info->svg = t + offset; - } else { - info->svg = 0; - } - } - return info->svg; -} - -static int stbtt_InitFont_internal(stbtt_fontinfo *info, unsigned char *data, int fontstart) -{ - stbtt_uint32 cmap, t; - stbtt_int32 i,numTables; - - info->data = data; - info->fontstart = fontstart; - info->cff = stbtt__new_buf(NULL, 0); - - cmap = stbtt__find_table(data, fontstart, "cmap"); /* required */ - info->loca = stbtt__find_table(data, fontstart, "loca"); /* required */ - info->head = stbtt__find_table(data, fontstart, "head"); /* required */ - info->glyf = stbtt__find_table(data, fontstart, "glyf"); /* required */ - info->hhea = stbtt__find_table(data, fontstart, "hhea"); /* required */ - info->hmtx = stbtt__find_table(data, fontstart, "hmtx"); /* required */ - info->kern = stbtt__find_table(data, fontstart, "kern"); /* not required */ - info->gpos = stbtt__find_table(data, fontstart, "GPOS"); /* not required */ - - if (!cmap || !info->head || !info->hhea || !info->hmtx) - return 0; - if (info->glyf) { - /* required for truetype */ - if (!info->loca) return 0; - } else { - /* initialization for CFF / Type2 fonts (OTF) */ - stbtt__buf b, topdict, topdictidx; - stbtt_uint32 cstype = 2, charstrings = 0, fdarrayoff = 0, fdselectoff = 0; - stbtt_uint32 cff; - - cff = stbtt__find_table(data, fontstart, "CFF "); - if (!cff) return 0; - - info->fontdicts = stbtt__new_buf(NULL, 0); - info->fdselect = stbtt__new_buf(NULL, 0); - - /* @TODO this should use size from table (not 512MB) */ - info->cff = stbtt__new_buf(data+cff, 512*1024*1024); - b = info->cff; - - /* read the header */ - stbtt__buf_skip(&b, 2); - stbtt__buf_seek(&b, stbtt__buf_get8(&b)); /* hdrsize */ - - /* @TODO the name INDEX could list multiple fonts, */ - /* but we just use the first one. */ - stbtt__cff_get_index(&b); /* name INDEX */ - topdictidx = stbtt__cff_get_index(&b); - topdict = stbtt__cff_index_get(topdictidx, 0); - stbtt__cff_get_index(&b); /* string INDEX */ - info->gsubrs = stbtt__cff_get_index(&b); - - stbtt__dict_get_ints(&topdict, 17, 1, &charstrings); - stbtt__dict_get_ints(&topdict, 0x100 | 6, 1, &cstype); - stbtt__dict_get_ints(&topdict, 0x100 | 36, 1, &fdarrayoff); - stbtt__dict_get_ints(&topdict, 0x100 | 37, 1, &fdselectoff); - info->subrs = stbtt__get_subrs(b, topdict); - - /* we only support Type 2 charstrings */ - if (cstype != 2) return 0; - if (charstrings == 0) return 0; - - if (fdarrayoff) { - /* looks like a CID font */ - if (!fdselectoff) return 0; - stbtt__buf_seek(&b, fdarrayoff); - info->fontdicts = stbtt__cff_get_index(&b); - info->fdselect = stbtt__buf_range(&b, fdselectoff, b.size-fdselectoff); - } - - stbtt__buf_seek(&b, charstrings); - info->charstrings = stbtt__cff_get_index(&b); - } - - t = stbtt__find_table(data, fontstart, "maxp"); - if (t) - info->numGlyphs = ttUSHORT(data+t+4); - else - info->numGlyphs = 0xffff; - - info->svg = -1; - - /* find a cmap encoding table we understand *now* to avoid searching */ - /* later. (todo: could make this installable) */ - /* the same regardless of glyph. */ - numTables = ttUSHORT(data + cmap + 2); - info->index_map = 0; - for (i=0; i < numTables; ++i) { - stbtt_uint32 encoding_record = cmap + 4 + 8 * i; - /* find an encoding we understand: */ - switch(ttUSHORT(data+encoding_record)) { - case STBTT_PLATFORM_ID_MICROSOFT: - switch (ttUSHORT(data+encoding_record+2)) { - case STBTT_MS_EID_UNICODE_BMP: - case STBTT_MS_EID_UNICODE_FULL: - /* MS/Unicode */ - info->index_map = cmap + ttULONG(data+encoding_record+4); - break; - } - break; - case STBTT_PLATFORM_ID_UNICODE: - /* Mac/iOS has these */ - /* all the encodingIDs are unicode, so we don't bother to check it */ - info->index_map = cmap + ttULONG(data+encoding_record+4); - break; - } - } - if (info->index_map == 0) - return 0; - - info->indexToLocFormat = ttUSHORT(data+info->head + 50); - return 1; -} - -STBTT_DEF int stbtt_FindGlyphIndex(const stbtt_fontinfo *info, int unicode_codepoint) -{ - stbtt_uint8 *data = info->data; - stbtt_uint32 index_map = info->index_map; - - stbtt_uint16 format = ttUSHORT(data + index_map + 0); - if (format == 0) { /* apple byte encoding */ - stbtt_int32 bytes = ttUSHORT(data + index_map + 2); - if (unicode_codepoint < bytes-6) - return ttBYTE(data + index_map + 6 + unicode_codepoint); - return 0; - } else if (format == 6) { - stbtt_uint32 first = ttUSHORT(data + index_map + 6); - stbtt_uint32 count = ttUSHORT(data + index_map + 8); - if ((stbtt_uint32) unicode_codepoint >= first && (stbtt_uint32) unicode_codepoint < first+count) - return ttUSHORT(data + index_map + 10 + (unicode_codepoint - first)*2); - return 0; - } else if (format == 2) { - STBTT_assert(0); /* @TODO: high-byte mapping for japanese/chinese/korean */ - return 0; - } else if (format == 4) { /* standard mapping for windows fonts: binary search collection of ranges */ - stbtt_uint16 segcount = ttUSHORT(data+index_map+6) >> 1; - stbtt_uint16 searchRange = ttUSHORT(data+index_map+8) >> 1; - stbtt_uint16 entrySelector = ttUSHORT(data+index_map+10); - stbtt_uint16 rangeShift = ttUSHORT(data+index_map+12) >> 1; - - /* do a binary search of the segments */ - stbtt_uint32 endCount = index_map + 14; - stbtt_uint32 search = endCount; - - if (unicode_codepoint > 0xffff) - return 0; - - /* they lie from endCount .. endCount + segCount */ - /* but searchRange is the nearest power of two, so... */ - if (unicode_codepoint >= ttUSHORT(data + search + rangeShift*2)) - search += rangeShift*2; - - /* now decrement to bias correctly to find smallest */ - search -= 2; - while (entrySelector) { - stbtt_uint16 end; - searchRange >>= 1; - end = ttUSHORT(data + search + searchRange*2); - if (unicode_codepoint > end) - search += searchRange*2; - --entrySelector; - } - search += 2; - - { - stbtt_uint16 offset, start, last; - stbtt_uint16 item = (stbtt_uint16) ((search - endCount) >> 1); - - start = ttUSHORT(data + index_map + 14 + segcount*2 + 2 + 2*item); - last = ttUSHORT(data + endCount + 2*item); - if (unicode_codepoint < start || unicode_codepoint > last) - return 0; - - offset = ttUSHORT(data + index_map + 14 + segcount*6 + 2 + 2*item); - if (offset == 0) - return (stbtt_uint16) (unicode_codepoint + ttSHORT(data + index_map + 14 + segcount*4 + 2 + 2*item)); - - return ttUSHORT(data + offset + (unicode_codepoint-start)*2 + index_map + 14 + segcount*6 + 2 + 2*item); - } - } else if (format == 12 || format == 13) { - stbtt_uint32 ngroups = ttULONG(data+index_map+12); - stbtt_int32 low,high; - low = 0; high = (stbtt_int32)ngroups; - /* Binary search the right group. */ - while (low < high) { - stbtt_int32 mid = low + ((high-low) >> 1); /* rounds down, so low <= mid < high */ - stbtt_uint32 start_char = ttULONG(data+index_map+16+mid*12); - stbtt_uint32 end_char = ttULONG(data+index_map+16+mid*12+4); - if ((stbtt_uint32) unicode_codepoint < start_char) - high = mid; - else if ((stbtt_uint32) unicode_codepoint > end_char) - low = mid+1; - else { - stbtt_uint32 start_glyph = ttULONG(data+index_map+16+mid*12+8); - if (format == 12) - return start_glyph + unicode_codepoint-start_char; - else /* format == 13 */ - return start_glyph; - } - } - return 0; /* not found */ - } - /* @TODO */ - STBTT_assert(0); - return 0; -} - -STBTT_DEF int stbtt_GetCodepointShape(const stbtt_fontinfo *info, int unicode_codepoint, stbtt_vertex **vertices) -{ - return stbtt_GetGlyphShape(info, stbtt_FindGlyphIndex(info, unicode_codepoint), vertices); -} - -static void stbtt_setvertex(stbtt_vertex *v, stbtt_uint8 type, stbtt_int32 x, stbtt_int32 y, stbtt_int32 cx, stbtt_int32 cy) -{ - v->type = type; - v->x = (stbtt_int16) x; - v->y = (stbtt_int16) y; - v->cx = (stbtt_int16) cx; - v->cy = (stbtt_int16) cy; -} - -static int stbtt__GetGlyfOffset(const stbtt_fontinfo *info, int glyph_index) -{ - int g1,g2; - - STBTT_assert(!info->cff.size); - - if (glyph_index >= info->numGlyphs) return -1; /* glyph index out of range */ - if (info->indexToLocFormat >= 2) return -1; /* unknown index->glyph map format */ - - if (info->indexToLocFormat == 0) { - g1 = info->glyf + ttUSHORT(info->data + info->loca + glyph_index * 2) * 2; - g2 = info->glyf + ttUSHORT(info->data + info->loca + glyph_index * 2 + 2) * 2; - } else { - g1 = info->glyf + ttULONG (info->data + info->loca + glyph_index * 4); - g2 = info->glyf + ttULONG (info->data + info->loca + glyph_index * 4 + 4); - } - - return g1==g2 ? -1 : g1; /* if length is 0, return -1 */ -} - -static int stbtt__GetGlyphInfoT2(const stbtt_fontinfo *info, int glyph_index, int *x0, int *y0, int *x1, int *y1); - -STBTT_DEF int stbtt_GetGlyphBox(const stbtt_fontinfo *info, int glyph_index, int *x0, int *y0, int *x1, int *y1) -{ - if (info->cff.size) { - stbtt__GetGlyphInfoT2(info, glyph_index, x0, y0, x1, y1); - } else { - int g = stbtt__GetGlyfOffset(info, glyph_index); - if (g < 0) return 0; - - if (x0) *x0 = ttSHORT(info->data + g + 2); - if (y0) *y0 = ttSHORT(info->data + g + 4); - if (x1) *x1 = ttSHORT(info->data + g + 6); - if (y1) *y1 = ttSHORT(info->data + g + 8); - } - return 1; -} - -STBTT_DEF int stbtt_GetCodepointBox(const stbtt_fontinfo *info, int codepoint, int *x0, int *y0, int *x1, int *y1) -{ - return stbtt_GetGlyphBox(info, stbtt_FindGlyphIndex(info,codepoint), x0,y0,x1,y1); -} - -STBTT_DEF int stbtt_IsGlyphEmpty(const stbtt_fontinfo *info, int glyph_index) -{ - stbtt_int16 numberOfContours; - int g; - if (info->cff.size) - return stbtt__GetGlyphInfoT2(info, glyph_index, NULL, NULL, NULL, NULL) == 0; - g = stbtt__GetGlyfOffset(info, glyph_index); - if (g < 0) return 1; - numberOfContours = ttSHORT(info->data + g); - return numberOfContours == 0; -} - -static int stbtt__close_shape(stbtt_vertex *vertices, int num_vertices, int was_off, int start_off, - stbtt_int32 sx, stbtt_int32 sy, stbtt_int32 scx, stbtt_int32 scy, stbtt_int32 cx, stbtt_int32 cy) -{ - if (start_off) { - if (was_off) - stbtt_setvertex(&vertices[num_vertices++], STBTT_vcurve, (cx+scx)>>1, (cy+scy)>>1, cx,cy); - stbtt_setvertex(&vertices[num_vertices++], STBTT_vcurve, sx,sy,scx,scy); - } else { - if (was_off) - stbtt_setvertex(&vertices[num_vertices++], STBTT_vcurve,sx,sy,cx,cy); - else - stbtt_setvertex(&vertices[num_vertices++], STBTT_vline,sx,sy,0,0); - } - return num_vertices; -} - -static int stbtt__GetGlyphShapeTT(const stbtt_fontinfo *info, int glyph_index, stbtt_vertex **pvertices) -{ - stbtt_int16 numberOfContours; - stbtt_uint8 *endPtsOfContours; - stbtt_uint8 *data = info->data; - stbtt_vertex *vertices=0; - int num_vertices=0; - int g = stbtt__GetGlyfOffset(info, glyph_index); - - *pvertices = NULL; - - if (g < 0) return 0; - - numberOfContours = ttSHORT(data + g); - - if (numberOfContours > 0) { - stbtt_uint8 flags=0,flagcount; - stbtt_int32 ins, i,j=0,m,n, next_move, was_off=0, off, start_off=0; - stbtt_int32 x,y,cx,cy,sx,sy, scx,scy; - stbtt_uint8 *points; - endPtsOfContours = (data + g + 10); - ins = ttUSHORT(data + g + 10 + numberOfContours * 2); - points = data + g + 10 + numberOfContours * 2 + 2 + ins; - - n = 1+ttUSHORT(endPtsOfContours + numberOfContours*2-2); - - m = n + 2*numberOfContours; /* a loose bound on how many vertices we might need */ - vertices = (stbtt_vertex *) STBTT_malloc(m * sizeof(vertices[0]), info->userdata); - if (vertices == 0) - return 0; - - next_move = 0; - flagcount=0; - - /* in first pass, we load uninterpreted data into the allocated array */ - /* above, shifted to the end of the array so we won't overwrite it when */ - /* we create our final data starting from the front */ - - off = m - n; /* starting offset for uninterpreted data, regardless of how m ends up being calculated */ - - /* first load flags */ - - for (i=0; i < n; ++i) { - if (flagcount == 0) { - flags = *points++; - if (flags & 8) - flagcount = *points++; - } else - --flagcount; - vertices[off+i].type = flags; - } - - /* now load x coordinates */ - x=0; - for (i=0; i < n; ++i) { - flags = vertices[off+i].type; - if (flags & 2) { - stbtt_int16 dx = *points++; - x += (flags & 16) ? dx : -dx; /* ??? */ - } else { - if (!(flags & 16)) { - x = x + (stbtt_int16) (points[0]*256 + points[1]); - points += 2; - } - } - vertices[off+i].x = (stbtt_int16) x; - } - - /* now load y coordinates */ - y=0; - for (i=0; i < n; ++i) { - flags = vertices[off+i].type; - if (flags & 4) { - stbtt_int16 dy = *points++; - y += (flags & 32) ? dy : -dy; /* ??? */ - } else { - if (!(flags & 32)) { - y = y + (stbtt_int16) (points[0]*256 + points[1]); - points += 2; - } - } - vertices[off+i].y = (stbtt_int16) y; - } - - /* now convert them to our format */ - num_vertices=0; - sx = sy = cx = cy = scx = scy = 0; - for (i=0; i < n; ++i) { - flags = vertices[off+i].type; - x = (stbtt_int16) vertices[off+i].x; - y = (stbtt_int16) vertices[off+i].y; - - if (next_move == i) { - if (i != 0) - num_vertices = stbtt__close_shape(vertices, num_vertices, was_off, start_off, sx,sy,scx,scy,cx,cy); - - /* now start the new one */ - start_off = !(flags & 1); - if (start_off) { - /* if we start off with an off-curve point, then when we need to find a point on the curve */ - /* where we can start, and we need to save some state for when we wraparound. */ - scx = x; - scy = y; - if (!(vertices[off+i+1].type & 1)) { - /* next point is also a curve point, so interpolate an on-point curve */ - sx = (x + (stbtt_int32) vertices[off+i+1].x) >> 1; - sy = (y + (stbtt_int32) vertices[off+i+1].y) >> 1; - } else { - /* otherwise just use the next point as our start point */ - sx = (stbtt_int32) vertices[off+i+1].x; - sy = (stbtt_int32) vertices[off+i+1].y; - ++i; /* we're using point i+1 as the starting point, so skip it */ - } - } else { - sx = x; - sy = y; - } - stbtt_setvertex(&vertices[num_vertices++], STBTT_vmove,sx,sy,0,0); - was_off = 0; - next_move = 1 + ttUSHORT(endPtsOfContours+j*2); - ++j; - } else { - if (!(flags & 1)) { /* if it's a curve */ - if (was_off) /* two off-curve control points in a row means interpolate an on-curve midpoint */ - stbtt_setvertex(&vertices[num_vertices++], STBTT_vcurve, (cx+x)>>1, (cy+y)>>1, cx, cy); - cx = x; - cy = y; - was_off = 1; - } else { - if (was_off) - stbtt_setvertex(&vertices[num_vertices++], STBTT_vcurve, x,y, cx, cy); - else - stbtt_setvertex(&vertices[num_vertices++], STBTT_vline, x,y,0,0); - was_off = 0; - } - } - } - num_vertices = stbtt__close_shape(vertices, num_vertices, was_off, start_off, sx,sy,scx,scy,cx,cy); - } else if (numberOfContours < 0) { - /* Compound shapes. */ - int more = 1; - stbtt_uint8 *comp = data + g + 10; - num_vertices = 0; - vertices = 0; - while (more) { - stbtt_uint16 flags, gidx; - int comp_num_verts = 0, i; - stbtt_vertex *comp_verts = 0, *tmp = 0; - float mtx[6] = {1,0,0,1,0,0}, m, n; - - flags = ttSHORT(comp); comp+=2; - gidx = ttSHORT(comp); comp+=2; - - if (flags & 2) { /* XY values */ - if (flags & 1) { /* shorts */ - mtx[4] = ttSHORT(comp); comp+=2; - mtx[5] = ttSHORT(comp); comp+=2; - } else { - mtx[4] = ttCHAR(comp); comp+=1; - mtx[5] = ttCHAR(comp); comp+=1; - } - } - else { - /* @TODO handle matching point */ - STBTT_assert(0); - } - if (flags & (1<<3)) { /* WE_HAVE_A_SCALE */ - mtx[0] = mtx[3] = ttSHORT(comp)/16384.0f; comp+=2; - mtx[1] = mtx[2] = 0; - } else if (flags & (1<<6)) { /* WE_HAVE_AN_X_AND_YSCALE */ - mtx[0] = ttSHORT(comp)/16384.0f; comp+=2; - mtx[1] = mtx[2] = 0; - mtx[3] = ttSHORT(comp)/16384.0f; comp+=2; - } else if (flags & (1<<7)) { /* WE_HAVE_A_TWO_BY_TWO */ - mtx[0] = ttSHORT(comp)/16384.0f; comp+=2; - mtx[1] = ttSHORT(comp)/16384.0f; comp+=2; - mtx[2] = ttSHORT(comp)/16384.0f; comp+=2; - mtx[3] = ttSHORT(comp)/16384.0f; comp+=2; - } - - /* Find transformation scales. */ - m = (float) STBTT_sqrt(mtx[0]*mtx[0] + mtx[1]*mtx[1]); - n = (float) STBTT_sqrt(mtx[2]*mtx[2] + mtx[3]*mtx[3]); - - /* Get indexed glyph. */ - comp_num_verts = stbtt_GetGlyphShape(info, gidx, &comp_verts); - if (comp_num_verts > 0) { - /* Transform vertices. */ - for (i = 0; i < comp_num_verts; ++i) { - stbtt_vertex* v = &comp_verts[i]; - stbtt_vertex_type x,y; - x=v->x; y=v->y; - v->x = (stbtt_vertex_type)(m * (mtx[0]*x + mtx[2]*y + mtx[4])); - v->y = (stbtt_vertex_type)(n * (mtx[1]*x + mtx[3]*y + mtx[5])); - x=v->cx; y=v->cy; - v->cx = (stbtt_vertex_type)(m * (mtx[0]*x + mtx[2]*y + mtx[4])); - v->cy = (stbtt_vertex_type)(n * (mtx[1]*x + mtx[3]*y + mtx[5])); - } - /* Append vertices. */ - tmp = (stbtt_vertex*)STBTT_malloc((num_vertices+comp_num_verts)*sizeof(stbtt_vertex), info->userdata); - if (!tmp) { - if (vertices) STBTT_free(vertices, info->userdata); - if (comp_verts) STBTT_free(comp_verts, info->userdata); - return 0; - } - if (num_vertices > 0 && vertices) STBTT_memcpy(tmp, vertices, num_vertices*sizeof(stbtt_vertex)); - STBTT_memcpy(tmp+num_vertices, comp_verts, comp_num_verts*sizeof(stbtt_vertex)); - if (vertices) STBTT_free(vertices, info->userdata); - vertices = tmp; - STBTT_free(comp_verts, info->userdata); - num_vertices += comp_num_verts; - } - /* More components ? */ - more = flags & (1<<5); - } - } else { - /* numberOfCounters == 0, do nothing */ - } - - *pvertices = vertices; - return num_vertices; -} - -typedef struct -{ - int bounds; - int started; - float first_x, first_y; - float x, y; - stbtt_int32 min_x, max_x, min_y, max_y; - - stbtt_vertex *pvertices; - int num_vertices; -} stbtt__csctx; - -#define STBTT__CSCTX_INIT(bounds) {bounds,0, 0,0, 0,0, 0,0,0,0, NULL, 0} - -static void stbtt__track_vertex(stbtt__csctx *c, stbtt_int32 x, stbtt_int32 y) -{ - if (x > c->max_x || !c->started) c->max_x = x; - if (y > c->max_y || !c->started) c->max_y = y; - if (x < c->min_x || !c->started) c->min_x = x; - if (y < c->min_y || !c->started) c->min_y = y; - c->started = 1; -} - -static void stbtt__csctx_v(stbtt__csctx *c, stbtt_uint8 type, stbtt_int32 x, stbtt_int32 y, stbtt_int32 cx, stbtt_int32 cy, stbtt_int32 cx1, stbtt_int32 cy1) -{ - if (c->bounds) { - stbtt__track_vertex(c, x, y); - if (type == STBTT_vcubic) { - stbtt__track_vertex(c, cx, cy); - stbtt__track_vertex(c, cx1, cy1); - } - } else { - stbtt_setvertex(&c->pvertices[c->num_vertices], type, x, y, cx, cy); - c->pvertices[c->num_vertices].cx1 = (stbtt_int16) cx1; - c->pvertices[c->num_vertices].cy1 = (stbtt_int16) cy1; - } - c->num_vertices++; -} - -static void stbtt__csctx_close_shape(stbtt__csctx *ctx) -{ - if (ctx->first_x != ctx->x || ctx->first_y != ctx->y) - stbtt__csctx_v(ctx, STBTT_vline, (int)ctx->first_x, (int)ctx->first_y, 0, 0, 0, 0); -} - -static void stbtt__csctx_rmove_to(stbtt__csctx *ctx, float dx, float dy) -{ - stbtt__csctx_close_shape(ctx); - ctx->first_x = ctx->x = ctx->x + dx; - ctx->first_y = ctx->y = ctx->y + dy; - stbtt__csctx_v(ctx, STBTT_vmove, (int)ctx->x, (int)ctx->y, 0, 0, 0, 0); -} - -static void stbtt__csctx_rline_to(stbtt__csctx *ctx, float dx, float dy) -{ - ctx->x += dx; - ctx->y += dy; - stbtt__csctx_v(ctx, STBTT_vline, (int)ctx->x, (int)ctx->y, 0, 0, 0, 0); -} - -static void stbtt__csctx_rccurve_to(stbtt__csctx *ctx, float dx1, float dy1, float dx2, float dy2, float dx3, float dy3) -{ - float cx1 = ctx->x + dx1; - float cy1 = ctx->y + dy1; - float cx2 = cx1 + dx2; - float cy2 = cy1 + dy2; - ctx->x = cx2 + dx3; - ctx->y = cy2 + dy3; - stbtt__csctx_v(ctx, STBTT_vcubic, (int)ctx->x, (int)ctx->y, (int)cx1, (int)cy1, (int)cx2, (int)cy2); -} - -static stbtt__buf stbtt__get_subr(stbtt__buf idx, int n) -{ - int count = stbtt__cff_index_count(&idx); - int bias = 107; - if (count >= 33900) - bias = 32768; - else if (count >= 1240) - bias = 1131; - n += bias; - if (n < 0 || n >= count) - return stbtt__new_buf(NULL, 0); - return stbtt__cff_index_get(idx, n); -} - -static stbtt__buf stbtt__cid_get_glyph_subrs(const stbtt_fontinfo *info, int glyph_index) -{ - stbtt__buf fdselect = info->fdselect; - int nranges, start, end, v, fmt, fdselector = -1, i; - - stbtt__buf_seek(&fdselect, 0); - fmt = stbtt__buf_get8(&fdselect); - if (fmt == 0) { - /* untested */ - stbtt__buf_skip(&fdselect, glyph_index); - fdselector = stbtt__buf_get8(&fdselect); - } else if (fmt == 3) { - nranges = stbtt__buf_get16(&fdselect); - start = stbtt__buf_get16(&fdselect); - for (i = 0; i < nranges; i++) { - v = stbtt__buf_get8(&fdselect); - end = stbtt__buf_get16(&fdselect); - if (glyph_index >= start && glyph_index < end) { - fdselector = v; - break; - } - start = end; - } - } - if (fdselector == -1) stbtt__new_buf(NULL, 0); - return stbtt__get_subrs(info->cff, stbtt__cff_index_get(info->fontdicts, fdselector)); -} - -static int stbtt__run_charstring(const stbtt_fontinfo *info, int glyph_index, stbtt__csctx *c) -{ - int in_header = 1, maskbits = 0, subr_stack_height = 0, sp = 0, v, i, b0; - int has_subrs = 0, clear_stack; - float s[48]; - stbtt__buf subr_stack[10], subrs = info->subrs, b; - float f; - -#define STBTT__CSERR(s) (0) - - /* this currently ignores the initial width value, which isn't needed if we have hmtx */ - b = stbtt__cff_index_get(info->charstrings, glyph_index); - while (b.cursor < b.size) { - i = 0; - clear_stack = 1; - b0 = stbtt__buf_get8(&b); - switch (b0) { - /* @TODO implement hinting */ - case 0x13: /* hintmask */ - case 0x14: /* cntrmask */ - if (in_header) - maskbits += (sp / 2); /* implicit "vstem" */ - in_header = 0; - stbtt__buf_skip(&b, (maskbits + 7) / 8); - break; - - case 0x01: /* hstem */ - case 0x03: /* vstem */ - case 0x12: /* hstemhm */ - case 0x17: /* vstemhm */ - maskbits += (sp / 2); - break; - - case 0x15: /* rmoveto */ - in_header = 0; - if (sp < 2) return STBTT__CSERR("rmoveto stack"); - stbtt__csctx_rmove_to(c, s[sp-2], s[sp-1]); - break; - case 0x04: /* vmoveto */ - in_header = 0; - if (sp < 1) return STBTT__CSERR("vmoveto stack"); - stbtt__csctx_rmove_to(c, 0, s[sp-1]); - break; - case 0x16: /* hmoveto */ - in_header = 0; - if (sp < 1) return STBTT__CSERR("hmoveto stack"); - stbtt__csctx_rmove_to(c, s[sp-1], 0); - break; - - case 0x05: /* rlineto */ - if (sp < 2) return STBTT__CSERR("rlineto stack"); - for (; i + 1 < sp; i += 2) - stbtt__csctx_rline_to(c, s[i], s[i+1]); - break; - - /* hlineto/vlineto and vhcurveto/hvcurveto alternate horizontal and vertical */ - /* starting from a different place. */ - - case 0x07: /* vlineto */ - if (sp < 1) return STBTT__CSERR("vlineto stack"); - goto vlineto; - case 0x06: /* hlineto */ - if (sp < 1) return STBTT__CSERR("hlineto stack"); - for (;;) { - if (i >= sp) break; - stbtt__csctx_rline_to(c, s[i], 0); - i++; - vlineto: - if (i >= sp) break; - stbtt__csctx_rline_to(c, 0, s[i]); - i++; - } - break; - - case 0x1F: /* hvcurveto */ - if (sp < 4) return STBTT__CSERR("hvcurveto stack"); - goto hvcurveto; - case 0x1E: /* vhcurveto */ - if (sp < 4) return STBTT__CSERR("vhcurveto stack"); - for (;;) { - if (i + 3 >= sp) break; - stbtt__csctx_rccurve_to(c, 0, s[i], s[i+1], s[i+2], s[i+3], (sp - i == 5) ? s[i + 4] : 0.0f); - i += 4; - hvcurveto: - if (i + 3 >= sp) break; - stbtt__csctx_rccurve_to(c, s[i], 0, s[i+1], s[i+2], (sp - i == 5) ? s[i+4] : 0.0f, s[i+3]); - i += 4; - } - break; - - case 0x08: /* rrcurveto */ - if (sp < 6) return STBTT__CSERR("rcurveline stack"); - for (; i + 5 < sp; i += 6) - stbtt__csctx_rccurve_to(c, s[i], s[i+1], s[i+2], s[i+3], s[i+4], s[i+5]); - break; - - case 0x18: /* rcurveline */ - if (sp < 8) return STBTT__CSERR("rcurveline stack"); - for (; i + 5 < sp - 2; i += 6) - stbtt__csctx_rccurve_to(c, s[i], s[i+1], s[i+2], s[i+3], s[i+4], s[i+5]); - if (i + 1 >= sp) return STBTT__CSERR("rcurveline stack"); - stbtt__csctx_rline_to(c, s[i], s[i+1]); - break; - - case 0x19: /* rlinecurve */ - if (sp < 8) return STBTT__CSERR("rlinecurve stack"); - for (; i + 1 < sp - 6; i += 2) - stbtt__csctx_rline_to(c, s[i], s[i+1]); - if (i + 5 >= sp) return STBTT__CSERR("rlinecurve stack"); - stbtt__csctx_rccurve_to(c, s[i], s[i+1], s[i+2], s[i+3], s[i+4], s[i+5]); - break; - - case 0x1A: /* vvcurveto */ - case 0x1B: /* hhcurveto */ - if (sp < 4) return STBTT__CSERR("(vv|hh)curveto stack"); - f = 0.0; - if (sp & 1) { f = s[i]; i++; } - for (; i + 3 < sp; i += 4) { - if (b0 == 0x1B) - stbtt__csctx_rccurve_to(c, s[i], f, s[i+1], s[i+2], s[i+3], 0.0); - else - stbtt__csctx_rccurve_to(c, f, s[i], s[i+1], s[i+2], 0.0, s[i+3]); - f = 0.0; - } - break; - - case 0x0A: /* callsubr */ - if (!has_subrs) { - if (info->fdselect.size) - subrs = stbtt__cid_get_glyph_subrs(info, glyph_index); - has_subrs = 1; - } - /* FALLTHROUGH */ - case 0x1D: /* callgsubr */ - if (sp < 1) return STBTT__CSERR("call(g|)subr stack"); - v = (int) s[--sp]; - if (subr_stack_height >= 10) return STBTT__CSERR("recursion limit"); - subr_stack[subr_stack_height++] = b; - b = stbtt__get_subr(b0 == 0x0A ? subrs : info->gsubrs, v); - if (b.size == 0) return STBTT__CSERR("subr not found"); - b.cursor = 0; - clear_stack = 0; - break; - - case 0x0B: /* return */ - if (subr_stack_height <= 0) return STBTT__CSERR("return outside subr"); - b = subr_stack[--subr_stack_height]; - clear_stack = 0; - break; - - case 0x0E: /* endchar */ - stbtt__csctx_close_shape(c); - return 1; - - case 0x0C: { /* two-byte escape */ - float dx1, dx2, dx3, dx4, dx5, dx6, dy1, dy2, dy3, dy4, dy5, dy6; - float dx, dy; - int b1 = stbtt__buf_get8(&b); - switch (b1) { - /* @TODO These "flex" implementations ignore the flex-depth and resolution, */ - /* and always draw beziers. */ - case 0x22: /* hflex */ - if (sp < 7) return STBTT__CSERR("hflex stack"); - dx1 = s[0]; - dx2 = s[1]; - dy2 = s[2]; - dx3 = s[3]; - dx4 = s[4]; - dx5 = s[5]; - dx6 = s[6]; - stbtt__csctx_rccurve_to(c, dx1, 0, dx2, dy2, dx3, 0); - stbtt__csctx_rccurve_to(c, dx4, 0, dx5, -dy2, dx6, 0); - break; - - case 0x23: /* flex */ - if (sp < 13) return STBTT__CSERR("flex stack"); - dx1 = s[0]; - dy1 = s[1]; - dx2 = s[2]; - dy2 = s[3]; - dx3 = s[4]; - dy3 = s[5]; - dx4 = s[6]; - dy4 = s[7]; - dx5 = s[8]; - dy5 = s[9]; - dx6 = s[10]; - dy6 = s[11]; - /* fd is s[12] */ - stbtt__csctx_rccurve_to(c, dx1, dy1, dx2, dy2, dx3, dy3); - stbtt__csctx_rccurve_to(c, dx4, dy4, dx5, dy5, dx6, dy6); - break; - - case 0x24: /* hflex1 */ - if (sp < 9) return STBTT__CSERR("hflex1 stack"); - dx1 = s[0]; - dy1 = s[1]; - dx2 = s[2]; - dy2 = s[3]; - dx3 = s[4]; - dx4 = s[5]; - dx5 = s[6]; - dy5 = s[7]; - dx6 = s[8]; - stbtt__csctx_rccurve_to(c, dx1, dy1, dx2, dy2, dx3, 0); - stbtt__csctx_rccurve_to(c, dx4, 0, dx5, dy5, dx6, -(dy1+dy2+dy5)); - break; - - case 0x25: /* flex1 */ - if (sp < 11) return STBTT__CSERR("flex1 stack"); - dx1 = s[0]; - dy1 = s[1]; - dx2 = s[2]; - dy2 = s[3]; - dx3 = s[4]; - dy3 = s[5]; - dx4 = s[6]; - dy4 = s[7]; - dx5 = s[8]; - dy5 = s[9]; - dx6 = dy6 = s[10]; - dx = dx1+dx2+dx3+dx4+dx5; - dy = dy1+dy2+dy3+dy4+dy5; - if (STBTT_fabs(dx) > STBTT_fabs(dy)) - dy6 = -dy; - else - dx6 = -dx; - stbtt__csctx_rccurve_to(c, dx1, dy1, dx2, dy2, dx3, dy3); - stbtt__csctx_rccurve_to(c, dx4, dy4, dx5, dy5, dx6, dy6); - break; - - default: - return STBTT__CSERR("unimplemented"); - } - } break; - - default: - if (b0 != 255 && b0 != 28 && b0 < 32) - return STBTT__CSERR("reserved operator"); - - /* push immediate */ - if (b0 == 255) { - f = (float)(stbtt_int32)stbtt__buf_get32(&b) / 0x10000; - } else { - stbtt__buf_skip(&b, -1); - f = (float)(stbtt_int16)stbtt__cff_int(&b); - } - if (sp >= 48) return STBTT__CSERR("push stack overflow"); - s[sp++] = f; - clear_stack = 0; - break; - } - if (clear_stack) sp = 0; - } - return STBTT__CSERR("no endchar"); - -#undef STBTT__CSERR -} - -static int stbtt__GetGlyphShapeT2(const stbtt_fontinfo *info, int glyph_index, stbtt_vertex **pvertices) -{ - /* runs the charstring twice, once to count and once to output (to avoid realloc) */ - stbtt__csctx count_ctx = STBTT__CSCTX_INIT(1); - stbtt__csctx output_ctx = STBTT__CSCTX_INIT(0); - if (stbtt__run_charstring(info, glyph_index, &count_ctx)) { - *pvertices = (stbtt_vertex*)STBTT_malloc(count_ctx.num_vertices*sizeof(stbtt_vertex), info->userdata); - output_ctx.pvertices = *pvertices; - if (stbtt__run_charstring(info, glyph_index, &output_ctx)) { - STBTT_assert(output_ctx.num_vertices == count_ctx.num_vertices); - return output_ctx.num_vertices; - } - } - *pvertices = NULL; - return 0; -} - -static int stbtt__GetGlyphInfoT2(const stbtt_fontinfo *info, int glyph_index, int *x0, int *y0, int *x1, int *y1) -{ - stbtt__csctx c = STBTT__CSCTX_INIT(1); - int r = stbtt__run_charstring(info, glyph_index, &c); - if (x0) *x0 = r ? c.min_x : 0; - if (y0) *y0 = r ? c.min_y : 0; - if (x1) *x1 = r ? c.max_x : 0; - if (y1) *y1 = r ? c.max_y : 0; - return r ? c.num_vertices : 0; -} - -STBTT_DEF int stbtt_GetGlyphShape(const stbtt_fontinfo *info, int glyph_index, stbtt_vertex **pvertices) -{ - if (!info->cff.size) - return stbtt__GetGlyphShapeTT(info, glyph_index, pvertices); - else - return stbtt__GetGlyphShapeT2(info, glyph_index, pvertices); -} - -STBTT_DEF void stbtt_GetGlyphHMetrics(const stbtt_fontinfo *info, int glyph_index, int *advanceWidth, int *leftSideBearing) -{ - stbtt_uint16 numOfLongHorMetrics = ttUSHORT(info->data+info->hhea + 34); - if (glyph_index < numOfLongHorMetrics) { - if (advanceWidth) *advanceWidth = ttSHORT(info->data + info->hmtx + 4*glyph_index); - if (leftSideBearing) *leftSideBearing = ttSHORT(info->data + info->hmtx + 4*glyph_index + 2); - } else { - if (advanceWidth) *advanceWidth = ttSHORT(info->data + info->hmtx + 4*(numOfLongHorMetrics-1)); - if (leftSideBearing) *leftSideBearing = ttSHORT(info->data + info->hmtx + 4*numOfLongHorMetrics + 2*(glyph_index - numOfLongHorMetrics)); - } -} - -STBTT_DEF int stbtt_GetKerningTableLength(const stbtt_fontinfo *info) -{ - stbtt_uint8 *data = info->data + info->kern; - - /* we only look at the first table. it must be 'horizontal' and format 0. */ - if (!info->kern) - return 0; - if (ttUSHORT(data+2) < 1) /* number of tables, need at least 1 */ - return 0; - if (ttUSHORT(data+8) != 1) /* horizontal flag must be set in format */ - return 0; - - return ttUSHORT(data+10); -} - -STBTT_DEF int stbtt_GetKerningTable(const stbtt_fontinfo *info, stbtt_kerningentry* table, int table_length) -{ - stbtt_uint8 *data = info->data + info->kern; - int k, length; - - /* we only look at the first table. it must be 'horizontal' and format 0. */ - if (!info->kern) - return 0; - if (ttUSHORT(data+2) < 1) /* number of tables, need at least 1 */ - return 0; - if (ttUSHORT(data+8) != 1) /* horizontal flag must be set in format */ - return 0; - - length = ttUSHORT(data+10); - if (table_length < length) - length = table_length; - - for (k = 0; k < length; k++) - { - table[k].glyph1 = ttUSHORT(data+18+(k*6)); - table[k].glyph2 = ttUSHORT(data+20+(k*6)); - table[k].advance = ttSHORT(data+22+(k*6)); - } - - return length; -} - -static int stbtt__GetGlyphKernInfoAdvance(const stbtt_fontinfo *info, int glyph1, int glyph2) -{ - stbtt_uint8 *data = info->data + info->kern; - stbtt_uint32 needle, straw; - int l, r, m; - - /* we only look at the first table. it must be 'horizontal' and format 0. */ - if (!info->kern) - return 0; - if (ttUSHORT(data+2) < 1) /* number of tables, need at least 1 */ - return 0; - if (ttUSHORT(data+8) != 1) /* horizontal flag must be set in format */ - return 0; - - l = 0; - r = ttUSHORT(data+10) - 1; - needle = glyph1 << 16 | glyph2; - while (l <= r) { - m = (l + r) >> 1; - straw = ttULONG(data+18+(m*6)); /* note: unaligned read */ - if (needle < straw) - r = m - 1; - else if (needle > straw) - l = m + 1; - else - return ttSHORT(data+22+(m*6)); - } - return 0; -} - -static stbtt_int32 stbtt__GetCoverageIndex(stbtt_uint8 *coverageTable, int glyph) -{ - stbtt_uint16 coverageFormat = ttUSHORT(coverageTable); - switch (coverageFormat) { - case 1: { - stbtt_uint16 glyphCount = ttUSHORT(coverageTable + 2); - - /* Binary search. */ - stbtt_int32 l=0, r=glyphCount-1, m; - int straw, needle=glyph; - while (l <= r) { - stbtt_uint8 *glyphArray = coverageTable + 4; - stbtt_uint16 glyphID; - m = (l + r) >> 1; - glyphID = ttUSHORT(glyphArray + 2 * m); - straw = glyphID; - if (needle < straw) - r = m - 1; - else if (needle > straw) - l = m + 1; - else { - return m; - } - } - break; - } - - case 2: { - stbtt_uint16 rangeCount = ttUSHORT(coverageTable + 2); - stbtt_uint8 *rangeArray = coverageTable + 4; - - /* Binary search. */ - stbtt_int32 l=0, r=rangeCount-1, m; - int strawStart, strawEnd, needle=glyph; - while (l <= r) { - stbtt_uint8 *rangeRecord; - m = (l + r) >> 1; - rangeRecord = rangeArray + 6 * m; - strawStart = ttUSHORT(rangeRecord); - strawEnd = ttUSHORT(rangeRecord + 2); - if (needle < strawStart) - r = m - 1; - else if (needle > strawEnd) - l = m + 1; - else { - stbtt_uint16 startCoverageIndex = ttUSHORT(rangeRecord + 4); - return startCoverageIndex + glyph - strawStart; - } - } - break; - } - - default: return -1; /* unsupported */ - } - - return -1; -} - -static stbtt_int32 stbtt__GetGlyphClass(stbtt_uint8 *classDefTable, int glyph) -{ - stbtt_uint16 classDefFormat = ttUSHORT(classDefTable); - switch (classDefFormat) - { - case 1: { - stbtt_uint16 startGlyphID = ttUSHORT(classDefTable + 2); - stbtt_uint16 glyphCount = ttUSHORT(classDefTable + 4); - stbtt_uint8 *classDef1ValueArray = classDefTable + 6; - - if (glyph >= startGlyphID && glyph < startGlyphID + glyphCount) - return (stbtt_int32)ttUSHORT(classDef1ValueArray + 2 * (glyph - startGlyphID)); - break; - } - - case 2: { - stbtt_uint16 classRangeCount = ttUSHORT(classDefTable + 2); - stbtt_uint8 *classRangeRecords = classDefTable + 4; - - /* Binary search. */ - stbtt_int32 l=0, r=classRangeCount-1, m; - int strawStart, strawEnd, needle=glyph; - while (l <= r) { - stbtt_uint8 *classRangeRecord; - m = (l + r) >> 1; - classRangeRecord = classRangeRecords + 6 * m; - strawStart = ttUSHORT(classRangeRecord); - strawEnd = ttUSHORT(classRangeRecord + 2); - if (needle < strawStart) - r = m - 1; - else if (needle > strawEnd) - l = m + 1; - else - return (stbtt_int32)ttUSHORT(classRangeRecord + 4); - } - break; - } - - default: - return -1; /* Unsupported definition type, return an error. */ - } - - /* "All glyphs not assigned to a class fall into class 0". (OpenType spec) */ - return 0; -} - -/* Define to STBTT_assert(x) if you want to break on unimplemented formats. */ -#define STBTT_GPOS_TODO_assert(x) - -static stbtt_int32 stbtt__GetGlyphGPOSInfoAdvance(const stbtt_fontinfo *info, int glyph1, int glyph2) -{ - stbtt_uint16 lookupListOffset; - stbtt_uint8 *lookupList; - stbtt_uint16 lookupCount; - stbtt_uint8 *data; - stbtt_int32 i, sti; - - if (!info->gpos) return 0; - - data = info->data + info->gpos; - - if (ttUSHORT(data+0) != 1) return 0; /* Major version 1 */ - if (ttUSHORT(data+2) != 0) return 0; /* Minor version 0 */ - - lookupListOffset = ttUSHORT(data+8); - lookupList = data + lookupListOffset; - lookupCount = ttUSHORT(lookupList); - - for (i=0; i= pairSetCount) return 0; - - needle=glyph2; - r=pairValueCount-1; - l=0; - - /* Binary search. */ - while (l <= r) { - stbtt_uint16 secondGlyph; - stbtt_uint8 *pairValue; - m = (l + r) >> 1; - pairValue = pairValueArray + (2 + valueRecordPairSizeInBytes) * m; - secondGlyph = ttUSHORT(pairValue); - straw = secondGlyph; - if (needle < straw) - r = m - 1; - else if (needle > straw) - l = m + 1; - else { - stbtt_int16 xAdvance = ttSHORT(pairValue + 2); - return xAdvance; - } - } - } else - return 0; - break; - } - - case 2: { - stbtt_uint16 valueFormat1 = ttUSHORT(table + 4); - stbtt_uint16 valueFormat2 = ttUSHORT(table + 6); - if (valueFormat1 == 4 && valueFormat2 == 0) { /* Support more formats? */ - stbtt_uint16 classDef1Offset = ttUSHORT(table + 8); - stbtt_uint16 classDef2Offset = ttUSHORT(table + 10); - int glyph1class = stbtt__GetGlyphClass(table + classDef1Offset, glyph1); - int glyph2class = stbtt__GetGlyphClass(table + classDef2Offset, glyph2); - - stbtt_uint16 class1Count = ttUSHORT(table + 12); - stbtt_uint16 class2Count = ttUSHORT(table + 14); - stbtt_uint8 *class1Records, *class2Records; - stbtt_int16 xAdvance; - - if (glyph1class < 0 || glyph1class >= class1Count) return 0; /* malformed */ - if (glyph2class < 0 || glyph2class >= class2Count) return 0; /* malformed */ - - class1Records = table + 16; - class2Records = class1Records + 2 * (glyph1class * class2Count); - xAdvance = ttSHORT(class2Records + 2 * glyph2class); - return xAdvance; - } else - return 0; - break; - } - - default: - return 0; /* Unsupported position format */ - } - } - } - - return 0; -} - -STBTT_DEF int stbtt_GetGlyphKernAdvance(const stbtt_fontinfo *info, int g1, int g2) -{ - int xAdvance = 0; - - if (info->gpos) - xAdvance += stbtt__GetGlyphGPOSInfoAdvance(info, g1, g2); - else if (info->kern) - xAdvance += stbtt__GetGlyphKernInfoAdvance(info, g1, g2); - - return xAdvance; -} - -STBTT_DEF int stbtt_GetCodepointKernAdvance(const stbtt_fontinfo *info, int ch1, int ch2) -{ - if (!info->kern && !info->gpos) /* if no kerning table, don't waste time looking up both codepoint->glyphs */ - return 0; - return stbtt_GetGlyphKernAdvance(info, stbtt_FindGlyphIndex(info,ch1), stbtt_FindGlyphIndex(info,ch2)); -} - -STBTT_DEF void stbtt_GetCodepointHMetrics(const stbtt_fontinfo *info, int codepoint, int *advanceWidth, int *leftSideBearing) -{ - stbtt_GetGlyphHMetrics(info, stbtt_FindGlyphIndex(info,codepoint), advanceWidth, leftSideBearing); -} - -STBTT_DEF void stbtt_GetFontVMetrics(const stbtt_fontinfo *info, int *ascent, int *descent, int *lineGap) -{ - if (ascent ) *ascent = ttSHORT(info->data+info->hhea + 4); - if (descent) *descent = ttSHORT(info->data+info->hhea + 6); - if (lineGap) *lineGap = ttSHORT(info->data+info->hhea + 8); -} - -STBTT_DEF int stbtt_GetFontVMetricsOS2(const stbtt_fontinfo *info, int *typoAscent, int *typoDescent, int *typoLineGap) -{ - int tab = stbtt__find_table(info->data, info->fontstart, "OS/2"); - if (!tab) - return 0; - if (typoAscent ) *typoAscent = ttSHORT(info->data+tab + 68); - if (typoDescent) *typoDescent = ttSHORT(info->data+tab + 70); - if (typoLineGap) *typoLineGap = ttSHORT(info->data+tab + 72); - return 1; -} - -STBTT_DEF void stbtt_GetFontBoundingBox(const stbtt_fontinfo *info, int *x0, int *y0, int *x1, int *y1) -{ - *x0 = ttSHORT(info->data + info->head + 36); - *y0 = ttSHORT(info->data + info->head + 38); - *x1 = ttSHORT(info->data + info->head + 40); - *y1 = ttSHORT(info->data + info->head + 42); -} - -STBTT_DEF float stbtt_ScaleForPixelHeight(const stbtt_fontinfo *info, float height) -{ - int fheight = ttSHORT(info->data + info->hhea + 4) - ttSHORT(info->data + info->hhea + 6); - return (float) height / fheight; -} - -STBTT_DEF float stbtt_ScaleForMappingEmToPixels(const stbtt_fontinfo *info, float pixels) -{ - int unitsPerEm = ttUSHORT(info->data + info->head + 18); - return pixels / unitsPerEm; -} - -STBTT_DEF void stbtt_FreeShape(const stbtt_fontinfo *info, stbtt_vertex *v) -{ - STBTT_free(v, info->userdata); -} - -STBTT_DEF stbtt_uint8 *stbtt_FindSVGDoc(const stbtt_fontinfo *info, int gl) -{ - int i; - stbtt_uint8 *data = info->data; - stbtt_uint8 *svg_doc_list = data + stbtt__get_svg((stbtt_fontinfo *) info); - - int numEntries = ttUSHORT(svg_doc_list); - stbtt_uint8 *svg_docs = svg_doc_list + 2; - - for(i=0; i= ttUSHORT(svg_doc)) && (gl <= ttUSHORT(svg_doc + 2))) - return svg_doc; - } - return 0; -} - -STBTT_DEF int stbtt_GetGlyphSVG(const stbtt_fontinfo *info, int gl, const char **svg) -{ - stbtt_uint8 *data = info->data; - stbtt_uint8 *svg_doc; - - if (info->svg == 0) - return 0; - - svg_doc = stbtt_FindSVGDoc(info, gl); - if (svg_doc != NULL) { - *svg = (char *) data + info->svg + ttULONG(svg_doc + 4); - return ttULONG(svg_doc + 8); - } else { - return 0; - } -} - -STBTT_DEF int stbtt_GetCodepointSVG(const stbtt_fontinfo *info, int unicode_codepoint, const char **svg) -{ - return stbtt_GetGlyphSVG(info, stbtt_FindGlyphIndex(info, unicode_codepoint), svg); -} - -/* //////////////////////////////////////////////////////////////////////////// */ -/* */ -/* antialiasing software rasterizer */ -/* */ - -STBTT_DEF void stbtt_GetGlyphBitmapBoxSubpixel(const stbtt_fontinfo *font, int glyph, float scale_x, float scale_y,float shift_x, float shift_y, int *ix0, int *iy0, int *ix1, int *iy1) -{ - int x0=0,y0=0,x1,y1; /* =0 suppresses compiler warning */ - if (!stbtt_GetGlyphBox(font, glyph, &x0,&y0,&x1,&y1)) { - /* e.g. space character */ - if (ix0) *ix0 = 0; - if (iy0) *iy0 = 0; - if (ix1) *ix1 = 0; - if (iy1) *iy1 = 0; - } else { - /* move to integral bboxes (treating pixels as little squares, what pixels get touched)? */ - if (ix0) *ix0 = STBTT_ifloor( x0 * scale_x + shift_x); - if (iy0) *iy0 = STBTT_ifloor(-y1 * scale_y + shift_y); - if (ix1) *ix1 = STBTT_iceil ( x1 * scale_x + shift_x); - if (iy1) *iy1 = STBTT_iceil (-y0 * scale_y + shift_y); - } -} - -STBTT_DEF void stbtt_GetGlyphBitmapBox(const stbtt_fontinfo *font, int glyph, float scale_x, float scale_y, int *ix0, int *iy0, int *ix1, int *iy1) -{ - stbtt_GetGlyphBitmapBoxSubpixel(font, glyph, scale_x, scale_y,0.0f,0.0f, ix0, iy0, ix1, iy1); -} - -STBTT_DEF void stbtt_GetCodepointBitmapBoxSubpixel(const stbtt_fontinfo *font, int codepoint, float scale_x, float scale_y, float shift_x, float shift_y, int *ix0, int *iy0, int *ix1, int *iy1) -{ - stbtt_GetGlyphBitmapBoxSubpixel(font, stbtt_FindGlyphIndex(font,codepoint), scale_x, scale_y,shift_x,shift_y, ix0,iy0,ix1,iy1); -} - -STBTT_DEF void stbtt_GetCodepointBitmapBox(const stbtt_fontinfo *font, int codepoint, float scale_x, float scale_y, int *ix0, int *iy0, int *ix1, int *iy1) -{ - stbtt_GetCodepointBitmapBoxSubpixel(font, codepoint, scale_x, scale_y,0.0f,0.0f, ix0,iy0,ix1,iy1); -} - -/* //////////////////////////////////////////////////////////////////////////// */ -/* */ -/* Rasterizer */ - -typedef struct stbtt__hheap_chunk -{ - struct stbtt__hheap_chunk *next; -} stbtt__hheap_chunk; - -typedef struct stbtt__hheap -{ - struct stbtt__hheap_chunk *head; - void *first_free; - int num_remaining_in_head_chunk; -} stbtt__hheap; - -static void *stbtt__hheap_alloc(stbtt__hheap *hh, size_t size, void *userdata) -{ - if (hh->first_free) { - void *p = hh->first_free; - hh->first_free = * (void **) p; - return p; - } else { - if (hh->num_remaining_in_head_chunk == 0) { - int count = (size < 32 ? 2000 : size < 128 ? 800 : 100); - stbtt__hheap_chunk *c = (stbtt__hheap_chunk *) STBTT_malloc(sizeof(stbtt__hheap_chunk) + size * count, userdata); - if (c == NULL) - return NULL; - c->next = hh->head; - hh->head = c; - hh->num_remaining_in_head_chunk = count; - } - --hh->num_remaining_in_head_chunk; - return (char *) (hh->head) + sizeof(stbtt__hheap_chunk) + size * hh->num_remaining_in_head_chunk; - } -} - -static void stbtt__hheap_free(stbtt__hheap *hh, void *p) -{ - *(void **) p = hh->first_free; - hh->first_free = p; -} - -static void stbtt__hheap_cleanup(stbtt__hheap *hh, void *userdata) -{ - stbtt__hheap_chunk *c = hh->head; - while (c) { - stbtt__hheap_chunk *n = c->next; - STBTT_free(c, userdata); - c = n; - } -} - -typedef struct stbtt__edge { - float x0,y0, x1,y1; - int invert; -} stbtt__edge; - - -typedef struct stbtt__active_edge -{ - struct stbtt__active_edge *next; - #if STBTT_RASTERIZER_VERSION==1 - int x,dx; - float ey; - int direction; - #elif STBTT_RASTERIZER_VERSION==2 - float fx,fdx,fdy; - float direction; - float sy; - float ey; - #else - #error "Unrecognized value of STBTT_RASTERIZER_VERSION" - #endif -} stbtt__active_edge; - -#if STBTT_RASTERIZER_VERSION == 1 -#define STBTT_FIXSHIFT 10 -#define STBTT_FIX (1 << STBTT_FIXSHIFT) -#define STBTT_FIXMASK (STBTT_FIX-1) - -static stbtt__active_edge *stbtt__new_active(stbtt__hheap *hh, stbtt__edge *e, int off_x, float start_point, void *userdata) -{ - stbtt__active_edge *z = (stbtt__active_edge *) stbtt__hheap_alloc(hh, sizeof(*z), userdata); - float dxdy = (e->x1 - e->x0) / (e->y1 - e->y0); - STBTT_assert(z != NULL); - if (!z) return z; - - /* round dx down to avoid overshooting */ - if (dxdy < 0) - z->dx = -STBTT_ifloor(STBTT_FIX * -dxdy); - else - z->dx = STBTT_ifloor(STBTT_FIX * dxdy); - - z->x = STBTT_ifloor(STBTT_FIX * e->x0 + z->dx * (start_point - e->y0)); /* use z->dx so when we offset later it's by the same amount */ - z->x -= off_x * STBTT_FIX; - - z->ey = e->y1; - z->next = 0; - z->direction = e->invert ? 1 : -1; - return z; -} -#elif STBTT_RASTERIZER_VERSION == 2 -static stbtt__active_edge *stbtt__new_active(stbtt__hheap *hh, stbtt__edge *e, int off_x, float start_point, void *userdata) -{ - stbtt__active_edge *z = (stbtt__active_edge *) stbtt__hheap_alloc(hh, sizeof(*z), userdata); - float dxdy = (e->x1 - e->x0) / (e->y1 - e->y0); - STBTT_assert(z != NULL); - /* STBTT_assert(e->y0 <= start_point); */ - if (!z) return z; - z->fdx = dxdy; - z->fdy = dxdy != 0.0f ? (1.0f/dxdy) : 0.0f; - z->fx = e->x0 + dxdy * (start_point - e->y0); - z->fx -= off_x; - z->direction = e->invert ? 1.0f : -1.0f; - z->sy = e->y0; - z->ey = e->y1; - z->next = 0; - return z; -} -#else -#error "Unrecognized value of STBTT_RASTERIZER_VERSION" -#endif - -#if STBTT_RASTERIZER_VERSION == 1 -/* 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 */ -static void stbtt__fill_active_edges(unsigned char *scanline, int len, stbtt__active_edge *e, int max_weight) -{ - /* non-zero winding fill */ - int x0=0, w=0; - - while (e) { - if (w == 0) { - /* if we're currently at zero, we need to record the edge start point */ - x0 = e->x; w += e->direction; - } else { - int x1 = e->x; w += e->direction; - /* if we went to zero, we need to draw */ - if (w == 0) { - int i = x0 >> STBTT_FIXSHIFT; - int j = x1 >> STBTT_FIXSHIFT; - - if (i < len && j >= 0) { - if (i == j) { - /* x0,x1 are the same pixel, so compute combined coverage */ - scanline[i] = scanline[i] + (stbtt_uint8) ((x1 - x0) * max_weight >> STBTT_FIXSHIFT); - } else { - if (i >= 0) /* add antialiasing for x0 */ - scanline[i] = scanline[i] + (stbtt_uint8) (((STBTT_FIX - (x0 & STBTT_FIXMASK)) * max_weight) >> STBTT_FIXSHIFT); - else - i = -1; /* clip */ - - if (j < len) /* add antialiasing for x1 */ - scanline[j] = scanline[j] + (stbtt_uint8) (((x1 & STBTT_FIXMASK) * max_weight) >> STBTT_FIXSHIFT); - else - j = len; /* clip */ - - for (++i; i < j; ++i) /* fill pixels between x0 and x1 */ - scanline[i] = scanline[i] + (stbtt_uint8) max_weight; - } - } - } - } - - e = e->next; - } -} - -static void stbtt__rasterize_sorted_edges(stbtt__bitmap *result, stbtt__edge *e, int n, int vsubsample, int off_x, int off_y, void *userdata) -{ - stbtt__hheap hh = { 0, 0, 0 }; - stbtt__active_edge *active = NULL; - int y,j=0; - int max_weight = (255 / vsubsample); /* weight per vertical scanline */ - int s; /* vertical subsample index */ - unsigned char scanline_data[512], *scanline; - - if (result->w > 512) - scanline = (unsigned char *) STBTT_malloc(result->w, userdata); - else - scanline = scanline_data; - - y = off_y * vsubsample; - e[n].y0 = (off_y + result->h) * (float) vsubsample + 1; - - while (j < result->h) { - STBTT_memset(scanline, 0, result->w); - for (s=0; s < vsubsample; ++s) { - /* find center of pixel for this scanline */ - float scan_y = y + 0.5f; - stbtt__active_edge **step = &active; - - /* update all active edges; */ - /* remove all active edges that terminate before the center of this scanline */ - while (*step) { - stbtt__active_edge * z = *step; - if (z->ey <= scan_y) { - *step = z->next; /* delete from list */ - STBTT_assert(z->direction); - z->direction = 0; - stbtt__hheap_free(&hh, 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) { - stbtt__active_edge *t = *step; - stbtt__active_edge *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->y0 <= scan_y) { - if (e->y1 > scan_y) { - stbtt__active_edge *z = stbtt__new_active(&hh, e, off_x, scan_y, userdata); - if (z != NULL) { - /* find insertion point */ - if (active == NULL) - active = z; - else if (z->x < active->x) { - /* insert at front */ - z->next = active; - active = z; - } else { - /* find thing to insert AFTER */ - stbtt__active_edge *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 XOR fashion */ - if (active) - stbtt__fill_active_edges(scanline, result->w, active, max_weight); - - ++y; - } - STBTT_memcpy(result->pixels + j * result->stride, scanline, result->w); - ++j; - } - - stbtt__hheap_cleanup(&hh, userdata); - - if (scanline != scanline_data) - STBTT_free(scanline, userdata); -} - -#elif STBTT_RASTERIZER_VERSION == 2 - -/* the edge passed in here does not cross the vertical line at x or the vertical line at x+1 */ -/* (i.e. it has already been clipped to those) */ -static void stbtt__handle_clipped_edge(float *scanline, int x, stbtt__active_edge *e, float x0, float y0, float x1, float y1) -{ - if (y0 == y1) return; - STBTT_assert(y0 < y1); - STBTT_assert(e->sy <= e->ey); - if (y0 > e->ey) return; - if (y1 < e->sy) return; - if (y0 < e->sy) { - x0 += (x1-x0) * (e->sy - y0) / (y1-y0); - y0 = e->sy; - } - if (y1 > e->ey) { - x1 += (x1-x0) * (e->ey - y1) / (y1-y0); - y1 = e->ey; - } - - if (x0 == x) - STBTT_assert(x1 <= x+1); - else if (x0 == x+1) - STBTT_assert(x1 >= x); - else if (x0 <= x) - STBTT_assert(x1 <= x); - else if (x0 >= x+1) - STBTT_assert(x1 >= x+1); - else - STBTT_assert(x1 >= x && x1 <= x+1); - - if (x0 <= x && x1 <= x) - scanline[x] += e->direction * (y1-y0); - else if (x0 >= x+1 && x1 >= x+1) - ; - else { - STBTT_assert(x0 >= x && x0 <= x+1 && x1 >= x && x1 <= x+1); - scanline[x] += e->direction * (y1-y0) * (1-((x0-x)+(x1-x))/2); /* coverage = 1 - average x position */ - } -} - -static float stbtt__sized_trapezoid_area(float height, float top_width, float bottom_width) -{ - STBTT_assert(top_width >= 0); - STBTT_assert(bottom_width >= 0); - return (top_width + bottom_width) / 2.0f * height; -} - -static float stbtt__position_trapezoid_area(float height, float tx0, float tx1, float bx0, float bx1) -{ - return stbtt__sized_trapezoid_area(height, tx1 - tx0, bx1 - bx0); -} - -static float stbtt__sized_triangle_area(float height, float width) -{ - return height * width / 2; -} - -static void stbtt__fill_active_edges_new(float *scanline, float *scanline_fill, int len, stbtt__active_edge *e, float y_top) -{ - float y_bottom = y_top+1; - - while (e) { - /* brute force every pixel */ - - /* compute intersection points with top & bottom */ - STBTT_assert(e->ey >= y_top); - - if (e->fdx == 0) { - float x0 = e->fx; - if (x0 < len) { - if (x0 >= 0) { - stbtt__handle_clipped_edge(scanline,(int) x0,e, x0,y_top, x0,y_bottom); - stbtt__handle_clipped_edge(scanline_fill-1,(int) x0+1,e, x0,y_top, x0,y_bottom); - } else { - stbtt__handle_clipped_edge(scanline_fill-1,0,e, x0,y_top, x0,y_bottom); - } - } - } else { - float x0 = e->fx; - float dx = e->fdx; - float xb = x0 + dx; - float x_top, x_bottom; - float sy0,sy1; - float dy = e->fdy; - STBTT_assert(e->sy <= y_bottom && e->ey >= y_top); - - /* compute endpoints of line segment clipped to this scanline (if the */ - /* line segment starts on this scanline. x0 is the intersection of the */ - /* line with y_top, but that may be off the line segment. */ - if (e->sy > y_top) { - x_top = x0 + dx * (e->sy - y_top); - sy0 = e->sy; - } else { - x_top = x0; - sy0 = y_top; - } - if (e->ey < y_bottom) { - x_bottom = x0 + dx * (e->ey - y_top); - sy1 = e->ey; - } else { - x_bottom = xb; - sy1 = y_bottom; - } - - if (x_top >= 0 && x_bottom >= 0 && x_top < len && x_bottom < len) { - /* from here on, we don't have to range check x values */ - - if ((int) x_top == (int) x_bottom) { - float height; - /* simple case, only spans one pixel */ - int x = (int) x_top; - height = (sy1 - sy0) * e->direction; - STBTT_assert(x >= 0 && x < len); - scanline[x] += stbtt__position_trapezoid_area(height, x_top, x+1.0f, x_bottom, x+1.0f); - scanline_fill[x] += height; /* everything right of this pixel is filled */ - } else { - int x,x1,x2; - float y_crossing, y_final, step, sign, area; - /* covers 2+ pixels */ - if (x_top > x_bottom) { - /* flip scanline vertically; signed area is the same */ - float t; - sy0 = y_bottom - (sy0 - y_top); - sy1 = y_bottom - (sy1 - y_top); - t = sy0, sy0 = sy1, sy1 = t; - t = x_bottom, x_bottom = x_top, x_top = t; - dx = -dx; - dy = -dy; - t = x0, x0 = xb, xb = t; - } - STBTT_assert(dy >= 0); - STBTT_assert(dx >= 0); - - x1 = (int) x_top; - x2 = (int) x_bottom; - /* compute intersection with y axis at x1+1 */ - y_crossing = y_top + dy * (x1+1 - x0); - - /* compute intersection with y axis at x2 */ - y_final = y_top + dy * (x2 - x0); - - /* x1 x_top x2 x_bottom */ - /* y_top +------|-----+------------+------------+--------|---+------------+ */ - /* | | | | | | */ - /* | | | | | | */ - /* sy0 | Txxxxx|............|............|............|............| */ - /* y_crossing | *xxxxx.......|............|............|............| */ - /* | | xxxxx..|............|............|............| */ - /* | | /- xx*xxxx........|............|............| */ - /* | | dy < | xxxxxx..|............|............| */ - /* y_final | | \- | xx*xxx.........|............| */ - /* sy1 | | | | xxxxxB...|............| */ - /* | | | | | | */ - /* | | | | | | */ - /* y_bottom +------------+------------+------------+------------+------------+ */ - /* */ - /* goal is to measure the area covered by '.' in each pixel */ - - /* if x2 is right at the right edge of x1, y_crossing can blow up, github #1057 */ - /* @TODO: maybe test against sy1 rather than y_bottom? */ - if (y_crossing > y_bottom) - y_crossing = y_bottom; - - sign = e->direction; - - /* area of the rectangle covered from sy0..y_crossing */ - area = sign * (y_crossing-sy0); - - /* area of the triangle (x_top,sy0), (x1+1,sy0), (x1+1,y_crossing) */ - scanline[x1] += stbtt__sized_triangle_area(area, x1+1 - x_top); - - /* check if final y_crossing is blown up; no test case for this */ - if (y_final > y_bottom) { - y_final = y_bottom; - dy = (y_final - y_crossing ) / (x2 - (x1+1)); /* if denom=0, y_final = y_crossing, so y_final <= y_bottom */ - } - - /* in second pixel, area covered by line segment found in first pixel */ - /* is always a rectangle 1 wide * the height of that line segment; this */ - /* is exactly what the variable 'area' stores. it also gets a contribution */ - /* from the line segment within it. the THIRD pixel will get the first */ - /* pixel's rectangle contribution, the second pixel's rectangle contribution, */ - /* and its own contribution. the 'own contribution' is the same in every pixel except */ - /* the leftmost and rightmost, a trapezoid that slides down in each pixel. */ - /* the second pixel's contribution to the third pixel will be the */ - /* rectangle 1 wide times the height change in the second pixel, which is dy. */ - - step = sign * dy * 1; /* dy is dy/dx, change in y for every 1 change in x, */ - /* which multiplied by 1-pixel-width is how much pixel area changes for each step in x */ - /* so the area advances by 'step' every time */ - - for (x = x1+1; x < x2; ++x) { - scanline[x] += area + step/2; /* area of trapezoid is 1*step/2 */ - area += step; - } - STBTT_assert(STBTT_fabs(area) <= 1.01f); /* accumulated error from area += step unless we round step down */ - STBTT_assert(sy1 > y_final-0.01f); - - /* area covered in the last pixel is the rectangle from all the pixels to the left, */ - /* plus the trapezoid filled by the line segment in this pixel all the way to the right edge */ - scanline[x2] += area + sign * stbtt__position_trapezoid_area(sy1-y_final, (float) x2, x2+1.0f, x_bottom, x2+1.0f); - - /* the rest of the line is filled based on the total height of the line segment in this pixel */ - scanline_fill[x2] += sign * (sy1-sy0); - } - } else { - /* if edge goes outside of box we're drawing, we require */ - /* clipping logic. since this does not match the intended use */ - /* of this library, we use a different, very slow brute */ - /* force implementation */ - /* note though that this does happen some of the time because */ - /* x_top and x_bottom can be extrapolated at the top & bottom of */ - /* the shape and actually lie outside the bounding box */ - int x; - for (x=0; x < len; ++x) { - /* cases: */ - /* */ - /* there can be up to two intersections with the pixel. any intersection */ - /* with left or right edges can be handled by splitting into two (or three) */ - /* regions. intersections with top & bottom do not necessitate case-wise logic. */ - /* */ - /* the old way of doing this found the intersections with the left & right edges, */ - /* then used some simple logic to produce up to three segments in sorted order */ - /* from top-to-bottom. however, this had a problem: if an x edge was epsilon */ - /* across the x border, then the corresponding y position might not be distinct */ - /* from the other y segment, and it might ignored as an empty segment. to avoid */ - /* that, we need to explicitly produce segments based on x positions. */ - - /* rename variables to clearly-defined pairs */ - float y0 = y_top; - float x1 = (float) (x); - float x2 = (float) (x+1); - float x3 = xb; - float y3 = y_bottom; - - /* x = e->x + e->dx * (y-y_top) */ - /* (y-y_top) = (x - e->x) / e->dx */ - /* y = (x - e->x) / e->dx + y_top */ - float y1 = (x - x0) / dx + y_top; - float y2 = (x+1 - x0) / dx + y_top; - - if (x0 < x1 && x3 > x2) { /* three segments descending down-right */ - stbtt__handle_clipped_edge(scanline,x,e, x0,y0, x1,y1); - stbtt__handle_clipped_edge(scanline,x,e, x1,y1, x2,y2); - stbtt__handle_clipped_edge(scanline,x,e, x2,y2, x3,y3); - } else if (x3 < x1 && x0 > x2) { /* three segments descending down-left */ - stbtt__handle_clipped_edge(scanline,x,e, x0,y0, x2,y2); - stbtt__handle_clipped_edge(scanline,x,e, x2,y2, x1,y1); - stbtt__handle_clipped_edge(scanline,x,e, x1,y1, x3,y3); - } else if (x0 < x1 && x3 > x1) { /* two segments across x, down-right */ - stbtt__handle_clipped_edge(scanline,x,e, x0,y0, x1,y1); - stbtt__handle_clipped_edge(scanline,x,e, x1,y1, x3,y3); - } else if (x3 < x1 && x0 > x1) { /* two segments across x, down-left */ - stbtt__handle_clipped_edge(scanline,x,e, x0,y0, x1,y1); - stbtt__handle_clipped_edge(scanline,x,e, x1,y1, x3,y3); - } else if (x0 < x2 && x3 > x2) { /* two segments across x+1, down-right */ - stbtt__handle_clipped_edge(scanline,x,e, x0,y0, x2,y2); - stbtt__handle_clipped_edge(scanline,x,e, x2,y2, x3,y3); - } else if (x3 < x2 && x0 > x2) { /* two segments across x+1, down-left */ - stbtt__handle_clipped_edge(scanline,x,e, x0,y0, x2,y2); - stbtt__handle_clipped_edge(scanline,x,e, x2,y2, x3,y3); - } else { /* one segment */ - stbtt__handle_clipped_edge(scanline,x,e, x0,y0, x3,y3); - } - } - } - } - e = e->next; - } -} - -/* directly AA rasterize edges w/o supersampling */ -static void stbtt__rasterize_sorted_edges(stbtt__bitmap *result, stbtt__edge *e, int n, int vsubsample, int off_x, int off_y, void *userdata) -{ - stbtt__hheap hh = { 0, 0, 0 }; - stbtt__active_edge *active = NULL; - int y,j=0, i; - float scanline_data[129], *scanline, *scanline2; - - STBTT__NOTUSED(vsubsample); - - if (result->w > 64) - scanline = (float *) STBTT_malloc((result->w*2+1) * sizeof(float), userdata); - else - scanline = scanline_data; - - scanline2 = scanline + result->w; - - y = off_y; - e[n].y0 = (float) (off_y + result->h) + 1; - - while (j < result->h) { - /* find center of pixel for this scanline */ - float scan_y_top = y + 0.0f; - float scan_y_bottom = y + 1.0f; - stbtt__active_edge **step = &active; - - STBTT_memset(scanline , 0, result->w*sizeof(scanline[0])); - STBTT_memset(scanline2, 0, (result->w+1)*sizeof(scanline[0])); - - /* update all active edges; */ - /* remove all active edges that terminate before the top of this scanline */ - while (*step) { - stbtt__active_edge * z = *step; - if (z->ey <= scan_y_top) { - *step = z->next; /* delete from list */ - STBTT_assert(z->direction); - z->direction = 0; - stbtt__hheap_free(&hh, z); - } else { - step = &((*step)->next); /* advance through list */ - } - } - - /* insert all edges that start before the bottom of this scanline */ - while (e->y0 <= scan_y_bottom) { - if (e->y0 != e->y1) { - stbtt__active_edge *z = stbtt__new_active(&hh, e, off_x, scan_y_top, userdata); - if (z != NULL) { - if (j == 0 && off_y != 0) { - if (z->ey < scan_y_top) { - /* this can happen due to subpixel positioning and some kind of fp rounding error i think */ - z->ey = scan_y_top; - } - } - STBTT_assert(z->ey >= scan_y_top); /* if we get really unlucky a tiny bit of an edge can be out of bounds */ - /* insert at front */ - z->next = active; - active = z; - } - } - ++e; - } - - /* now process all active edges */ - if (active) - stbtt__fill_active_edges_new(scanline, scanline2+1, result->w, active, scan_y_top); - - { - float sum = 0; - for (i=0; i < result->w; ++i) { - float k; - int m; - sum += scanline2[i]; - k = scanline[i] + sum; - k = (float) STBTT_fabs(k)*255 + 0.5f; - m = (int) k; - if (m > 255) m = 255; - result->pixels[j*result->stride + i] = (unsigned char) m; - } - } - /* advance all the edges */ - step = &active; - while (*step) { - stbtt__active_edge *z = *step; - z->fx += z->fdx; /* advance to position for current scanline */ - step = &((*step)->next); /* advance through list */ - } - - ++y; - ++j; - } - - stbtt__hheap_cleanup(&hh, userdata); - - if (scanline != scanline_data) - STBTT_free(scanline, userdata); -} -#else -#error "Unrecognized value of STBTT_RASTERIZER_VERSION" -#endif - -#define STBTT__COMPARE(a,b) ((a)->y0 < (b)->y0) - -static void stbtt__sort_edges_ins_sort(stbtt__edge *p, int n) -{ - int i,j; - for (i=1; i < n; ++i) { - stbtt__edge t = p[i], *a = &t; - j = i; - while (j > 0) { - stbtt__edge *b = &p[j-1]; - int c = STBTT__COMPARE(a,b); - if (!c) break; - p[j] = p[j-1]; - --j; - } - if (i != j) - p[j] = t; - } -} - -static void stbtt__sort_edges_quicksort(stbtt__edge *p, int n) -{ - /* threshold for transitioning to insertion sort */ - while (n > 12) { - stbtt__edge t; - int c01,c12,c,m,i,j; - - /* compute median of three */ - m = n >> 1; - c01 = STBTT__COMPARE(&p[0],&p[m]); - c12 = STBTT__COMPARE(&p[m],&p[n-1]); - /* if 0 >= mid >= end, or 0 < mid < end, then use mid */ - if (c01 != c12) { - /* otherwise, we'll need to swap something else to middle */ - int z; - c = STBTT__COMPARE(&p[0],&p[n-1]); - /* 0>mid && midn => n; 0 0 */ - /* 0n: 0>n => 0; 0 n */ - z = (c == c12) ? 0 : n-1; - t = p[z]; - p[z] = p[m]; - p[m] = t; - } - /* now p[m] is the median-of-three */ - /* swap it to the beginning so it won't move around */ - t = p[0]; - p[0] = p[m]; - p[m] = t; - - /* partition loop */ - i=1; - j=n-1; - for(;;) { - /* handling of equality is crucial here */ - /* for sentinels & efficiency with duplicates */ - for (;;++i) { - if (!STBTT__COMPARE(&p[i], &p[0])) break; - } - for (;;--j) { - if (!STBTT__COMPARE(&p[0], &p[j])) break; - } - /* make sure we haven't crossed */ - if (i >= j) break; - t = p[i]; - p[i] = p[j]; - p[j] = t; - - ++i; - --j; - } - /* recurse on smaller side, iterate on larger */ - if (j < (n-i)) { - stbtt__sort_edges_quicksort(p,j); - p = p+i; - n = n-i; - } else { - stbtt__sort_edges_quicksort(p+i, n-i); - n = j; - } - } -} - -static void stbtt__sort_edges(stbtt__edge *p, int n) -{ - stbtt__sort_edges_quicksort(p, n); - stbtt__sort_edges_ins_sort(p, n); -} - -typedef struct -{ - float x,y; -} stbtt__point; - -static void stbtt__rasterize(stbtt__bitmap *result, stbtt__point *pts, int *wcount, int windings, float scale_x, float scale_y, float shift_x, float shift_y, int off_x, int off_y, int invert, void *userdata) -{ - float y_scale_inv = invert ? -scale_y : scale_y; - stbtt__edge *e; - int n,i,j,k,m; -#if STBTT_RASTERIZER_VERSION == 1 - int vsubsample = result->h < 8 ? 15 : 5; -#elif STBTT_RASTERIZER_VERSION == 2 - int vsubsample = 1; -#else - #error "Unrecognized value of STBTT_RASTERIZER_VERSION" -#endif - /* vsubsample should divide 255 evenly; otherwise we won't reach full opacity */ - - /* now we have to blow out the windings into explicit edge lists */ - n = 0; - for (i=0; i < windings; ++i) - n += wcount[i]; - - e = (stbtt__edge *) STBTT_malloc(sizeof(*e) * (n+1), userdata); /* add an extra one as a sentinel */ - if (e == 0) return; - n = 0; - - m=0; - for (i=0; i < windings; ++i) { - stbtt__point *p = pts + m; - m += wcount[i]; - j = wcount[i]-1; - for (k=0; k < wcount[i]; j=k++) { - int a=k,b=j; - /* skip the edge if horizontal */ - if (p[j].y == p[k].y) - continue; - /* add edge from j to k to the list */ - e[n].invert = 0; - if (invert ? p[j].y > p[k].y : p[j].y < p[k].y) { - e[n].invert = 1; - a=j,b=k; - } - e[n].x0 = p[a].x * scale_x + shift_x; - e[n].y0 = (p[a].y * y_scale_inv + shift_y) * vsubsample; - e[n].x1 = p[b].x * scale_x + shift_x; - e[n].y1 = (p[b].y * y_scale_inv + shift_y) * vsubsample; - ++n; - } - } - - /* now sort the edges by their highest point (should snap to integer, and then by x) */ - /* STBTT_sort(e, n, sizeof(e[0]), stbtt__edge_compare); */ - stbtt__sort_edges(e, n); - - /* now, traverse the scanlines and find the intersections on each scanline, use xor winding rule */ - stbtt__rasterize_sorted_edges(result, e, n, vsubsample, off_x, off_y, userdata); - - STBTT_free(e, userdata); -} - -static void stbtt__add_point(stbtt__point *points, int n, float x, float y) -{ - if (!points) return; /* during first pass, it's unallocated */ - points[n].x = x; - points[n].y = y; -} - -/* tessellate until threshold p is happy... @TODO warped to compensate for non-linear stretching */ -static int stbtt__tesselate_curve(stbtt__point *points, int *num_points, float x0, float y0, float x1, float y1, float x2, float y2, float objspace_flatness_squared, int n) -{ - /* midpoint */ - float mx = (x0 + 2*x1 + x2)/4; - float my = (y0 + 2*y1 + y2)/4; - /* versus directly drawn line */ - float dx = (x0+x2)/2 - mx; - float dy = (y0+y2)/2 - my; - if (n > 16) /* 65536 segments on one curve better be enough! */ - return 1; - if (dx*dx+dy*dy > objspace_flatness_squared) { /* half-pixel error allowed... need to be smaller if AA */ - stbtt__tesselate_curve(points, num_points, x0,y0, (x0+x1)/2.0f,(y0+y1)/2.0f, mx,my, objspace_flatness_squared,n+1); - stbtt__tesselate_curve(points, num_points, mx,my, (x1+x2)/2.0f,(y1+y2)/2.0f, x2,y2, objspace_flatness_squared,n+1); - } else { - stbtt__add_point(points, *num_points,x2,y2); - *num_points = *num_points+1; - } - return 1; -} - -static void stbtt__tesselate_cubic(stbtt__point *points, int *num_points, float x0, float y0, float x1, float y1, float x2, float y2, float x3, float y3, float objspace_flatness_squared, int n) -{ - /* @TODO this "flatness" calculation is just made-up nonsense that seems to work well enough */ - float dx0 = x1-x0; - float dy0 = y1-y0; - float dx1 = x2-x1; - float dy1 = y2-y1; - float dx2 = x3-x2; - float dy2 = y3-y2; - float dx = x3-x0; - float dy = y3-y0; - float longlen = (float) (STBTT_sqrt(dx0*dx0+dy0*dy0)+STBTT_sqrt(dx1*dx1+dy1*dy1)+STBTT_sqrt(dx2*dx2+dy2*dy2)); - float shortlen = (float) STBTT_sqrt(dx*dx+dy*dy); - float flatness_squared = longlen*longlen-shortlen*shortlen; - - if (n > 16) /* 65536 segments on one curve better be enough! */ - return; - - if (flatness_squared > objspace_flatness_squared) { - float x01 = (x0+x1)/2; - float y01 = (y0+y1)/2; - float x12 = (x1+x2)/2; - float y12 = (y1+y2)/2; - float x23 = (x2+x3)/2; - float y23 = (y2+y3)/2; - - float xa = (x01+x12)/2; - float ya = (y01+y12)/2; - float xb = (x12+x23)/2; - float yb = (y12+y23)/2; - - float mx = (xa+xb)/2; - float my = (ya+yb)/2; - - stbtt__tesselate_cubic(points, num_points, x0,y0, x01,y01, xa,ya, mx,my, objspace_flatness_squared,n+1); - stbtt__tesselate_cubic(points, num_points, mx,my, xb,yb, x23,y23, x3,y3, objspace_flatness_squared,n+1); - } else { - stbtt__add_point(points, *num_points,x3,y3); - *num_points = *num_points+1; - } -} - -/* returns number of contours */ -static stbtt__point *stbtt_FlattenCurves(stbtt_vertex *vertices, int num_verts, float objspace_flatness, int **contour_lengths, int *num_contours, void *userdata) -{ - stbtt__point *points=0; - int num_points=0; - - float objspace_flatness_squared = objspace_flatness * objspace_flatness; - int i,n=0,start=0, pass; - - /* count how many "moves" there are to get the contour count */ - for (i=0; i < num_verts; ++i) - if (vertices[i].type == STBTT_vmove) - ++n; - - *num_contours = n; - if (n == 0) return 0; - - *contour_lengths = (int *) STBTT_malloc(sizeof(**contour_lengths) * n, userdata); - - if (*contour_lengths == 0) { - *num_contours = 0; - return 0; - } - - /* make two passes through the points so we don't need to realloc */ - for (pass=0; pass < 2; ++pass) { - float x=0,y=0; - if (pass == 1) { - points = (stbtt__point *) STBTT_malloc(num_points * sizeof(points[0]), userdata); - if (points == NULL) goto error; - } - num_points = 0; - n= -1; - for (i=0; i < num_verts; ++i) { - switch (vertices[i].type) { - case STBTT_vmove: - /* start the next contour */ - if (n >= 0) - (*contour_lengths)[n] = num_points - start; - ++n; - start = num_points; - - x = vertices[i].x, y = vertices[i].y; - stbtt__add_point(points, num_points++, x,y); - break; - case STBTT_vline: - x = vertices[i].x, y = vertices[i].y; - stbtt__add_point(points, num_points++, x, y); - break; - case STBTT_vcurve: - stbtt__tesselate_curve(points, &num_points, x,y, - vertices[i].cx, vertices[i].cy, - vertices[i].x, vertices[i].y, - objspace_flatness_squared, 0); - x = vertices[i].x, y = vertices[i].y; - break; - case STBTT_vcubic: - stbtt__tesselate_cubic(points, &num_points, x,y, - vertices[i].cx, vertices[i].cy, - vertices[i].cx1, vertices[i].cy1, - vertices[i].x, vertices[i].y, - objspace_flatness_squared, 0); - x = vertices[i].x, y = vertices[i].y; - break; - } - } - (*contour_lengths)[n] = num_points - start; - } - - return points; -error: - STBTT_free(points, userdata); - STBTT_free(*contour_lengths, userdata); - *contour_lengths = 0; - *num_contours = 0; - return NULL; -} - -STBTT_DEF void stbtt_Rasterize(stbtt__bitmap *result, float flatness_in_pixels, stbtt_vertex *vertices, int num_verts, float scale_x, float scale_y, float shift_x, float shift_y, int x_off, int y_off, int invert, void *userdata) -{ - float scale = scale_x > scale_y ? scale_y : scale_x; - int winding_count = 0; - int *winding_lengths = NULL; - stbtt__point *windings = stbtt_FlattenCurves(vertices, num_verts, flatness_in_pixels / scale, &winding_lengths, &winding_count, userdata); - if (windings) { - stbtt__rasterize(result, windings, winding_lengths, winding_count, scale_x, scale_y, shift_x, shift_y, x_off, y_off, invert, userdata); - STBTT_free(winding_lengths, userdata); - STBTT_free(windings, userdata); - } -} - -STBTT_DEF void stbtt_FreeBitmap(unsigned char *bitmap, void *userdata) -{ - STBTT_free(bitmap, userdata); -} - -STBTT_DEF unsigned char *stbtt_GetGlyphBitmapSubpixel(const stbtt_fontinfo *info, float scale_x, float scale_y, float shift_x, float shift_y, int glyph, int *width, int *height, int *xoff, int *yoff) -{ - int ix0,iy0,ix1,iy1; - stbtt__bitmap gbm; - stbtt_vertex *vertices; - int num_verts = stbtt_GetGlyphShape(info, glyph, &vertices); - - if (scale_x == 0) scale_x = scale_y; - if (scale_y == 0) { - if (scale_x == 0) { - STBTT_free(vertices, info->userdata); - return NULL; - } - scale_y = scale_x; - } - - stbtt_GetGlyphBitmapBoxSubpixel(info, glyph, scale_x, scale_y, shift_x, shift_y, &ix0,&iy0,&ix1,&iy1); - - /* now we get the size */ - gbm.w = (ix1 - ix0); - gbm.h = (iy1 - iy0); - gbm.pixels = NULL; /* in case we error */ - - if (width ) *width = gbm.w; - if (height) *height = gbm.h; - if (xoff ) *xoff = ix0; - if (yoff ) *yoff = iy0; - - if (gbm.w && gbm.h) { - gbm.pixels = (unsigned char *) STBTT_malloc(gbm.w * gbm.h, info->userdata); - if (gbm.pixels) { - gbm.stride = gbm.w; - - stbtt_Rasterize(&gbm, 0.35f, vertices, num_verts, scale_x, scale_y, shift_x, shift_y, ix0, iy0, 1, info->userdata); - } - } - STBTT_free(vertices, info->userdata); - return gbm.pixels; -} - -STBTT_DEF unsigned char *stbtt_GetGlyphBitmap(const stbtt_fontinfo *info, float scale_x, float scale_y, int glyph, int *width, int *height, int *xoff, int *yoff) -{ - return stbtt_GetGlyphBitmapSubpixel(info, scale_x, scale_y, 0.0f, 0.0f, glyph, width, height, xoff, yoff); -} - -STBTT_DEF void stbtt_MakeGlyphBitmapSubpixel(const stbtt_fontinfo *info, unsigned char *output, int out_w, int out_h, int out_stride, float scale_x, float scale_y, float shift_x, float shift_y, int glyph) -{ - int ix0,iy0; - stbtt_vertex *vertices; - int num_verts = stbtt_GetGlyphShape(info, glyph, &vertices); - stbtt__bitmap gbm; - - stbtt_GetGlyphBitmapBoxSubpixel(info, glyph, scale_x, scale_y, shift_x, shift_y, &ix0,&iy0,0,0); - gbm.pixels = output; - gbm.w = out_w; - gbm.h = out_h; - gbm.stride = out_stride; - - if (gbm.w && gbm.h) - stbtt_Rasterize(&gbm, 0.35f, vertices, num_verts, scale_x, scale_y, shift_x, shift_y, ix0,iy0, 1, info->userdata); - - STBTT_free(vertices, info->userdata); -} - -STBTT_DEF void stbtt_MakeGlyphBitmap(const stbtt_fontinfo *info, unsigned char *output, int out_w, int out_h, int out_stride, float scale_x, float scale_y, int glyph) -{ - stbtt_MakeGlyphBitmapSubpixel(info, output, out_w, out_h, out_stride, scale_x, scale_y, 0.0f,0.0f, glyph); -} - -STBTT_DEF unsigned char *stbtt_GetCodepointBitmapSubpixel(const stbtt_fontinfo *info, float scale_x, float scale_y, float shift_x, float shift_y, int codepoint, int *width, int *height, int *xoff, int *yoff) -{ - return stbtt_GetGlyphBitmapSubpixel(info, scale_x, scale_y,shift_x,shift_y, stbtt_FindGlyphIndex(info,codepoint), width,height,xoff,yoff); -} - -STBTT_DEF void stbtt_MakeCodepointBitmapSubpixelPrefilter(const stbtt_fontinfo *info, unsigned char *output, int out_w, int out_h, int out_stride, float scale_x, float scale_y, float shift_x, float shift_y, int oversample_x, int oversample_y, float *sub_x, float *sub_y, int codepoint) -{ - stbtt_MakeGlyphBitmapSubpixelPrefilter(info, output, out_w, out_h, out_stride, scale_x, scale_y, shift_x, shift_y, oversample_x, oversample_y, sub_x, sub_y, stbtt_FindGlyphIndex(info,codepoint)); -} - -STBTT_DEF void stbtt_MakeCodepointBitmapSubpixel(const stbtt_fontinfo *info, unsigned char *output, int out_w, int out_h, int out_stride, float scale_x, float scale_y, float shift_x, float shift_y, int codepoint) -{ - stbtt_MakeGlyphBitmapSubpixel(info, output, out_w, out_h, out_stride, scale_x, scale_y, shift_x, shift_y, stbtt_FindGlyphIndex(info,codepoint)); -} - -STBTT_DEF unsigned char *stbtt_GetCodepointBitmap(const stbtt_fontinfo *info, float scale_x, float scale_y, int codepoint, int *width, int *height, int *xoff, int *yoff) -{ - return stbtt_GetCodepointBitmapSubpixel(info, scale_x, scale_y, 0.0f,0.0f, codepoint, width,height,xoff,yoff); -} - -STBTT_DEF void stbtt_MakeCodepointBitmap(const stbtt_fontinfo *info, unsigned char *output, int out_w, int out_h, int out_stride, float scale_x, float scale_y, int codepoint) -{ - stbtt_MakeCodepointBitmapSubpixel(info, output, out_w, out_h, out_stride, scale_x, scale_y, 0.0f,0.0f, codepoint); -} - -/* //////////////////////////////////////////////////////////////////////////// */ -/* */ -/* bitmap baking */ -/* */ -/* This is SUPER-CRAPPY packing to keep source code small */ - -static int stbtt_BakeFontBitmap_internal(unsigned char *data, int offset, /* font location (use offset=0 for plain .ttf) */ - float pixel_height, /* height of font in pixels */ - unsigned char *pixels, int pw, int ph, /* bitmap to be filled in */ - int first_char, int num_chars, /* characters to bake */ - stbtt_bakedchar *chardata) -{ - float scale; - int x,y,bottom_y, i; - stbtt_fontinfo f; - f.userdata = NULL; - if (!stbtt_InitFont(&f, data, offset)) - return -1; - STBTT_memset(pixels, 0, pw*ph); /* background of 0 around pixels */ - x=y=1; - bottom_y = 1; - - scale = stbtt_ScaleForPixelHeight(&f, pixel_height); - - for (i=0; i < num_chars; ++i) { - int advance, lsb, x0,y0,x1,y1,gw,gh; - int g = stbtt_FindGlyphIndex(&f, first_char + i); - stbtt_GetGlyphHMetrics(&f, g, &advance, &lsb); - stbtt_GetGlyphBitmapBox(&f, g, scale,scale, &x0,&y0,&x1,&y1); - gw = x1-x0; - gh = y1-y0; - if (x + gw + 1 >= pw) - y = bottom_y, x = 1; /* advance to next row */ - if (y + gh + 1 >= ph) /* check if it fits vertically AFTER potentially moving to next row */ - return -i; - STBTT_assert(x+gw < pw); - STBTT_assert(y+gh < ph); - stbtt_MakeGlyphBitmap(&f, pixels+x+y*pw, gw,gh,pw, scale,scale, g); - chardata[i].x0 = (stbtt_int16) x; - chardata[i].y0 = (stbtt_int16) y; - chardata[i].x1 = (stbtt_int16) (x + gw); - chardata[i].y1 = (stbtt_int16) (y + gh); - chardata[i].xadvance = scale * advance; - chardata[i].xoff = (float) x0; - chardata[i].yoff = (float) y0; - x = x + gw + 1; - if (y+gh+1 > bottom_y) - bottom_y = y+gh+1; - } - return bottom_y; -} - -STBTT_DEF void stbtt_GetBakedQuad(const stbtt_bakedchar *chardata, int pw, int ph, int char_index, float *xpos, float *ypos, stbtt_aligned_quad *q, int opengl_fillrule) -{ - float d3d_bias = opengl_fillrule ? 0 : -0.5f; - float ipw = 1.0f / pw, iph = 1.0f / ph; - const stbtt_bakedchar *b = chardata + char_index; - int round_x = STBTT_ifloor((*xpos + b->xoff) + 0.5f); - int round_y = STBTT_ifloor((*ypos + b->yoff) + 0.5f); - - q->x0 = round_x + d3d_bias; - q->y0 = round_y + d3d_bias; - q->x1 = round_x + b->x1 - b->x0 + d3d_bias; - q->y1 = round_y + b->y1 - b->y0 + d3d_bias; - - q->s0 = b->x0 * ipw; - q->t0 = b->y0 * iph; - q->s1 = b->x1 * ipw; - q->t1 = b->y1 * iph; - - *xpos += b->xadvance; -} - -/* //////////////////////////////////////////////////////////////////////////// */ -/* */ -/* rectangle packing replacement routines if you don't have stb_rect_pack.h */ -/* */ - -#ifndef STB_RECT_PACK_VERSION - -typedef int stbrp_coord; - -/* ////////////////////////////////////////////////////////////////////////////////// */ -/* // */ -/* // */ -/* COMPILER WARNING ?!?!? // */ -/* // */ -/* // */ -/* if you get a compile warning due to these symbols being defined more than // */ -/* once, move #include "stb_rect_pack.h" before #include "stb_truetype.h" // */ -/* // */ -/* ////////////////////////////////////////////////////////////////////////////////// */ - -typedef struct -{ - int width,height; - int x,y,bottom_y; -} stbrp_context; - -typedef struct -{ - unsigned char x; -} stbrp_node; - -struct stbrp_rect -{ - stbrp_coord x,y; - int id,w,h,was_packed; -}; - -static void stbrp_init_target(stbrp_context *con, int pw, int ph, stbrp_node *nodes, int num_nodes) -{ - con->width = pw; - con->height = ph; - con->x = 0; - con->y = 0; - con->bottom_y = 0; - STBTT__NOTUSED(nodes); - STBTT__NOTUSED(num_nodes); -} - -static void stbrp_pack_rects(stbrp_context *con, stbrp_rect *rects, int num_rects) -{ - int i; - for (i=0; i < num_rects; ++i) { - if (con->x + rects[i].w > con->width) { - con->x = 0; - con->y = con->bottom_y; - } - if (con->y + rects[i].h > con->height) - break; - rects[i].x = con->x; - rects[i].y = con->y; - rects[i].was_packed = 1; - con->x += rects[i].w; - if (con->y + rects[i].h > con->bottom_y) - con->bottom_y = con->y + rects[i].h; - } - for ( ; i < num_rects; ++i) - rects[i].was_packed = 0; -} -#endif - -/* //////////////////////////////////////////////////////////////////////////// */ -/* */ -/* bitmap baking */ -/* */ -/* This is SUPER-AWESOME (tm Ryan Gordon) packing using stb_rect_pack.h. If */ -/* stb_rect_pack.h isn't available, it uses the BakeFontBitmap strategy. */ - -STBTT_DEF int stbtt_PackBegin(stbtt_pack_context *spc, unsigned char *pixels, int pw, int ph, int stride_in_bytes, int padding, void *alloc_context) -{ - stbrp_context *context = (stbrp_context *) STBTT_malloc(sizeof(*context) ,alloc_context); - int num_nodes = pw - padding; - stbrp_node *nodes = (stbrp_node *) STBTT_malloc(sizeof(*nodes ) * num_nodes,alloc_context); - - if (context == NULL || nodes == NULL) { - if (context != NULL) STBTT_free(context, alloc_context); - if (nodes != NULL) STBTT_free(nodes , alloc_context); - return 0; - } - - spc->user_allocator_context = alloc_context; - spc->width = pw; - spc->height = ph; - spc->pixels = pixels; - spc->pack_info = context; - spc->nodes = nodes; - spc->padding = padding; - spc->stride_in_bytes = stride_in_bytes != 0 ? stride_in_bytes : pw; - spc->h_oversample = 1; - spc->v_oversample = 1; - spc->skip_missing = 0; - - stbrp_init_target(context, pw-padding, ph-padding, nodes, num_nodes); - - if (pixels) - STBTT_memset(pixels, 0, pw*ph); /* background of 0 around pixels */ - - return 1; -} - -STBTT_DEF void stbtt_PackEnd (stbtt_pack_context *spc) -{ - STBTT_free(spc->nodes , spc->user_allocator_context); - STBTT_free(spc->pack_info, spc->user_allocator_context); -} - -STBTT_DEF void stbtt_PackSetOversampling(stbtt_pack_context *spc, unsigned int h_oversample, unsigned int v_oversample) -{ - STBTT_assert(h_oversample <= STBTT_MAX_OVERSAMPLE); - STBTT_assert(v_oversample <= STBTT_MAX_OVERSAMPLE); - if (h_oversample <= STBTT_MAX_OVERSAMPLE) - spc->h_oversample = h_oversample; - if (v_oversample <= STBTT_MAX_OVERSAMPLE) - spc->v_oversample = v_oversample; -} - -STBTT_DEF void stbtt_PackSetSkipMissingCodepoints(stbtt_pack_context *spc, int skip) -{ - spc->skip_missing = skip; -} - -#define STBTT__OVER_MASK (STBTT_MAX_OVERSAMPLE-1) - -static void stbtt__h_prefilter(unsigned char *pixels, int w, int h, int stride_in_bytes, unsigned int kernel_width) -{ - unsigned char buffer[STBTT_MAX_OVERSAMPLE]; - int safe_w = w - kernel_width; - int j; - STBTT_memset(buffer, 0, STBTT_MAX_OVERSAMPLE); /* suppress bogus warning from VS2013 -analyze */ - for (j=0; j < h; ++j) { - int i; - unsigned int total; - STBTT_memset(buffer, 0, kernel_width); - - total = 0; - - /* make kernel_width a constant in common cases so compiler can optimize out the divide */ - switch (kernel_width) { - case 2: - for (i=0; i <= safe_w; ++i) { - total += pixels[i] - buffer[i & STBTT__OVER_MASK]; - buffer[(i+kernel_width) & STBTT__OVER_MASK] = pixels[i]; - pixels[i] = (unsigned char) (total / 2); - } - break; - case 3: - for (i=0; i <= safe_w; ++i) { - total += pixels[i] - buffer[i & STBTT__OVER_MASK]; - buffer[(i+kernel_width) & STBTT__OVER_MASK] = pixels[i]; - pixels[i] = (unsigned char) (total / 3); - } - break; - case 4: - for (i=0; i <= safe_w; ++i) { - total += pixels[i] - buffer[i & STBTT__OVER_MASK]; - buffer[(i+kernel_width) & STBTT__OVER_MASK] = pixels[i]; - pixels[i] = (unsigned char) (total / 4); - } - break; - case 5: - for (i=0; i <= safe_w; ++i) { - total += pixels[i] - buffer[i & STBTT__OVER_MASK]; - buffer[(i+kernel_width) & STBTT__OVER_MASK] = pixels[i]; - pixels[i] = (unsigned char) (total / 5); - } - break; - default: - for (i=0; i <= safe_w; ++i) { - total += pixels[i] - buffer[i & STBTT__OVER_MASK]; - buffer[(i+kernel_width) & STBTT__OVER_MASK] = pixels[i]; - pixels[i] = (unsigned char) (total / kernel_width); - } - break; - } - - for (; i < w; ++i) { - STBTT_assert(pixels[i] == 0); - total -= buffer[i & STBTT__OVER_MASK]; - pixels[i] = (unsigned char) (total / kernel_width); - } - - pixels += stride_in_bytes; - } -} - -static void stbtt__v_prefilter(unsigned char *pixels, int w, int h, int stride_in_bytes, unsigned int kernel_width) -{ - unsigned char buffer[STBTT_MAX_OVERSAMPLE]; - int safe_h = h - kernel_width; - int j; - STBTT_memset(buffer, 0, STBTT_MAX_OVERSAMPLE); /* suppress bogus warning from VS2013 -analyze */ - for (j=0; j < w; ++j) { - int i; - unsigned int total; - STBTT_memset(buffer, 0, kernel_width); - - total = 0; - - /* make kernel_width a constant in common cases so compiler can optimize out the divide */ - switch (kernel_width) { - case 2: - for (i=0; i <= safe_h; ++i) { - total += pixels[i*stride_in_bytes] - buffer[i & STBTT__OVER_MASK]; - buffer[(i+kernel_width) & STBTT__OVER_MASK] = pixels[i*stride_in_bytes]; - pixels[i*stride_in_bytes] = (unsigned char) (total / 2); - } - break; - case 3: - for (i=0; i <= safe_h; ++i) { - total += pixels[i*stride_in_bytes] - buffer[i & STBTT__OVER_MASK]; - buffer[(i+kernel_width) & STBTT__OVER_MASK] = pixels[i*stride_in_bytes]; - pixels[i*stride_in_bytes] = (unsigned char) (total / 3); - } - break; - case 4: - for (i=0; i <= safe_h; ++i) { - total += pixels[i*stride_in_bytes] - buffer[i & STBTT__OVER_MASK]; - buffer[(i+kernel_width) & STBTT__OVER_MASK] = pixels[i*stride_in_bytes]; - pixels[i*stride_in_bytes] = (unsigned char) (total / 4); - } - break; - case 5: - for (i=0; i <= safe_h; ++i) { - total += pixels[i*stride_in_bytes] - buffer[i & STBTT__OVER_MASK]; - buffer[(i+kernel_width) & STBTT__OVER_MASK] = pixels[i*stride_in_bytes]; - pixels[i*stride_in_bytes] = (unsigned char) (total / 5); - } - break; - default: - for (i=0; i <= safe_h; ++i) { - total += pixels[i*stride_in_bytes] - buffer[i & STBTT__OVER_MASK]; - buffer[(i+kernel_width) & STBTT__OVER_MASK] = pixels[i*stride_in_bytes]; - pixels[i*stride_in_bytes] = (unsigned char) (total / kernel_width); - } - break; - } - - for (; i < h; ++i) { - STBTT_assert(pixels[i*stride_in_bytes] == 0); - total -= buffer[i & STBTT__OVER_MASK]; - pixels[i*stride_in_bytes] = (unsigned char) (total / kernel_width); - } - - pixels += 1; - } -} - -static float stbtt__oversample_shift(int oversample) -{ - if (!oversample) - return 0.0f; - - /* The prefilter is a box filter of width "oversample", */ - /* which shifts phase by (oversample - 1)/2 pixels in */ - /* oversampled space. We want to shift in the opposite */ - /* direction to counter this. */ - return (float)-(oversample - 1) / (2.0f * (float)oversample); -} - -/* rects array must be big enough to accommodate all characters in the given ranges */ -STBTT_DEF int stbtt_PackFontRangesGatherRects(stbtt_pack_context *spc, const stbtt_fontinfo *info, stbtt_pack_range *ranges, int num_ranges, stbrp_rect *rects) -{ - int i,j,k; - int missing_glyph_added = 0; - - k=0; - for (i=0; i < num_ranges; ++i) { - float fh = ranges[i].font_size; - float scale = fh > 0 ? stbtt_ScaleForPixelHeight(info, fh) : stbtt_ScaleForMappingEmToPixels(info, -fh); - ranges[i].h_oversample = (unsigned char) spc->h_oversample; - ranges[i].v_oversample = (unsigned char) spc->v_oversample; - for (j=0; j < ranges[i].num_chars; ++j) { - int x0,y0,x1,y1; - int codepoint = ranges[i].array_of_unicode_codepoints == NULL ? ranges[i].first_unicode_codepoint_in_range + j : ranges[i].array_of_unicode_codepoints[j]; - int glyph = stbtt_FindGlyphIndex(info, codepoint); - if (glyph == 0 && (spc->skip_missing || missing_glyph_added)) { - rects[k].w = rects[k].h = 0; - } else { - stbtt_GetGlyphBitmapBoxSubpixel(info,glyph, - scale * spc->h_oversample, - scale * spc->v_oversample, - 0,0, - &x0,&y0,&x1,&y1); - rects[k].w = (stbrp_coord) (x1-x0 + spc->padding + spc->h_oversample-1); - rects[k].h = (stbrp_coord) (y1-y0 + spc->padding + spc->v_oversample-1); - if (glyph == 0) - missing_glyph_added = 1; - } - ++k; - } - } - - return k; -} - -STBTT_DEF void stbtt_MakeGlyphBitmapSubpixelPrefilter(const stbtt_fontinfo *info, unsigned char *output, int out_w, int out_h, int out_stride, float scale_x, float scale_y, float shift_x, float shift_y, int prefilter_x, int prefilter_y, float *sub_x, float *sub_y, int glyph) -{ - stbtt_MakeGlyphBitmapSubpixel(info, - output, - out_w - (prefilter_x - 1), - out_h - (prefilter_y - 1), - out_stride, - scale_x, - scale_y, - shift_x, - shift_y, - glyph); - - if (prefilter_x > 1) - stbtt__h_prefilter(output, out_w, out_h, out_stride, prefilter_x); - - if (prefilter_y > 1) - stbtt__v_prefilter(output, out_w, out_h, out_stride, prefilter_y); - - *sub_x = stbtt__oversample_shift(prefilter_x); - *sub_y = stbtt__oversample_shift(prefilter_y); -} - -/* rects array must be big enough to accommodate all characters in the given ranges */ -STBTT_DEF int stbtt_PackFontRangesRenderIntoRects(stbtt_pack_context *spc, const stbtt_fontinfo *info, stbtt_pack_range *ranges, int num_ranges, stbrp_rect *rects) -{ - int i,j,k, missing_glyph = -1, return_value = 1; - - /* save current values */ - int old_h_over = spc->h_oversample; - int old_v_over = spc->v_oversample; - - k = 0; - for (i=0; i < num_ranges; ++i) { - float fh = ranges[i].font_size; - float scale = fh > 0 ? stbtt_ScaleForPixelHeight(info, fh) : stbtt_ScaleForMappingEmToPixels(info, -fh); - float recip_h,recip_v,sub_x,sub_y; - spc->h_oversample = ranges[i].h_oversample; - spc->v_oversample = ranges[i].v_oversample; - recip_h = 1.0f / spc->h_oversample; - recip_v = 1.0f / spc->v_oversample; - sub_x = stbtt__oversample_shift(spc->h_oversample); - sub_y = stbtt__oversample_shift(spc->v_oversample); - for (j=0; j < ranges[i].num_chars; ++j) { - stbrp_rect *r = &rects[k]; - if (r->was_packed && r->w != 0 && r->h != 0) { - stbtt_packedchar *bc = &ranges[i].chardata_for_range[j]; - int advance, lsb, x0,y0,x1,y1; - int codepoint = ranges[i].array_of_unicode_codepoints == NULL ? ranges[i].first_unicode_codepoint_in_range + j : ranges[i].array_of_unicode_codepoints[j]; - int glyph = stbtt_FindGlyphIndex(info, codepoint); - stbrp_coord pad = (stbrp_coord) spc->padding; - - /* pad on left and top */ - r->x += pad; - r->y += pad; - r->w -= pad; - r->h -= pad; - stbtt_GetGlyphHMetrics(info, glyph, &advance, &lsb); - stbtt_GetGlyphBitmapBox(info, glyph, - scale * spc->h_oversample, - scale * spc->v_oversample, - &x0,&y0,&x1,&y1); - stbtt_MakeGlyphBitmapSubpixel(info, - spc->pixels + r->x + r->y*spc->stride_in_bytes, - r->w - spc->h_oversample+1, - r->h - spc->v_oversample+1, - spc->stride_in_bytes, - scale * spc->h_oversample, - scale * spc->v_oversample, - 0,0, - glyph); - - if (spc->h_oversample > 1) - stbtt__h_prefilter(spc->pixels + r->x + r->y*spc->stride_in_bytes, - r->w, r->h, spc->stride_in_bytes, - spc->h_oversample); - - if (spc->v_oversample > 1) - stbtt__v_prefilter(spc->pixels + r->x + r->y*spc->stride_in_bytes, - r->w, r->h, spc->stride_in_bytes, - spc->v_oversample); - - bc->x0 = (stbtt_int16) r->x; - bc->y0 = (stbtt_int16) r->y; - bc->x1 = (stbtt_int16) (r->x + r->w); - bc->y1 = (stbtt_int16) (r->y + r->h); - bc->xadvance = scale * advance; - bc->xoff = (float) x0 * recip_h + sub_x; - bc->yoff = (float) y0 * recip_v + sub_y; - bc->xoff2 = (x0 + r->w) * recip_h + sub_x; - bc->yoff2 = (y0 + r->h) * recip_v + sub_y; - - if (glyph == 0) - missing_glyph = j; - } else if (spc->skip_missing) { - return_value = 0; - } else if (r->was_packed && r->w == 0 && r->h == 0 && missing_glyph >= 0) { - ranges[i].chardata_for_range[j] = ranges[i].chardata_for_range[missing_glyph]; - } else { - return_value = 0; /* if any fail, report failure */ - } - - ++k; - } - } - - /* restore original values */ - spc->h_oversample = old_h_over; - spc->v_oversample = old_v_over; - - return return_value; -} - -STBTT_DEF void stbtt_PackFontRangesPackRects(stbtt_pack_context *spc, stbrp_rect *rects, int num_rects) -{ - stbrp_pack_rects((stbrp_context *) spc->pack_info, rects, num_rects); -} - -STBTT_DEF int stbtt_PackFontRanges(stbtt_pack_context *spc, const unsigned char *fontdata, int font_index, stbtt_pack_range *ranges, int num_ranges) -{ - stbtt_fontinfo info; - int i,j,n, return_value = 1; - /* stbrp_context *context = (stbrp_context *) spc->pack_info; */ - stbrp_rect *rects; - - /* flag all characters as NOT packed */ - for (i=0; i < num_ranges; ++i) - for (j=0; j < ranges[i].num_chars; ++j) - ranges[i].chardata_for_range[j].x0 = - ranges[i].chardata_for_range[j].y0 = - ranges[i].chardata_for_range[j].x1 = - ranges[i].chardata_for_range[j].y1 = 0; - - n = 0; - for (i=0; i < num_ranges; ++i) - n += ranges[i].num_chars; - - rects = (stbrp_rect *) STBTT_malloc(sizeof(*rects) * n, spc->user_allocator_context); - if (rects == NULL) - return 0; - - info.userdata = spc->user_allocator_context; - stbtt_InitFont(&info, fontdata, stbtt_GetFontOffsetForIndex(fontdata,font_index)); - - n = stbtt_PackFontRangesGatherRects(spc, &info, ranges, num_ranges, rects); - - stbtt_PackFontRangesPackRects(spc, rects, n); - - return_value = stbtt_PackFontRangesRenderIntoRects(spc, &info, ranges, num_ranges, rects); - - STBTT_free(rects, spc->user_allocator_context); - return return_value; -} - -STBTT_DEF int stbtt_PackFontRange(stbtt_pack_context *spc, const unsigned char *fontdata, int font_index, float font_size, - int first_unicode_codepoint_in_range, int num_chars_in_range, stbtt_packedchar *chardata_for_range) -{ - stbtt_pack_range range; - range.first_unicode_codepoint_in_range = first_unicode_codepoint_in_range; - range.array_of_unicode_codepoints = NULL; - range.num_chars = num_chars_in_range; - range.chardata_for_range = chardata_for_range; - range.font_size = font_size; - return stbtt_PackFontRanges(spc, fontdata, font_index, &range, 1); -} - -STBTT_DEF void stbtt_GetScaledFontVMetrics(const unsigned char *fontdata, int index, float size, float *ascent, float *descent, float *lineGap) -{ - int i_ascent, i_descent, i_lineGap; - float scale; - stbtt_fontinfo info; - stbtt_InitFont(&info, fontdata, stbtt_GetFontOffsetForIndex(fontdata, index)); - scale = size > 0 ? stbtt_ScaleForPixelHeight(&info, size) : stbtt_ScaleForMappingEmToPixels(&info, -size); - stbtt_GetFontVMetrics(&info, &i_ascent, &i_descent, &i_lineGap); - *ascent = (float) i_ascent * scale; - *descent = (float) i_descent * scale; - *lineGap = (float) i_lineGap * scale; -} - -STBTT_DEF void stbtt_GetPackedQuad(const stbtt_packedchar *chardata, int pw, int ph, int char_index, float *xpos, float *ypos, stbtt_aligned_quad *q, int align_to_integer) -{ - float ipw = 1.0f / pw, iph = 1.0f / ph; - const stbtt_packedchar *b = chardata + char_index; - - if (align_to_integer) { - float x = (float) STBTT_ifloor((*xpos + b->xoff) + 0.5f); - float y = (float) STBTT_ifloor((*ypos + b->yoff) + 0.5f); - q->x0 = x; - q->y0 = y; - q->x1 = x + b->xoff2 - b->xoff; - q->y1 = y + b->yoff2 - b->yoff; - } else { - q->x0 = *xpos + b->xoff; - q->y0 = *ypos + b->yoff; - q->x1 = *xpos + b->xoff2; - q->y1 = *ypos + b->yoff2; - } - - q->s0 = b->x0 * ipw; - q->t0 = b->y0 * iph; - q->s1 = b->x1 * ipw; - q->t1 = b->y1 * iph; - - *xpos += b->xadvance; -} - -/* //////////////////////////////////////////////////////////////////////////// */ -/* */ -/* sdf computation */ -/* */ - -#define STBTT_min(a,b) ((a) < (b) ? (a) : (b)) -#define STBTT_max(a,b) ((a) < (b) ? (b) : (a)) - -static int stbtt__ray_intersect_bezier(float orig[2], float ray[2], float q0[2], float q1[2], float q2[2], float hits[2][2]) -{ - float q0perp = q0[1]*ray[0] - q0[0]*ray[1]; - float q1perp = q1[1]*ray[0] - q1[0]*ray[1]; - float q2perp = q2[1]*ray[0] - q2[0]*ray[1]; - float roperp = orig[1]*ray[0] - orig[0]*ray[1]; - - float a = q0perp - 2*q1perp + q2perp; - float b = q1perp - q0perp; - float c = q0perp - roperp; - - float s0 = 0., s1 = 0.; - int num_s = 0; - - if (a != 0.0) { - float discr = b*b - a*c; - if (discr > 0.0) { - float rcpna = -1 / a; - float d = (float) STBTT_sqrt(discr); - s0 = (b+d) * rcpna; - s1 = (b-d) * rcpna; - if (s0 >= 0.0 && s0 <= 1.0) - num_s = 1; - if (d > 0.0 && s1 >= 0.0 && s1 <= 1.0) { - if (num_s == 0) s0 = s1; - ++num_s; - } - } - } else { - /* 2*b*s + c = 0 */ - /* s = -c / (2*b) */ - s0 = c / (-2 * b); - if (s0 >= 0.0 && s0 <= 1.0) - num_s = 1; - } - - if (num_s == 0) - return 0; - else { - float rcp_len2 = 1 / (ray[0]*ray[0] + ray[1]*ray[1]); - float rayn_x = ray[0] * rcp_len2, rayn_y = ray[1] * rcp_len2; - - float q0d = q0[0]*rayn_x + q0[1]*rayn_y; - float q1d = q1[0]*rayn_x + q1[1]*rayn_y; - float q2d = q2[0]*rayn_x + q2[1]*rayn_y; - float rod = orig[0]*rayn_x + orig[1]*rayn_y; - - float q10d = q1d - q0d; - float q20d = q2d - q0d; - float q0rd = q0d - rod; - - hits[0][0] = q0rd + s0*(2.0f - 2.0f*s0)*q10d + s0*s0*q20d; - hits[0][1] = a*s0+b; - - if (num_s > 1) { - hits[1][0] = q0rd + s1*(2.0f - 2.0f*s1)*q10d + s1*s1*q20d; - hits[1][1] = a*s1+b; - return 2; - } else { - return 1; - } - } -} - -static int equal(float *a, float *b) -{ - return (a[0] == b[0] && a[1] == b[1]); -} - -static int stbtt__compute_crossings_x(float x, float y, int nverts, stbtt_vertex *verts) -{ - int i; - float orig[2], ray[2] = { 1, 0 }; - float y_frac; - int winding = 0; - - /* make sure y never passes through a vertex of the shape */ - y_frac = (float) STBTT_fmod(y, 1.0f); - if (y_frac < 0.01f) - y += 0.01f; - else if (y_frac > 0.99f) - y -= 0.01f; - - orig[0] = x; - orig[1] = y; - - /* test a ray from (-infinity,y) to (x,y) */ - for (i=0; i < nverts; ++i) { - if (verts[i].type == STBTT_vline) { - int x0 = (int) verts[i-1].x, y0 = (int) verts[i-1].y; - int x1 = (int) verts[i ].x, y1 = (int) verts[i ].y; - if (y > STBTT_min(y0,y1) && y < STBTT_max(y0,y1) && x > STBTT_min(x0,x1)) { - float x_inter = (y - y0) / (y1 - y0) * (x1-x0) + x0; - if (x_inter < x) - winding += (y0 < y1) ? 1 : -1; - } - } - if (verts[i].type == STBTT_vcurve) { - int x0 = (int) verts[i-1].x , y0 = (int) verts[i-1].y ; - int x1 = (int) verts[i ].cx, y1 = (int) verts[i ].cy; - int x2 = (int) verts[i ].x , y2 = (int) verts[i ].y ; - int ax = STBTT_min(x0,STBTT_min(x1,x2)), ay = STBTT_min(y0,STBTT_min(y1,y2)); - int by = STBTT_max(y0,STBTT_max(y1,y2)); - if (y > ay && y < by && x > ax) { - float q0[2],q1[2],q2[2]; - float hits[2][2]; - q0[0] = (float)x0; - q0[1] = (float)y0; - q1[0] = (float)x1; - q1[1] = (float)y1; - q2[0] = (float)x2; - q2[1] = (float)y2; - if (equal(q0,q1) || equal(q1,q2)) { - x0 = (int)verts[i-1].x; - y0 = (int)verts[i-1].y; - x1 = (int)verts[i ].x; - y1 = (int)verts[i ].y; - if (y > STBTT_min(y0,y1) && y < STBTT_max(y0,y1) && x > STBTT_min(x0,x1)) { - float x_inter = (y - y0) / (y1 - y0) * (x1-x0) + x0; - if (x_inter < x) - winding += (y0 < y1) ? 1 : -1; - } - } else { - int num_hits = stbtt__ray_intersect_bezier(orig, ray, q0, q1, q2, hits); - if (num_hits >= 1) - if (hits[0][0] < 0) - winding += (hits[0][1] < 0 ? -1 : 1); - if (num_hits >= 2) - if (hits[1][0] < 0) - winding += (hits[1][1] < 0 ? -1 : 1); - } - } - } - } - return winding; -} - -static float stbtt__cuberoot( float x ) -{ - if (x<0) - return -(float) STBTT_pow(-x,1.0f/3.0f); - else - return (float) STBTT_pow( x,1.0f/3.0f); -} - -/* x^3 + a*x^2 + b*x + c = 0 */ -static int stbtt__solve_cubic(float a, float b, float c, float* r) -{ - float s = -a / 3; - float p = b - a*a / 3; - float q = a * (2*a*a - 9*b) / 27 + c; - float p3 = p*p*p; - float d = q*q + 4*p3 / 27; - if (d >= 0) { - float z = (float) STBTT_sqrt(d); - float u = (-q + z) / 2; - float v = (-q - z) / 2; - u = stbtt__cuberoot(u); - v = stbtt__cuberoot(v); - r[0] = s + u + v; - return 1; - } else { - float u = (float) STBTT_sqrt(-p/3); - float v = (float) STBTT_acos(-STBTT_sqrt(-27/p3) * q / 2) / 3; /* p3 must be negative, since d is negative */ - float m = (float) STBTT_cos(v); - float n = (float) STBTT_cos(v-3.141592/2)*1.732050808f; - r[0] = s + u * 2 * m; - r[1] = s - u * (m + n); - r[2] = s - u * (m - n); - - /* STBTT_assert( STBTT_fabs(((r[0]+a)*r[0]+b)*r[0]+c) < 0.05f); // these asserts may not be safe at all scales, though they're in bezier t parameter units so maybe? */ - /* STBTT_assert( STBTT_fabs(((r[1]+a)*r[1]+b)*r[1]+c) < 0.05f); */ - /* STBTT_assert( STBTT_fabs(((r[2]+a)*r[2]+b)*r[2]+c) < 0.05f); */ - return 3; - } -} - -STBTT_DEF unsigned char * stbtt_GetGlyphSDF(const stbtt_fontinfo *info, float scale, int glyph, int padding, unsigned char onedge_value, float pixel_dist_scale, int *width, int *height, int *xoff, int *yoff) -{ - float scale_x = scale, scale_y = scale; - int ix0,iy0,ix1,iy1; - int w,h; - unsigned char *data; - - if (scale == 0) return NULL; - - stbtt_GetGlyphBitmapBoxSubpixel(info, glyph, scale, scale, 0.0f,0.0f, &ix0,&iy0,&ix1,&iy1); - - /* if empty, return NULL */ - if (ix0 == ix1 || iy0 == iy1) - return NULL; - - ix0 -= padding; - iy0 -= padding; - ix1 += padding; - iy1 += padding; - - w = (ix1 - ix0); - h = (iy1 - iy0); - - if (width ) *width = w; - if (height) *height = h; - if (xoff ) *xoff = ix0; - if (yoff ) *yoff = iy0; - - /* invert for y-downwards bitmaps */ - scale_y = -scale_y; - - { - int x,y,i,j; - float *precompute; - stbtt_vertex *verts; - int num_verts = stbtt_GetGlyphShape(info, glyph, &verts); - data = (unsigned char *) STBTT_malloc(w * h, info->userdata); - precompute = (float *) STBTT_malloc(num_verts * sizeof(float), info->userdata); - - for (i=0,j=num_verts-1; i < num_verts; j=i++) { - if (verts[i].type == STBTT_vline) { - float x0 = verts[i].x*scale_x, y0 = verts[i].y*scale_y; - float x1 = verts[j].x*scale_x, y1 = verts[j].y*scale_y; - float dist = (float) STBTT_sqrt((x1-x0)*(x1-x0) + (y1-y0)*(y1-y0)); - precompute[i] = (dist == 0) ? 0.0f : 1.0f / dist; - } else if (verts[i].type == STBTT_vcurve) { - float x2 = verts[j].x *scale_x, y2 = verts[j].y *scale_y; - float x1 = verts[i].cx*scale_x, y1 = verts[i].cy*scale_y; - float x0 = verts[i].x *scale_x, y0 = verts[i].y *scale_y; - float bx = x0 - 2*x1 + x2, by = y0 - 2*y1 + y2; - float len2 = bx*bx + by*by; - if (len2 != 0.0f) - precompute[i] = 1.0f / (bx*bx + by*by); - else - precompute[i] = 0.0f; - } else - precompute[i] = 0.0f; - } - - for (y=iy0; y < iy1; ++y) { - for (x=ix0; x < ix1; ++x) { - float val; - float min_dist = 999999.0f; - float sx = (float) x + 0.5f; - float sy = (float) y + 0.5f; - float x_gspace = (sx / scale_x); - float y_gspace = (sy / scale_y); - - int winding = stbtt__compute_crossings_x(x_gspace, y_gspace, num_verts, verts); /* @OPTIMIZE: this could just be a rasterization, but needs to be line vs. non-tesselated curves so a new path */ - - for (i=0; i < num_verts; ++i) { - float x0 = verts[i].x*scale_x, y0 = verts[i].y*scale_y; - - if (verts[i].type == STBTT_vline && precompute[i] != 0.0f) { - float x1 = verts[i-1].x*scale_x, y1 = verts[i-1].y*scale_y; - - float dist,dist2 = (x0-sx)*(x0-sx) + (y0-sy)*(y0-sy); - if (dist2 < min_dist*min_dist) - min_dist = (float) STBTT_sqrt(dist2); - - /* coarse culling against bbox */ - /* if (sx > STBTT_min(x0,x1)-min_dist && sx < STBTT_max(x0,x1)+min_dist && */ - /* sy > STBTT_min(y0,y1)-min_dist && sy < STBTT_max(y0,y1)+min_dist) */ - dist = (float) STBTT_fabs((x1-x0)*(y0-sy) - (y1-y0)*(x0-sx)) * precompute[i]; - STBTT_assert(i != 0); - if (dist < min_dist) { - /* check position along line */ - /* x' = x0 + t*(x1-x0), y' = y0 + t*(y1-y0) */ - /* minimize (x'-sx)*(x'-sx)+(y'-sy)*(y'-sy) */ - float dx = x1-x0, dy = y1-y0; - float px = x0-sx, py = y0-sy; - /* minimize (px+t*dx)^2 + (py+t*dy)^2 = px*px + 2*px*dx*t + t^2*dx*dx + py*py + 2*py*dy*t + t^2*dy*dy */ - /* derivative: 2*px*dx + 2*py*dy + (2*dx*dx+2*dy*dy)*t, set to 0 and solve */ - float t = -(px*dx + py*dy) / (dx*dx + dy*dy); - if (t >= 0.0f && t <= 1.0f) - min_dist = dist; - } - } else if (verts[i].type == STBTT_vcurve) { - float x2 = verts[i-1].x *scale_x, y2 = verts[i-1].y *scale_y; - float x1 = verts[i ].cx*scale_x, y1 = verts[i ].cy*scale_y; - float box_x0 = STBTT_min(STBTT_min(x0,x1),x2); - float box_y0 = STBTT_min(STBTT_min(y0,y1),y2); - float box_x1 = STBTT_max(STBTT_max(x0,x1),x2); - float box_y1 = STBTT_max(STBTT_max(y0,y1),y2); - /* coarse culling against bbox to avoid computing cubic unnecessarily */ - if (sx > box_x0-min_dist && sx < box_x1+min_dist && sy > box_y0-min_dist && sy < box_y1+min_dist) { - int num=0; - float ax = x1-x0, ay = y1-y0; - float bx = x0 - 2*x1 + x2, by = y0 - 2*y1 + y2; - float mx = x0 - sx, my = y0 - sy; - float res[3] = {0.f,0.f,0.f}; - float px,py,t,it,dist2; - float a_inv = precompute[i]; - if (a_inv == 0.0) { /* if a_inv is 0, it's 2nd degree so use quadratic formula */ - float a = 3*(ax*bx + ay*by); - float b = 2*(ax*ax + ay*ay) + (mx*bx+my*by); - float c = mx*ax+my*ay; - if (a == 0.0) { /* if a is 0, it's linear */ - if (b != 0.0) { - res[num++] = -c/b; - } - } else { - float discriminant = b*b - 4*a*c; - if (discriminant < 0) - num = 0; - else { - float root = (float) STBTT_sqrt(discriminant); - res[0] = (-b - root)/(2*a); - res[1] = (-b + root)/(2*a); - num = 2; /* don't bother distinguishing 1-solution case, as code below will still work */ - } - } - } else { - float b = 3*(ax*bx + ay*by) * a_inv; /* could precompute this as it doesn't depend on sample point */ - float c = (2*(ax*ax + ay*ay) + (mx*bx+my*by)) * a_inv; - float d = (mx*ax+my*ay) * a_inv; - num = stbtt__solve_cubic(b, c, d, res); - } - dist2 = (x0-sx)*(x0-sx) + (y0-sy)*(y0-sy); - if (dist2 < min_dist*min_dist) - min_dist = (float) STBTT_sqrt(dist2); - - if (num >= 1 && res[0] >= 0.0f && res[0] <= 1.0f) { - t = res[0], it = 1.0f - t; - px = it*it*x0 + 2*t*it*x1 + t*t*x2; - py = it*it*y0 + 2*t*it*y1 + t*t*y2; - dist2 = (px-sx)*(px-sx) + (py-sy)*(py-sy); - if (dist2 < min_dist * min_dist) - min_dist = (float) STBTT_sqrt(dist2); - } - if (num >= 2 && res[1] >= 0.0f && res[1] <= 1.0f) { - t = res[1], it = 1.0f - t; - px = it*it*x0 + 2*t*it*x1 + t*t*x2; - py = it*it*y0 + 2*t*it*y1 + t*t*y2; - dist2 = (px-sx)*(px-sx) + (py-sy)*(py-sy); - if (dist2 < min_dist * min_dist) - min_dist = (float) STBTT_sqrt(dist2); - } - if (num >= 3 && res[2] >= 0.0f && res[2] <= 1.0f) { - t = res[2], it = 1.0f - t; - px = it*it*x0 + 2*t*it*x1 + t*t*x2; - py = it*it*y0 + 2*t*it*y1 + t*t*y2; - dist2 = (px-sx)*(px-sx) + (py-sy)*(py-sy); - if (dist2 < min_dist * min_dist) - min_dist = (float) STBTT_sqrt(dist2); - } - } - } - } - if (winding == 0) - min_dist = -min_dist; /* if outside the shape, value is negative */ - val = onedge_value + pixel_dist_scale * min_dist; - if (val < 0) - val = 0; - else if (val > 255) - val = 255; - data[(y-iy0)*w+(x-ix0)] = (unsigned char) val; - } - } - STBTT_free(precompute, info->userdata); - STBTT_free(verts, info->userdata); - } - return data; -} - -STBTT_DEF unsigned char * stbtt_GetCodepointSDF(const stbtt_fontinfo *info, float scale, int codepoint, int padding, unsigned char onedge_value, float pixel_dist_scale, int *width, int *height, int *xoff, int *yoff) -{ - return stbtt_GetGlyphSDF(info, scale, stbtt_FindGlyphIndex(info, codepoint), padding, onedge_value, pixel_dist_scale, width, height, xoff, yoff); -} - -STBTT_DEF void stbtt_FreeSDF(unsigned char *bitmap, void *userdata) -{ - STBTT_free(bitmap, userdata); -} - -/* //////////////////////////////////////////////////////////////////////////// */ -/* */ -/* font name matching -- recommended not to use this */ -/* */ - -/* check if a utf8 string contains a prefix which is the utf16 string; if so return length of matching utf8 string */ -static stbtt_int32 stbtt__CompareUTF8toUTF16_bigendian_prefix(stbtt_uint8 *s1, stbtt_int32 len1, stbtt_uint8 *s2, stbtt_int32 len2) -{ - stbtt_int32 i=0; - - /* convert utf16 to utf8 and compare the results while converting */ - while (len2) { - stbtt_uint16 ch = s2[0]*256 + s2[1]; - if (ch < 0x80) { - if (i >= len1) return -1; - if (s1[i++] != ch) return -1; - } else if (ch < 0x800) { - if (i+1 >= len1) return -1; - if (s1[i++] != 0xc0 + (ch >> 6)) return -1; - if (s1[i++] != 0x80 + (ch & 0x3f)) return -1; - } else if (ch >= 0xd800 && ch < 0xdc00) { - stbtt_uint32 c; - stbtt_uint16 ch2 = s2[2]*256 + s2[3]; - if (i+3 >= len1) return -1; - c = ((ch - 0xd800) << 10) + (ch2 - 0xdc00) + 0x10000; - if (s1[i++] != 0xf0 + (c >> 18)) return -1; - if (s1[i++] != 0x80 + ((c >> 12) & 0x3f)) return -1; - if (s1[i++] != 0x80 + ((c >> 6) & 0x3f)) return -1; - if (s1[i++] != 0x80 + ((c ) & 0x3f)) return -1; - s2 += 2; /* plus another 2 below */ - len2 -= 2; - } else if (ch >= 0xdc00 && ch < 0xe000) { - return -1; - } else { - if (i+2 >= len1) return -1; - if (s1[i++] != 0xe0 + (ch >> 12)) return -1; - if (s1[i++] != 0x80 + ((ch >> 6) & 0x3f)) return -1; - if (s1[i++] != 0x80 + ((ch ) & 0x3f)) return -1; - } - s2 += 2; - len2 -= 2; - } - return i; -} - -static int stbtt_CompareUTF8toUTF16_bigendian_internal(char *s1, int len1, char *s2, int len2) -{ - return len1 == stbtt__CompareUTF8toUTF16_bigendian_prefix((stbtt_uint8*) s1, len1, (stbtt_uint8*) s2, len2); -} - -/* returns results in whatever encoding you request... but note that 2-byte encodings */ -/* will be BIG-ENDIAN... use stbtt_CompareUTF8toUTF16_bigendian() to compare */ -STBTT_DEF const char *stbtt_GetFontNameString(const stbtt_fontinfo *font, int *length, int platformID, int encodingID, int languageID, int nameID) -{ - stbtt_int32 i,count,stringOffset; - stbtt_uint8 *fc = font->data; - stbtt_uint32 offset = font->fontstart; - stbtt_uint32 nm = stbtt__find_table(fc, offset, "name"); - if (!nm) return NULL; - - count = ttUSHORT(fc+nm+2); - stringOffset = nm + ttUSHORT(fc+nm+4); - for (i=0; i < count; ++i) { - stbtt_uint32 loc = nm + 6 + 12 * i; - if (platformID == ttUSHORT(fc+loc+0) && encodingID == ttUSHORT(fc+loc+2) - && languageID == ttUSHORT(fc+loc+4) && nameID == ttUSHORT(fc+loc+6)) { - *length = ttUSHORT(fc+loc+8); - return (const char *) (fc+stringOffset+ttUSHORT(fc+loc+10)); - } - } - return NULL; -} - -static int stbtt__matchpair(stbtt_uint8 *fc, stbtt_uint32 nm, stbtt_uint8 *name, stbtt_int32 nlen, stbtt_int32 target_id, stbtt_int32 next_id) -{ - stbtt_int32 i; - stbtt_int32 count = ttUSHORT(fc+nm+2); - stbtt_int32 stringOffset = nm + ttUSHORT(fc+nm+4); - - for (i=0; i < count; ++i) { - stbtt_uint32 loc = nm + 6 + 12 * i; - stbtt_int32 id = ttUSHORT(fc+loc+6); - if (id == target_id) { - /* find the encoding */ - stbtt_int32 platform = ttUSHORT(fc+loc+0), encoding = ttUSHORT(fc+loc+2), language = ttUSHORT(fc+loc+4); - - /* is this a Unicode encoding? */ - if (platform == 0 || (platform == 3 && encoding == 1) || (platform == 3 && encoding == 10)) { - stbtt_int32 slen = ttUSHORT(fc+loc+8); - stbtt_int32 off = ttUSHORT(fc+loc+10); - - /* check if there's a prefix match */ - stbtt_int32 matchlen = stbtt__CompareUTF8toUTF16_bigendian_prefix(name, nlen, fc+stringOffset+off,slen); - if (matchlen >= 0) { - /* check for target_id+1 immediately following, with same encoding & language */ - if (i+1 < count && ttUSHORT(fc+loc+12+6) == next_id && ttUSHORT(fc+loc+12) == platform && ttUSHORT(fc+loc+12+2) == encoding && ttUSHORT(fc+loc+12+4) == language) { - slen = ttUSHORT(fc+loc+12+8); - off = ttUSHORT(fc+loc+12+10); - if (slen == 0) { - if (matchlen == nlen) - return 1; - } else if (matchlen < nlen && name[matchlen] == ' ') { - ++matchlen; - if (stbtt_CompareUTF8toUTF16_bigendian_internal((char*) (name+matchlen), nlen-matchlen, (char*)(fc+stringOffset+off),slen)) - return 1; - } - } else { - /* if nothing immediately following */ - if (matchlen == nlen) - return 1; - } - } - } - - /* @TODO handle other encodings */ - } - } - return 0; -} - -static int stbtt__matches(stbtt_uint8 *fc, stbtt_uint32 offset, stbtt_uint8 *name, stbtt_int32 flags) -{ - stbtt_int32 nlen = (stbtt_int32) STBTT_strlen((char *) name); - stbtt_uint32 nm,hd; - if (!stbtt__isfont(fc+offset)) return 0; - - /* check italics/bold/underline flags in macStyle... */ - if (flags) { - hd = stbtt__find_table(fc, offset, "head"); - if ((ttUSHORT(fc+hd+44) & 7) != (flags & 7)) return 0; - } - - nm = stbtt__find_table(fc, offset, "name"); - if (!nm) return 0; - - if (flags) { - /* if we checked the macStyle flags, then just check the family and ignore the subfamily */ - if (stbtt__matchpair(fc, nm, name, nlen, 16, -1)) return 1; - if (stbtt__matchpair(fc, nm, name, nlen, 1, -1)) return 1; - if (stbtt__matchpair(fc, nm, name, nlen, 3, -1)) return 1; - } else { - if (stbtt__matchpair(fc, nm, name, nlen, 16, 17)) return 1; - if (stbtt__matchpair(fc, nm, name, nlen, 1, 2)) return 1; - if (stbtt__matchpair(fc, nm, name, nlen, 3, -1)) return 1; - } - - return 0; -} - -static int stbtt_FindMatchingFont_internal(unsigned char *font_collection, char *name_utf8, stbtt_int32 flags) -{ - stbtt_int32 i; - for (i=0;;++i) { - stbtt_int32 off = stbtt_GetFontOffsetForIndex(font_collection, i); - if (off < 0) return off; - if (stbtt__matches((stbtt_uint8 *) font_collection, off, (stbtt_uint8*) name_utf8, flags)) - return off; - } -} - -#if defined(__GNUC__) || defined(__clang__) -#pragma GCC diagnostic push -#pragma GCC diagnostic ignored "-Wcast-qual" -#endif - -STBTT_DEF int stbtt_BakeFontBitmap(const unsigned char *data, int offset, - float pixel_height, unsigned char *pixels, int pw, int ph, - int first_char, int num_chars, stbtt_bakedchar *chardata) -{ - return stbtt_BakeFontBitmap_internal((unsigned char *) data, offset, pixel_height, pixels, pw, ph, first_char, num_chars, chardata); -} - -STBTT_DEF int stbtt_GetFontOffsetForIndex(const unsigned char *data, int index) -{ - return stbtt_GetFontOffsetForIndex_internal((unsigned char *) data, index); -} - -STBTT_DEF int stbtt_GetNumberOfFonts(const unsigned char *data) -{ - return stbtt_GetNumberOfFonts_internal((unsigned char *) data); -} - -STBTT_DEF int stbtt_InitFont(stbtt_fontinfo *info, const unsigned char *data, int offset) -{ - return stbtt_InitFont_internal(info, (unsigned char *) data, offset); -} - -STBTT_DEF int stbtt_FindMatchingFont(const unsigned char *fontdata, const char *name, int flags) -{ - return stbtt_FindMatchingFont_internal((unsigned char *) fontdata, (char *) name, flags); -} - -STBTT_DEF int stbtt_CompareUTF8toUTF16_bigendian(const char *s1, int len1, const char *s2, int len2) -{ - return stbtt_CompareUTF8toUTF16_bigendian_internal((char *) s1, len1, (char *) s2, len2); -} - -#if defined(__GNUC__) || defined(__clang__) -#pragma GCC diagnostic pop -#endif - -#endif /* STB_TRUETYPE_IMPLEMENTATION */ - - -/* FULL VERSION HISTORY */ -/* */ -/* 1.25 (2021-07-11) many fixes */ -/* 1.24 (2020-02-05) fix warning */ -/* 1.23 (2020-02-02) query SVG data for glyphs; query whole kerning table (but only kern not GPOS) */ -/* 1.22 (2019-08-11) minimize missing-glyph duplication; fix kerning if both 'GPOS' and 'kern' are defined */ -/* 1.21 (2019-02-25) fix warning */ -/* 1.20 (2019-02-07) PackFontRange skips missing codepoints; GetScaleFontVMetrics() */ -/* 1.19 (2018-02-11) OpenType GPOS kerning (horizontal only), STBTT_fmod */ -/* 1.18 (2018-01-29) add missing function */ -/* 1.17 (2017-07-23) make more arguments const; doc fix */ -/* 1.16 (2017-07-12) SDF support */ -/* 1.15 (2017-03-03) make more arguments const */ -/* 1.14 (2017-01-16) num-fonts-in-TTC function */ -/* 1.13 (2017-01-02) support OpenType fonts, certain Apple fonts */ -/* 1.12 (2016-10-25) suppress warnings about casting away const with -Wcast-qual */ -/* 1.11 (2016-04-02) fix unused-variable warning */ -/* 1.10 (2016-04-02) allow user-defined fabs() replacement */ -/* fix memory leak if fontsize=0.0 */ -/* fix warning from duplicate typedef */ -/* 1.09 (2016-01-16) warning fix; avoid crash on outofmem; use alloc userdata for PackFontRanges */ -/* 1.08 (2015-09-13) document stbtt_Rasterize(); fixes for vertical & horizontal edges */ -/* 1.07 (2015-08-01) allow PackFontRanges to accept arrays of sparse codepoints; */ -/* allow PackFontRanges to pack and render in separate phases; */ -/* fix stbtt_GetFontOFfsetForIndex (never worked for non-0 input?); */ -/* fixed an assert() bug in the new rasterizer */ -/* replace assert() with STBTT_assert() in new rasterizer */ -/* 1.06 (2015-07-14) performance improvements (~35% faster on x86 and x64 on test machine) */ -/* also more precise AA rasterizer, except if shapes overlap */ -/* remove need for STBTT_sort */ -/* 1.05 (2015-04-15) fix misplaced definitions for STBTT_STATIC */ -/* 1.04 (2015-04-15) typo in example */ -/* 1.03 (2015-04-12) STBTT_STATIC, fix memory leak in new packing, various fixes */ -/* 1.02 (2014-12-10) fix various warnings & compile issues w/ stb_rect_pack, C++ */ -/* 1.01 (2014-12-08) fix subpixel position when oversampling to exactly match */ -/* non-oversampled; STBTT_POINT_SIZE for packed case only */ -/* 1.00 (2014-12-06) add new PackBegin etc. API, w/ support for oversampling */ -/* 0.99 (2014-09-18) fix multiple bugs with subpixel rendering (ryg) */ -/* 0.9 (2014-08-07) support certain mac/iOS fonts without an MS platformID */ -/* 0.8b (2014-07-07) fix a warning */ -/* 0.8 (2014-05-25) fix a few more warnings */ -/* 0.7 (2013-09-25) bugfix: subpixel glyph bug fixed in 0.5 had come back */ -/* 0.6c (2012-07-24) improve documentation */ -/* 0.6b (2012-07-20) fix a few more warnings */ -/* 0.6 (2012-07-17) fix warnings; added stbtt_ScaleForMappingEmToPixels, */ -/* stbtt_GetFontBoundingBox, stbtt_IsGlyphEmpty */ -/* 0.5 (2011-12-09) bugfixes: */ -/* subpixel glyph renderer computed wrong bounding box */ -/* first vertex of shape can be off-curve (FreeSans) */ -/* 0.4b (2011-12-03) fixed an error in the font baking example */ -/* 0.4 (2011-12-01) kerning, subpixel rendering (tor) */ -/* bugfixes for: */ -/* codepoint-to-glyph conversion using table fmt=12 */ -/* codepoint-to-glyph conversion using table fmt=4 */ -/* stbtt_GetBakedQuad with non-square texture (Zer) */ -/* updated Hello World! sample to use kerning and subpixel */ -/* fixed some warnings */ -/* 0.3 (2009-06-24) cmap fmt=12, compound shapes (MM) */ -/* userdata, malloc-from-userdata, non-zero fill (stb) */ -/* 0.2 (2009-03-11) Fix unsigned/signed char warnings */ -/* 0.1 (2009-03-09) First public release */ -/* */ - -/* ------------------------------------------------------------------------------- -This software is available under 2 licenses -- choose whichever you prefer. ------------------------------------------------------------------------------- -ALTERNATIVE A - MIT License -Copyright (c) 2017 Sean Barrett -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. ------------------------------------------------------------------------------- -ALTERNATIVE B - Public Domain (www.unlicense.org) -This is free and unencumbered software released into the public domain. -Anyone is free to copy, modify, publish, use, compile, sell, or distribute this -software, either in source code form or as a compiled binary, for any purpose, -commercial or non-commercial, and by any means. -In jurisdictions that recognize copyright laws, the author or authors of this -software dedicate any and all copyright interest in the software to the public -domain. We make this dedication for the benefit of the public at large and to -the detriment of our heirs and successors. We intend this dedication to be an -overt act of relinquishment in perpetuity of all present and future rights to -this software under copyright law. -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 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. ------------------------------------------------------------------------------- -*/ - - - - -#ifdef NK_INCLUDE_FONT_BAKING -/* ------------------------------------------------------------- - * - * RECT PACK - * - * --------------------------------------------------------------*/ - - - -/* - * ============================================================== - * - * TRUETYPE - * - * =============================================================== - */ -#define STBTT_MAX_OVERSAMPLE 8 - - -/* ------------------------------------------------------------- - * - * FONT BAKING - * - * --------------------------------------------------------------*/ -struct nk_font_bake_data { - struct stbtt_fontinfo info; - struct stbrp_rect *rects; - stbtt_pack_range *ranges; - nk_rune range_count; -}; - -struct nk_font_baker { - struct nk_allocator alloc; - struct stbtt_pack_context spc; - struct nk_font_bake_data *build; - stbtt_packedchar *packed_chars; - struct stbrp_rect *rects; - stbtt_pack_range *ranges; -}; - -NK_GLOBAL const nk_size nk_rect_align = NK_ALIGNOF(struct stbrp_rect); -NK_GLOBAL const nk_size nk_range_align = NK_ALIGNOF(stbtt_pack_range); -NK_GLOBAL const nk_size nk_char_align = NK_ALIGNOF(stbtt_packedchar); -NK_GLOBAL const nk_size nk_build_align = NK_ALIGNOF(struct nk_font_bake_data); -NK_GLOBAL const nk_size nk_baker_align = NK_ALIGNOF(struct nk_font_baker); - -NK_INTERN int -nk_range_count(const nk_rune *range) -{ - const nk_rune *iter = range; - NK_ASSERT(range); - if (!range) return 0; - while (*(iter++) != 0); - return (iter == range) ? 0 : (int)((iter - range)/2); -} -NK_INTERN int -nk_range_glyph_count(const nk_rune *range, int count) -{ - int i = 0; - int total_glyphs = 0; - for (i = 0; i < count; ++i) { - int diff; - nk_rune f = range[(i*2)+0]; - nk_rune t = range[(i*2)+1]; - NK_ASSERT(t >= f); - diff = (int)((t - f) + 1); - total_glyphs += diff; - } - return total_glyphs; -} -NK_API const nk_rune* -nk_font_default_glyph_ranges(void) -{ - NK_STORAGE const nk_rune ranges[] = {0x0020, 0x00FF, 0}; - return ranges; -} -NK_API const nk_rune* -nk_font_chinese_glyph_ranges(void) -{ - NK_STORAGE const nk_rune ranges[] = { - 0x0020, 0x00FF, - 0x3000, 0x30FF, - 0x31F0, 0x31FF, - 0xFF00, 0xFFEF, - 0x4E00, 0x9FAF, - 0 - }; - return ranges; -} -NK_API const nk_rune* -nk_font_cyrillic_glyph_ranges(void) -{ - NK_STORAGE const nk_rune ranges[] = { - 0x0020, 0x00FF, - 0x0400, 0x052F, - 0x2DE0, 0x2DFF, - 0xA640, 0xA69F, - 0 - }; - return ranges; -} -NK_API const nk_rune* -nk_font_korean_glyph_ranges(void) -{ - NK_STORAGE const nk_rune ranges[] = { - 0x0020, 0x00FF, - 0x3131, 0x3163, - 0xAC00, 0xD79D, - 0 - }; - return ranges; -} -NK_INTERN void -nk_font_baker_memory(nk_size *temp, int *glyph_count, - struct nk_font_config *config_list, int count) -{ - int range_count = 0; - int total_range_count = 0; - struct nk_font_config *iter, *i; - - NK_ASSERT(config_list); - NK_ASSERT(glyph_count); - if (!config_list) { - *temp = 0; - *glyph_count = 0; - return; - } - *glyph_count = 0; - for (iter = config_list; iter; iter = iter->next) { - i = iter; - do {if (!i->range) iter->range = nk_font_default_glyph_ranges(); - range_count = nk_range_count(i->range); - total_range_count += range_count; - *glyph_count += nk_range_glyph_count(i->range, range_count); - } while ((i = i->n) != iter); - } - *temp = (nk_size)*glyph_count * sizeof(struct stbrp_rect); - *temp += (nk_size)total_range_count * sizeof(stbtt_pack_range); - *temp += (nk_size)*glyph_count * sizeof(stbtt_packedchar); - *temp += (nk_size)count * sizeof(struct nk_font_bake_data); - *temp += sizeof(struct nk_font_baker); - *temp += nk_rect_align + nk_range_align + nk_char_align; - *temp += nk_build_align + nk_baker_align; -} -NK_INTERN struct nk_font_baker* -nk_font_baker(void *memory, int glyph_count, int count, struct nk_allocator *alloc) -{ - struct nk_font_baker *baker; - if (!memory) return 0; - /* setup baker inside a memory block */ - baker = (struct nk_font_baker*)NK_ALIGN_PTR(memory, nk_baker_align); - baker->build = (struct nk_font_bake_data*)NK_ALIGN_PTR((baker + 1), nk_build_align); - baker->packed_chars = (stbtt_packedchar*)NK_ALIGN_PTR((baker->build + count), nk_char_align); - baker->rects = (struct stbrp_rect*)NK_ALIGN_PTR((baker->packed_chars + glyph_count), nk_rect_align); - baker->ranges = (stbtt_pack_range*)NK_ALIGN_PTR((baker->rects + glyph_count), nk_range_align); - baker->alloc = *alloc; - return baker; -} -NK_INTERN int -nk_font_bake_pack(struct nk_font_baker *baker, - nk_size *image_memory, int *width, int *height, struct nk_recti *custom, - const struct nk_font_config *config_list, int count, - struct nk_allocator *alloc) -{ - NK_STORAGE const nk_size max_height = 1024 * 32; - const struct nk_font_config *config_iter, *it; - int total_glyph_count = 0; - int total_range_count = 0; - int range_count = 0; - int i = 0; - - NK_ASSERT(image_memory); - NK_ASSERT(width); - NK_ASSERT(height); - NK_ASSERT(config_list); - NK_ASSERT(count); - NK_ASSERT(alloc); - - if (!image_memory || !width || !height || !config_list || !count) return nk_false; - for (config_iter = config_list; config_iter; config_iter = config_iter->next) { - it = config_iter; - do {range_count = nk_range_count(it->range); - total_range_count += range_count; - total_glyph_count += nk_range_glyph_count(it->range, range_count); - } while ((it = it->n) != config_iter); - } - /* setup font baker from temporary memory */ - for (config_iter = config_list; config_iter; config_iter = config_iter->next) { - it = config_iter; - do { - struct stbtt_fontinfo *font_info = &baker->build[i++].info; - font_info->userdata = alloc; - - if (!stbtt_InitFont(font_info, (const unsigned char*)it->ttf_blob, stbtt_GetFontOffsetForIndex((const unsigned char*)it->ttf_blob, 0))) - return nk_false; - } while ((it = it->n) != config_iter); - } - *height = 0; - *width = (total_glyph_count > 1000) ? 1024 : 512; - stbtt_PackBegin(&baker->spc, 0, (int)*width, (int)max_height, 0, 1, alloc); - { - int input_i = 0; - int range_n = 0; - int rect_n = 0; - int char_n = 0; - - if (custom) { - /* pack custom user data first so it will be in the upper left corner*/ - struct stbrp_rect custom_space; - nk_zero(&custom_space, sizeof(custom_space)); - custom_space.w = (stbrp_coord)(custom->w); - custom_space.h = (stbrp_coord)(custom->h); - - stbtt_PackSetOversampling(&baker->spc, 1, 1); - stbrp_pack_rects((struct stbrp_context*)baker->spc.pack_info, &custom_space, 1); - *height = NK_MAX(*height, (int)(custom_space.y + custom_space.h)); - - custom->x = (short)custom_space.x; - custom->y = (short)custom_space.y; - custom->w = (short)custom_space.w; - custom->h = (short)custom_space.h; - } - - /* first font pass: pack all glyphs */ - for (input_i = 0, config_iter = config_list; input_i < count && config_iter; - config_iter = config_iter->next) { - it = config_iter; - do {int n = 0; - int glyph_count; - const nk_rune *in_range; - const struct nk_font_config *cfg = it; - struct nk_font_bake_data *tmp = &baker->build[input_i++]; - - /* count glyphs + ranges in current font */ - glyph_count = 0; range_count = 0; - for (in_range = cfg->range; in_range[0] && in_range[1]; in_range += 2) { - glyph_count += (int)(in_range[1] - in_range[0]) + 1; - range_count++; - } - - /* setup ranges */ - tmp->ranges = baker->ranges + range_n; - tmp->range_count = (nk_rune)range_count; - range_n += range_count; - for (i = 0; i < range_count; ++i) { - in_range = &cfg->range[i * 2]; - tmp->ranges[i].font_size = cfg->size; - tmp->ranges[i].first_unicode_codepoint_in_range = (int)in_range[0]; - tmp->ranges[i].num_chars = (int)(in_range[1]- in_range[0]) + 1; - tmp->ranges[i].chardata_for_range = baker->packed_chars + char_n; - char_n += tmp->ranges[i].num_chars; - } - - /* pack */ - tmp->rects = baker->rects + rect_n; - rect_n += glyph_count; - stbtt_PackSetOversampling(&baker->spc, cfg->oversample_h, cfg->oversample_v); - n = stbtt_PackFontRangesGatherRects(&baker->spc, &tmp->info, - tmp->ranges, (int)tmp->range_count, tmp->rects); - stbrp_pack_rects((struct stbrp_context*)baker->spc.pack_info, tmp->rects, (int)n); - - /* texture height */ - for (i = 0; i < n; ++i) { - if (tmp->rects[i].was_packed) - *height = NK_MAX(*height, tmp->rects[i].y + tmp->rects[i].h); - } - } while ((it = it->n) != config_iter); - } - NK_ASSERT(rect_n == total_glyph_count); - NK_ASSERT(char_n == total_glyph_count); - NK_ASSERT(range_n == total_range_count); - } - *height = (int)nk_round_up_pow2((nk_uint)*height); - *image_memory = (nk_size)(*width) * (nk_size)(*height); - return nk_true; -} -NK_INTERN void -nk_font_bake(struct nk_font_baker *baker, void *image_memory, int width, int height, - struct nk_font_glyph *glyphs, int glyphs_count, - const struct nk_font_config *config_list, int font_count) -{ - int input_i = 0; - nk_rune glyph_n = 0; - const struct nk_font_config *config_iter; - const struct nk_font_config *it; - - NK_ASSERT(image_memory); - NK_ASSERT(width); - NK_ASSERT(height); - NK_ASSERT(config_list); - NK_ASSERT(baker); - NK_ASSERT(font_count); - NK_ASSERT(glyphs_count); - if (!image_memory || !width || !height || !config_list || - !font_count || !glyphs || !glyphs_count) - return; - - /* second font pass: render glyphs */ - nk_zero(image_memory, (nk_size)((nk_size)width * (nk_size)height)); - baker->spc.pixels = (unsigned char*)image_memory; - baker->spc.height = (int)height; - for (input_i = 0, config_iter = config_list; input_i < font_count && config_iter; - config_iter = config_iter->next) { - it = config_iter; - do {const struct nk_font_config *cfg = it; - struct nk_font_bake_data *tmp = &baker->build[input_i++]; - stbtt_PackSetOversampling(&baker->spc, cfg->oversample_h, cfg->oversample_v); - stbtt_PackFontRangesRenderIntoRects(&baker->spc, &tmp->info, tmp->ranges, (int)tmp->range_count, tmp->rects); - } while ((it = it->n) != config_iter); - } stbtt_PackEnd(&baker->spc); - - /* third pass: setup font and glyphs */ - for (input_i = 0, config_iter = config_list; input_i < font_count && config_iter; - config_iter = config_iter->next) { - it = config_iter; - do {nk_size i = 0; - int char_idx = 0; - nk_rune glyph_count = 0; - const struct nk_font_config *cfg = it; - struct nk_font_bake_data *tmp = &baker->build[input_i++]; - struct nk_baked_font *dst_font = cfg->font; - - float font_scale = stbtt_ScaleForPixelHeight(&tmp->info, cfg->size); - int unscaled_ascent, unscaled_descent, unscaled_line_gap; - stbtt_GetFontVMetrics(&tmp->info, &unscaled_ascent, &unscaled_descent, - &unscaled_line_gap); - - /* fill baked font */ - if (!cfg->merge_mode) { - dst_font->ranges = cfg->range; - dst_font->height = cfg->size; - dst_font->ascent = ((float)unscaled_ascent * font_scale); - dst_font->descent = ((float)unscaled_descent * font_scale); - dst_font->glyph_offset = glyph_n; - /* - Need to zero this, or it will carry over from a previous - bake, and cause a segfault when accessing glyphs[]. - */ - dst_font->glyph_count = 0; - } - - /* fill own baked font glyph array */ - for (i = 0; i < tmp->range_count; ++i) { - stbtt_pack_range *range = &tmp->ranges[i]; - for (char_idx = 0; char_idx < range->num_chars; char_idx++) - { - nk_rune codepoint = 0; - float dummy_x = 0, dummy_y = 0; - stbtt_aligned_quad q; - struct nk_font_glyph *glyph; - - /* query glyph bounds from stb_truetype */ - const stbtt_packedchar *pc = &range->chardata_for_range[char_idx]; - codepoint = (nk_rune)(range->first_unicode_codepoint_in_range + char_idx); - stbtt_GetPackedQuad(range->chardata_for_range, (int)width, - (int)height, char_idx, &dummy_x, &dummy_y, &q, 0); - - /* fill own glyph type with data */ - glyph = &glyphs[dst_font->glyph_offset + dst_font->glyph_count + (unsigned int)glyph_count]; - glyph->codepoint = codepoint; - glyph->x0 = q.x0; glyph->y0 = q.y0; - glyph->x1 = q.x1; glyph->y1 = q.y1; - glyph->y0 += (dst_font->ascent + 0.5f); - glyph->y1 += (dst_font->ascent + 0.5f); - glyph->w = glyph->x1 - glyph->x0 + 0.5f; - glyph->h = glyph->y1 - glyph->y0; - - if (cfg->coord_type == NK_COORD_PIXEL) { - glyph->u0 = q.s0 * (float)width; - glyph->v0 = q.t0 * (float)height; - glyph->u1 = q.s1 * (float)width; - glyph->v1 = q.t1 * (float)height; - } else { - glyph->u0 = q.s0; - glyph->v0 = q.t0; - glyph->u1 = q.s1; - glyph->v1 = q.t1; - } - glyph->xadvance = (pc->xadvance + cfg->spacing.x); - if (cfg->pixel_snap) - glyph->xadvance = (float)(int)(glyph->xadvance + 0.5f); - glyph_count++; - } - } - dst_font->glyph_count += glyph_count; - glyph_n += glyph_count; - } while ((it = it->n) != config_iter); - } -} -NK_INTERN void -nk_font_bake_custom_data(void *img_memory, int img_width, int img_height, - struct nk_recti img_dst, const char *texture_data_mask, int tex_width, - int tex_height, char white, char black) -{ - nk_byte *pixels; - int y = 0; - int x = 0; - int n = 0; - - NK_ASSERT(img_memory); - NK_ASSERT(img_width); - NK_ASSERT(img_height); - NK_ASSERT(texture_data_mask); - NK_UNUSED(tex_height); - if (!img_memory || !img_width || !img_height || !texture_data_mask) - return; - - pixels = (nk_byte*)img_memory; - for (y = 0, n = 0; y < tex_height; ++y) { - for (x = 0; x < tex_width; ++x, ++n) { - const int off0 = ((img_dst.x + x) + (img_dst.y + y) * img_width); - const int off1 = off0 + 1 + tex_width; - pixels[off0] = (texture_data_mask[n] == white) ? 0xFF : 0x00; - pixels[off1] = (texture_data_mask[n] == black) ? 0xFF : 0x00; - } - } -} -NK_INTERN void -nk_font_bake_convert(void *out_memory, int img_width, int img_height, - const void *in_memory) -{ - int n = 0; - nk_rune *dst; - const nk_byte *src; - - NK_ASSERT(out_memory); - NK_ASSERT(in_memory); - NK_ASSERT(img_width); - NK_ASSERT(img_height); - if (!out_memory || !in_memory || !img_height || !img_width) return; - - dst = (nk_rune*)out_memory; - src = (const nk_byte*)in_memory; - for (n = (int)(img_width * img_height); n > 0; n--) - *dst++ = ((nk_rune)(*src++) << 24) | 0x00FFFFFF; -} - -/* ------------------------------------------------------------- - * - * FONT - * - * --------------------------------------------------------------*/ -NK_INTERN float -nk_font_text_width(nk_handle handle, float height, const char *text, int len) -{ - nk_rune unicode; - int text_len = 0; - float text_width = 0; - int glyph_len = 0; - float scale = 0; - - struct nk_font *font = (struct nk_font*)handle.ptr; - NK_ASSERT(font); - NK_ASSERT(font->glyphs); - if (!font || !text || !len) - return 0; - - scale = height/font->info.height; - glyph_len = text_len = nk_utf_decode(text, &unicode, (int)len); - if (!glyph_len) return 0; - while (text_len <= (int)len && glyph_len) { - const struct nk_font_glyph *g; - if (unicode == NK_UTF_INVALID) break; - - /* query currently drawn glyph information */ - g = nk_font_find_glyph(font, unicode); - text_width += g->xadvance * scale; - - /* offset next glyph */ - glyph_len = nk_utf_decode(text + text_len, &unicode, (int)len - text_len); - text_len += glyph_len; - } - return text_width; -} -#ifdef NK_INCLUDE_VERTEX_BUFFER_OUTPUT -NK_INTERN void -nk_font_query_font_glyph(nk_handle handle, float height, - struct nk_user_font_glyph *glyph, nk_rune codepoint, nk_rune next_codepoint) -{ - float scale; - const struct nk_font_glyph *g; - struct nk_font *font; - - NK_ASSERT(glyph); - NK_UNUSED(next_codepoint); - - font = (struct nk_font*)handle.ptr; - NK_ASSERT(font); - NK_ASSERT(font->glyphs); - if (!font || !glyph) - return; - - scale = height/font->info.height; - g = nk_font_find_glyph(font, codepoint); - glyph->width = (g->x1 - g->x0) * scale; - glyph->height = (g->y1 - g->y0) * scale; - glyph->offset = nk_vec2(g->x0 * scale, g->y0 * scale); - glyph->xadvance = (g->xadvance * scale); - glyph->uv[0] = nk_vec2(g->u0, g->v0); - glyph->uv[1] = nk_vec2(g->u1, g->v1); -} -#endif -NK_API const struct nk_font_glyph* -nk_font_find_glyph(struct nk_font *font, nk_rune unicode) -{ - int i = 0; - int count; - int total_glyphs = 0; - const struct nk_font_glyph *glyph = 0; - const struct nk_font_config *iter = 0; - - NK_ASSERT(font); - NK_ASSERT(font->glyphs); - NK_ASSERT(font->info.ranges); - if (!font || !font->glyphs) return 0; - - glyph = font->fallback; - iter = font->config; - do {count = nk_range_count(iter->range); - for (i = 0; i < count; ++i) { - nk_rune f = iter->range[(i*2)+0]; - nk_rune t = iter->range[(i*2)+1]; - int diff = (int)((t - f) + 1); - if (unicode >= f && unicode <= t) - return &font->glyphs[((nk_rune)total_glyphs + (unicode - f))]; - total_glyphs += diff; - } - } while ((iter = iter->n) != font->config); - return glyph; -} -NK_INTERN void -nk_font_init(struct nk_font *font, float pixel_height, - nk_rune fallback_codepoint, struct nk_font_glyph *glyphs, - const struct nk_baked_font *baked_font, nk_handle atlas) -{ - struct nk_baked_font baked; - NK_ASSERT(font); - NK_ASSERT(glyphs); - NK_ASSERT(baked_font); - if (!font || !glyphs || !baked_font) - return; - - baked = *baked_font; - font->fallback = 0; - font->info = baked; - font->scale = (float)pixel_height / (float)font->info.height; - font->glyphs = &glyphs[baked_font->glyph_offset]; - font->texture = atlas; - font->fallback_codepoint = fallback_codepoint; - font->fallback = nk_font_find_glyph(font, fallback_codepoint); - - font->handle.height = font->info.height * font->scale; - font->handle.width = nk_font_text_width; - font->handle.userdata.ptr = font; -#ifdef NK_INCLUDE_VERTEX_BUFFER_OUTPUT - font->handle.query = nk_font_query_font_glyph; - font->handle.texture = font->texture; -#endif -} - -/* --------------------------------------------------------------------------- - * - * DEFAULT FONT - * - * ProggyClean.ttf - * Copyright (c) 2004, 2005 Tristan Grimmer - * MIT license (see License.txt in http://www.upperbounds.net/download/ProggyClean.ttf.zip) - * Download and more information at http://upperbounds.net - *-----------------------------------------------------------------------------*/ -#ifdef __clang__ -#pragma clang diagnostic push -#pragma clang diagnostic ignored "-Woverlength-strings" -#elif defined(__GNUC__) || defined(__GNUG__) -#pragma GCC diagnostic push -#pragma GCC diagnostic ignored "-Woverlength-strings" -#endif - -#ifdef NK_INCLUDE_DEFAULT_FONT - -NK_GLOBAL const char nk_proggy_clean_ttf_compressed_data_base85[11980+1] = - "7])#######hV0qs'/###[),##/l:$#Q6>##5[n42>c-TH`->>#/e>11NNV=Bv(*:.F?uu#(gRU.o0XGH`$vhLG1hxt9?W`#,5LsCp#-i>.r$<$6pD>Lb';9Crc6tgXmKVeU2cD4Eo3R/" - "2*>]b(MC;$jPfY.;h^`IWM9Qo#t'X#(v#Y9w0#1D$CIf;W'#pWUPXOuxXuU(H9M(1=Ke$$'5F%)]0^#0X@U.a$FBjVQTSDgEKnIS7EM9>ZY9w0#L;>>#Mx&4Mvt//L[MkA#W@lK.N'[0#7RL_&#w+F%HtG9M#XL`N&.,GM4Pg;--VsM.M0rJfLH2eTM`*oJMHRC`N" - "kfimM2J,W-jXS:)r0wK#@Fge$U>`w'N7G#$#fB#$E^$#:9:hk+eOe--6x)F7*E%?76%^GMHePW-Z5l'&GiF#$956:rS?dA#fiK:)Yr+`�j@'DbG&#^$PG.Ll+DNa&VZ>1i%h1S9u5o@YaaW$e+bROPOpxTO7Stwi1::iB1q)C_=dV26J;2,]7op$]uQr@_V7$q^%lQwtuHY]=DX,n3L#0PHDO4f9>dC@O>HBuKPpP*E,N+b3L#lpR/MrTEH.IAQk.a>D[.e;mc." - "x]Ip.PH^'/aqUO/$1WxLoW0[iLAw=4h(9.`G" - "CRUxHPeR`5Mjol(dUWxZa(>STrPkrJiWx`5U7F#.g*jrohGg`cg:lSTvEY/EV_7H4Q9[Z%cnv;JQYZ5q.l7Zeas:HOIZOB?Ggv:[7MI2k).'2($5FNP&EQ(,)" - "U]W]+fh18.vsai00);D3@4ku5P?DP8aJt+;qUM]=+b'8@;mViBKx0DE[-auGl8:PJ&Dj+M6OC]O^((##]`0i)drT;-7X`=-H3[igUnPG-NZlo.#k@h#=Ork$m>a>$-?Tm$UV(?#P6YY#" - "'/###xe7q.73rI3*pP/$1>s9)W,JrM7SN]'/4C#v$U`0#V.[0>xQsH$fEmPMgY2u7Kh(G%siIfLSoS+MK2eTM$=5,M8p`A.;_R%#u[K#$x4AG8.kK/HSB==-'Ie/QTtG?-.*^N-4B/ZM" - "_3YlQC7(p7q)&](`6_c)$/*JL(L-^(]$wIM`dPtOdGA,U3:w2M-0+WomX2u7lqM2iEumMTcsF?-aT=Z-97UEnXglEn1K-bnEO`gu" - "Ft(c%=;Am_Qs@jLooI&NX;]0#j4#F14;gl8-GQpgwhrq8'=l_f-b49'UOqkLu7-##oDY2L(te+Mch&gLYtJ,MEtJfLh'x'M=$CS-ZZ%P]8bZ>#S?YY#%Q&q'3^Fw&?D)UDNrocM3A76/" - "/oL?#h7gl85[qW/NDOk%16ij;+:1a'iNIdb-ou8.P*w,v5#EI$TWS>Pot-R*H'-SEpA:g)f+O$%%`kA#G=8RMmG1&O`>to8bC]T&$,n.LoO>29sp3dt-52U%VM#q7'DHpg+#Z9%H[Ket`e;)f#Km8&+DC$I46>#Kr]]u-[=99tts1.qb#q72g1WJO81q+eN'03'eM>&1XxY-caEnO" - "j%2n8)),?ILR5^.Ibn<-X-Mq7[a82Lq:F&#ce+S9wsCK*x`569E8ew'He]h:sI[2LM$[guka3ZRd6:t%IG:;$%YiJ:Nq=?eAw;/:nnDq0(CYcMpG)qLN4$##&J-XTt,%OVU4)S1+R-#dg0/Nn?Ku1^0f$B*P:Rowwm-`0PKjYDDM'3]d39VZHEl4,.j']Pk-M.h^&:0FACm$maq-&sgw0t7/6(^xtk%" - "LuH88Fj-ekm>GA#_>568x6(OFRl-IZp`&b,_P'$MhLbxfc$mj`,O;&%W2m`Zh:/)Uetw:aJ%]K9h:TcF]u_-Sj9,VK3M.*'&0D[Ca]J9gp8,kAW]" - "%(?A%R$f<->Zts'^kn=-^@c4%-pY6qI%J%1IGxfLU9CP8cbPlXv);C=b),<2mOvP8up,UVf3839acAWAW-W?#ao/^#%KYo8fRULNd2.>%m]UK:n%r$'sw]J;5pAoO_#2mO3n,'=H5(et" - "Hg*`+RLgv>=4U8guD$I%D:W>-r5V*%j*W:Kvej.Lp$'?;++O'>()jLR-^u68PHm8ZFWe+ej8h:9r6L*0//c&iH&R8pRbA#Kjm%upV1g:" - "a_#Ur7FuA#(tRh#.Y5K+@?3<-8m0$PEn;J:rh6?I6uG<-`wMU'ircp0LaE_OtlMb&1#6T.#FDKu#1Lw%u%+GM+X'e?YLfjM[VO0MbuFp7;>Q&#WIo)0@F%q7c#4XAXN-U&VBpqB>0ie&jhZ[?iLR@@_AvA-iQC(=ksRZRVp7`.=+NpBC%rh&3]R:8XDmE5^V8O(x<-+k?'(^](H.aREZSi,#1:[IXaZFOm<-ui#qUq2$##Ri;u75OK#(RtaW-K-F`S+cF]uN`-KMQ%rP/Xri.LRcB##=YL3BgM/3M" - "D?@f&1'BW-)Ju#bmmWCMkk&#TR`C,5d>g)F;t,4:@_l8G/5h4vUd%&%950:VXD'QdWoY-F$BtUwmfe$YqL'8(PWX(" - "P?^@Po3$##`MSs?DWBZ/S>+4%>fX,VWv/w'KD`LP5IbH;rTV>n3cEK8U#bX]l-/V+^lj3;vlMb&[5YQ8#pekX9JP3XUC72L,,?+Ni&co7ApnO*5NK,((W-i:$,kp'UDAO(G0Sq7MVjJs" - "bIu)'Z,*[>br5fX^:FPAWr-m2KgLQ_nN6'8uTGT5g)uLv:873UpTLgH+#FgpH'_o1780Ph8KmxQJ8#H72L4@768@Tm&Q" - "h4CB/5OvmA&,Q&QbUoi$a_%3M01H)4x7I^&KQVgtFnV+;[Pc>[m4k//,]1?#`VY[Jr*3&&slRfLiVZJ:]?=K3Sw=[$=uRB?3xk48@aege0jT6'N#(q%.O=?2S]u*(m<-" - "V8J'(1)G][68hW$5'q[GC&5j`TE?m'esFGNRM)j,ffZ?-qx8;->g4t*:CIP/[Qap7/9'#(1sao7w-.qNUdkJ)tCF&#B^;xGvn2r9FEPFFFcL@.iFNkTve$m%#QvQS8U@)2Z+3K:AKM5i" - "sZ88+dKQ)W6>J%CL`.d*(B`-n8D9oK-XV1q['-5k'cAZ69e;D_?$ZPP&s^+7])$*$#@QYi9,5P r+$%CE=68>K8r0=dSC%%(@p7" - ".m7jilQ02'0-VWAg
    TlGW'b)Tq7VT9q^*^$$.:&N@@" - "$&)WHtPm*5_rO0&e%K&#-30j(E4#'Zb.o/(Tpm$>K'f@[PvFl,hfINTNU6u'0pao7%XUp9]5.>%h`8_=VYbxuel.NTSsJfLacFu3B'lQSu/m6-Oqem8T+oE--$0a/k]uj9EwsG>%veR*" - "hv^BFpQj:K'#SJ,sB-'#](j.Lg92rTw-*n%@/;39rrJF,l#qV%OrtBeC6/,;qB3ebNW[?,Hqj2L.1NP&GjUR=1D8QaS3Up&@*9wP?+lo7b?@%'k4`p0Z$22%K3+iCZj?XJN4Nm&+YF]u" - "@-W$U%VEQ/,,>>#)D#%8cY#YZ?=,`Wdxu/ae&#" - "w6)R89tI#6@s'(6Bf7a&?S=^ZI_kS&ai`&=tE72L_D,;^R)7[$so8lKN%5/$(vdfq7+ebA#" - "u1p]ovUKW&Y%q]'>$1@-[xfn$7ZTp7mM,G,Ko7a&Gu%G[RMxJs[0MM%wci.LFDK)(%:_i2B5CsR8&9Z&#=mPEnm0f`<&c)QL5uJ#%u%lJj+D-r;BoFDoS97h5g)E#o:&S4weDF,9^Hoe`h*L+_a*NrLW-1pG_&2UdB8" - "6e%B/:=>)N4xeW.*wft-;$'58-ESqr#U`'6AQ]m&6/`Z>#S?YY#Vc;r7U2&326d=w&H####?TZ`*4?&.MK?LP8Vxg>$[QXc%QJv92.(Db*B)gb*BM9dM*hJMAo*c&#" - "b0v=Pjer]$gG&JXDf->'StvU7505l9$AFvgYRI^&<^b68?j#q9QX4SM'RO#&sL1IM.rJfLUAj221]d##DW=m83u5;'bYx,*Sl0hL(W;;$doB&O/TQ:(Z^xBdLjLV#*8U_72Lh+2Q8Cj0i:6hp&$C/:p(HK>T8Y[gHQ4`4)'$Ab(Nof%V'8hL&#SfD07&6D@M.*J:;$-rv29'M]8qMv-tLp,'886iaC=Hb*YJoKJ,(j%K=H`K.v9HggqBIiZu'QvBT.#=)0ukruV&.)3=(^1`o*Pj4<-#MJ+gLq9-##@HuZPN0]u:h7.T..G:;$/Usj(T7`Q8tT72LnYl<-qx8;-HV7Q-&Xdx%1a,hC=0u+HlsV>nuIQL-5" - "_>@kXQtMacfD.m-VAb8;IReM3$wf0''hra*so568'Ip&vRs849'MRYSp%:t:h5qSgwpEr$B>Q,;s(C#$)`svQuF$##-D,##,g68@2[T;.XSdN9Qe)rpt._K-#5wF)sP'##p#C0c%-Gb%" - "hd+<-j'Ai*x&&HMkT]C'OSl##5RG[JXaHN;d'uA#x._U;.`PU@(Z3dt4r152@:v,'R.Sj'w#0<-;kPI)FfJ&#AYJ&#//)>-k=m=*XnK$>=)72L]0I%>.G690a:$##<,);?;72#?x9+d;" - "^V'9;jY@;)br#q^YQpx:X#Te$Z^'=-=bGhLf:D6&bNwZ9-ZD#n^9HhLMr5G;']d&6'wYmTFmLq9wI>P(9mI[>kC-ekLC/R&CH+s'B;K-M6$EB%is00:" - "+A4[7xks.LrNk0&E)wILYF@2L'0Nb$+pv<(2.768/FrY&h$^3i&@+G%JT'<-,v`3;_)I9M^AE]CN?Cl2AZg+%4iTpT3$U4O]GKx'm9)b@p7YsvK3w^YR-" - "CdQ*:Ir<($u&)#(&?L9Rg3H)4fiEp^iI9O8KnTj,]H?D*r7'M;PwZ9K0E^k&-cpI;.p/6_vwoFMV<->#%Xi.LxVnrU(4&8/P+:hLSKj$#U%]49t'I:rgMi'FL@a:0Y-uA[39',(vbma*" - "hU%<-SRF`Tt:542R_VV$p@[p8DV[A,?1839FWdFTi1O*H&#(AL8[_P%.M>v^-))qOT*F5Cq0`Ye%+$B6i:7@0IXSsDiWP,##P`%/L-" - "S(qw%sf/@%#B6;/U7K]uZbi^Oc^2n%t<)'mEVE''n`WnJra$^TKvX5B>;_aSEK',(hwa0:i4G?.Bci.(X[?b*($,=-n<.Q%`(X=?+@Am*Js0&=3bh8K]mL69=Lb,OcZV/);TTm8VI;?%OtJ<(b4mq7M6:u?KRdFl*:xP?Yb.5)%w_I?7uk5JC+FS(m#i'k.'a0i)9<7b'fs'59hq$*5Uhv##pi^8+hIEBF`nvo`;'l0.^S1<-wUK2/Coh58KKhLj" - "M=SO*rfO`+qC`W-On.=AJ56>>i2@2LH6A:&5q`?9I3@@'04&p2/LVa*T-4<-i3;M9UvZd+N7>b*eIwg:CC)c<>nO&#$(>.Z-I&J(Q0Hd5Q%7Co-b`-cP)hI;*_F]u`Rb[.j8_Q/<&>uu+VsH$sM9TA%?)(vmJ80),P7E>)tjD%2L=-t#fK[%`v=Q8WlA2);Sa" - ">gXm8YB`1d@K#n]76-a$U,mF%Ul:#/'xoFM9QX-$.QN'>" - "[%$Z$uF6pA6Ki2O5:8w*vP1<-1`[G,)-m#>0`P&#eb#.3i)rtB61(o'$?X3B2Qft^ae_5tKL9MUe9b*sLEQ95C&`=G?@Mj=wh*'3E>=-<)Gt*Iw)'QG:`@I" - "wOf7&]1i'S01B+Ev/Nac#9S;=;YQpg_6U`*kVY39xK,[/6Aj7:'1Bm-_1EYfa1+o&o4hp7KN_Q(OlIo@S%;jVdn0'1h19w,WQhLI)3S#f$2(eb,jr*b;3Vw]*7NH%$c4Vs,eD9>XW8?N]o+(*pgC%/72LV-uW%iewS8W6m2rtCpo'RS1R84=@paTKt)>=%&1[)*vp'u+x,VrwN;&]kuO9JDbg=pO$J*.jVe;u'm0dr9l,<*wMK*Oe=g8lV_KEBFkO'oU]^=[-792#ok,)" - "i]lR8qQ2oA8wcRCZ^7w/Njh;?.stX?Q1>S1q4Bn$)K1<-rGdO'$Wr.Lc.CG)$/*JL4tNR/,SVO3,aUw'DJN:)Ss;wGn9A32ijw%FL+Z0Fn.U9;reSq)bmI32U==5ALuG&#Vf1398/pVo" - "1*c-(aY168o<`JsSbk-,1N;$>0:OUas(3:8Z972LSfF8eb=c-;>SPw7.6hn3m`9^Xkn(r.qS[0;T%&Qc=+STRxX'q1BNk3&*eu2;&8q$&x>Q#Q7^Tf+6<(d%ZVmj2bDi%.3L2n+4W'$P" - "iDDG)g,r%+?,$@?uou5tSe2aN_AQU*'IAO" - "URQ##V^Fv-XFbGM7Fl(N<3DhLGF%q.1rC$#:T__&Pi68%0xi_&[qFJ(77j_&JWoF.V735&T,[R*:xFR*K5>>#`bW-?4Ne_&6Ne_&6Ne_&n`kr-#GJcM6X;uM6X;uM(.a..^2TkL%oR(#" - ";u.T%fAr%4tJ8&><1=GHZ_+m9/#H1F^R#SC#*N=BA9(D?v[UiFY>>^8p,KKF.W]L29uLkLlu/+4T" - "w$)F./^n3+rlo+DB;5sIYGNk+i1t-69Jg--0pao7Sm#K)pdHW&;LuDNH@H>#/X-TI(;P>#,Gc>#0Su>#4`1?#8lC?#xL$#B.`$#F:r$#JF.%#NR@%#R_R%#Vke%#Zww%#_-4^Rh%Sflr-k'MS.o?.5/sWel/wpEM0%3'/1)K^f1-d>G21&v(35>V`39V7A4=onx4" - "A1OY5EI0;6Ibgr6M$HS7Q<)58C5w,;WoA*#[%T*#`1g*#d=#+#hI5+#lUG+#pbY+#tnl+#x$),#&1;,#*=M,#.I`,#2Ur,#6b.-#;w[H#iQtA#m^0B#qjBB#uvTB##-hB#'9$C#+E6C#" - "/QHC#3^ZC#7jmC#;v)D#?,)4kMYD4lVu`4m`:&5niUA5@(A5BA1]PBB:xlBCC=2CDLXMCEUtiCf&0g2'tN?PGT4CPGT4CPGT4CPGT4CPGT4CPGT4CPGT4CP" - "GT4CPGT4CPGT4CPGT4CPGT4CPGT4CP-qekC`.9kEg^+F$kwViFJTB&5KTB&5KTB&5KTB&5KTB&5KTB&5KTB&5KTB&5KTB&5KTB&5KTB&5KTB&5KTB&5KTB&5KTB&5o,^<-28ZI'O?;xp" - "O?;xpO?;xpO?;xpO?;xpO?;xpO?;xpO?;xpO?;xpO?;xpO?;xpO?;xpO?;xpO?;xp;7q-#lLYI:xvD=#"; - -#endif /* NK_INCLUDE_DEFAULT_FONT */ - -#define NK_CURSOR_DATA_W 90 -#define NK_CURSOR_DATA_H 27 -NK_GLOBAL const char nk_custom_cursor_data[NK_CURSOR_DATA_W * NK_CURSOR_DATA_H + 1] = -{ - "..- -XXXXXXX- X - X -XXXXXXX - XXXXXXX" - "..- -X.....X- X.X - X.X -X.....X - X.....X" - "--- -XXX.XXX- X...X - X...X -X....X - X....X" - "X - X.X - X.....X - X.....X -X...X - X...X" - "XX - X.X -X.......X- X.......X -X..X.X - X.X..X" - "X.X - X.X -XXXX.XXXX- XXXX.XXXX -X.X X.X - X.X X.X" - "X..X - X.X - X.X - X.X -XX X.X - X.X XX" - "X...X - X.X - X.X - XX X.X XX - X.X - X.X " - "X....X - X.X - X.X - X.X X.X X.X - X.X - X.X " - "X.....X - X.X - X.X - X..X X.X X..X - X.X - X.X " - "X......X - X.X - X.X - X...XXXXXX.XXXXXX...X - X.X XX-XX X.X " - "X.......X - X.X - X.X -X.....................X- X.X X.X-X.X X.X " - "X........X - X.X - X.X - X...XXXXXX.XXXXXX...X - X.X..X-X..X.X " - "X.........X -XXX.XXX- X.X - X..X X.X X..X - X...X-X...X " - "X..........X-X.....X- X.X - X.X X.X X.X - X....X-X....X " - "X......XXXXX-XXXXXXX- X.X - XX X.X XX - X.....X-X.....X " - "X...X..X --------- X.X - X.X - XXXXXXX-XXXXXXX " - "X..X X..X - -XXXX.XXXX- XXXX.XXXX ------------------------------------" - "X.X X..X - -X.......X- X.......X - XX XX - " - "XX X..X - - X.....X - X.....X - X.X X.X - " - " X..X - X...X - X...X - X..X X..X - " - " XX - X.X - X.X - X...XXXXXXXXXXXXX...X - " - "------------ - X - X -X.....................X- " - " ----------------------------------- X...XXXXXXXXXXXXX...X - " - " - X..X X..X - " - " - X.X X.X - " - " - XX XX - " -}; - -#ifdef __clang__ -#pragma clang diagnostic pop -#elif defined(__GNUC__) || defined(__GNUG__) -#pragma GCC diagnostic pop -#endif - -NK_GLOBAL unsigned char *nk__barrier; -NK_GLOBAL unsigned char *nk__barrier2; -NK_GLOBAL unsigned char *nk__barrier3; -NK_GLOBAL unsigned char *nk__barrier4; -NK_GLOBAL unsigned char *nk__dout; - -NK_INTERN unsigned int -nk_decompress_length(unsigned char *input) -{ - return (unsigned int)((input[8] << 24) + (input[9] << 16) + (input[10] << 8) + input[11]); -} -NK_INTERN void -nk__match(unsigned char *data, unsigned int length) -{ - /* INVERSE of memmove... write each byte before copying the next...*/ - NK_ASSERT (nk__dout + length <= nk__barrier); - if (nk__dout + length > nk__barrier) { nk__dout += length; return; } - if (data < nk__barrier4) { nk__dout = nk__barrier+1; return; } - while (length--) *nk__dout++ = *data++; -} -NK_INTERN void -nk__lit(unsigned char *data, unsigned int length) -{ - NK_ASSERT (nk__dout + length <= nk__barrier); - if (nk__dout + length > nk__barrier) { nk__dout += length; return; } - if (data < nk__barrier2) { nk__dout = nk__barrier+1; return; } - NK_MEMCPY(nk__dout, data, length); - nk__dout += length; -} -NK_INTERN unsigned char* -nk_decompress_token(unsigned char *i) -{ - #define nk__in2(x) ((i[x] << 8) + i[(x)+1]) - #define nk__in3(x) ((i[x] << 16) + nk__in2((x)+1)) - #define nk__in4(x) ((i[x] << 24) + nk__in3((x)+1)) - - if (*i >= 0x20) { /* use fewer if's for cases that expand small */ - if (*i >= 0x80) nk__match(nk__dout-i[1]-1, (unsigned int)i[0] - 0x80 + 1), i += 2; - else if (*i >= 0x40) nk__match(nk__dout-(nk__in2(0) - 0x4000 + 1), (unsigned int)i[2]+1), i += 3; - else /* *i >= 0x20 */ nk__lit(i+1, (unsigned int)i[0] - 0x20 + 1), i += 1 + (i[0] - 0x20 + 1); - } else { /* more ifs for cases that expand large, since overhead is amortized */ - if (*i >= 0x18) nk__match(nk__dout-(unsigned int)(nk__in3(0) - 0x180000 + 1), (unsigned int)i[3]+1), i += 4; - else if (*i >= 0x10) nk__match(nk__dout-(unsigned int)(nk__in3(0) - 0x100000 + 1), (unsigned int)nk__in2(3)+1), i += 5; - else if (*i >= 0x08) nk__lit(i+2, (unsigned int)nk__in2(0) - 0x0800 + 1), i += 2 + (nk__in2(0) - 0x0800 + 1); - else if (*i == 0x07) nk__lit(i+3, (unsigned int)nk__in2(1) + 1), i += 3 + (nk__in2(1) + 1); - else if (*i == 0x06) nk__match(nk__dout-(unsigned int)(nk__in3(1)+1), i[4]+1u), i += 5; - else if (*i == 0x04) nk__match(nk__dout-(unsigned int)(nk__in3(1)+1), (unsigned int)nk__in2(4)+1u), i += 6; - } - return i; -} -NK_INTERN unsigned int -nk_adler32(unsigned int adler32, unsigned char *buffer, unsigned int buflen) -{ - const unsigned long ADLER_MOD = 65521; - unsigned long s1 = adler32 & 0xffff, s2 = adler32 >> 16; - unsigned long blocklen, i; - - blocklen = buflen % 5552; - while (buflen) { - for (i=0; i + 7 < blocklen; i += 8) { - s1 += buffer[0]; s2 += s1; - s1 += buffer[1]; s2 += s1; - s1 += buffer[2]; s2 += s1; - s1 += buffer[3]; s2 += s1; - s1 += buffer[4]; s2 += s1; - s1 += buffer[5]; s2 += s1; - s1 += buffer[6]; s2 += s1; - s1 += buffer[7]; s2 += s1; - buffer += 8; - } - for (; i < blocklen; ++i) { - s1 += *buffer++; s2 += s1; - } - - s1 %= ADLER_MOD; s2 %= ADLER_MOD; - buflen -= (unsigned int)blocklen; - blocklen = 5552; - } - return (unsigned int)(s2 << 16) + (unsigned int)s1; -} -NK_INTERN unsigned int -nk_decompress(unsigned char *output, unsigned char *i, unsigned int length) -{ - unsigned int olen; - if (nk__in4(0) != 0x57bC0000) return 0; - if (nk__in4(4) != 0) return 0; /* error! stream is > 4GB */ - olen = nk_decompress_length(i); - nk__barrier2 = i; - nk__barrier3 = i+length; - nk__barrier = output + olen; - nk__barrier4 = output; - i += 16; - - nk__dout = output; - for (;;) { - unsigned char *old_i = i; - i = nk_decompress_token(i); - if (i == old_i) { - if (*i == 0x05 && i[1] == 0xfa) { - NK_ASSERT(nk__dout == output + olen); - if (nk__dout != output + olen) return 0; - if (nk_adler32(1, output, olen) != (unsigned int) nk__in4(2)) - return 0; - return olen; - } else { - NK_ASSERT(0); /* NOTREACHED */ - return 0; - } - } - NK_ASSERT(nk__dout <= output + olen); - if (nk__dout > output + olen) - return 0; - } -} -NK_INTERN unsigned int -nk_decode_85_byte(char c) -{ - return (unsigned int)((c >= '\\') ? c-36 : c-35); -} -NK_INTERN void -nk_decode_85(unsigned char* dst, const unsigned char* src) -{ - while (*src) - { - unsigned int tmp = - nk_decode_85_byte((char)src[0]) + - 85 * (nk_decode_85_byte((char)src[1]) + - 85 * (nk_decode_85_byte((char)src[2]) + - 85 * (nk_decode_85_byte((char)src[3]) + - 85 * nk_decode_85_byte((char)src[4])))); - - /* we can't assume little-endianess. */ - dst[0] = (unsigned char)((tmp >> 0) & 0xFF); - dst[1] = (unsigned char)((tmp >> 8) & 0xFF); - dst[2] = (unsigned char)((tmp >> 16) & 0xFF); - dst[3] = (unsigned char)((tmp >> 24) & 0xFF); - - src += 5; - dst += 4; - } -} - -/* ------------------------------------------------------------- - * - * FONT ATLAS - * - * --------------------------------------------------------------*/ -NK_API struct nk_font_config -nk_font_config(float pixel_height) -{ - struct nk_font_config cfg; - nk_zero_struct(cfg); - cfg.ttf_blob = 0; - cfg.ttf_size = 0; - cfg.ttf_data_owned_by_atlas = 0; - cfg.size = pixel_height; - cfg.oversample_h = 3; - cfg.oversample_v = 1; - cfg.pixel_snap = 0; - cfg.coord_type = NK_COORD_UV; - cfg.spacing = nk_vec2(0,0); - cfg.range = nk_font_default_glyph_ranges(); - cfg.merge_mode = 0; - cfg.fallback_glyph = '?'; - cfg.font = 0; - cfg.n = 0; - return cfg; -} -#ifdef NK_INCLUDE_DEFAULT_ALLOCATOR -NK_API void -nk_font_atlas_init_default(struct nk_font_atlas *atlas) -{ - NK_ASSERT(atlas); - if (!atlas) return; - nk_zero_struct(*atlas); - atlas->temporary.userdata.ptr = 0; - atlas->temporary.alloc = nk_malloc; - atlas->temporary.free = nk_mfree; - atlas->permanent.userdata.ptr = 0; - atlas->permanent.alloc = nk_malloc; - atlas->permanent.free = nk_mfree; -} -#endif -NK_API void -nk_font_atlas_init(struct nk_font_atlas *atlas, struct nk_allocator *alloc) -{ - NK_ASSERT(atlas); - NK_ASSERT(alloc); - if (!atlas || !alloc) return; - nk_zero_struct(*atlas); - atlas->permanent = *alloc; - atlas->temporary = *alloc; -} -NK_API void -nk_font_atlas_init_custom(struct nk_font_atlas *atlas, - struct nk_allocator *permanent, struct nk_allocator *temporary) -{ - NK_ASSERT(atlas); - NK_ASSERT(permanent); - NK_ASSERT(temporary); - if (!atlas || !permanent || !temporary) return; - nk_zero_struct(*atlas); - atlas->permanent = *permanent; - atlas->temporary = *temporary; -} -NK_API void -nk_font_atlas_begin(struct nk_font_atlas *atlas) -{ - NK_ASSERT(atlas); - NK_ASSERT(atlas->temporary.alloc && atlas->temporary.free); - NK_ASSERT(atlas->permanent.alloc && atlas->permanent.free); - if (!atlas || !atlas->permanent.alloc || !atlas->permanent.free || - !atlas->temporary.alloc || !atlas->temporary.free) return; - if (atlas->glyphs) { - atlas->permanent.free(atlas->permanent.userdata, atlas->glyphs); - atlas->glyphs = 0; - } - if (atlas->pixel) { - atlas->permanent.free(atlas->permanent.userdata, atlas->pixel); - atlas->pixel = 0; - } -} -NK_API struct nk_font* -nk_font_atlas_add(struct nk_font_atlas *atlas, const struct nk_font_config *config) -{ - struct nk_font *font = 0; - struct nk_font_config *cfg; - - NK_ASSERT(atlas); - NK_ASSERT(atlas->permanent.alloc); - NK_ASSERT(atlas->permanent.free); - NK_ASSERT(atlas->temporary.alloc); - NK_ASSERT(atlas->temporary.free); - - NK_ASSERT(config); - NK_ASSERT(config->ttf_blob); - NK_ASSERT(config->ttf_size); - NK_ASSERT(config->size > 0.0f); - - if (!atlas || !config || !config->ttf_blob || !config->ttf_size || config->size <= 0.0f|| - !atlas->permanent.alloc || !atlas->permanent.free || - !atlas->temporary.alloc || !atlas->temporary.free) - return 0; - - /* allocate font config */ - cfg = (struct nk_font_config*) - atlas->permanent.alloc(atlas->permanent.userdata,0, sizeof(struct nk_font_config)); - NK_MEMCPY(cfg, config, sizeof(*config)); - cfg->n = cfg; - cfg->p = cfg; - - if (!config->merge_mode) { - /* insert font config into list */ - if (!atlas->config) { - atlas->config = cfg; - cfg->next = 0; - } else { - struct nk_font_config *i = atlas->config; - while (i->next) i = i->next; - i->next = cfg; - cfg->next = 0; - } - /* allocate new font */ - font = (struct nk_font*) - atlas->permanent.alloc(atlas->permanent.userdata,0, sizeof(struct nk_font)); - NK_ASSERT(font); - nk_zero(font, sizeof(*font)); - if (!font) return 0; - font->config = cfg; - - /* insert font into list */ - if (!atlas->fonts) { - atlas->fonts = font; - font->next = 0; - } else { - struct nk_font *i = atlas->fonts; - while (i->next) i = i->next; - i->next = font; - font->next = 0; - } - cfg->font = &font->info; - } else { - /* extend previously added font */ - struct nk_font *f = 0; - struct nk_font_config *c = 0; - NK_ASSERT(atlas->font_num); - f = atlas->fonts; - c = f->config; - cfg->font = &f->info; - - cfg->n = c; - cfg->p = c->p; - c->p->n = cfg; - c->p = cfg; - } - /* create own copy of .TTF font blob */ - if (!config->ttf_data_owned_by_atlas) { - cfg->ttf_blob = atlas->permanent.alloc(atlas->permanent.userdata,0, cfg->ttf_size); - NK_ASSERT(cfg->ttf_blob); - if (!cfg->ttf_blob) { - atlas->font_num++; - return 0; - } - NK_MEMCPY(cfg->ttf_blob, config->ttf_blob, cfg->ttf_size); - cfg->ttf_data_owned_by_atlas = 1; - } - atlas->font_num++; - return font; -} -NK_API struct nk_font* -nk_font_atlas_add_from_memory(struct nk_font_atlas *atlas, void *memory, - nk_size size, float height, const struct nk_font_config *config) -{ - struct nk_font_config cfg; - NK_ASSERT(memory); - NK_ASSERT(size); - - NK_ASSERT(atlas); - NK_ASSERT(atlas->temporary.alloc); - NK_ASSERT(atlas->temporary.free); - NK_ASSERT(atlas->permanent.alloc); - NK_ASSERT(atlas->permanent.free); - if (!atlas || !atlas->temporary.alloc || !atlas->temporary.free || !memory || !size || - !atlas->permanent.alloc || !atlas->permanent.free) - return 0; - - cfg = (config) ? *config: nk_font_config(height); - cfg.ttf_blob = memory; - cfg.ttf_size = size; - cfg.size = height; - cfg.ttf_data_owned_by_atlas = 0; - return nk_font_atlas_add(atlas, &cfg); -} -#ifdef NK_INCLUDE_STANDARD_IO -NK_API struct nk_font* -nk_font_atlas_add_from_file(struct nk_font_atlas *atlas, const char *file_path, - float height, const struct nk_font_config *config) -{ - nk_size size; - char *memory; - struct nk_font_config cfg; - - NK_ASSERT(atlas); - NK_ASSERT(atlas->temporary.alloc); - NK_ASSERT(atlas->temporary.free); - NK_ASSERT(atlas->permanent.alloc); - NK_ASSERT(atlas->permanent.free); - - if (!atlas || !file_path) return 0; - memory = nk_file_load(file_path, &size, &atlas->permanent); - if (!memory) return 0; - - cfg = (config) ? *config: nk_font_config(height); - cfg.ttf_blob = memory; - cfg.ttf_size = size; - cfg.size = height; - cfg.ttf_data_owned_by_atlas = 1; - return nk_font_atlas_add(atlas, &cfg); -} -#endif -NK_API struct nk_font* -nk_font_atlas_add_compressed(struct nk_font_atlas *atlas, - void *compressed_data, nk_size compressed_size, float height, - const struct nk_font_config *config) -{ - unsigned int decompressed_size; - void *decompressed_data; - struct nk_font_config cfg; - - NK_ASSERT(atlas); - NK_ASSERT(atlas->temporary.alloc); - NK_ASSERT(atlas->temporary.free); - NK_ASSERT(atlas->permanent.alloc); - NK_ASSERT(atlas->permanent.free); - - NK_ASSERT(compressed_data); - NK_ASSERT(compressed_size); - if (!atlas || !compressed_data || !atlas->temporary.alloc || !atlas->temporary.free || - !atlas->permanent.alloc || !atlas->permanent.free) - return 0; - - decompressed_size = nk_decompress_length((unsigned char*)compressed_data); - decompressed_data = atlas->permanent.alloc(atlas->permanent.userdata,0,decompressed_size); - NK_ASSERT(decompressed_data); - if (!decompressed_data) return 0; - nk_decompress((unsigned char*)decompressed_data, (unsigned char*)compressed_data, - (unsigned int)compressed_size); - - cfg = (config) ? *config: nk_font_config(height); - cfg.ttf_blob = decompressed_data; - cfg.ttf_size = decompressed_size; - cfg.size = height; - cfg.ttf_data_owned_by_atlas = 1; - return nk_font_atlas_add(atlas, &cfg); -} -NK_API struct nk_font* -nk_font_atlas_add_compressed_base85(struct nk_font_atlas *atlas, - const char *data_base85, float height, const struct nk_font_config *config) -{ - int compressed_size; - void *compressed_data; - struct nk_font *font; - - NK_ASSERT(atlas); - NK_ASSERT(atlas->temporary.alloc); - NK_ASSERT(atlas->temporary.free); - NK_ASSERT(atlas->permanent.alloc); - NK_ASSERT(atlas->permanent.free); - - NK_ASSERT(data_base85); - if (!atlas || !data_base85 || !atlas->temporary.alloc || !atlas->temporary.free || - !atlas->permanent.alloc || !atlas->permanent.free) - return 0; - - compressed_size = (((int)nk_strlen(data_base85) + 4) / 5) * 4; - compressed_data = atlas->temporary.alloc(atlas->temporary.userdata,0, (nk_size)compressed_size); - NK_ASSERT(compressed_data); - if (!compressed_data) return 0; - nk_decode_85((unsigned char*)compressed_data, (const unsigned char*)data_base85); - font = nk_font_atlas_add_compressed(atlas, compressed_data, - (nk_size)compressed_size, height, config); - atlas->temporary.free(atlas->temporary.userdata, compressed_data); - return font; -} - -#ifdef NK_INCLUDE_DEFAULT_FONT -NK_API struct nk_font* -nk_font_atlas_add_default(struct nk_font_atlas *atlas, - float pixel_height, const struct nk_font_config *config) -{ - NK_ASSERT(atlas); - NK_ASSERT(atlas->temporary.alloc); - NK_ASSERT(atlas->temporary.free); - NK_ASSERT(atlas->permanent.alloc); - NK_ASSERT(atlas->permanent.free); - return nk_font_atlas_add_compressed_base85(atlas, - nk_proggy_clean_ttf_compressed_data_base85, pixel_height, config); -} -#endif -NK_API const void* -nk_font_atlas_bake(struct nk_font_atlas *atlas, int *width, int *height, - enum nk_font_atlas_format fmt) -{ - int i = 0; - void *tmp = 0; - nk_size tmp_size, img_size; - struct nk_font *font_iter; - struct nk_font_baker *baker; - - NK_ASSERT(atlas); - NK_ASSERT(atlas->temporary.alloc); - NK_ASSERT(atlas->temporary.free); - NK_ASSERT(atlas->permanent.alloc); - NK_ASSERT(atlas->permanent.free); - - NK_ASSERT(width); - NK_ASSERT(height); - if (!atlas || !width || !height || - !atlas->temporary.alloc || !atlas->temporary.free || - !atlas->permanent.alloc || !atlas->permanent.free) - return 0; - -#ifdef NK_INCLUDE_DEFAULT_FONT - /* no font added so just use default font */ - if (!atlas->font_num) - atlas->default_font = nk_font_atlas_add_default(atlas, 13.0f, 0); -#endif - NK_ASSERT(atlas->font_num); - if (!atlas->font_num) return 0; - - /* allocate temporary baker memory required for the baking process */ - nk_font_baker_memory(&tmp_size, &atlas->glyph_count, atlas->config, atlas->font_num); - tmp = atlas->temporary.alloc(atlas->temporary.userdata,0, tmp_size); - NK_ASSERT(tmp); - if (!tmp) goto failed; - NK_MEMSET(tmp,0,tmp_size); - - /* allocate glyph memory for all fonts */ - baker = nk_font_baker(tmp, atlas->glyph_count, atlas->font_num, &atlas->temporary); - atlas->glyphs = (struct nk_font_glyph*)atlas->permanent.alloc( - atlas->permanent.userdata,0, sizeof(struct nk_font_glyph)*(nk_size)atlas->glyph_count); - NK_ASSERT(atlas->glyphs); - if (!atlas->glyphs) - goto failed; - - /* pack all glyphs into a tight fit space */ - atlas->custom.w = (NK_CURSOR_DATA_W*2)+1; - atlas->custom.h = NK_CURSOR_DATA_H + 1; - if (!nk_font_bake_pack(baker, &img_size, width, height, &atlas->custom, - atlas->config, atlas->font_num, &atlas->temporary)) - goto failed; - - /* allocate memory for the baked image font atlas */ - atlas->pixel = atlas->temporary.alloc(atlas->temporary.userdata,0, img_size); - NK_ASSERT(atlas->pixel); - if (!atlas->pixel) - goto failed; - - /* bake glyphs and custom white pixel into image */ - nk_font_bake(baker, atlas->pixel, *width, *height, - atlas->glyphs, atlas->glyph_count, atlas->config, atlas->font_num); - nk_font_bake_custom_data(atlas->pixel, *width, *height, atlas->custom, - nk_custom_cursor_data, NK_CURSOR_DATA_W, NK_CURSOR_DATA_H, '.', 'X'); - - if (fmt == NK_FONT_ATLAS_RGBA32) { - /* convert alpha8 image into rgba32 image */ - void *img_rgba = atlas->temporary.alloc(atlas->temporary.userdata,0, - (nk_size)(*width * *height * 4)); - NK_ASSERT(img_rgba); - if (!img_rgba) goto failed; - nk_font_bake_convert(img_rgba, *width, *height, atlas->pixel); - atlas->temporary.free(atlas->temporary.userdata, atlas->pixel); - atlas->pixel = img_rgba; - } - atlas->tex_width = *width; - atlas->tex_height = *height; - - /* initialize each font */ - for (font_iter = atlas->fonts; font_iter; font_iter = font_iter->next) { - struct nk_font *font = font_iter; - struct nk_font_config *config = font->config; - nk_font_init(font, config->size, config->fallback_glyph, atlas->glyphs, - config->font, nk_handle_ptr(0)); - } - - /* initialize each cursor */ - {NK_STORAGE const struct nk_vec2 nk_cursor_data[NK_CURSOR_COUNT][3] = { - /* Pos Size Offset */ - {{ 0, 3}, {12,19}, { 0, 0}}, - {{13, 0}, { 7,16}, { 4, 8}}, - {{31, 0}, {23,23}, {11,11}}, - {{21, 0}, { 9, 23}, { 5,11}}, - {{55,18}, {23, 9}, {11, 5}}, - {{73, 0}, {17,17}, { 9, 9}}, - {{55, 0}, {17,17}, { 9, 9}} - }; - for (i = 0; i < NK_CURSOR_COUNT; ++i) { - struct nk_cursor *cursor = &atlas->cursors[i]; - cursor->img.w = (unsigned short)*width; - cursor->img.h = (unsigned short)*height; - cursor->img.region[0] = (unsigned short)(atlas->custom.x + nk_cursor_data[i][0].x); - cursor->img.region[1] = (unsigned short)(atlas->custom.y + nk_cursor_data[i][0].y); - cursor->img.region[2] = (unsigned short)nk_cursor_data[i][1].x; - cursor->img.region[3] = (unsigned short)nk_cursor_data[i][1].y; - cursor->size = nk_cursor_data[i][1]; - cursor->offset = nk_cursor_data[i][2]; - }} - /* free temporary memory */ - atlas->temporary.free(atlas->temporary.userdata, tmp); - return atlas->pixel; - -failed: - /* error so cleanup all memory */ - if (tmp) atlas->temporary.free(atlas->temporary.userdata, tmp); - if (atlas->glyphs) { - atlas->permanent.free(atlas->permanent.userdata, atlas->glyphs); - atlas->glyphs = 0; - } - if (atlas->pixel) { - atlas->temporary.free(atlas->temporary.userdata, atlas->pixel); - atlas->pixel = 0; - } - return 0; -} -NK_API void -nk_font_atlas_end(struct nk_font_atlas *atlas, nk_handle texture, - struct nk_draw_null_texture *tex_null) -{ - int i = 0; - struct nk_font *font_iter; - NK_ASSERT(atlas); - if (!atlas) { - if (!tex_null) return; - tex_null->texture = texture; - tex_null->uv = nk_vec2(0.5f,0.5f); - } - if (tex_null) { - tex_null->texture = texture; - tex_null->uv.x = (atlas->custom.x + 0.5f)/(float)atlas->tex_width; - tex_null->uv.y = (atlas->custom.y + 0.5f)/(float)atlas->tex_height; - } - for (font_iter = atlas->fonts; font_iter; font_iter = font_iter->next) { - font_iter->texture = texture; -#ifdef NK_INCLUDE_VERTEX_BUFFER_OUTPUT - font_iter->handle.texture = texture; -#endif - } - for (i = 0; i < NK_CURSOR_COUNT; ++i) - atlas->cursors[i].img.handle = texture; - - atlas->temporary.free(atlas->temporary.userdata, atlas->pixel); - atlas->pixel = 0; - atlas->tex_width = 0; - atlas->tex_height = 0; - atlas->custom.x = 0; - atlas->custom.y = 0; - atlas->custom.w = 0; - atlas->custom.h = 0; -} -NK_API void -nk_font_atlas_cleanup(struct nk_font_atlas *atlas) -{ - NK_ASSERT(atlas); - NK_ASSERT(atlas->temporary.alloc); - NK_ASSERT(atlas->temporary.free); - NK_ASSERT(atlas->permanent.alloc); - NK_ASSERT(atlas->permanent.free); - if (!atlas || !atlas->permanent.alloc || !atlas->permanent.free) return; - if (atlas->config) { - struct nk_font_config *iter; - for (iter = atlas->config; iter; iter = iter->next) { - struct nk_font_config *i; - for (i = iter->n; i != iter; i = i->n) { - atlas->permanent.free(atlas->permanent.userdata, i->ttf_blob); - i->ttf_blob = 0; - } - atlas->permanent.free(atlas->permanent.userdata, iter->ttf_blob); - iter->ttf_blob = 0; - } - } -} -NK_API void -nk_font_atlas_clear(struct nk_font_atlas *atlas) -{ - NK_ASSERT(atlas); - NK_ASSERT(atlas->temporary.alloc); - NK_ASSERT(atlas->temporary.free); - NK_ASSERT(atlas->permanent.alloc); - NK_ASSERT(atlas->permanent.free); - if (!atlas || !atlas->permanent.alloc || !atlas->permanent.free) return; - - if (atlas->config) { - struct nk_font_config *iter, *next; - for (iter = atlas->config; iter; iter = next) { - struct nk_font_config *i, *n; - for (i = iter->n; i != iter; i = n) { - n = i->n; - if (i->ttf_blob) - atlas->permanent.free(atlas->permanent.userdata, i->ttf_blob); - atlas->permanent.free(atlas->permanent.userdata, i); - } - next = iter->next; - if (i->ttf_blob) - atlas->permanent.free(atlas->permanent.userdata, iter->ttf_blob); - atlas->permanent.free(atlas->permanent.userdata, iter); - } - atlas->config = 0; - } - if (atlas->fonts) { - struct nk_font *iter, *next; - for (iter = atlas->fonts; iter; iter = next) { - next = iter->next; - atlas->permanent.free(atlas->permanent.userdata, iter); - } - atlas->fonts = 0; - } - if (atlas->glyphs) - atlas->permanent.free(atlas->permanent.userdata, atlas->glyphs); - nk_zero_struct(*atlas); -} -#endif - - - - - -/* =============================================================== - * - * INPUT - * - * ===============================================================*/ -NK_API void -nk_input_begin(struct nk_context *ctx) -{ - int i; - struct nk_input *in; - NK_ASSERT(ctx); - if (!ctx) return; - in = &ctx->input; - for (i = 0; i < NK_BUTTON_MAX; ++i) - in->mouse.buttons[i].clicked = 0; - - in->keyboard.text_len = 0; - in->mouse.scroll_delta = nk_vec2(0,0); - in->mouse.prev.x = in->mouse.pos.x; - in->mouse.prev.y = in->mouse.pos.y; - in->mouse.delta.x = 0; - in->mouse.delta.y = 0; - for (i = 0; i < NK_KEY_MAX; i++) - in->keyboard.keys[i].clicked = 0; -} -NK_API void -nk_input_end(struct nk_context *ctx) -{ - struct nk_input *in; - NK_ASSERT(ctx); - if (!ctx) return; - in = &ctx->input; - if (in->mouse.grab) - in->mouse.grab = 0; - if (in->mouse.ungrab) { - in->mouse.grabbed = 0; - in->mouse.ungrab = 0; - in->mouse.grab = 0; - } -} -NK_API void -nk_input_motion(struct nk_context *ctx, int x, int y) -{ - struct nk_input *in; - NK_ASSERT(ctx); - if (!ctx) return; - in = &ctx->input; - in->mouse.pos.x = (float)x; - in->mouse.pos.y = (float)y; - in->mouse.delta.x = in->mouse.pos.x - in->mouse.prev.x; - in->mouse.delta.y = in->mouse.pos.y - in->mouse.prev.y; -} -NK_API void -nk_input_key(struct nk_context *ctx, enum nk_keys key, nk_bool down) -{ - struct nk_input *in; - NK_ASSERT(ctx); - if (!ctx) return; - in = &ctx->input; -#ifdef NK_KEYSTATE_BASED_INPUT - if (in->keyboard.keys[key].down != down) - in->keyboard.keys[key].clicked++; -#else - in->keyboard.keys[key].clicked++; -#endif - in->keyboard.keys[key].down = down; -} -NK_API void -nk_input_button(struct nk_context *ctx, enum nk_buttons id, int x, int y, nk_bool down) -{ - struct nk_mouse_button *btn; - struct nk_input *in; - NK_ASSERT(ctx); - if (!ctx) return; - in = &ctx->input; - if (in->mouse.buttons[id].down == down) return; - - btn = &in->mouse.buttons[id]; - btn->clicked_pos.x = (float)x; - btn->clicked_pos.y = (float)y; - btn->down = down; - btn->clicked++; - - /* Fix Click-Drag for touch events. */ - in->mouse.delta.x = 0; - in->mouse.delta.y = 0; -#ifdef NK_BUTTON_TRIGGER_ON_RELEASE - if (down == 1 && id == NK_BUTTON_LEFT) - { - in->mouse.down_pos.x = btn->clicked_pos.x; - in->mouse.down_pos.y = btn->clicked_pos.y; - } -#endif -} -NK_API void -nk_input_scroll(struct nk_context *ctx, struct nk_vec2 val) -{ - NK_ASSERT(ctx); - if (!ctx) return; - ctx->input.mouse.scroll_delta.x += val.x; - ctx->input.mouse.scroll_delta.y += val.y; -} -NK_API void -nk_input_glyph(struct nk_context *ctx, const nk_glyph glyph) -{ - int len = 0; - nk_rune unicode; - struct nk_input *in; - - NK_ASSERT(ctx); - if (!ctx) return; - in = &ctx->input; - - len = nk_utf_decode(glyph, &unicode, NK_UTF_SIZE); - if (len && ((in->keyboard.text_len + len) < NK_INPUT_MAX)) { - nk_utf_encode(unicode, &in->keyboard.text[in->keyboard.text_len], - NK_INPUT_MAX - in->keyboard.text_len); - in->keyboard.text_len += len; - } -} -NK_API void -nk_input_char(struct nk_context *ctx, char c) -{ - nk_glyph glyph; - NK_ASSERT(ctx); - if (!ctx) return; - glyph[0] = c; - nk_input_glyph(ctx, glyph); -} -NK_API void -nk_input_unicode(struct nk_context *ctx, nk_rune unicode) -{ - nk_glyph rune; - NK_ASSERT(ctx); - if (!ctx) return; - nk_utf_encode(unicode, rune, NK_UTF_SIZE); - nk_input_glyph(ctx, rune); -} -NK_API nk_bool -nk_input_has_mouse_click(const struct nk_input *i, enum nk_buttons id) -{ - const struct nk_mouse_button *btn; - if (!i) return nk_false; - btn = &i->mouse.buttons[id]; - return (btn->clicked && btn->down == nk_false) ? nk_true : nk_false; -} -NK_API nk_bool -nk_input_has_mouse_click_in_rect(const struct nk_input *i, enum nk_buttons id, - struct nk_rect b) -{ - const struct nk_mouse_button *btn; - if (!i) return nk_false; - btn = &i->mouse.buttons[id]; - if (!NK_INBOX(btn->clicked_pos.x,btn->clicked_pos.y,b.x,b.y,b.w,b.h)) - return nk_false; - return nk_true; -} -NK_API nk_bool -nk_input_has_mouse_click_in_button_rect(const struct nk_input *i, enum nk_buttons id, - struct nk_rect b) -{ - const struct nk_mouse_button *btn; - if (!i) return nk_false; - btn = &i->mouse.buttons[id]; -#ifdef NK_BUTTON_TRIGGER_ON_RELEASE - if (!NK_INBOX(btn->clicked_pos.x,btn->clicked_pos.y,b.x,b.y,b.w,b.h) - || !NK_INBOX(i->mouse.down_pos.x,i->mouse.down_pos.y,b.x,b.y,b.w,b.h)) -#else - if (!NK_INBOX(btn->clicked_pos.x,btn->clicked_pos.y,b.x,b.y,b.w,b.h)) -#endif - return nk_false; - return nk_true; -} -NK_API nk_bool -nk_input_has_mouse_click_down_in_rect(const struct nk_input *i, enum nk_buttons id, - struct nk_rect b, nk_bool down) -{ - const struct nk_mouse_button *btn; - if (!i) return nk_false; - btn = &i->mouse.buttons[id]; - return nk_input_has_mouse_click_in_rect(i, id, b) && (btn->down == down); -} -NK_API nk_bool -nk_input_is_mouse_click_in_rect(const struct nk_input *i, enum nk_buttons id, - struct nk_rect b) -{ - const struct nk_mouse_button *btn; - if (!i) return nk_false; - btn = &i->mouse.buttons[id]; - return (nk_input_has_mouse_click_down_in_rect(i, id, b, nk_false) && - btn->clicked) ? nk_true : nk_false; -} -NK_API nk_bool -nk_input_is_mouse_click_down_in_rect(const struct nk_input *i, enum nk_buttons id, - struct nk_rect b, nk_bool down) -{ - const struct nk_mouse_button *btn; - if (!i) return nk_false; - btn = &i->mouse.buttons[id]; - return (nk_input_has_mouse_click_down_in_rect(i, id, b, down) && - btn->clicked) ? nk_true : nk_false; -} -NK_API nk_bool -nk_input_any_mouse_click_in_rect(const struct nk_input *in, struct nk_rect b) -{ - int i, down = 0; - for (i = 0; i < NK_BUTTON_MAX; ++i) - down = down || nk_input_is_mouse_click_in_rect(in, (enum nk_buttons)i, b); - return down; -} -NK_API nk_bool -nk_input_is_mouse_hovering_rect(const struct nk_input *i, struct nk_rect rect) -{ - if (!i) return nk_false; - return NK_INBOX(i->mouse.pos.x, i->mouse.pos.y, rect.x, rect.y, rect.w, rect.h); -} -NK_API nk_bool -nk_input_is_mouse_prev_hovering_rect(const struct nk_input *i, struct nk_rect rect) -{ - if (!i) return nk_false; - return NK_INBOX(i->mouse.prev.x, i->mouse.prev.y, rect.x, rect.y, rect.w, rect.h); -} -NK_API nk_bool -nk_input_mouse_clicked(const struct nk_input *i, enum nk_buttons id, struct nk_rect rect) -{ - if (!i) return nk_false; - if (!nk_input_is_mouse_hovering_rect(i, rect)) return nk_false; - return nk_input_is_mouse_click_in_rect(i, id, rect); -} -NK_API nk_bool -nk_input_is_mouse_down(const struct nk_input *i, enum nk_buttons id) -{ - if (!i) return nk_false; - return i->mouse.buttons[id].down; -} -NK_API nk_bool -nk_input_is_mouse_pressed(const struct nk_input *i, enum nk_buttons id) -{ - const struct nk_mouse_button *b; - if (!i) return nk_false; - b = &i->mouse.buttons[id]; - if (b->down && b->clicked) - return nk_true; - return nk_false; -} -NK_API nk_bool -nk_input_is_mouse_released(const struct nk_input *i, enum nk_buttons id) -{ - if (!i) return nk_false; - return (!i->mouse.buttons[id].down && i->mouse.buttons[id].clicked); -} -NK_API nk_bool -nk_input_is_key_pressed(const struct nk_input *i, enum nk_keys key) -{ - const struct nk_key *k; - if (!i) return nk_false; - k = &i->keyboard.keys[key]; - if ((k->down && k->clicked) || (!k->down && k->clicked >= 2)) - return nk_true; - return nk_false; -} -NK_API nk_bool -nk_input_is_key_released(const struct nk_input *i, enum nk_keys key) -{ - const struct nk_key *k; - if (!i) return nk_false; - k = &i->keyboard.keys[key]; - if ((!k->down && k->clicked) || (k->down && k->clicked >= 2)) - return nk_true; - return nk_false; -} -NK_API nk_bool -nk_input_is_key_down(const struct nk_input *i, enum nk_keys key) -{ - const struct nk_key *k; - if (!i) return nk_false; - k = &i->keyboard.keys[key]; - if (k->down) return nk_true; - return nk_false; -} - - - - - -/* =============================================================== - * - * STYLE - * - * ===============================================================*/ -NK_API void nk_style_default(struct nk_context *ctx){nk_style_from_table(ctx, 0);} -#define NK_COLOR_MAP(NK_COLOR)\ - NK_COLOR(NK_COLOR_TEXT, 175,175,175,255) \ - NK_COLOR(NK_COLOR_WINDOW, 45, 45, 45, 255) \ - NK_COLOR(NK_COLOR_HEADER, 40, 40, 40, 255) \ - NK_COLOR(NK_COLOR_BORDER, 65, 65, 65, 255) \ - NK_COLOR(NK_COLOR_BUTTON, 50, 50, 50, 255) \ - NK_COLOR(NK_COLOR_BUTTON_HOVER, 40, 40, 40, 255) \ - NK_COLOR(NK_COLOR_BUTTON_ACTIVE, 35, 35, 35, 255) \ - NK_COLOR(NK_COLOR_TOGGLE, 100,100,100,255) \ - NK_COLOR(NK_COLOR_TOGGLE_HOVER, 120,120,120,255) \ - NK_COLOR(NK_COLOR_TOGGLE_CURSOR, 45, 45, 45, 255) \ - NK_COLOR(NK_COLOR_SELECT, 45, 45, 45, 255) \ - NK_COLOR(NK_COLOR_SELECT_ACTIVE, 35, 35, 35,255) \ - NK_COLOR(NK_COLOR_SLIDER, 38, 38, 38, 255) \ - NK_COLOR(NK_COLOR_SLIDER_CURSOR, 100,100,100,255) \ - NK_COLOR(NK_COLOR_SLIDER_CURSOR_HOVER, 120,120,120,255) \ - NK_COLOR(NK_COLOR_SLIDER_CURSOR_ACTIVE, 150,150,150,255) \ - NK_COLOR(NK_COLOR_PROPERTY, 38, 38, 38, 255) \ - NK_COLOR(NK_COLOR_EDIT, 38, 38, 38, 255) \ - NK_COLOR(NK_COLOR_EDIT_CURSOR, 175,175,175,255) \ - NK_COLOR(NK_COLOR_COMBO, 45, 45, 45, 255) \ - NK_COLOR(NK_COLOR_CHART, 120,120,120,255) \ - NK_COLOR(NK_COLOR_CHART_COLOR, 45, 45, 45, 255) \ - NK_COLOR(NK_COLOR_CHART_COLOR_HIGHLIGHT, 255, 0, 0, 255) \ - NK_COLOR(NK_COLOR_SCROLLBAR, 40, 40, 40, 255) \ - NK_COLOR(NK_COLOR_SCROLLBAR_CURSOR, 100,100,100,255) \ - NK_COLOR(NK_COLOR_SCROLLBAR_CURSOR_HOVER, 120,120,120,255) \ - NK_COLOR(NK_COLOR_SCROLLBAR_CURSOR_ACTIVE, 150,150,150,255) \ - NK_COLOR(NK_COLOR_TAB_HEADER, 40, 40, 40,255) - -NK_GLOBAL const struct nk_color -nk_default_color_style[NK_COLOR_COUNT] = { -#define NK_COLOR(a,b,c,d,e) {b,c,d,e}, - NK_COLOR_MAP(NK_COLOR) -#undef NK_COLOR -}; -NK_GLOBAL const char *nk_color_names[NK_COLOR_COUNT] = { -#define NK_COLOR(a,b,c,d,e) #a, - NK_COLOR_MAP(NK_COLOR) -#undef NK_COLOR -}; - -NK_API const char* -nk_style_get_color_by_name(enum nk_style_colors c) -{ - return nk_color_names[c]; -} -NK_API struct nk_style_item -nk_style_item_color(struct nk_color col) -{ - struct nk_style_item i; - i.type = NK_STYLE_ITEM_COLOR; - i.data.color = col; - return i; -} -NK_API struct nk_style_item -nk_style_item_image(struct nk_image img) -{ - struct nk_style_item i; - i.type = NK_STYLE_ITEM_IMAGE; - i.data.image = img; - return i; -} -NK_API struct nk_style_item -nk_style_item_nine_slice(struct nk_nine_slice slice) -{ - struct nk_style_item i; - i.type = NK_STYLE_ITEM_NINE_SLICE; - i.data.slice = slice; - return i; -} -NK_API struct nk_style_item -nk_style_item_hide(void) -{ - struct nk_style_item i; - i.type = NK_STYLE_ITEM_COLOR; - i.data.color = nk_rgba(0,0,0,0); - return i; -} -NK_API void -nk_style_from_table(struct nk_context *ctx, const struct nk_color *table) -{ - struct nk_style *style; - struct nk_style_text *text; - struct nk_style_button *button; - struct nk_style_toggle *toggle; - struct nk_style_selectable *select; - struct nk_style_slider *slider; - struct nk_style_progress *prog; - struct nk_style_scrollbar *scroll; - struct nk_style_edit *edit; - struct nk_style_property *property; - struct nk_style_combo *combo; - struct nk_style_chart *chart; - struct nk_style_tab *tab; - struct nk_style_window *win; - - NK_ASSERT(ctx); - if (!ctx) return; - style = &ctx->style; - table = (!table) ? nk_default_color_style: table; - - /* default text */ - text = &style->text; - text->color = table[NK_COLOR_TEXT]; - text->padding = nk_vec2(0,0); - - /* default button */ - button = &style->button; - nk_zero_struct(*button); - button->normal = nk_style_item_color(table[NK_COLOR_BUTTON]); - button->hover = nk_style_item_color(table[NK_COLOR_BUTTON_HOVER]); - button->active = nk_style_item_color(table[NK_COLOR_BUTTON_ACTIVE]); - button->border_color = table[NK_COLOR_BORDER]; - button->text_background = table[NK_COLOR_BUTTON]; - button->text_normal = table[NK_COLOR_TEXT]; - button->text_hover = table[NK_COLOR_TEXT]; - button->text_active = table[NK_COLOR_TEXT]; - button->padding = nk_vec2(2.0f,2.0f); - button->image_padding = nk_vec2(0.0f,0.0f); - button->touch_padding = nk_vec2(0.0f, 0.0f); - button->userdata = nk_handle_ptr(0); - button->text_alignment = NK_TEXT_CENTERED; - button->border = 1.0f; - button->rounding = 4.0f; - button->draw_begin = 0; - button->draw_end = 0; - - /* contextual button */ - button = &style->contextual_button; - nk_zero_struct(*button); - button->normal = nk_style_item_color(table[NK_COLOR_WINDOW]); - button->hover = nk_style_item_color(table[NK_COLOR_BUTTON_HOVER]); - button->active = nk_style_item_color(table[NK_COLOR_BUTTON_ACTIVE]); - button->border_color = table[NK_COLOR_WINDOW]; - button->text_background = table[NK_COLOR_WINDOW]; - button->text_normal = table[NK_COLOR_TEXT]; - button->text_hover = table[NK_COLOR_TEXT]; - button->text_active = table[NK_COLOR_TEXT]; - button->padding = nk_vec2(2.0f,2.0f); - button->touch_padding = nk_vec2(0.0f,0.0f); - button->userdata = nk_handle_ptr(0); - button->text_alignment = NK_TEXT_CENTERED; - button->border = 0.0f; - button->rounding = 0.0f; - button->draw_begin = 0; - button->draw_end = 0; - - /* menu button */ - button = &style->menu_button; - nk_zero_struct(*button); - button->normal = nk_style_item_color(table[NK_COLOR_WINDOW]); - button->hover = nk_style_item_color(table[NK_COLOR_WINDOW]); - button->active = nk_style_item_color(table[NK_COLOR_WINDOW]); - button->border_color = table[NK_COLOR_WINDOW]; - button->text_background = table[NK_COLOR_WINDOW]; - button->text_normal = table[NK_COLOR_TEXT]; - button->text_hover = table[NK_COLOR_TEXT]; - button->text_active = table[NK_COLOR_TEXT]; - button->padding = nk_vec2(2.0f,2.0f); - button->touch_padding = nk_vec2(0.0f,0.0f); - button->userdata = nk_handle_ptr(0); - button->text_alignment = NK_TEXT_CENTERED; - button->border = 0.0f; - button->rounding = 1.0f; - button->draw_begin = 0; - button->draw_end = 0; - - /* checkbox toggle */ - toggle = &style->checkbox; - nk_zero_struct(*toggle); - toggle->normal = nk_style_item_color(table[NK_COLOR_TOGGLE]); - toggle->hover = nk_style_item_color(table[NK_COLOR_TOGGLE_HOVER]); - toggle->active = nk_style_item_color(table[NK_COLOR_TOGGLE_HOVER]); - toggle->cursor_normal = nk_style_item_color(table[NK_COLOR_TOGGLE_CURSOR]); - toggle->cursor_hover = nk_style_item_color(table[NK_COLOR_TOGGLE_CURSOR]); - toggle->userdata = nk_handle_ptr(0); - toggle->text_background = table[NK_COLOR_WINDOW]; - toggle->text_normal = table[NK_COLOR_TEXT]; - toggle->text_hover = table[NK_COLOR_TEXT]; - toggle->text_active = table[NK_COLOR_TEXT]; - toggle->padding = nk_vec2(2.0f, 2.0f); - toggle->touch_padding = nk_vec2(0,0); - toggle->border_color = nk_rgba(0,0,0,0); - toggle->border = 0.0f; - toggle->spacing = 4; - - /* option toggle */ - toggle = &style->option; - nk_zero_struct(*toggle); - toggle->normal = nk_style_item_color(table[NK_COLOR_TOGGLE]); - toggle->hover = nk_style_item_color(table[NK_COLOR_TOGGLE_HOVER]); - toggle->active = nk_style_item_color(table[NK_COLOR_TOGGLE_HOVER]); - toggle->cursor_normal = nk_style_item_color(table[NK_COLOR_TOGGLE_CURSOR]); - toggle->cursor_hover = nk_style_item_color(table[NK_COLOR_TOGGLE_CURSOR]); - toggle->userdata = nk_handle_ptr(0); - toggle->text_background = table[NK_COLOR_WINDOW]; - toggle->text_normal = table[NK_COLOR_TEXT]; - toggle->text_hover = table[NK_COLOR_TEXT]; - toggle->text_active = table[NK_COLOR_TEXT]; - toggle->padding = nk_vec2(3.0f, 3.0f); - toggle->touch_padding = nk_vec2(0,0); - toggle->border_color = nk_rgba(0,0,0,0); - toggle->border = 0.0f; - toggle->spacing = 4; - - /* selectable */ - select = &style->selectable; - nk_zero_struct(*select); - select->normal = nk_style_item_color(table[NK_COLOR_SELECT]); - select->hover = nk_style_item_color(table[NK_COLOR_SELECT]); - select->pressed = nk_style_item_color(table[NK_COLOR_SELECT]); - select->normal_active = nk_style_item_color(table[NK_COLOR_SELECT_ACTIVE]); - select->hover_active = nk_style_item_color(table[NK_COLOR_SELECT_ACTIVE]); - select->pressed_active = nk_style_item_color(table[NK_COLOR_SELECT_ACTIVE]); - select->text_normal = table[NK_COLOR_TEXT]; - select->text_hover = table[NK_COLOR_TEXT]; - select->text_pressed = table[NK_COLOR_TEXT]; - select->text_normal_active = table[NK_COLOR_TEXT]; - select->text_hover_active = table[NK_COLOR_TEXT]; - select->text_pressed_active = table[NK_COLOR_TEXT]; - select->padding = nk_vec2(2.0f,2.0f); - select->image_padding = nk_vec2(2.0f,2.0f); - select->touch_padding = nk_vec2(0,0); - select->userdata = nk_handle_ptr(0); - select->rounding = 0.0f; - select->draw_begin = 0; - select->draw_end = 0; - - /* slider */ - slider = &style->slider; - nk_zero_struct(*slider); - slider->normal = nk_style_item_hide(); - slider->hover = nk_style_item_hide(); - slider->active = nk_style_item_hide(); - slider->bar_normal = table[NK_COLOR_SLIDER]; - slider->bar_hover = table[NK_COLOR_SLIDER]; - slider->bar_active = table[NK_COLOR_SLIDER]; - slider->bar_filled = table[NK_COLOR_SLIDER_CURSOR]; - slider->cursor_normal = nk_style_item_color(table[NK_COLOR_SLIDER_CURSOR]); - slider->cursor_hover = nk_style_item_color(table[NK_COLOR_SLIDER_CURSOR_HOVER]); - slider->cursor_active = nk_style_item_color(table[NK_COLOR_SLIDER_CURSOR_ACTIVE]); - slider->inc_symbol = NK_SYMBOL_TRIANGLE_RIGHT; - slider->dec_symbol = NK_SYMBOL_TRIANGLE_LEFT; - slider->cursor_size = nk_vec2(16,16); - slider->padding = nk_vec2(2,2); - slider->spacing = nk_vec2(2,2); - slider->userdata = nk_handle_ptr(0); - slider->show_buttons = nk_false; - slider->bar_height = 8; - slider->rounding = 0; - slider->draw_begin = 0; - slider->draw_end = 0; - - /* slider buttons */ - button = &style->slider.inc_button; - button->normal = nk_style_item_color(nk_rgb(40,40,40)); - button->hover = nk_style_item_color(nk_rgb(42,42,42)); - button->active = nk_style_item_color(nk_rgb(44,44,44)); - button->border_color = nk_rgb(65,65,65); - button->text_background = nk_rgb(40,40,40); - button->text_normal = nk_rgb(175,175,175); - button->text_hover = nk_rgb(175,175,175); - button->text_active = nk_rgb(175,175,175); - button->padding = nk_vec2(8.0f,8.0f); - button->touch_padding = nk_vec2(0.0f,0.0f); - button->userdata = nk_handle_ptr(0); - button->text_alignment = NK_TEXT_CENTERED; - button->border = 1.0f; - button->rounding = 0.0f; - button->draw_begin = 0; - button->draw_end = 0; - style->slider.dec_button = style->slider.inc_button; - - /* progressbar */ - prog = &style->progress; - nk_zero_struct(*prog); - prog->normal = nk_style_item_color(table[NK_COLOR_SLIDER]); - prog->hover = nk_style_item_color(table[NK_COLOR_SLIDER]); - prog->active = nk_style_item_color(table[NK_COLOR_SLIDER]); - prog->cursor_normal = nk_style_item_color(table[NK_COLOR_SLIDER_CURSOR]); - prog->cursor_hover = nk_style_item_color(table[NK_COLOR_SLIDER_CURSOR_HOVER]); - prog->cursor_active = nk_style_item_color(table[NK_COLOR_SLIDER_CURSOR_ACTIVE]); - prog->border_color = nk_rgba(0,0,0,0); - prog->cursor_border_color = nk_rgba(0,0,0,0); - prog->userdata = nk_handle_ptr(0); - prog->padding = nk_vec2(4,4); - prog->rounding = 0; - prog->border = 0; - prog->cursor_rounding = 0; - prog->cursor_border = 0; - prog->draw_begin = 0; - prog->draw_end = 0; - - /* scrollbars */ - scroll = &style->scrollh; - nk_zero_struct(*scroll); - scroll->normal = nk_style_item_color(table[NK_COLOR_SCROLLBAR]); - scroll->hover = nk_style_item_color(table[NK_COLOR_SCROLLBAR]); - scroll->active = nk_style_item_color(table[NK_COLOR_SCROLLBAR]); - scroll->cursor_normal = nk_style_item_color(table[NK_COLOR_SCROLLBAR_CURSOR]); - scroll->cursor_hover = nk_style_item_color(table[NK_COLOR_SCROLLBAR_CURSOR_HOVER]); - scroll->cursor_active = nk_style_item_color(table[NK_COLOR_SCROLLBAR_CURSOR_ACTIVE]); - scroll->dec_symbol = NK_SYMBOL_CIRCLE_SOLID; - scroll->inc_symbol = NK_SYMBOL_CIRCLE_SOLID; - scroll->userdata = nk_handle_ptr(0); - scroll->border_color = table[NK_COLOR_SCROLLBAR]; - scroll->cursor_border_color = table[NK_COLOR_SCROLLBAR]; - scroll->padding = nk_vec2(0,0); - scroll->show_buttons = nk_false; - scroll->border = 0; - scroll->rounding = 0; - scroll->border_cursor = 0; - scroll->rounding_cursor = 0; - scroll->draw_begin = 0; - scroll->draw_end = 0; - style->scrollv = style->scrollh; - - /* scrollbars buttons */ - button = &style->scrollh.inc_button; - button->normal = nk_style_item_color(nk_rgb(40,40,40)); - button->hover = nk_style_item_color(nk_rgb(42,42,42)); - button->active = nk_style_item_color(nk_rgb(44,44,44)); - button->border_color = nk_rgb(65,65,65); - button->text_background = nk_rgb(40,40,40); - button->text_normal = nk_rgb(175,175,175); - button->text_hover = nk_rgb(175,175,175); - button->text_active = nk_rgb(175,175,175); - button->padding = nk_vec2(4.0f,4.0f); - button->touch_padding = nk_vec2(0.0f,0.0f); - button->userdata = nk_handle_ptr(0); - button->text_alignment = NK_TEXT_CENTERED; - button->border = 1.0f; - button->rounding = 0.0f; - button->draw_begin = 0; - button->draw_end = 0; - style->scrollh.dec_button = style->scrollh.inc_button; - style->scrollv.inc_button = style->scrollh.inc_button; - style->scrollv.dec_button = style->scrollh.inc_button; - - /* edit */ - edit = &style->edit; - nk_zero_struct(*edit); - edit->normal = nk_style_item_color(table[NK_COLOR_EDIT]); - edit->hover = nk_style_item_color(table[NK_COLOR_EDIT]); - edit->active = nk_style_item_color(table[NK_COLOR_EDIT]); - edit->cursor_normal = table[NK_COLOR_TEXT]; - edit->cursor_hover = table[NK_COLOR_TEXT]; - edit->cursor_text_normal= table[NK_COLOR_EDIT]; - edit->cursor_text_hover = table[NK_COLOR_EDIT]; - edit->border_color = table[NK_COLOR_BORDER]; - edit->text_normal = table[NK_COLOR_TEXT]; - edit->text_hover = table[NK_COLOR_TEXT]; - edit->text_active = table[NK_COLOR_TEXT]; - edit->selected_normal = table[NK_COLOR_TEXT]; - edit->selected_hover = table[NK_COLOR_TEXT]; - edit->selected_text_normal = table[NK_COLOR_EDIT]; - edit->selected_text_hover = table[NK_COLOR_EDIT]; - edit->scrollbar_size = nk_vec2(10,10); - edit->scrollbar = style->scrollv; - edit->padding = nk_vec2(4,4); - edit->row_padding = 2; - edit->cursor_size = 4; - edit->border = 1; - edit->rounding = 0; - - /* property */ - property = &style->property; - nk_zero_struct(*property); - property->normal = nk_style_item_color(table[NK_COLOR_PROPERTY]); - property->hover = nk_style_item_color(table[NK_COLOR_PROPERTY]); - property->active = nk_style_item_color(table[NK_COLOR_PROPERTY]); - property->border_color = table[NK_COLOR_BORDER]; - property->label_normal = table[NK_COLOR_TEXT]; - property->label_hover = table[NK_COLOR_TEXT]; - property->label_active = table[NK_COLOR_TEXT]; - property->sym_left = NK_SYMBOL_TRIANGLE_LEFT; - property->sym_right = NK_SYMBOL_TRIANGLE_RIGHT; - property->userdata = nk_handle_ptr(0); - property->padding = nk_vec2(4,4); - property->border = 1; - property->rounding = 10; - property->draw_begin = 0; - property->draw_end = 0; - - /* property buttons */ - button = &style->property.dec_button; - nk_zero_struct(*button); - button->normal = nk_style_item_color(table[NK_COLOR_PROPERTY]); - button->hover = nk_style_item_color(table[NK_COLOR_PROPERTY]); - button->active = nk_style_item_color(table[NK_COLOR_PROPERTY]); - button->border_color = nk_rgba(0,0,0,0); - button->text_background = table[NK_COLOR_PROPERTY]; - button->text_normal = table[NK_COLOR_TEXT]; - button->text_hover = table[NK_COLOR_TEXT]; - button->text_active = table[NK_COLOR_TEXT]; - button->padding = nk_vec2(0.0f,0.0f); - button->touch_padding = nk_vec2(0.0f,0.0f); - button->userdata = nk_handle_ptr(0); - button->text_alignment = NK_TEXT_CENTERED; - button->border = 0.0f; - button->rounding = 0.0f; - button->draw_begin = 0; - button->draw_end = 0; - style->property.inc_button = style->property.dec_button; - - /* property edit */ - edit = &style->property.edit; - nk_zero_struct(*edit); - edit->normal = nk_style_item_color(table[NK_COLOR_PROPERTY]); - edit->hover = nk_style_item_color(table[NK_COLOR_PROPERTY]); - edit->active = nk_style_item_color(table[NK_COLOR_PROPERTY]); - edit->border_color = nk_rgba(0,0,0,0); - edit->cursor_normal = table[NK_COLOR_TEXT]; - edit->cursor_hover = table[NK_COLOR_TEXT]; - edit->cursor_text_normal= table[NK_COLOR_EDIT]; - edit->cursor_text_hover = table[NK_COLOR_EDIT]; - edit->text_normal = table[NK_COLOR_TEXT]; - edit->text_hover = table[NK_COLOR_TEXT]; - edit->text_active = table[NK_COLOR_TEXT]; - edit->selected_normal = table[NK_COLOR_TEXT]; - edit->selected_hover = table[NK_COLOR_TEXT]; - edit->selected_text_normal = table[NK_COLOR_EDIT]; - edit->selected_text_hover = table[NK_COLOR_EDIT]; - edit->padding = nk_vec2(0,0); - edit->cursor_size = 8; - edit->border = 0; - edit->rounding = 0; - - /* chart */ - chart = &style->chart; - nk_zero_struct(*chart); - chart->background = nk_style_item_color(table[NK_COLOR_CHART]); - chart->border_color = table[NK_COLOR_BORDER]; - chart->selected_color = table[NK_COLOR_CHART_COLOR_HIGHLIGHT]; - chart->color = table[NK_COLOR_CHART_COLOR]; - chart->padding = nk_vec2(4,4); - chart->border = 0; - chart->rounding = 0; - - /* combo */ - combo = &style->combo; - combo->normal = nk_style_item_color(table[NK_COLOR_COMBO]); - combo->hover = nk_style_item_color(table[NK_COLOR_COMBO]); - combo->active = nk_style_item_color(table[NK_COLOR_COMBO]); - combo->border_color = table[NK_COLOR_BORDER]; - combo->label_normal = table[NK_COLOR_TEXT]; - combo->label_hover = table[NK_COLOR_TEXT]; - combo->label_active = table[NK_COLOR_TEXT]; - combo->sym_normal = NK_SYMBOL_TRIANGLE_DOWN; - combo->sym_hover = NK_SYMBOL_TRIANGLE_DOWN; - combo->sym_active = NK_SYMBOL_TRIANGLE_DOWN; - combo->content_padding = nk_vec2(4,4); - combo->button_padding = nk_vec2(0,4); - combo->spacing = nk_vec2(4,0); - combo->border = 1; - combo->rounding = 0; - - /* combo button */ - button = &style->combo.button; - nk_zero_struct(*button); - button->normal = nk_style_item_color(table[NK_COLOR_COMBO]); - button->hover = nk_style_item_color(table[NK_COLOR_COMBO]); - button->active = nk_style_item_color(table[NK_COLOR_COMBO]); - button->border_color = nk_rgba(0,0,0,0); - button->text_background = table[NK_COLOR_COMBO]; - button->text_normal = table[NK_COLOR_TEXT]; - button->text_hover = table[NK_COLOR_TEXT]; - button->text_active = table[NK_COLOR_TEXT]; - button->padding = nk_vec2(2.0f,2.0f); - button->touch_padding = nk_vec2(0.0f,0.0f); - button->userdata = nk_handle_ptr(0); - button->text_alignment = NK_TEXT_CENTERED; - button->border = 0.0f; - button->rounding = 0.0f; - button->draw_begin = 0; - button->draw_end = 0; - - /* tab */ - tab = &style->tab; - tab->background = nk_style_item_color(table[NK_COLOR_TAB_HEADER]); - tab->border_color = table[NK_COLOR_BORDER]; - tab->text = table[NK_COLOR_TEXT]; - tab->sym_minimize = NK_SYMBOL_TRIANGLE_RIGHT; - tab->sym_maximize = NK_SYMBOL_TRIANGLE_DOWN; - tab->padding = nk_vec2(4,4); - tab->spacing = nk_vec2(4,4); - tab->indent = 10.0f; - tab->border = 1; - tab->rounding = 0; - - /* tab button */ - button = &style->tab.tab_minimize_button; - nk_zero_struct(*button); - button->normal = nk_style_item_color(table[NK_COLOR_TAB_HEADER]); - button->hover = nk_style_item_color(table[NK_COLOR_TAB_HEADER]); - button->active = nk_style_item_color(table[NK_COLOR_TAB_HEADER]); - button->border_color = nk_rgba(0,0,0,0); - button->text_background = table[NK_COLOR_TAB_HEADER]; - button->text_normal = table[NK_COLOR_TEXT]; - button->text_hover = table[NK_COLOR_TEXT]; - button->text_active = table[NK_COLOR_TEXT]; - button->padding = nk_vec2(2.0f,2.0f); - button->touch_padding = nk_vec2(0.0f,0.0f); - button->userdata = nk_handle_ptr(0); - button->text_alignment = NK_TEXT_CENTERED; - button->border = 0.0f; - button->rounding = 0.0f; - button->draw_begin = 0; - button->draw_end = 0; - style->tab.tab_maximize_button =*button; - - /* node button */ - button = &style->tab.node_minimize_button; - nk_zero_struct(*button); - button->normal = nk_style_item_color(table[NK_COLOR_WINDOW]); - button->hover = nk_style_item_color(table[NK_COLOR_WINDOW]); - button->active = nk_style_item_color(table[NK_COLOR_WINDOW]); - button->border_color = nk_rgba(0,0,0,0); - button->text_background = table[NK_COLOR_TAB_HEADER]; - button->text_normal = table[NK_COLOR_TEXT]; - button->text_hover = table[NK_COLOR_TEXT]; - button->text_active = table[NK_COLOR_TEXT]; - button->padding = nk_vec2(2.0f,2.0f); - button->touch_padding = nk_vec2(0.0f,0.0f); - button->userdata = nk_handle_ptr(0); - button->text_alignment = NK_TEXT_CENTERED; - button->border = 0.0f; - button->rounding = 0.0f; - button->draw_begin = 0; - button->draw_end = 0; - style->tab.node_maximize_button =*button; - - /* window header */ - win = &style->window; - win->header.align = NK_HEADER_RIGHT; - win->header.close_symbol = NK_SYMBOL_X; - win->header.minimize_symbol = NK_SYMBOL_MINUS; - win->header.maximize_symbol = NK_SYMBOL_PLUS; - win->header.normal = nk_style_item_color(table[NK_COLOR_HEADER]); - win->header.hover = nk_style_item_color(table[NK_COLOR_HEADER]); - win->header.active = nk_style_item_color(table[NK_COLOR_HEADER]); - win->header.label_normal = table[NK_COLOR_TEXT]; - win->header.label_hover = table[NK_COLOR_TEXT]; - win->header.label_active = table[NK_COLOR_TEXT]; - win->header.label_padding = nk_vec2(4,4); - win->header.padding = nk_vec2(4,4); - win->header.spacing = nk_vec2(0,0); - - /* window header close button */ - button = &style->window.header.close_button; - nk_zero_struct(*button); - button->normal = nk_style_item_color(table[NK_COLOR_HEADER]); - button->hover = nk_style_item_color(table[NK_COLOR_HEADER]); - button->active = nk_style_item_color(table[NK_COLOR_HEADER]); - button->border_color = nk_rgba(0,0,0,0); - button->text_background = table[NK_COLOR_HEADER]; - button->text_normal = table[NK_COLOR_TEXT]; - button->text_hover = table[NK_COLOR_TEXT]; - button->text_active = table[NK_COLOR_TEXT]; - button->padding = nk_vec2(0.0f,0.0f); - button->touch_padding = nk_vec2(0.0f,0.0f); - button->userdata = nk_handle_ptr(0); - button->text_alignment = NK_TEXT_CENTERED; - button->border = 0.0f; - button->rounding = 0.0f; - button->draw_begin = 0; - button->draw_end = 0; - - /* window header minimize button */ - button = &style->window.header.minimize_button; - nk_zero_struct(*button); - button->normal = nk_style_item_color(table[NK_COLOR_HEADER]); - button->hover = nk_style_item_color(table[NK_COLOR_HEADER]); - button->active = nk_style_item_color(table[NK_COLOR_HEADER]); - button->border_color = nk_rgba(0,0,0,0); - button->text_background = table[NK_COLOR_HEADER]; - button->text_normal = table[NK_COLOR_TEXT]; - button->text_hover = table[NK_COLOR_TEXT]; - button->text_active = table[NK_COLOR_TEXT]; - button->padding = nk_vec2(0.0f,0.0f); - button->touch_padding = nk_vec2(0.0f,0.0f); - button->userdata = nk_handle_ptr(0); - button->text_alignment = NK_TEXT_CENTERED; - button->border = 0.0f; - button->rounding = 0.0f; - button->draw_begin = 0; - button->draw_end = 0; - - /* window */ - win->background = table[NK_COLOR_WINDOW]; - win->fixed_background = nk_style_item_color(table[NK_COLOR_WINDOW]); - win->border_color = table[NK_COLOR_BORDER]; - win->popup_border_color = table[NK_COLOR_BORDER]; - win->combo_border_color = table[NK_COLOR_BORDER]; - win->contextual_border_color = table[NK_COLOR_BORDER]; - win->menu_border_color = table[NK_COLOR_BORDER]; - win->group_border_color = table[NK_COLOR_BORDER]; - win->tooltip_border_color = table[NK_COLOR_BORDER]; - win->scaler = nk_style_item_color(table[NK_COLOR_TEXT]); - - win->rounding = 0.0f; - win->spacing = nk_vec2(4,4); - win->scrollbar_size = nk_vec2(10,10); - win->min_size = nk_vec2(64,64); - - win->combo_border = 1.0f; - win->contextual_border = 1.0f; - win->menu_border = 1.0f; - win->group_border = 1.0f; - win->tooltip_border = 1.0f; - win->popup_border = 1.0f; - win->border = 2.0f; - win->min_row_height_padding = 8; - - win->padding = nk_vec2(4,4); - win->group_padding = nk_vec2(4,4); - win->popup_padding = nk_vec2(4,4); - win->combo_padding = nk_vec2(4,4); - win->contextual_padding = nk_vec2(4,4); - win->menu_padding = nk_vec2(4,4); - win->tooltip_padding = nk_vec2(4,4); -} -NK_API void -nk_style_set_font(struct nk_context *ctx, const struct nk_user_font *font) -{ - struct nk_style *style; - NK_ASSERT(ctx); - - if (!ctx) return; - style = &ctx->style; - style->font = font; - ctx->stacks.fonts.head = 0; - if (ctx->current) - nk_layout_reset_min_row_height(ctx); -} -NK_API nk_bool -nk_style_push_font(struct nk_context *ctx, const struct nk_user_font *font) -{ - struct nk_config_stack_user_font *font_stack; - struct nk_config_stack_user_font_element *element; - - NK_ASSERT(ctx); - if (!ctx) return 0; - - font_stack = &ctx->stacks.fonts; - NK_ASSERT(font_stack->head < (int)NK_LEN(font_stack->elements)); - if (font_stack->head >= (int)NK_LEN(font_stack->elements)) - return 0; - - element = &font_stack->elements[font_stack->head++]; - element->address = &ctx->style.font; - element->old_value = ctx->style.font; - ctx->style.font = font; - return 1; -} -NK_API nk_bool -nk_style_pop_font(struct nk_context *ctx) -{ - struct nk_config_stack_user_font *font_stack; - struct nk_config_stack_user_font_element *element; - - NK_ASSERT(ctx); - if (!ctx) return 0; - - font_stack = &ctx->stacks.fonts; - NK_ASSERT(font_stack->head > 0); - if (font_stack->head < 1) - return 0; - - element = &font_stack->elements[--font_stack->head]; - *element->address = element->old_value; - return 1; -} -#define NK_STYLE_PUSH_IMPLEMENATION(prefix, type, stack) \ -nk_style_push_##type(struct nk_context *ctx, prefix##_##type *address, prefix##_##type value)\ -{\ - struct nk_config_stack_##type * type_stack;\ - struct nk_config_stack_##type##_element *element;\ - NK_ASSERT(ctx);\ - if (!ctx) return 0;\ - type_stack = &ctx->stacks.stack;\ - NK_ASSERT(type_stack->head < (int)NK_LEN(type_stack->elements));\ - if (type_stack->head >= (int)NK_LEN(type_stack->elements))\ - return 0;\ - element = &type_stack->elements[type_stack->head++];\ - element->address = address;\ - element->old_value = *address;\ - *address = value;\ - return 1;\ -} -#define NK_STYLE_POP_IMPLEMENATION(type, stack) \ -nk_style_pop_##type(struct nk_context *ctx)\ -{\ - struct nk_config_stack_##type *type_stack;\ - struct nk_config_stack_##type##_element *element;\ - NK_ASSERT(ctx);\ - if (!ctx) return 0;\ - type_stack = &ctx->stacks.stack;\ - NK_ASSERT(type_stack->head > 0);\ - if (type_stack->head < 1)\ - return 0;\ - element = &type_stack->elements[--type_stack->head];\ - *element->address = element->old_value;\ - return 1;\ -} -NK_API nk_bool NK_STYLE_PUSH_IMPLEMENATION(struct nk, style_item, style_items) -NK_API nk_bool NK_STYLE_PUSH_IMPLEMENATION(nk,float, floats) -NK_API nk_bool NK_STYLE_PUSH_IMPLEMENATION(struct nk, vec2, vectors) -NK_API nk_bool NK_STYLE_PUSH_IMPLEMENATION(nk,flags, flags) -NK_API nk_bool NK_STYLE_PUSH_IMPLEMENATION(struct nk,color, colors) - -NK_API nk_bool NK_STYLE_POP_IMPLEMENATION(style_item, style_items) -NK_API nk_bool NK_STYLE_POP_IMPLEMENATION(float,floats) -NK_API nk_bool NK_STYLE_POP_IMPLEMENATION(vec2, vectors) -NK_API nk_bool NK_STYLE_POP_IMPLEMENATION(flags,flags) -NK_API nk_bool NK_STYLE_POP_IMPLEMENATION(color,colors) - -NK_API nk_bool -nk_style_set_cursor(struct nk_context *ctx, enum nk_style_cursor c) -{ - struct nk_style *style; - NK_ASSERT(ctx); - if (!ctx) return 0; - style = &ctx->style; - if (style->cursors[c]) { - style->cursor_active = style->cursors[c]; - return 1; - } - return 0; -} -NK_API void -nk_style_show_cursor(struct nk_context *ctx) -{ - ctx->style.cursor_visible = nk_true; -} -NK_API void -nk_style_hide_cursor(struct nk_context *ctx) -{ - ctx->style.cursor_visible = nk_false; -} -NK_API void -nk_style_load_cursor(struct nk_context *ctx, enum nk_style_cursor cursor, - const struct nk_cursor *c) -{ - struct nk_style *style; - NK_ASSERT(ctx); - if (!ctx) return; - style = &ctx->style; - style->cursors[cursor] = c; -} -NK_API void -nk_style_load_all_cursors(struct nk_context *ctx, struct nk_cursor *cursors) -{ - int i = 0; - struct nk_style *style; - NK_ASSERT(ctx); - if (!ctx) return; - style = &ctx->style; - for (i = 0; i < NK_CURSOR_COUNT; ++i) - style->cursors[i] = &cursors[i]; - style->cursor_visible = nk_true; -} - - - - - -/* ============================================================== - * - * CONTEXT - * - * ===============================================================*/ -NK_INTERN void -nk_setup(struct nk_context *ctx, const struct nk_user_font *font) -{ - NK_ASSERT(ctx); - if (!ctx) return; - nk_zero_struct(*ctx); - nk_style_default(ctx); - ctx->seq = 1; - if (font) ctx->style.font = font; -#ifdef NK_INCLUDE_VERTEX_BUFFER_OUTPUT - nk_draw_list_init(&ctx->draw_list); -#endif -} -#ifdef NK_INCLUDE_DEFAULT_ALLOCATOR -NK_API nk_bool -nk_init_default(struct nk_context *ctx, const struct nk_user_font *font) -{ - struct nk_allocator alloc; - alloc.userdata.ptr = 0; - alloc.alloc = nk_malloc; - alloc.free = nk_mfree; - return nk_init(ctx, &alloc, font); -} -#endif -NK_API nk_bool -nk_init_fixed(struct nk_context *ctx, void *memory, nk_size size, - const struct nk_user_font *font) -{ - NK_ASSERT(memory); - if (!memory) return 0; - nk_setup(ctx, font); - nk_buffer_init_fixed(&ctx->memory, memory, size); - ctx->use_pool = nk_false; - return 1; -} -NK_API nk_bool -nk_init_custom(struct nk_context *ctx, struct nk_buffer *cmds, - struct nk_buffer *pool, const struct nk_user_font *font) -{ - NK_ASSERT(cmds); - NK_ASSERT(pool); - if (!cmds || !pool) return 0; - - nk_setup(ctx, font); - ctx->memory = *cmds; - if (pool->type == NK_BUFFER_FIXED) { - /* take memory from buffer and alloc fixed pool */ - nk_pool_init_fixed(&ctx->pool, pool->memory.ptr, pool->memory.size); - } else { - /* create dynamic pool from buffer allocator */ - struct nk_allocator *alloc = &pool->pool; - nk_pool_init(&ctx->pool, alloc, NK_POOL_DEFAULT_CAPACITY); - } - ctx->use_pool = nk_true; - return 1; -} -NK_API nk_bool -nk_init(struct nk_context *ctx, struct nk_allocator *alloc, - const struct nk_user_font *font) -{ - NK_ASSERT(alloc); - if (!alloc) return 0; - nk_setup(ctx, font); - nk_buffer_init(&ctx->memory, alloc, NK_DEFAULT_COMMAND_BUFFER_SIZE); - nk_pool_init(&ctx->pool, alloc, NK_POOL_DEFAULT_CAPACITY); - ctx->use_pool = nk_true; - return 1; -} -#ifdef NK_INCLUDE_COMMAND_USERDATA -NK_API void -nk_set_user_data(struct nk_context *ctx, nk_handle handle) -{ - if (!ctx) return; - ctx->userdata = handle; - if (ctx->current) - ctx->current->buffer.userdata = handle; -} -#endif -NK_API void -nk_free(struct nk_context *ctx) -{ - NK_ASSERT(ctx); - if (!ctx) return; - nk_buffer_free(&ctx->memory); - if (ctx->use_pool) - nk_pool_free(&ctx->pool); - - nk_zero(&ctx->input, sizeof(ctx->input)); - nk_zero(&ctx->style, sizeof(ctx->style)); - nk_zero(&ctx->memory, sizeof(ctx->memory)); - - ctx->seq = 0; - ctx->build = 0; - ctx->begin = 0; - ctx->end = 0; - ctx->active = 0; - ctx->current = 0; - ctx->freelist = 0; - ctx->count = 0; -} -NK_API void -nk_clear(struct nk_context *ctx) -{ - struct nk_window *iter; - struct nk_window *next; - NK_ASSERT(ctx); - - if (!ctx) return; - if (ctx->use_pool) - nk_buffer_clear(&ctx->memory); - else nk_buffer_reset(&ctx->memory, NK_BUFFER_FRONT); - - ctx->build = 0; - ctx->memory.calls = 0; - ctx->last_widget_state = 0; - ctx->style.cursor_active = ctx->style.cursors[NK_CURSOR_ARROW]; - NK_MEMSET(&ctx->overlay, 0, sizeof(ctx->overlay)); - - /* garbage collector */ - iter = ctx->begin; - while (iter) { - /* make sure valid minimized windows do not get removed */ - if ((iter->flags & NK_WINDOW_MINIMIZED) && - !(iter->flags & NK_WINDOW_CLOSED) && - iter->seq == ctx->seq) { - iter = iter->next; - continue; - } - /* remove hotness from hidden or closed windows*/ - if (((iter->flags & NK_WINDOW_HIDDEN) || - (iter->flags & NK_WINDOW_CLOSED)) && - iter == ctx->active) { - ctx->active = iter->prev; - ctx->end = iter->prev; - if (!ctx->end) - ctx->begin = 0; - if (ctx->active) - ctx->active->flags &= ~(unsigned)NK_WINDOW_ROM; - } - /* free unused popup windows */ - if (iter->popup.win && iter->popup.win->seq != ctx->seq) { - nk_free_window(ctx, iter->popup.win); - iter->popup.win = 0; - } - /* remove unused window state tables */ - {struct nk_table *n, *it = iter->tables; - while (it) { - n = it->next; - if (it->seq != ctx->seq) { - nk_remove_table(iter, it); - nk_zero(it, sizeof(union nk_page_data)); - nk_free_table(ctx, it); - if (it == iter->tables) - iter->tables = n; - } it = n; - }} - /* window itself is not used anymore so free */ - if (iter->seq != ctx->seq || iter->flags & NK_WINDOW_CLOSED) { - next = iter->next; - nk_remove_window(ctx, iter); - nk_free_window(ctx, iter); - iter = next; - } else iter = iter->next; - } - ctx->seq++; -} -NK_LIB void -nk_start_buffer(struct nk_context *ctx, struct nk_command_buffer *buffer) -{ - NK_ASSERT(ctx); - NK_ASSERT(buffer); - if (!ctx || !buffer) return; - buffer->begin = ctx->memory.allocated; - buffer->end = buffer->begin; - buffer->last = buffer->begin; - buffer->clip = nk_null_rect; -} -NK_LIB void -nk_start(struct nk_context *ctx, struct nk_window *win) -{ - NK_ASSERT(ctx); - NK_ASSERT(win); - nk_start_buffer(ctx, &win->buffer); -} -NK_LIB void -nk_start_popup(struct nk_context *ctx, struct nk_window *win) -{ - struct nk_popup_buffer *buf; - NK_ASSERT(ctx); - NK_ASSERT(win); - if (!ctx || !win) return; - - /* save buffer fill state for popup */ - buf = &win->popup.buf; - buf->begin = win->buffer.end; - buf->end = win->buffer.end; - buf->parent = win->buffer.last; - buf->last = buf->begin; - buf->active = nk_true; -} -NK_LIB void -nk_finish_popup(struct nk_context *ctx, struct nk_window *win) -{ - struct nk_popup_buffer *buf; - NK_ASSERT(ctx); - NK_ASSERT(win); - if (!ctx || !win) return; - - buf = &win->popup.buf; - buf->last = win->buffer.last; - buf->end = win->buffer.end; -} -NK_LIB void -nk_finish_buffer(struct nk_context *ctx, struct nk_command_buffer *buffer) -{ - NK_ASSERT(ctx); - NK_ASSERT(buffer); - if (!ctx || !buffer) return; - buffer->end = ctx->memory.allocated; -} -NK_LIB void -nk_finish(struct nk_context *ctx, struct nk_window *win) -{ - struct nk_popup_buffer *buf; - struct nk_command *parent_last; - void *memory; - - NK_ASSERT(ctx); - NK_ASSERT(win); - if (!ctx || !win) return; - nk_finish_buffer(ctx, &win->buffer); - if (!win->popup.buf.active) return; - - buf = &win->popup.buf; - memory = ctx->memory.memory.ptr; - parent_last = nk_ptr_add(struct nk_command, memory, buf->parent); - parent_last->next = buf->end; -} -NK_LIB void -nk_build(struct nk_context *ctx) -{ - struct nk_window *it = 0; - struct nk_command *cmd = 0; - nk_byte *buffer = 0; - - /* draw cursor overlay */ - if (!ctx->style.cursor_active) - ctx->style.cursor_active = ctx->style.cursors[NK_CURSOR_ARROW]; - if (ctx->style.cursor_active && !ctx->input.mouse.grabbed && ctx->style.cursor_visible) { - struct nk_rect mouse_bounds; - const struct nk_cursor *cursor = ctx->style.cursor_active; - nk_command_buffer_init(&ctx->overlay, &ctx->memory, NK_CLIPPING_OFF); - nk_start_buffer(ctx, &ctx->overlay); - - mouse_bounds.x = ctx->input.mouse.pos.x - cursor->offset.x; - mouse_bounds.y = ctx->input.mouse.pos.y - cursor->offset.y; - mouse_bounds.w = cursor->size.x; - mouse_bounds.h = cursor->size.y; - - nk_draw_image(&ctx->overlay, mouse_bounds, &cursor->img, nk_white); - nk_finish_buffer(ctx, &ctx->overlay); - } - /* build one big draw command list out of all window buffers */ - it = ctx->begin; - buffer = (nk_byte*)ctx->memory.memory.ptr; - while (it != 0) { - struct nk_window *next = it->next; - if (it->buffer.last == it->buffer.begin || (it->flags & NK_WINDOW_HIDDEN)|| - it->seq != ctx->seq) - goto cont; - - cmd = nk_ptr_add(struct nk_command, buffer, it->buffer.last); - while (next && ((next->buffer.last == next->buffer.begin) || - (next->flags & NK_WINDOW_HIDDEN) || next->seq != ctx->seq)) - next = next->next; /* skip empty command buffers */ - - if (next) cmd->next = next->buffer.begin; - cont: it = next; - } - /* append all popup draw commands into lists */ - it = ctx->begin; - while (it != 0) { - struct nk_window *next = it->next; - struct nk_popup_buffer *buf; - if (!it->popup.buf.active) - goto skip; - - buf = &it->popup.buf; - cmd->next = buf->begin; - cmd = nk_ptr_add(struct nk_command, buffer, buf->last); - buf->active = nk_false; - skip: it = next; - } - if (cmd) { - /* append overlay commands */ - if (ctx->overlay.end != ctx->overlay.begin) - cmd->next = ctx->overlay.begin; - else cmd->next = ctx->memory.allocated; - } -} -NK_API const struct nk_command* -nk__begin(struct nk_context *ctx) -{ - struct nk_window *iter; - nk_byte *buffer; - NK_ASSERT(ctx); - if (!ctx) return 0; - if (!ctx->count) return 0; - - buffer = (nk_byte*)ctx->memory.memory.ptr; - if (!ctx->build) { - nk_build(ctx); - ctx->build = nk_true; - } - iter = ctx->begin; - while (iter && ((iter->buffer.begin == iter->buffer.end) || - (iter->flags & NK_WINDOW_HIDDEN) || iter->seq != ctx->seq)) - iter = iter->next; - if (!iter) return 0; - return nk_ptr_add_const(struct nk_command, buffer, iter->buffer.begin); -} - -NK_API const struct nk_command* -nk__next(struct nk_context *ctx, const struct nk_command *cmd) -{ - nk_byte *buffer; - const struct nk_command *next; - NK_ASSERT(ctx); - if (!ctx || !cmd || !ctx->count) return 0; - if (cmd->next >= ctx->memory.allocated) return 0; - buffer = (nk_byte*)ctx->memory.memory.ptr; - next = nk_ptr_add_const(struct nk_command, buffer, cmd->next); - return next; -} - - - - - - -/* =============================================================== - * - * POOL - * - * ===============================================================*/ -NK_LIB void -nk_pool_init(struct nk_pool *pool, struct nk_allocator *alloc, - unsigned int capacity) -{ - NK_ASSERT(capacity >= 1); - nk_zero(pool, sizeof(*pool)); - pool->alloc = *alloc; - pool->capacity = capacity; - pool->type = NK_BUFFER_DYNAMIC; - pool->pages = 0; -} -NK_LIB void -nk_pool_free(struct nk_pool *pool) -{ - struct nk_page *iter; - if (!pool) return; - iter = pool->pages; - if (pool->type == NK_BUFFER_FIXED) return; - while (iter) { - struct nk_page *next = iter->next; - pool->alloc.free(pool->alloc.userdata, iter); - iter = next; - } -} -NK_LIB void -nk_pool_init_fixed(struct nk_pool *pool, void *memory, nk_size size) -{ - nk_zero(pool, sizeof(*pool)); - NK_ASSERT(size >= sizeof(struct nk_page)); - if (size < sizeof(struct nk_page)) return; - /* first nk_page_element is embedded in nk_page, additional elements follow in adjacent space */ - pool->capacity = (unsigned)(1 + (size - sizeof(struct nk_page)) / sizeof(struct nk_page_element)); - pool->pages = (struct nk_page*)memory; - pool->type = NK_BUFFER_FIXED; - pool->size = size; -} -NK_LIB struct nk_page_element* -nk_pool_alloc(struct nk_pool *pool) -{ - if (!pool->pages || pool->pages->size >= pool->capacity) { - /* allocate new page */ - struct nk_page *page; - if (pool->type == NK_BUFFER_FIXED) { - NK_ASSERT(pool->pages); - if (!pool->pages) return 0; - NK_ASSERT(pool->pages->size < pool->capacity); - return 0; - } else { - nk_size size = sizeof(struct nk_page); - size += (pool->capacity - 1) * sizeof(struct nk_page_element); - page = (struct nk_page*)pool->alloc.alloc(pool->alloc.userdata,0, size); - page->next = pool->pages; - pool->pages = page; - page->size = 0; - } - } return &pool->pages->win[pool->pages->size++]; -} - - - - - -/* =============================================================== - * - * PAGE ELEMENT - * - * ===============================================================*/ -NK_LIB struct nk_page_element* -nk_create_page_element(struct nk_context *ctx) -{ - struct nk_page_element *elem; - if (ctx->freelist) { - /* unlink page element from free list */ - elem = ctx->freelist; - ctx->freelist = elem->next; - } else if (ctx->use_pool) { - /* allocate page element from memory pool */ - elem = nk_pool_alloc(&ctx->pool); - NK_ASSERT(elem); - if (!elem) return 0; - } else { - /* allocate new page element from back of fixed size memory buffer */ - NK_STORAGE const nk_size size = sizeof(struct nk_page_element); - NK_STORAGE const nk_size align = NK_ALIGNOF(struct nk_page_element); - elem = (struct nk_page_element*)nk_buffer_alloc(&ctx->memory, NK_BUFFER_BACK, size, align); - NK_ASSERT(elem); - if (!elem) return 0; - } - nk_zero_struct(*elem); - elem->next = 0; - elem->prev = 0; - return elem; -} -NK_LIB void -nk_link_page_element_into_freelist(struct nk_context *ctx, - struct nk_page_element *elem) -{ - /* link table into freelist */ - if (!ctx->freelist) { - ctx->freelist = elem; - } else { - elem->next = ctx->freelist; - ctx->freelist = elem; - } -} -NK_LIB void -nk_free_page_element(struct nk_context *ctx, struct nk_page_element *elem) -{ - /* we have a pool so just add to free list */ - if (ctx->use_pool) { - nk_link_page_element_into_freelist(ctx, elem); - return; - } - /* if possible remove last element from back of fixed memory buffer */ - {void *elem_end = (void*)(elem + 1); - void *buffer_end = (nk_byte*)ctx->memory.memory.ptr + ctx->memory.size; - if (elem_end == buffer_end) - ctx->memory.size -= sizeof(struct nk_page_element); - else nk_link_page_element_into_freelist(ctx, elem);} -} - - - - - -/* =============================================================== - * - * TABLE - * - * ===============================================================*/ -NK_LIB struct nk_table* -nk_create_table(struct nk_context *ctx) -{ - struct nk_page_element *elem; - elem = nk_create_page_element(ctx); - if (!elem) return 0; - nk_zero_struct(*elem); - return &elem->data.tbl; -} -NK_LIB void -nk_free_table(struct nk_context *ctx, struct nk_table *tbl) -{ - union nk_page_data *pd = NK_CONTAINER_OF(tbl, union nk_page_data, tbl); - struct nk_page_element *pe = NK_CONTAINER_OF(pd, struct nk_page_element, data); - nk_free_page_element(ctx, pe); -} -NK_LIB void -nk_push_table(struct nk_window *win, struct nk_table *tbl) -{ - if (!win->tables) { - win->tables = tbl; - tbl->next = 0; - tbl->prev = 0; - tbl->size = 0; - win->table_count = 1; - return; - } - win->tables->prev = tbl; - tbl->next = win->tables; - tbl->prev = 0; - tbl->size = 0; - win->tables = tbl; - win->table_count++; -} -NK_LIB void -nk_remove_table(struct nk_window *win, struct nk_table *tbl) -{ - if (win->tables == tbl) - win->tables = tbl->next; - if (tbl->next) - tbl->next->prev = tbl->prev; - if (tbl->prev) - tbl->prev->next = tbl->next; - tbl->next = 0; - tbl->prev = 0; -} -NK_LIB nk_uint* -nk_add_value(struct nk_context *ctx, struct nk_window *win, - nk_hash name, nk_uint value) -{ - NK_ASSERT(ctx); - NK_ASSERT(win); - if (!win || !ctx) return 0; - if (!win->tables || win->tables->size >= NK_VALUE_PAGE_CAPACITY) { - struct nk_table *tbl = nk_create_table(ctx); - NK_ASSERT(tbl); - if (!tbl) return 0; - nk_push_table(win, tbl); - } - win->tables->seq = win->seq; - win->tables->keys[win->tables->size] = name; - win->tables->values[win->tables->size] = value; - return &win->tables->values[win->tables->size++]; -} -NK_LIB nk_uint* -nk_find_value(struct nk_window *win, nk_hash name) -{ - struct nk_table *iter = win->tables; - while (iter) { - unsigned int i = 0; - unsigned int size = iter->size; - for (i = 0; i < size; ++i) { - if (iter->keys[i] == name) { - iter->seq = win->seq; - return &iter->values[i]; - } - } size = NK_VALUE_PAGE_CAPACITY; - iter = iter->next; - } - return 0; -} - - - - - -/* =============================================================== - * - * PANEL - * - * ===============================================================*/ -NK_LIB void* -nk_create_panel(struct nk_context *ctx) -{ - struct nk_page_element *elem; - elem = nk_create_page_element(ctx); - if (!elem) return 0; - nk_zero_struct(*elem); - return &elem->data.pan; -} -NK_LIB void -nk_free_panel(struct nk_context *ctx, struct nk_panel *pan) -{ - union nk_page_data *pd = NK_CONTAINER_OF(pan, union nk_page_data, pan); - struct nk_page_element *pe = NK_CONTAINER_OF(pd, struct nk_page_element, data); - nk_free_page_element(ctx, pe); -} -NK_LIB nk_bool -nk_panel_has_header(nk_flags flags, const char *title) -{ - nk_bool active = 0; - active = (flags & (NK_WINDOW_CLOSABLE|NK_WINDOW_MINIMIZABLE)); - active = active || (flags & NK_WINDOW_TITLE); - active = active && !(flags & NK_WINDOW_HIDDEN) && title; - return active; -} -NK_LIB struct nk_vec2 -nk_panel_get_padding(const struct nk_style *style, enum nk_panel_type type) -{ - switch (type) { - default: - case NK_PANEL_WINDOW: return style->window.padding; - case NK_PANEL_GROUP: return style->window.group_padding; - case NK_PANEL_POPUP: return style->window.popup_padding; - case NK_PANEL_CONTEXTUAL: return style->window.contextual_padding; - case NK_PANEL_COMBO: return style->window.combo_padding; - case NK_PANEL_MENU: return style->window.menu_padding; - case NK_PANEL_TOOLTIP: return style->window.menu_padding;} -} -NK_LIB float -nk_panel_get_border(const struct nk_style *style, nk_flags flags, - enum nk_panel_type type) -{ - if (flags & NK_WINDOW_BORDER) { - switch (type) { - default: - case NK_PANEL_WINDOW: return style->window.border; - case NK_PANEL_GROUP: return style->window.group_border; - case NK_PANEL_POPUP: return style->window.popup_border; - case NK_PANEL_CONTEXTUAL: return style->window.contextual_border; - case NK_PANEL_COMBO: return style->window.combo_border; - case NK_PANEL_MENU: return style->window.menu_border; - case NK_PANEL_TOOLTIP: return style->window.menu_border; - }} else return 0; -} -NK_LIB struct nk_color -nk_panel_get_border_color(const struct nk_style *style, enum nk_panel_type type) -{ - switch (type) { - default: - case NK_PANEL_WINDOW: return style->window.border_color; - case NK_PANEL_GROUP: return style->window.group_border_color; - case NK_PANEL_POPUP: return style->window.popup_border_color; - case NK_PANEL_CONTEXTUAL: return style->window.contextual_border_color; - case NK_PANEL_COMBO: return style->window.combo_border_color; - case NK_PANEL_MENU: return style->window.menu_border_color; - case NK_PANEL_TOOLTIP: return style->window.menu_border_color;} -} -NK_LIB nk_bool -nk_panel_is_sub(enum nk_panel_type type) -{ - return (type & NK_PANEL_SET_SUB)?1:0; -} -NK_LIB nk_bool -nk_panel_is_nonblock(enum nk_panel_type type) -{ - return (type & NK_PANEL_SET_NONBLOCK)?1:0; -} -NK_LIB nk_bool -nk_panel_begin(struct nk_context *ctx, const char *title, enum nk_panel_type panel_type) -{ - struct nk_input *in; - struct nk_window *win; - struct nk_panel *layout; - struct nk_command_buffer *out; - const struct nk_style *style; - const struct nk_user_font *font; - - struct nk_vec2 scrollbar_size; - struct nk_vec2 panel_padding; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) return 0; - nk_zero(ctx->current->layout, sizeof(*ctx->current->layout)); - if ((ctx->current->flags & NK_WINDOW_HIDDEN) || (ctx->current->flags & NK_WINDOW_CLOSED)) { - nk_zero(ctx->current->layout, sizeof(struct nk_panel)); - ctx->current->layout->type = panel_type; - return 0; - } - /* pull state into local stack */ - style = &ctx->style; - font = style->font; - win = ctx->current; - layout = win->layout; - out = &win->buffer; - in = (win->flags & NK_WINDOW_NO_INPUT) ? 0: &ctx->input; -#ifdef NK_INCLUDE_COMMAND_USERDATA - win->buffer.userdata = ctx->userdata; -#endif - /* pull style configuration into local stack */ - scrollbar_size = style->window.scrollbar_size; - panel_padding = nk_panel_get_padding(style, panel_type); - - /* window movement */ - if ((win->flags & NK_WINDOW_MOVABLE) && !(win->flags & NK_WINDOW_ROM)) { - nk_bool left_mouse_down; - unsigned int left_mouse_clicked; - int left_mouse_click_in_cursor; - - /* calculate draggable window space */ - struct nk_rect header; - header.x = win->bounds.x; - header.y = win->bounds.y; - header.w = win->bounds.w; - if (nk_panel_has_header(win->flags, title)) { - header.h = font->height + 2.0f * style->window.header.padding.y; - header.h += 2.0f * style->window.header.label_padding.y; - } else header.h = panel_padding.y; - - /* window movement by dragging */ - left_mouse_down = in->mouse.buttons[NK_BUTTON_LEFT].down; - left_mouse_clicked = in->mouse.buttons[NK_BUTTON_LEFT].clicked; - left_mouse_click_in_cursor = nk_input_has_mouse_click_down_in_rect(in, - NK_BUTTON_LEFT, header, nk_true); - if (left_mouse_down && left_mouse_click_in_cursor && !left_mouse_clicked) { - win->bounds.x = win->bounds.x + in->mouse.delta.x; - win->bounds.y = win->bounds.y + in->mouse.delta.y; - in->mouse.buttons[NK_BUTTON_LEFT].clicked_pos.x += in->mouse.delta.x; - in->mouse.buttons[NK_BUTTON_LEFT].clicked_pos.y += in->mouse.delta.y; - ctx->style.cursor_active = ctx->style.cursors[NK_CURSOR_MOVE]; - } - } - - /* setup panel */ - layout->type = panel_type; - layout->flags = win->flags; - layout->bounds = win->bounds; - layout->bounds.x += panel_padding.x; - layout->bounds.w -= 2*panel_padding.x; - if (win->flags & NK_WINDOW_BORDER) { - layout->border = nk_panel_get_border(style, win->flags, panel_type); - layout->bounds = nk_shrink_rect(layout->bounds, layout->border); - } else layout->border = 0; - layout->at_y = layout->bounds.y; - layout->at_x = layout->bounds.x; - layout->max_x = 0; - layout->header_height = 0; - layout->footer_height = 0; - nk_layout_reset_min_row_height(ctx); - layout->row.index = 0; - layout->row.columns = 0; - layout->row.ratio = 0; - layout->row.item_width = 0; - layout->row.tree_depth = 0; - layout->row.height = panel_padding.y; - layout->has_scrolling = nk_true; - if (!(win->flags & NK_WINDOW_NO_SCROLLBAR)) - layout->bounds.w -= scrollbar_size.x; - if (!nk_panel_is_nonblock(panel_type)) { - layout->footer_height = 0; - if (!(win->flags & NK_WINDOW_NO_SCROLLBAR) || win->flags & NK_WINDOW_SCALABLE) - layout->footer_height = scrollbar_size.y; - layout->bounds.h -= layout->footer_height; - } - - /* panel header */ - if (nk_panel_has_header(win->flags, title)) - { - struct nk_text text; - struct nk_rect header; - const struct nk_style_item *background = 0; - - /* calculate header bounds */ - header.x = win->bounds.x; - header.y = win->bounds.y; - header.w = win->bounds.w; - header.h = font->height + 2.0f * style->window.header.padding.y; - header.h += (2.0f * style->window.header.label_padding.y); - - /* shrink panel by header */ - layout->header_height = header.h; - layout->bounds.y += header.h; - layout->bounds.h -= header.h; - layout->at_y += header.h; - - /* select correct header background and text color */ - if (ctx->active == win) { - background = &style->window.header.active; - text.text = style->window.header.label_active; - } else if (nk_input_is_mouse_hovering_rect(&ctx->input, header)) { - background = &style->window.header.hover; - text.text = style->window.header.label_hover; - } else { - background = &style->window.header.normal; - text.text = style->window.header.label_normal; - } - - /* draw header background */ - header.h += 1.0f; - - switch(background->type) { - case NK_STYLE_ITEM_IMAGE: - text.background = nk_rgba(0,0,0,0); - nk_draw_image(&win->buffer, header, &background->data.image, nk_white); - break; - case NK_STYLE_ITEM_NINE_SLICE: - text.background = nk_rgba(0, 0, 0, 0); - nk_draw_nine_slice(&win->buffer, header, &background->data.slice, nk_white); - break; - case NK_STYLE_ITEM_COLOR: - text.background = background->data.color; - nk_fill_rect(out, header, 0, background->data.color); - break; - } - - /* window close button */ - {struct nk_rect button; - button.y = header.y + style->window.header.padding.y; - button.h = header.h - 2 * style->window.header.padding.y; - button.w = button.h; - if (win->flags & NK_WINDOW_CLOSABLE) { - nk_flags ws = 0; - if (style->window.header.align == NK_HEADER_RIGHT) { - button.x = (header.w + header.x) - (button.w + style->window.header.padding.x); - header.w -= button.w + style->window.header.spacing.x + style->window.header.padding.x; - } else { - button.x = header.x + style->window.header.padding.x; - header.x += button.w + style->window.header.spacing.x + style->window.header.padding.x; - } - - if (nk_do_button_symbol(&ws, &win->buffer, button, - style->window.header.close_symbol, NK_BUTTON_DEFAULT, - &style->window.header.close_button, in, style->font) && !(win->flags & NK_WINDOW_ROM)) - { - layout->flags |= NK_WINDOW_HIDDEN; - layout->flags &= (nk_flags)~NK_WINDOW_MINIMIZED; - } - } - - /* window minimize button */ - if (win->flags & NK_WINDOW_MINIMIZABLE) { - nk_flags ws = 0; - if (style->window.header.align == NK_HEADER_RIGHT) { - button.x = (header.w + header.x) - button.w; - if (!(win->flags & NK_WINDOW_CLOSABLE)) { - button.x -= style->window.header.padding.x; - header.w -= style->window.header.padding.x; - } - header.w -= button.w + style->window.header.spacing.x; - } else { - button.x = header.x; - header.x += button.w + style->window.header.spacing.x + style->window.header.padding.x; - } - if (nk_do_button_symbol(&ws, &win->buffer, button, (layout->flags & NK_WINDOW_MINIMIZED)? - style->window.header.maximize_symbol: style->window.header.minimize_symbol, - NK_BUTTON_DEFAULT, &style->window.header.minimize_button, in, style->font) && !(win->flags & NK_WINDOW_ROM)) - layout->flags = (layout->flags & NK_WINDOW_MINIMIZED) ? - layout->flags & (nk_flags)~NK_WINDOW_MINIMIZED: - layout->flags | NK_WINDOW_MINIMIZED; - }} - - {/* window header title */ - int text_len = nk_strlen(title); - struct nk_rect label = {0,0,0,0}; - float t = font->width(font->userdata, font->height, title, text_len); - text.padding = nk_vec2(0,0); - - label.x = header.x + style->window.header.padding.x; - label.x += style->window.header.label_padding.x; - label.y = header.y + style->window.header.label_padding.y; - label.h = font->height + 2 * style->window.header.label_padding.y; - label.w = t + 2 * style->window.header.spacing.x; - label.w = NK_CLAMP(0, label.w, header.x + header.w - label.x); - nk_widget_text(out, label, (const char*)title, text_len, &text, NK_TEXT_LEFT, font);} - } - - /* draw window background */ - if (!(layout->flags & NK_WINDOW_MINIMIZED) && !(layout->flags & NK_WINDOW_DYNAMIC)) { - struct nk_rect body; - body.x = win->bounds.x; - body.w = win->bounds.w; - body.y = (win->bounds.y + layout->header_height); - body.h = (win->bounds.h - layout->header_height); - - switch(style->window.fixed_background.type) { - case NK_STYLE_ITEM_IMAGE: - nk_draw_image(out, body, &style->window.fixed_background.data.image, nk_white); - break; - case NK_STYLE_ITEM_NINE_SLICE: - nk_draw_nine_slice(out, body, &style->window.fixed_background.data.slice, nk_white); - break; - case NK_STYLE_ITEM_COLOR: - nk_fill_rect(out, body, 0, style->window.fixed_background.data.color); - break; - } - } - - /* set clipping rectangle */ - {struct nk_rect clip; - layout->clip = layout->bounds; - nk_unify(&clip, &win->buffer.clip, layout->clip.x, layout->clip.y, - layout->clip.x + layout->clip.w, layout->clip.y + layout->clip.h); - nk_push_scissor(out, clip); - layout->clip = clip;} - return !(layout->flags & NK_WINDOW_HIDDEN) && !(layout->flags & NK_WINDOW_MINIMIZED); -} -NK_LIB void -nk_panel_end(struct nk_context *ctx) -{ - struct nk_input *in; - struct nk_window *window; - struct nk_panel *layout; - const struct nk_style *style; - struct nk_command_buffer *out; - - struct nk_vec2 scrollbar_size; - struct nk_vec2 panel_padding; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return; - - window = ctx->current; - layout = window->layout; - style = &ctx->style; - out = &window->buffer; - in = (layout->flags & NK_WINDOW_ROM || layout->flags & NK_WINDOW_NO_INPUT) ? 0 :&ctx->input; - if (!nk_panel_is_sub(layout->type)) - nk_push_scissor(out, nk_null_rect); - - /* cache configuration data */ - scrollbar_size = style->window.scrollbar_size; - panel_padding = nk_panel_get_padding(style, layout->type); - - /* update the current cursor Y-position to point over the last added widget */ - layout->at_y += layout->row.height; - - /* dynamic panels */ - if (layout->flags & NK_WINDOW_DYNAMIC && !(layout->flags & NK_WINDOW_MINIMIZED)) - { - /* update panel height to fit dynamic growth */ - struct nk_rect empty_space; - if (layout->at_y < (layout->bounds.y + layout->bounds.h)) - layout->bounds.h = layout->at_y - layout->bounds.y; - - /* fill top empty space */ - empty_space.x = window->bounds.x; - empty_space.y = layout->bounds.y; - empty_space.h = panel_padding.y; - empty_space.w = window->bounds.w; - nk_fill_rect(out, empty_space, 0, style->window.background); - - /* fill left empty space */ - empty_space.x = window->bounds.x; - empty_space.y = layout->bounds.y; - empty_space.w = panel_padding.x + layout->border; - empty_space.h = layout->bounds.h; - nk_fill_rect(out, empty_space, 0, style->window.background); - - /* fill right empty space */ - empty_space.x = layout->bounds.x + layout->bounds.w; - empty_space.y = layout->bounds.y; - empty_space.w = panel_padding.x + layout->border; - empty_space.h = layout->bounds.h; - if (*layout->offset_y == 0 && !(layout->flags & NK_WINDOW_NO_SCROLLBAR)) - empty_space.w += scrollbar_size.x; - nk_fill_rect(out, empty_space, 0, style->window.background); - - /* fill bottom empty space */ - if (layout->footer_height > 0) { - empty_space.x = window->bounds.x; - empty_space.y = layout->bounds.y + layout->bounds.h; - empty_space.w = window->bounds.w; - empty_space.h = layout->footer_height; - nk_fill_rect(out, empty_space, 0, style->window.background); - } - } - - /* scrollbars */ - if (!(layout->flags & NK_WINDOW_NO_SCROLLBAR) && - !(layout->flags & NK_WINDOW_MINIMIZED) && - window->scrollbar_hiding_timer < NK_SCROLLBAR_HIDING_TIMEOUT) - { - struct nk_rect scroll; - int scroll_has_scrolling; - float scroll_target; - float scroll_offset; - float scroll_step; - float scroll_inc; - - /* mouse wheel scrolling */ - if (nk_panel_is_sub(layout->type)) - { - /* sub-window mouse wheel scrolling */ - struct nk_window *root_window = window; - struct nk_panel *root_panel = window->layout; - while (root_panel->parent) - root_panel = root_panel->parent; - while (root_window->parent) - root_window = root_window->parent; - - /* only allow scrolling if parent window is active */ - scroll_has_scrolling = 0; - if ((root_window == ctx->active) && layout->has_scrolling) { - /* and panel is being hovered and inside clip rect*/ - if (nk_input_is_mouse_hovering_rect(in, layout->bounds) && - NK_INTERSECT(layout->bounds.x, layout->bounds.y, layout->bounds.w, layout->bounds.h, - root_panel->clip.x, root_panel->clip.y, root_panel->clip.w, root_panel->clip.h)) - { - /* deactivate all parent scrolling */ - root_panel = window->layout; - while (root_panel->parent) { - root_panel->has_scrolling = nk_false; - root_panel = root_panel->parent; - } - root_panel->has_scrolling = nk_false; - scroll_has_scrolling = nk_true; - } - } - } else if (!nk_panel_is_sub(layout->type)) { - /* window mouse wheel scrolling */ - scroll_has_scrolling = (window == ctx->active) && layout->has_scrolling; - if (in && (in->mouse.scroll_delta.y > 0 || in->mouse.scroll_delta.x > 0) && scroll_has_scrolling) - window->scrolled = nk_true; - else window->scrolled = nk_false; - } else scroll_has_scrolling = nk_false; - - { - /* vertical scrollbar */ - nk_flags state = 0; - scroll.x = layout->bounds.x + layout->bounds.w + panel_padding.x; - scroll.y = layout->bounds.y; - scroll.w = scrollbar_size.x; - scroll.h = layout->bounds.h; - - scroll_offset = (float)*layout->offset_y; - scroll_step = scroll.h * 0.10f; - scroll_inc = scroll.h * 0.01f; - scroll_target = (float)(int)(layout->at_y - scroll.y); - scroll_offset = nk_do_scrollbarv(&state, out, scroll, scroll_has_scrolling, - scroll_offset, scroll_target, scroll_step, scroll_inc, - &ctx->style.scrollv, in, style->font); - *layout->offset_y = (nk_uint)scroll_offset; - if (in && scroll_has_scrolling) - in->mouse.scroll_delta.y = 0; - } - { - /* horizontal scrollbar */ - nk_flags state = 0; - scroll.x = layout->bounds.x; - scroll.y = layout->bounds.y + layout->bounds.h; - scroll.w = layout->bounds.w; - scroll.h = scrollbar_size.y; - - scroll_offset = (float)*layout->offset_x; - scroll_target = (float)(int)(layout->max_x - scroll.x); - scroll_step = layout->max_x * 0.05f; - scroll_inc = layout->max_x * 0.005f; - scroll_offset = nk_do_scrollbarh(&state, out, scroll, scroll_has_scrolling, - scroll_offset, scroll_target, scroll_step, scroll_inc, - &ctx->style.scrollh, in, style->font); - *layout->offset_x = (nk_uint)scroll_offset; - } - } - - /* hide scroll if no user input */ - if (window->flags & NK_WINDOW_SCROLL_AUTO_HIDE) { - int has_input = ctx->input.mouse.delta.x != 0 || ctx->input.mouse.delta.y != 0 || ctx->input.mouse.scroll_delta.y != 0; - int is_window_hovered = nk_window_is_hovered(ctx); - int any_item_active = (ctx->last_widget_state & NK_WIDGET_STATE_MODIFIED); - if ((!has_input && is_window_hovered) || (!is_window_hovered && !any_item_active)) - window->scrollbar_hiding_timer += ctx->delta_time_seconds; - else window->scrollbar_hiding_timer = 0; - } else window->scrollbar_hiding_timer = 0; - - /* window border */ - if (layout->flags & NK_WINDOW_BORDER) - { - struct nk_color border_color = nk_panel_get_border_color(style, layout->type); - const float padding_y = (layout->flags & NK_WINDOW_MINIMIZED) - ? (style->window.border + window->bounds.y + layout->header_height) - : ((layout->flags & NK_WINDOW_DYNAMIC) - ? (layout->bounds.y + layout->bounds.h + layout->footer_height) - : (window->bounds.y + window->bounds.h)); - struct nk_rect b = window->bounds; - b.h = padding_y - window->bounds.y; - nk_stroke_rect(out, b, 0, layout->border, border_color); - } - - /* scaler */ - if ((layout->flags & NK_WINDOW_SCALABLE) && in && !(layout->flags & NK_WINDOW_MINIMIZED)) - { - /* calculate scaler bounds */ - struct nk_rect scaler; - scaler.w = scrollbar_size.x; - scaler.h = scrollbar_size.y; - scaler.y = layout->bounds.y + layout->bounds.h; - if (layout->flags & NK_WINDOW_SCALE_LEFT) - scaler.x = layout->bounds.x - panel_padding.x * 0.5f; - else scaler.x = layout->bounds.x + layout->bounds.w + panel_padding.x; - if (layout->flags & NK_WINDOW_NO_SCROLLBAR) - scaler.x -= scaler.w; - - /* draw scaler */ - {const struct nk_style_item *item = &style->window.scaler; - if (item->type == NK_STYLE_ITEM_IMAGE) - nk_draw_image(out, scaler, &item->data.image, nk_white); - else { - if (layout->flags & NK_WINDOW_SCALE_LEFT) { - nk_fill_triangle(out, scaler.x, scaler.y, scaler.x, - scaler.y + scaler.h, scaler.x + scaler.w, - scaler.y + scaler.h, item->data.color); - } else { - nk_fill_triangle(out, scaler.x + scaler.w, scaler.y, scaler.x + scaler.w, - scaler.y + scaler.h, scaler.x, scaler.y + scaler.h, item->data.color); - } - }} - - /* do window scaling */ - if (!(window->flags & NK_WINDOW_ROM)) { - struct nk_vec2 window_size = style->window.min_size; - int left_mouse_down = in->mouse.buttons[NK_BUTTON_LEFT].down; - int left_mouse_click_in_scaler = nk_input_has_mouse_click_down_in_rect(in, - NK_BUTTON_LEFT, scaler, nk_true); - - if (left_mouse_down && left_mouse_click_in_scaler) { - float delta_x = in->mouse.delta.x; - if (layout->flags & NK_WINDOW_SCALE_LEFT) { - delta_x = -delta_x; - window->bounds.x += in->mouse.delta.x; - } - /* dragging in x-direction */ - if (window->bounds.w + delta_x >= window_size.x) { - if ((delta_x < 0) || (delta_x > 0 && in->mouse.pos.x >= scaler.x)) { - window->bounds.w = window->bounds.w + delta_x; - scaler.x += in->mouse.delta.x; - } - } - /* dragging in y-direction (only possible if static window) */ - if (!(layout->flags & NK_WINDOW_DYNAMIC)) { - if (window_size.y < window->bounds.h + in->mouse.delta.y) { - if ((in->mouse.delta.y < 0) || (in->mouse.delta.y > 0 && in->mouse.pos.y >= scaler.y)) { - window->bounds.h = window->bounds.h + in->mouse.delta.y; - scaler.y += in->mouse.delta.y; - } - } - } - ctx->style.cursor_active = ctx->style.cursors[NK_CURSOR_RESIZE_TOP_RIGHT_DOWN_LEFT]; - in->mouse.buttons[NK_BUTTON_LEFT].clicked_pos.x = scaler.x + scaler.w/2.0f; - in->mouse.buttons[NK_BUTTON_LEFT].clicked_pos.y = scaler.y + scaler.h/2.0f; - } - } - } - if (!nk_panel_is_sub(layout->type)) { - /* window is hidden so clear command buffer */ - if (layout->flags & NK_WINDOW_HIDDEN) - nk_command_buffer_reset(&window->buffer); - /* window is visible and not tab */ - else nk_finish(ctx, window); - } - - /* NK_WINDOW_REMOVE_ROM flag was set so remove NK_WINDOW_ROM */ - if (layout->flags & NK_WINDOW_REMOVE_ROM) { - layout->flags &= ~(nk_flags)NK_WINDOW_ROM; - layout->flags &= ~(nk_flags)NK_WINDOW_REMOVE_ROM; - } - window->flags = layout->flags; - - /* property garbage collector */ - if (window->property.active && window->property.old != window->property.seq && - window->property.active == window->property.prev) { - nk_zero(&window->property, sizeof(window->property)); - } else { - window->property.old = window->property.seq; - window->property.prev = window->property.active; - window->property.seq = 0; - } - /* edit garbage collector */ - if (window->edit.active && window->edit.old != window->edit.seq && - window->edit.active == window->edit.prev) { - nk_zero(&window->edit, sizeof(window->edit)); - } else { - window->edit.old = window->edit.seq; - window->edit.prev = window->edit.active; - window->edit.seq = 0; - } - /* contextual garbage collector */ - if (window->popup.active_con && window->popup.con_old != window->popup.con_count) { - window->popup.con_count = 0; - window->popup.con_old = 0; - window->popup.active_con = 0; - } else { - window->popup.con_old = window->popup.con_count; - window->popup.con_count = 0; - } - window->popup.combo_count = 0; - /* helper to make sure you have a 'nk_tree_push' for every 'nk_tree_pop' */ - NK_ASSERT(!layout->row.tree_depth); -} - - - - - -/* =============================================================== - * - * WINDOW - * - * ===============================================================*/ -NK_LIB void* -nk_create_window(struct nk_context *ctx) -{ - struct nk_page_element *elem; - elem = nk_create_page_element(ctx); - if (!elem) return 0; - elem->data.win.seq = ctx->seq; - return &elem->data.win; -} -NK_LIB void -nk_free_window(struct nk_context *ctx, struct nk_window *win) -{ - /* unlink windows from list */ - struct nk_table *it = win->tables; - if (win->popup.win) { - nk_free_window(ctx, win->popup.win); - win->popup.win = 0; - } - win->next = 0; - win->prev = 0; - - while (it) { - /*free window state tables */ - struct nk_table *n = it->next; - nk_remove_table(win, it); - nk_free_table(ctx, it); - if (it == win->tables) - win->tables = n; - it = n; - } - - /* link windows into freelist */ - {union nk_page_data *pd = NK_CONTAINER_OF(win, union nk_page_data, win); - struct nk_page_element *pe = NK_CONTAINER_OF(pd, struct nk_page_element, data); - nk_free_page_element(ctx, pe);} -} -NK_LIB struct nk_window* -nk_find_window(struct nk_context *ctx, nk_hash hash, const char *name) -{ - struct nk_window *iter; - iter = ctx->begin; - while (iter) { - NK_ASSERT(iter != iter->next); - if (iter->name == hash) { - int max_len = nk_strlen(iter->name_string); - if (!nk_stricmpn(iter->name_string, name, max_len)) - return iter; - } - iter = iter->next; - } - return 0; -} -NK_LIB void -nk_insert_window(struct nk_context *ctx, struct nk_window *win, - enum nk_window_insert_location loc) -{ - const struct nk_window *iter; - NK_ASSERT(ctx); - NK_ASSERT(win); - if (!win || !ctx) return; - - iter = ctx->begin; - while (iter) { - NK_ASSERT(iter != iter->next); - NK_ASSERT(iter != win); - if (iter == win) return; - iter = iter->next; - } - - if (!ctx->begin) { - win->next = 0; - win->prev = 0; - ctx->begin = win; - ctx->end = win; - ctx->count = 1; - return; - } - if (loc == NK_INSERT_BACK) { - struct nk_window *end; - end = ctx->end; - end->flags |= NK_WINDOW_ROM; - end->next = win; - win->prev = ctx->end; - win->next = 0; - ctx->end = win; - ctx->active = ctx->end; - ctx->end->flags &= ~(nk_flags)NK_WINDOW_ROM; - } else { - /*ctx->end->flags |= NK_WINDOW_ROM;*/ - ctx->begin->prev = win; - win->next = ctx->begin; - win->prev = 0; - ctx->begin = win; - ctx->begin->flags &= ~(nk_flags)NK_WINDOW_ROM; - } - ctx->count++; -} -NK_LIB void -nk_remove_window(struct nk_context *ctx, struct nk_window *win) -{ - if (win == ctx->begin || win == ctx->end) { - if (win == ctx->begin) { - ctx->begin = win->next; - if (win->next) - win->next->prev = 0; - } - if (win == ctx->end) { - ctx->end = win->prev; - if (win->prev) - win->prev->next = 0; - } - } else { - if (win->next) - win->next->prev = win->prev; - if (win->prev) - win->prev->next = win->next; - } - if (win == ctx->active || !ctx->active) { - ctx->active = ctx->end; - if (ctx->end) - ctx->end->flags &= ~(nk_flags)NK_WINDOW_ROM; - } - win->next = 0; - win->prev = 0; - ctx->count--; -} -NK_API nk_bool -nk_begin(struct nk_context *ctx, const char *title, - struct nk_rect bounds, nk_flags flags) -{ - return nk_begin_titled(ctx, title, title, bounds, flags); -} -NK_API nk_bool -nk_begin_titled(struct nk_context *ctx, const char *name, const char *title, - struct nk_rect bounds, nk_flags flags) -{ - struct nk_window *win; - struct nk_style *style; - nk_hash name_hash; - int name_len; - int ret = 0; - - NK_ASSERT(ctx); - NK_ASSERT(name); - NK_ASSERT(title); - NK_ASSERT(ctx->style.font && ctx->style.font->width && "if this triggers you forgot to add a font"); - NK_ASSERT(!ctx->current && "if this triggers you missed a `nk_end` call"); - if (!ctx || ctx->current || !title || !name) - return 0; - - /* find or create window */ - style = &ctx->style; - name_len = (int)nk_strlen(name); - name_hash = nk_murmur_hash(name, (int)name_len, NK_WINDOW_TITLE); - win = nk_find_window(ctx, name_hash, name); - if (!win) { - /* create new window */ - nk_size name_length = (nk_size)name_len; - win = (struct nk_window*)nk_create_window(ctx); - NK_ASSERT(win); - if (!win) return 0; - - if (flags & NK_WINDOW_BACKGROUND) - nk_insert_window(ctx, win, NK_INSERT_FRONT); - else nk_insert_window(ctx, win, NK_INSERT_BACK); - nk_command_buffer_init(&win->buffer, &ctx->memory, NK_CLIPPING_ON); - - win->flags = flags; - win->bounds = bounds; - win->name = name_hash; - name_length = NK_MIN(name_length, NK_WINDOW_MAX_NAME-1); - NK_MEMCPY(win->name_string, name, name_length); - win->name_string[name_length] = 0; - win->popup.win = 0; - if (!ctx->active) - ctx->active = win; - } else { - /* update window */ - win->flags &= ~(nk_flags)(NK_WINDOW_PRIVATE-1); - win->flags |= flags; - if (!(win->flags & (NK_WINDOW_MOVABLE | NK_WINDOW_SCALABLE))) - win->bounds = bounds; - /* If this assert triggers you either: - * - * I.) Have more than one window with the same name or - * II.) You forgot to actually draw the window. - * More specific you did not call `nk_clear` (nk_clear will be - * automatically called for you if you are using one of the - * provided demo backends). */ - NK_ASSERT(win->seq != ctx->seq); - win->seq = ctx->seq; - if (!ctx->active && !(win->flags & NK_WINDOW_HIDDEN)) { - ctx->active = win; - ctx->end = win; - } - } - if (win->flags & NK_WINDOW_HIDDEN) { - ctx->current = win; - win->layout = 0; - return 0; - } else nk_start(ctx, win); - - /* window overlapping */ - if (!(win->flags & NK_WINDOW_HIDDEN) && !(win->flags & NK_WINDOW_NO_INPUT)) - { - int inpanel, ishovered; - struct nk_window *iter = win; - float h = ctx->style.font->height + 2.0f * style->window.header.padding.y + - (2.0f * style->window.header.label_padding.y); - struct nk_rect win_bounds = (!(win->flags & NK_WINDOW_MINIMIZED))? - win->bounds: nk_rect(win->bounds.x, win->bounds.y, win->bounds.w, h); - - /* activate window if hovered and no other window is overlapping this window */ - inpanel = nk_input_has_mouse_click_down_in_rect(&ctx->input, NK_BUTTON_LEFT, win_bounds, nk_true); - inpanel = inpanel && ctx->input.mouse.buttons[NK_BUTTON_LEFT].clicked; - ishovered = nk_input_is_mouse_hovering_rect(&ctx->input, win_bounds); - if ((win != ctx->active) && ishovered && !ctx->input.mouse.buttons[NK_BUTTON_LEFT].down) { - iter = win->next; - while (iter) { - struct nk_rect iter_bounds = (!(iter->flags & NK_WINDOW_MINIMIZED))? - iter->bounds: nk_rect(iter->bounds.x, iter->bounds.y, iter->bounds.w, h); - if (NK_INTERSECT(win_bounds.x, win_bounds.y, win_bounds.w, win_bounds.h, - iter_bounds.x, iter_bounds.y, iter_bounds.w, iter_bounds.h) && - (!(iter->flags & NK_WINDOW_HIDDEN))) - break; - - if (iter->popup.win && iter->popup.active && !(iter->flags & NK_WINDOW_HIDDEN) && - NK_INTERSECT(win->bounds.x, win_bounds.y, win_bounds.w, win_bounds.h, - iter->popup.win->bounds.x, iter->popup.win->bounds.y, - iter->popup.win->bounds.w, iter->popup.win->bounds.h)) - break; - iter = iter->next; - } - } - - /* activate window if clicked */ - if (iter && inpanel && (win != ctx->end)) { - iter = win->next; - while (iter) { - /* try to find a panel with higher priority in the same position */ - struct nk_rect iter_bounds = (!(iter->flags & NK_WINDOW_MINIMIZED))? - iter->bounds: nk_rect(iter->bounds.x, iter->bounds.y, iter->bounds.w, h); - if (NK_INBOX(ctx->input.mouse.pos.x, ctx->input.mouse.pos.y, - iter_bounds.x, iter_bounds.y, iter_bounds.w, iter_bounds.h) && - !(iter->flags & NK_WINDOW_HIDDEN)) - break; - if (iter->popup.win && iter->popup.active && !(iter->flags & NK_WINDOW_HIDDEN) && - NK_INTERSECT(win_bounds.x, win_bounds.y, win_bounds.w, win_bounds.h, - iter->popup.win->bounds.x, iter->popup.win->bounds.y, - iter->popup.win->bounds.w, iter->popup.win->bounds.h)) - break; - iter = iter->next; - } - } - if (iter && !(win->flags & NK_WINDOW_ROM) && (win->flags & NK_WINDOW_BACKGROUND)) { - win->flags |= (nk_flags)NK_WINDOW_ROM; - iter->flags &= ~(nk_flags)NK_WINDOW_ROM; - ctx->active = iter; - if (!(iter->flags & NK_WINDOW_BACKGROUND)) { - /* current window is active in that position so transfer to top - * at the highest priority in stack */ - nk_remove_window(ctx, iter); - nk_insert_window(ctx, iter, NK_INSERT_BACK); - } - } else { - if (!iter && ctx->end != win) { - if (!(win->flags & NK_WINDOW_BACKGROUND)) { - /* current window is active in that position so transfer to top - * at the highest priority in stack */ - nk_remove_window(ctx, win); - nk_insert_window(ctx, win, NK_INSERT_BACK); - } - win->flags &= ~(nk_flags)NK_WINDOW_ROM; - ctx->active = win; - } - if (ctx->end != win && !(win->flags & NK_WINDOW_BACKGROUND)) - win->flags |= NK_WINDOW_ROM; - } - } - win->layout = (struct nk_panel*)nk_create_panel(ctx); - ctx->current = win; - ret = nk_panel_begin(ctx, title, NK_PANEL_WINDOW); - win->layout->offset_x = &win->scrollbar.x; - win->layout->offset_y = &win->scrollbar.y; - return ret; -} -NK_API void -nk_end(struct nk_context *ctx) -{ - struct nk_panel *layout; - NK_ASSERT(ctx); - NK_ASSERT(ctx->current && "if this triggers you forgot to call `nk_begin`"); - if (!ctx || !ctx->current) - return; - - layout = ctx->current->layout; - if (!layout || (layout->type == NK_PANEL_WINDOW && (ctx->current->flags & NK_WINDOW_HIDDEN))) { - ctx->current = 0; - return; - } - nk_panel_end(ctx); - nk_free_panel(ctx, ctx->current->layout); - ctx->current = 0; -} -NK_API struct nk_rect -nk_window_get_bounds(const struct nk_context *ctx) -{ - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - if (!ctx || !ctx->current) return nk_rect(0,0,0,0); - return ctx->current->bounds; -} -NK_API struct nk_vec2 -nk_window_get_position(const struct nk_context *ctx) -{ - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - if (!ctx || !ctx->current) return nk_vec2(0,0); - return nk_vec2(ctx->current->bounds.x, ctx->current->bounds.y); -} -NK_API struct nk_vec2 -nk_window_get_size(const struct nk_context *ctx) -{ - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - if (!ctx || !ctx->current) return nk_vec2(0,0); - return nk_vec2(ctx->current->bounds.w, ctx->current->bounds.h); -} -NK_API float -nk_window_get_width(const struct nk_context *ctx) -{ - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - if (!ctx || !ctx->current) return 0; - return ctx->current->bounds.w; -} -NK_API float -nk_window_get_height(const struct nk_context *ctx) -{ - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - if (!ctx || !ctx->current) return 0; - return ctx->current->bounds.h; -} -NK_API struct nk_rect -nk_window_get_content_region(struct nk_context *ctx) -{ - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - if (!ctx || !ctx->current) return nk_rect(0,0,0,0); - return ctx->current->layout->clip; -} -NK_API struct nk_vec2 -nk_window_get_content_region_min(struct nk_context *ctx) -{ - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current) return nk_vec2(0,0); - return nk_vec2(ctx->current->layout->clip.x, ctx->current->layout->clip.y); -} -NK_API struct nk_vec2 -nk_window_get_content_region_max(struct nk_context *ctx) -{ - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current) return nk_vec2(0,0); - return nk_vec2(ctx->current->layout->clip.x + ctx->current->layout->clip.w, - ctx->current->layout->clip.y + ctx->current->layout->clip.h); -} -NK_API struct nk_vec2 -nk_window_get_content_region_size(struct nk_context *ctx) -{ - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current) return nk_vec2(0,0); - return nk_vec2(ctx->current->layout->clip.w, ctx->current->layout->clip.h); -} -NK_API struct nk_command_buffer* -nk_window_get_canvas(struct nk_context *ctx) -{ - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current) return 0; - return &ctx->current->buffer; -} -NK_API struct nk_panel* -nk_window_get_panel(struct nk_context *ctx) -{ - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - if (!ctx || !ctx->current) return 0; - return ctx->current->layout; -} -NK_API void -nk_window_get_scroll(struct nk_context *ctx, nk_uint *offset_x, nk_uint *offset_y) -{ - struct nk_window *win; - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - if (!ctx || !ctx->current) - return ; - win = ctx->current; - if (offset_x) - *offset_x = win->scrollbar.x; - if (offset_y) - *offset_y = win->scrollbar.y; -} -NK_API nk_bool -nk_window_has_focus(const struct nk_context *ctx) -{ - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current) return 0; - return ctx->current == ctx->active; -} -NK_API nk_bool -nk_window_is_hovered(struct nk_context *ctx) -{ - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - if (!ctx || !ctx->current || (ctx->current->flags & NK_WINDOW_HIDDEN)) - return 0; - else { - struct nk_rect actual_bounds = ctx->current->bounds; - if (ctx->begin->flags & NK_WINDOW_MINIMIZED) { - actual_bounds.h = ctx->current->layout->header_height; - } - return nk_input_is_mouse_hovering_rect(&ctx->input, actual_bounds); - } -} -NK_API nk_bool -nk_window_is_any_hovered(struct nk_context *ctx) -{ - struct nk_window *iter; - NK_ASSERT(ctx); - if (!ctx) return 0; - iter = ctx->begin; - while (iter) { - /* check if window is being hovered */ - if(!(iter->flags & NK_WINDOW_HIDDEN)) { - /* check if window popup is being hovered */ - if (iter->popup.active && iter->popup.win && nk_input_is_mouse_hovering_rect(&ctx->input, iter->popup.win->bounds)) - return 1; - - if (iter->flags & NK_WINDOW_MINIMIZED) { - struct nk_rect header = iter->bounds; - header.h = ctx->style.font->height + 2 * ctx->style.window.header.padding.y; - if (nk_input_is_mouse_hovering_rect(&ctx->input, header)) - return 1; - } else if (nk_input_is_mouse_hovering_rect(&ctx->input, iter->bounds)) { - return 1; - } - } - iter = iter->next; - } - return 0; -} -NK_API nk_bool -nk_item_is_any_active(struct nk_context *ctx) -{ - int any_hovered = nk_window_is_any_hovered(ctx); - int any_active = (ctx->last_widget_state & NK_WIDGET_STATE_MODIFIED); - return any_hovered || any_active; -} -NK_API nk_bool -nk_window_is_collapsed(struct nk_context *ctx, const char *name) -{ - int title_len; - nk_hash title_hash; - struct nk_window *win; - NK_ASSERT(ctx); - if (!ctx) return 0; - - title_len = (int)nk_strlen(name); - title_hash = nk_murmur_hash(name, (int)title_len, NK_WINDOW_TITLE); - win = nk_find_window(ctx, title_hash, name); - if (!win) return 0; - return win->flags & NK_WINDOW_MINIMIZED; -} -NK_API nk_bool -nk_window_is_closed(struct nk_context *ctx, const char *name) -{ - int title_len; - nk_hash title_hash; - struct nk_window *win; - NK_ASSERT(ctx); - if (!ctx) return 1; - - title_len = (int)nk_strlen(name); - title_hash = nk_murmur_hash(name, (int)title_len, NK_WINDOW_TITLE); - win = nk_find_window(ctx, title_hash, name); - if (!win) return 1; - return (win->flags & NK_WINDOW_CLOSED); -} -NK_API nk_bool -nk_window_is_hidden(struct nk_context *ctx, const char *name) -{ - int title_len; - nk_hash title_hash; - struct nk_window *win; - NK_ASSERT(ctx); - if (!ctx) return 1; - - title_len = (int)nk_strlen(name); - title_hash = nk_murmur_hash(name, (int)title_len, NK_WINDOW_TITLE); - win = nk_find_window(ctx, title_hash, name); - if (!win) return 1; - return (win->flags & NK_WINDOW_HIDDEN); -} -NK_API nk_bool -nk_window_is_active(struct nk_context *ctx, const char *name) -{ - int title_len; - nk_hash title_hash; - struct nk_window *win; - NK_ASSERT(ctx); - if (!ctx) return 0; - - title_len = (int)nk_strlen(name); - title_hash = nk_murmur_hash(name, (int)title_len, NK_WINDOW_TITLE); - win = nk_find_window(ctx, title_hash, name); - if (!win) return 0; - return win == ctx->active; -} -NK_API struct nk_window* -nk_window_find(struct nk_context *ctx, const char *name) -{ - int title_len; - nk_hash title_hash; - title_len = (int)nk_strlen(name); - title_hash = nk_murmur_hash(name, (int)title_len, NK_WINDOW_TITLE); - return nk_find_window(ctx, title_hash, name); -} -NK_API void -nk_window_close(struct nk_context *ctx, const char *name) -{ - struct nk_window *win; - NK_ASSERT(ctx); - if (!ctx) return; - win = nk_window_find(ctx, name); - if (!win) return; - NK_ASSERT(ctx->current != win && "You cannot close a currently active window"); - if (ctx->current == win) return; - win->flags |= NK_WINDOW_HIDDEN; - win->flags |= NK_WINDOW_CLOSED; -} -NK_API void -nk_window_set_bounds(struct nk_context *ctx, - const char *name, struct nk_rect bounds) -{ - struct nk_window *win; - NK_ASSERT(ctx); - if (!ctx) return; - win = nk_window_find(ctx, name); - if (!win) return; - NK_ASSERT(ctx->current != win && "You cannot update a currently in procecss window"); - win->bounds = bounds; -} -NK_API void -nk_window_set_position(struct nk_context *ctx, - const char *name, struct nk_vec2 pos) -{ - struct nk_window *win = nk_window_find(ctx, name); - if (!win) return; - win->bounds.x = pos.x; - win->bounds.y = pos.y; -} -NK_API void -nk_window_set_size(struct nk_context *ctx, - const char *name, struct nk_vec2 size) -{ - struct nk_window *win = nk_window_find(ctx, name); - if (!win) return; - win->bounds.w = size.x; - win->bounds.h = size.y; -} -NK_API void -nk_window_set_scroll(struct nk_context *ctx, nk_uint offset_x, nk_uint offset_y) -{ - struct nk_window *win; - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - if (!ctx || !ctx->current) - return; - win = ctx->current; - win->scrollbar.x = offset_x; - win->scrollbar.y = offset_y; -} -NK_API void -nk_window_collapse(struct nk_context *ctx, const char *name, - enum nk_collapse_states c) -{ - int title_len; - nk_hash title_hash; - struct nk_window *win; - NK_ASSERT(ctx); - if (!ctx) return; - - title_len = (int)nk_strlen(name); - title_hash = nk_murmur_hash(name, (int)title_len, NK_WINDOW_TITLE); - win = nk_find_window(ctx, title_hash, name); - if (!win) return; - if (c == NK_MINIMIZED) - win->flags |= NK_WINDOW_MINIMIZED; - else win->flags &= ~(nk_flags)NK_WINDOW_MINIMIZED; -} -NK_API void -nk_window_collapse_if(struct nk_context *ctx, const char *name, - enum nk_collapse_states c, int cond) -{ - NK_ASSERT(ctx); - if (!ctx || !cond) return; - nk_window_collapse(ctx, name, c); -} -NK_API void -nk_window_show(struct nk_context *ctx, const char *name, enum nk_show_states s) -{ - int title_len; - nk_hash title_hash; - struct nk_window *win; - NK_ASSERT(ctx); - if (!ctx) return; - - title_len = (int)nk_strlen(name); - title_hash = nk_murmur_hash(name, (int)title_len, NK_WINDOW_TITLE); - win = nk_find_window(ctx, title_hash, name); - if (!win) return; - if (s == NK_HIDDEN) { - win->flags |= NK_WINDOW_HIDDEN; - } else win->flags &= ~(nk_flags)NK_WINDOW_HIDDEN; -} -NK_API void -nk_window_show_if(struct nk_context *ctx, const char *name, - enum nk_show_states s, int cond) -{ - NK_ASSERT(ctx); - if (!ctx || !cond) return; - nk_window_show(ctx, name, s); -} - -NK_API void -nk_window_set_focus(struct nk_context *ctx, const char *name) -{ - int title_len; - nk_hash title_hash; - struct nk_window *win; - NK_ASSERT(ctx); - if (!ctx) return; - - title_len = (int)nk_strlen(name); - title_hash = nk_murmur_hash(name, (int)title_len, NK_WINDOW_TITLE); - win = nk_find_window(ctx, title_hash, name); - if (win && ctx->end != win) { - nk_remove_window(ctx, win); - nk_insert_window(ctx, win, NK_INSERT_BACK); - } - ctx->active = win; -} - - - - -/* =============================================================== - * - * POPUP - * - * ===============================================================*/ -NK_API nk_bool -nk_popup_begin(struct nk_context *ctx, enum nk_popup_type type, - const char *title, nk_flags flags, struct nk_rect rect) -{ - struct nk_window *popup; - struct nk_window *win; - struct nk_panel *panel; - - int title_len; - nk_hash title_hash; - nk_size allocated; - - NK_ASSERT(ctx); - NK_ASSERT(title); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return 0; - - win = ctx->current; - panel = win->layout; - NK_ASSERT(!(panel->type & NK_PANEL_SET_POPUP) && "popups are not allowed to have popups"); - (void)panel; - title_len = (int)nk_strlen(title); - title_hash = nk_murmur_hash(title, (int)title_len, NK_PANEL_POPUP); - - popup = win->popup.win; - if (!popup) { - popup = (struct nk_window*)nk_create_window(ctx); - popup->parent = win; - win->popup.win = popup; - win->popup.active = 0; - win->popup.type = NK_PANEL_POPUP; - } - - /* make sure we have correct popup */ - if (win->popup.name != title_hash) { - if (!win->popup.active) { - nk_zero(popup, sizeof(*popup)); - win->popup.name = title_hash; - win->popup.active = 1; - win->popup.type = NK_PANEL_POPUP; - } else return 0; - } - - /* popup position is local to window */ - ctx->current = popup; - rect.x += win->layout->clip.x; - rect.y += win->layout->clip.y; - - /* setup popup data */ - popup->parent = win; - popup->bounds = rect; - popup->seq = ctx->seq; - popup->layout = (struct nk_panel*)nk_create_panel(ctx); - popup->flags = flags; - popup->flags |= NK_WINDOW_BORDER; - if (type == NK_POPUP_DYNAMIC) - popup->flags |= NK_WINDOW_DYNAMIC; - - popup->buffer = win->buffer; - nk_start_popup(ctx, win); - allocated = ctx->memory.allocated; - nk_push_scissor(&popup->buffer, nk_null_rect); - - if (nk_panel_begin(ctx, title, NK_PANEL_POPUP)) { - /* popup is running therefore invalidate parent panels */ - struct nk_panel *root; - root = win->layout; - while (root) { - root->flags |= NK_WINDOW_ROM; - root->flags &= ~(nk_flags)NK_WINDOW_REMOVE_ROM; - root = root->parent; - } - win->popup.active = 1; - popup->layout->offset_x = &popup->scrollbar.x; - popup->layout->offset_y = &popup->scrollbar.y; - popup->layout->parent = win->layout; - return 1; - } else { - /* popup was closed/is invalid so cleanup */ - struct nk_panel *root; - root = win->layout; - while (root) { - root->flags |= NK_WINDOW_REMOVE_ROM; - root = root->parent; - } - win->popup.buf.active = 0; - win->popup.active = 0; - ctx->memory.allocated = allocated; - ctx->current = win; - nk_free_panel(ctx, popup->layout); - popup->layout = 0; - return 0; - } -} -NK_LIB nk_bool -nk_nonblock_begin(struct nk_context *ctx, - nk_flags flags, struct nk_rect body, struct nk_rect header, - enum nk_panel_type panel_type) -{ - struct nk_window *popup; - struct nk_window *win; - struct nk_panel *panel; - int is_active = nk_true; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return 0; - - /* popups cannot have popups */ - win = ctx->current; - panel = win->layout; - NK_ASSERT(!(panel->type & NK_PANEL_SET_POPUP)); - (void)panel; - popup = win->popup.win; - if (!popup) { - /* create window for nonblocking popup */ - popup = (struct nk_window*)nk_create_window(ctx); - popup->parent = win; - win->popup.win = popup; - win->popup.type = panel_type; - nk_command_buffer_init(&popup->buffer, &ctx->memory, NK_CLIPPING_ON); - } else { - /* close the popup if user pressed outside or in the header */ - int pressed, in_body, in_header; -#ifdef NK_BUTTON_TRIGGER_ON_RELEASE - pressed = nk_input_is_mouse_released(&ctx->input, NK_BUTTON_LEFT); -#else - pressed = nk_input_is_mouse_pressed(&ctx->input, NK_BUTTON_LEFT); -#endif - in_body = nk_input_is_mouse_hovering_rect(&ctx->input, body); - in_header = nk_input_is_mouse_hovering_rect(&ctx->input, header); - if (pressed && (!in_body || in_header)) - is_active = nk_false; - } - win->popup.header = header; - - if (!is_active) { - /* remove read only mode from all parent panels */ - struct nk_panel *root = win->layout; - while (root) { - root->flags |= NK_WINDOW_REMOVE_ROM; - root = root->parent; - } - return is_active; - } - popup->bounds = body; - popup->parent = win; - popup->layout = (struct nk_panel*)nk_create_panel(ctx); - popup->flags = flags; - popup->flags |= NK_WINDOW_BORDER; - popup->flags |= NK_WINDOW_DYNAMIC; - popup->seq = ctx->seq; - win->popup.active = 1; - NK_ASSERT(popup->layout); - - nk_start_popup(ctx, win); - popup->buffer = win->buffer; - nk_push_scissor(&popup->buffer, nk_null_rect); - ctx->current = popup; - - nk_panel_begin(ctx, 0, panel_type); - win->buffer = popup->buffer; - popup->layout->parent = win->layout; - popup->layout->offset_x = &popup->scrollbar.x; - popup->layout->offset_y = &popup->scrollbar.y; - - /* set read only mode to all parent panels */ - {struct nk_panel *root; - root = win->layout; - while (root) { - root->flags |= NK_WINDOW_ROM; - root = root->parent; - }} - return is_active; -} -NK_API void -nk_popup_close(struct nk_context *ctx) -{ - struct nk_window *popup; - NK_ASSERT(ctx); - if (!ctx || !ctx->current) return; - - popup = ctx->current; - NK_ASSERT(popup->parent); - NK_ASSERT(popup->layout->type & NK_PANEL_SET_POPUP); - popup->flags |= NK_WINDOW_HIDDEN; -} -NK_API void -nk_popup_end(struct nk_context *ctx) -{ - struct nk_window *win; - struct nk_window *popup; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return; - - popup = ctx->current; - if (!popup->parent) return; - win = popup->parent; - if (popup->flags & NK_WINDOW_HIDDEN) { - struct nk_panel *root; - root = win->layout; - while (root) { - root->flags |= NK_WINDOW_REMOVE_ROM; - root = root->parent; - } - win->popup.active = 0; - } - nk_push_scissor(&popup->buffer, nk_null_rect); - nk_end(ctx); - - win->buffer = popup->buffer; - nk_finish_popup(ctx, win); - ctx->current = win; - nk_push_scissor(&win->buffer, win->layout->clip); -} -NK_API void -nk_popup_get_scroll(struct nk_context *ctx, nk_uint *offset_x, nk_uint *offset_y) -{ - struct nk_window *popup; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return; - - popup = ctx->current; - if (offset_x) - *offset_x = popup->scrollbar.x; - if (offset_y) - *offset_y = popup->scrollbar.y; -} -NK_API void -nk_popup_set_scroll(struct nk_context *ctx, nk_uint offset_x, nk_uint offset_y) -{ - struct nk_window *popup; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return; - - popup = ctx->current; - popup->scrollbar.x = offset_x; - popup->scrollbar.y = offset_y; -} - - - - -/* ============================================================== - * - * CONTEXTUAL - * - * ===============================================================*/ -NK_API nk_bool -nk_contextual_begin(struct nk_context *ctx, nk_flags flags, struct nk_vec2 size, - struct nk_rect trigger_bounds) -{ - struct nk_window *win; - struct nk_window *popup; - struct nk_rect body; - - NK_STORAGE const struct nk_rect null_rect = {-1,-1,0,0}; - int is_clicked = 0; - int is_open = 0; - int ret = 0; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return 0; - - win = ctx->current; - ++win->popup.con_count; - if (ctx->current != ctx->active) - return 0; - - /* check if currently active contextual is active */ - popup = win->popup.win; - is_open = (popup && win->popup.type == NK_PANEL_CONTEXTUAL); - is_clicked = nk_input_mouse_clicked(&ctx->input, NK_BUTTON_RIGHT, trigger_bounds); - if (win->popup.active_con && win->popup.con_count != win->popup.active_con) - return 0; - if (!is_open && win->popup.active_con) - win->popup.active_con = 0; - if ((!is_open && !is_clicked)) - return 0; - - /* calculate contextual position on click */ - win->popup.active_con = win->popup.con_count; - if (is_clicked) { - body.x = ctx->input.mouse.pos.x; - body.y = ctx->input.mouse.pos.y; - } else { - body.x = popup->bounds.x; - body.y = popup->bounds.y; - } - body.w = size.x; - body.h = size.y; - - /* start nonblocking contextual popup */ - ret = nk_nonblock_begin(ctx, flags|NK_WINDOW_NO_SCROLLBAR, body, - null_rect, NK_PANEL_CONTEXTUAL); - if (ret) win->popup.type = NK_PANEL_CONTEXTUAL; - else { - win->popup.active_con = 0; - win->popup.type = NK_PANEL_NONE; - if (win->popup.win) - win->popup.win->flags = 0; - } - return ret; -} -NK_API nk_bool -nk_contextual_item_text(struct nk_context *ctx, const char *text, int len, - nk_flags alignment) -{ - struct nk_window *win; - const struct nk_input *in; - const struct nk_style *style; - - struct nk_rect bounds; - enum nk_widget_layout_states state; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return 0; - - win = ctx->current; - style = &ctx->style; - state = nk_widget_fitting(&bounds, ctx, style->contextual_button.padding); - if (!state) return nk_false; - - in = (state == NK_WIDGET_ROM || win->layout->flags & NK_WINDOW_ROM) ? 0 : &ctx->input; - if (nk_do_button_text(&ctx->last_widget_state, &win->buffer, bounds, - text, len, alignment, NK_BUTTON_DEFAULT, &style->contextual_button, in, style->font)) { - nk_contextual_close(ctx); - return nk_true; - } - return nk_false; -} -NK_API nk_bool -nk_contextual_item_label(struct nk_context *ctx, const char *label, nk_flags align) -{ - return nk_contextual_item_text(ctx, label, nk_strlen(label), align); -} -NK_API nk_bool -nk_contextual_item_image_text(struct nk_context *ctx, struct nk_image img, - const char *text, int len, nk_flags align) -{ - struct nk_window *win; - const struct nk_input *in; - const struct nk_style *style; - - struct nk_rect bounds; - enum nk_widget_layout_states state; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return 0; - - win = ctx->current; - style = &ctx->style; - state = nk_widget_fitting(&bounds, ctx, style->contextual_button.padding); - if (!state) return nk_false; - - in = (state == NK_WIDGET_ROM || win->layout->flags & NK_WINDOW_ROM) ? 0 : &ctx->input; - if (nk_do_button_text_image(&ctx->last_widget_state, &win->buffer, bounds, - img, text, len, align, NK_BUTTON_DEFAULT, &style->contextual_button, style->font, in)){ - nk_contextual_close(ctx); - return nk_true; - } - return nk_false; -} -NK_API nk_bool -nk_contextual_item_image_label(struct nk_context *ctx, struct nk_image img, - const char *label, nk_flags align) -{ - return nk_contextual_item_image_text(ctx, img, label, nk_strlen(label), align); -} -NK_API nk_bool -nk_contextual_item_symbol_text(struct nk_context *ctx, enum nk_symbol_type symbol, - const char *text, int len, nk_flags align) -{ - struct nk_window *win; - const struct nk_input *in; - const struct nk_style *style; - - struct nk_rect bounds; - enum nk_widget_layout_states state; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return 0; - - win = ctx->current; - style = &ctx->style; - state = nk_widget_fitting(&bounds, ctx, style->contextual_button.padding); - if (!state) return nk_false; - - in = (state == NK_WIDGET_ROM || win->layout->flags & NK_WINDOW_ROM) ? 0 : &ctx->input; - if (nk_do_button_text_symbol(&ctx->last_widget_state, &win->buffer, bounds, - symbol, text, len, align, NK_BUTTON_DEFAULT, &style->contextual_button, style->font, in)) { - nk_contextual_close(ctx); - return nk_true; - } - return nk_false; -} -NK_API nk_bool -nk_contextual_item_symbol_label(struct nk_context *ctx, enum nk_symbol_type symbol, - const char *text, nk_flags align) -{ - return nk_contextual_item_symbol_text(ctx, symbol, text, nk_strlen(text), align); -} -NK_API void -nk_contextual_close(struct nk_context *ctx) -{ - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) return; - nk_popup_close(ctx); -} -NK_API void -nk_contextual_end(struct nk_context *ctx) -{ - struct nk_window *popup; - struct nk_panel *panel; - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - if (!ctx || !ctx->current) return; - - popup = ctx->current; - panel = popup->layout; - NK_ASSERT(popup->parent); - NK_ASSERT(panel->type & NK_PANEL_SET_POPUP); - if (panel->flags & NK_WINDOW_DYNAMIC) { - /* Close behavior - This is a bit of a hack solution since we do not know before we end our popup - how big it will be. We therefore do not directly know when a - click outside the non-blocking popup must close it at that direct frame. - Instead it will be closed in the next frame.*/ - struct nk_rect body = {0,0,0,0}; - if (panel->at_y < (panel->bounds.y + panel->bounds.h)) { - struct nk_vec2 padding = nk_panel_get_padding(&ctx->style, panel->type); - body = panel->bounds; - body.y = (panel->at_y + panel->footer_height + panel->border + padding.y + panel->row.height); - body.h = (panel->bounds.y + panel->bounds.h) - body.y; - } - {int pressed = nk_input_is_mouse_pressed(&ctx->input, NK_BUTTON_LEFT); - int in_body = nk_input_is_mouse_hovering_rect(&ctx->input, body); - if (pressed && in_body) - popup->flags |= NK_WINDOW_HIDDEN; - } - } - if (popup->flags & NK_WINDOW_HIDDEN) - popup->seq = 0; - nk_popup_end(ctx); - return; -} - - - - - -/* =============================================================== - * - * MENU - * - * ===============================================================*/ -NK_API void -nk_menubar_begin(struct nk_context *ctx) -{ - struct nk_panel *layout; - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return; - - layout = ctx->current->layout; - NK_ASSERT(layout->at_y == layout->bounds.y); - /* if this assert triggers you allocated space between nk_begin and nk_menubar_begin. - If you want a menubar the first nuklear function after `nk_begin` has to be a - `nk_menubar_begin` call. Inside the menubar you then have to allocate space for - widgets (also supports multiple rows). - Example: - if (nk_begin(...)) { - nk_menubar_begin(...); - nk_layout_xxxx(...); - nk_button(...); - nk_layout_xxxx(...); - nk_button(...); - nk_menubar_end(...); - } - nk_end(...); - */ - if (layout->flags & NK_WINDOW_HIDDEN || layout->flags & NK_WINDOW_MINIMIZED) - return; - - layout->menu.x = layout->at_x; - layout->menu.y = layout->at_y + layout->row.height; - layout->menu.w = layout->bounds.w; - layout->menu.offset.x = *layout->offset_x; - layout->menu.offset.y = *layout->offset_y; - *layout->offset_y = 0; -} -NK_API void -nk_menubar_end(struct nk_context *ctx) -{ - struct nk_window *win; - struct nk_panel *layout; - struct nk_command_buffer *out; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return; - - win = ctx->current; - out = &win->buffer; - layout = win->layout; - if (layout->flags & NK_WINDOW_HIDDEN || layout->flags & NK_WINDOW_MINIMIZED) - return; - - layout->menu.h = layout->at_y - layout->menu.y; - layout->menu.h += layout->row.height + ctx->style.window.spacing.y; - - layout->bounds.y += layout->menu.h; - layout->bounds.h -= layout->menu.h; - - *layout->offset_x = layout->menu.offset.x; - *layout->offset_y = layout->menu.offset.y; - layout->at_y = layout->bounds.y - layout->row.height; - - layout->clip.y = layout->bounds.y; - layout->clip.h = layout->bounds.h; - nk_push_scissor(out, layout->clip); -} -NK_INTERN int -nk_menu_begin(struct nk_context *ctx, struct nk_window *win, - const char *id, int is_clicked, struct nk_rect header, struct nk_vec2 size) -{ - int is_open = 0; - int is_active = 0; - struct nk_rect body; - struct nk_window *popup; - nk_hash hash = nk_murmur_hash(id, (int)nk_strlen(id), NK_PANEL_MENU); - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return 0; - - body.x = header.x; - body.w = size.x; - body.y = header.y + header.h; - body.h = size.y; - - popup = win->popup.win; - is_open = popup ? nk_true : nk_false; - is_active = (popup && (win->popup.name == hash) && win->popup.type == NK_PANEL_MENU); - if ((is_clicked && is_open && !is_active) || (is_open && !is_active) || - (!is_open && !is_active && !is_clicked)) return 0; - if (!nk_nonblock_begin(ctx, NK_WINDOW_NO_SCROLLBAR, body, header, NK_PANEL_MENU)) - return 0; - - win->popup.type = NK_PANEL_MENU; - win->popup.name = hash; - return 1; -} -NK_API nk_bool -nk_menu_begin_text(struct nk_context *ctx, const char *title, int len, - nk_flags align, struct nk_vec2 size) -{ - struct nk_window *win; - const struct nk_input *in; - struct nk_rect header; - int is_clicked = nk_false; - nk_flags state; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return 0; - - win = ctx->current; - state = nk_widget(&header, ctx); - if (!state) return 0; - in = (state == NK_WIDGET_ROM || win->flags & NK_WINDOW_ROM) ? 0 : &ctx->input; - if (nk_do_button_text(&ctx->last_widget_state, &win->buffer, header, - title, len, align, NK_BUTTON_DEFAULT, &ctx->style.menu_button, in, ctx->style.font)) - is_clicked = nk_true; - return nk_menu_begin(ctx, win, title, is_clicked, header, size); -} -NK_API nk_bool nk_menu_begin_label(struct nk_context *ctx, - const char *text, nk_flags align, struct nk_vec2 size) -{ - return nk_menu_begin_text(ctx, text, nk_strlen(text), align, size); -} -NK_API nk_bool -nk_menu_begin_image(struct nk_context *ctx, const char *id, struct nk_image img, - struct nk_vec2 size) -{ - struct nk_window *win; - struct nk_rect header; - const struct nk_input *in; - int is_clicked = nk_false; - nk_flags state; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return 0; - - win = ctx->current; - state = nk_widget(&header, ctx); - if (!state) return 0; - in = (state == NK_WIDGET_ROM || win->layout->flags & NK_WINDOW_ROM) ? 0 : &ctx->input; - if (nk_do_button_image(&ctx->last_widget_state, &win->buffer, header, - img, NK_BUTTON_DEFAULT, &ctx->style.menu_button, in)) - is_clicked = nk_true; - return nk_menu_begin(ctx, win, id, is_clicked, header, size); -} -NK_API nk_bool -nk_menu_begin_symbol(struct nk_context *ctx, const char *id, - enum nk_symbol_type sym, struct nk_vec2 size) -{ - struct nk_window *win; - const struct nk_input *in; - struct nk_rect header; - int is_clicked = nk_false; - nk_flags state; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return 0; - - win = ctx->current; - state = nk_widget(&header, ctx); - if (!state) return 0; - in = (state == NK_WIDGET_ROM || win->layout->flags & NK_WINDOW_ROM) ? 0 : &ctx->input; - if (nk_do_button_symbol(&ctx->last_widget_state, &win->buffer, header, - sym, NK_BUTTON_DEFAULT, &ctx->style.menu_button, in, ctx->style.font)) - is_clicked = nk_true; - return nk_menu_begin(ctx, win, id, is_clicked, header, size); -} -NK_API nk_bool -nk_menu_begin_image_text(struct nk_context *ctx, const char *title, int len, - nk_flags align, struct nk_image img, struct nk_vec2 size) -{ - struct nk_window *win; - struct nk_rect header; - const struct nk_input *in; - int is_clicked = nk_false; - nk_flags state; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return 0; - - win = ctx->current; - state = nk_widget(&header, ctx); - if (!state) return 0; - in = (state == NK_WIDGET_ROM || win->layout->flags & NK_WINDOW_ROM) ? 0 : &ctx->input; - if (nk_do_button_text_image(&ctx->last_widget_state, &win->buffer, - header, img, title, len, align, NK_BUTTON_DEFAULT, &ctx->style.menu_button, - ctx->style.font, in)) - is_clicked = nk_true; - return nk_menu_begin(ctx, win, title, is_clicked, header, size); -} -NK_API nk_bool -nk_menu_begin_image_label(struct nk_context *ctx, - const char *title, nk_flags align, struct nk_image img, struct nk_vec2 size) -{ - return nk_menu_begin_image_text(ctx, title, nk_strlen(title), align, img, size); -} -NK_API nk_bool -nk_menu_begin_symbol_text(struct nk_context *ctx, const char *title, int len, - nk_flags align, enum nk_symbol_type sym, struct nk_vec2 size) -{ - struct nk_window *win; - struct nk_rect header; - const struct nk_input *in; - int is_clicked = nk_false; - nk_flags state; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return 0; - - win = ctx->current; - state = nk_widget(&header, ctx); - if (!state) return 0; - - in = (state == NK_WIDGET_ROM || win->layout->flags & NK_WINDOW_ROM) ? 0 : &ctx->input; - if (nk_do_button_text_symbol(&ctx->last_widget_state, &win->buffer, - header, sym, title, len, align, NK_BUTTON_DEFAULT, &ctx->style.menu_button, - ctx->style.font, in)) is_clicked = nk_true; - return nk_menu_begin(ctx, win, title, is_clicked, header, size); -} -NK_API nk_bool -nk_menu_begin_symbol_label(struct nk_context *ctx, - const char *title, nk_flags align, enum nk_symbol_type sym, struct nk_vec2 size ) -{ - return nk_menu_begin_symbol_text(ctx, title, nk_strlen(title), align,sym,size); -} -NK_API nk_bool -nk_menu_item_text(struct nk_context *ctx, const char *title, int len, nk_flags align) -{ - return nk_contextual_item_text(ctx, title, len, align); -} -NK_API nk_bool -nk_menu_item_label(struct nk_context *ctx, const char *label, nk_flags align) -{ - return nk_contextual_item_label(ctx, label, align); -} -NK_API nk_bool -nk_menu_item_image_label(struct nk_context *ctx, struct nk_image img, - const char *label, nk_flags align) -{ - return nk_contextual_item_image_label(ctx, img, label, align); -} -NK_API nk_bool -nk_menu_item_image_text(struct nk_context *ctx, struct nk_image img, - const char *text, int len, nk_flags align) -{ - return nk_contextual_item_image_text(ctx, img, text, len, align); -} -NK_API nk_bool nk_menu_item_symbol_text(struct nk_context *ctx, enum nk_symbol_type sym, - const char *text, int len, nk_flags align) -{ - return nk_contextual_item_symbol_text(ctx, sym, text, len, align); -} -NK_API nk_bool nk_menu_item_symbol_label(struct nk_context *ctx, enum nk_symbol_type sym, - const char *label, nk_flags align) -{ - return nk_contextual_item_symbol_label(ctx, sym, label, align); -} -NK_API void nk_menu_close(struct nk_context *ctx) -{ - nk_contextual_close(ctx); -} -NK_API void -nk_menu_end(struct nk_context *ctx) -{ - nk_contextual_end(ctx); -} - - - - - -/* =============================================================== - * - * LAYOUT - * - * ===============================================================*/ -NK_API void -nk_layout_set_min_row_height(struct nk_context *ctx, float height) -{ - struct nk_window *win; - struct nk_panel *layout; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return; - - win = ctx->current; - layout = win->layout; - layout->row.min_height = height; -} -NK_API void -nk_layout_reset_min_row_height(struct nk_context *ctx) -{ - struct nk_window *win; - struct nk_panel *layout; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return; - - win = ctx->current; - layout = win->layout; - layout->row.min_height = ctx->style.font->height; - layout->row.min_height += ctx->style.text.padding.y*2; - layout->row.min_height += ctx->style.window.min_row_height_padding*2; -} -NK_LIB float -nk_layout_row_calculate_usable_space(const struct nk_style *style, enum nk_panel_type type, - float total_space, int columns) -{ - float panel_spacing; - float panel_space; - - struct nk_vec2 spacing; - - NK_UNUSED(type); - - spacing = style->window.spacing; - - /* calculate the usable panel space */ - panel_spacing = (float)NK_MAX(columns - 1, 0) * spacing.x; - panel_space = total_space - panel_spacing; - return panel_space; -} -NK_LIB void -nk_panel_layout(const struct nk_context *ctx, struct nk_window *win, - float height, int cols) -{ - struct nk_panel *layout; - const struct nk_style *style; - struct nk_command_buffer *out; - - struct nk_vec2 item_spacing; - struct nk_color color; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return; - - /* prefetch some configuration data */ - layout = win->layout; - style = &ctx->style; - out = &win->buffer; - color = style->window.background; - item_spacing = style->window.spacing; - - /* if one of these triggers you forgot to add an `if` condition around either - a window, group, popup, combobox or contextual menu `begin` and `end` block. - Example: - if (nk_begin(...) {...} nk_end(...); or - if (nk_group_begin(...) { nk_group_end(...);} */ - NK_ASSERT(!(layout->flags & NK_WINDOW_MINIMIZED)); - NK_ASSERT(!(layout->flags & NK_WINDOW_HIDDEN)); - NK_ASSERT(!(layout->flags & NK_WINDOW_CLOSED)); - - /* update the current row and set the current row layout */ - layout->row.index = 0; - layout->at_y += layout->row.height; - layout->row.columns = cols; - if (height == 0.0f) - layout->row.height = NK_MAX(height, layout->row.min_height) + item_spacing.y; - else layout->row.height = height + item_spacing.y; - - layout->row.item_offset = 0; - if (layout->flags & NK_WINDOW_DYNAMIC) { - /* draw background for dynamic panels */ - struct nk_rect background; - background.x = win->bounds.x; - background.w = win->bounds.w; - background.y = layout->at_y - 1.0f; - background.h = layout->row.height + 1.0f; - nk_fill_rect(out, background, 0, color); - } -} -NK_LIB void -nk_row_layout(struct nk_context *ctx, enum nk_layout_format fmt, - float height, int cols, int width) -{ - /* update the current row and set the current row layout */ - struct nk_window *win; - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return; - - win = ctx->current; - nk_panel_layout(ctx, win, height, cols); - if (fmt == NK_DYNAMIC) - win->layout->row.type = NK_LAYOUT_DYNAMIC_FIXED; - else win->layout->row.type = NK_LAYOUT_STATIC_FIXED; - - win->layout->row.ratio = 0; - win->layout->row.filled = 0; - win->layout->row.item_offset = 0; - win->layout->row.item_width = (float)width; -} -NK_API float -nk_layout_ratio_from_pixel(struct nk_context *ctx, float pixel_width) -{ - struct nk_window *win; - NK_ASSERT(ctx); - NK_ASSERT(pixel_width); - if (!ctx || !ctx->current || !ctx->current->layout) return 0; - win = ctx->current; - return NK_CLAMP(0.0f, pixel_width/win->bounds.x, 1.0f); -} -NK_API void -nk_layout_row_dynamic(struct nk_context *ctx, float height, int cols) -{ - nk_row_layout(ctx, NK_DYNAMIC, height, cols, 0); -} -NK_API void -nk_layout_row_static(struct nk_context *ctx, float height, int item_width, int cols) -{ - nk_row_layout(ctx, NK_STATIC, height, cols, item_width); -} -NK_API void -nk_layout_row_begin(struct nk_context *ctx, enum nk_layout_format fmt, - float row_height, int cols) -{ - struct nk_window *win; - struct nk_panel *layout; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return; - - win = ctx->current; - layout = win->layout; - nk_panel_layout(ctx, win, row_height, cols); - if (fmt == NK_DYNAMIC) - layout->row.type = NK_LAYOUT_DYNAMIC_ROW; - else layout->row.type = NK_LAYOUT_STATIC_ROW; - - layout->row.ratio = 0; - layout->row.filled = 0; - layout->row.item_width = 0; - layout->row.item_offset = 0; - layout->row.columns = cols; -} -NK_API void -nk_layout_row_push(struct nk_context *ctx, float ratio_or_width) -{ - struct nk_window *win; - struct nk_panel *layout; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return; - - win = ctx->current; - layout = win->layout; - NK_ASSERT(layout->row.type == NK_LAYOUT_STATIC_ROW || layout->row.type == NK_LAYOUT_DYNAMIC_ROW); - if (layout->row.type != NK_LAYOUT_STATIC_ROW && layout->row.type != NK_LAYOUT_DYNAMIC_ROW) - return; - - if (layout->row.type == NK_LAYOUT_DYNAMIC_ROW) { - float ratio = ratio_or_width; - if ((ratio + layout->row.filled) > 1.0f) return; - if (ratio > 0.0f) - layout->row.item_width = NK_SATURATE(ratio); - else layout->row.item_width = 1.0f - layout->row.filled; - } else layout->row.item_width = ratio_or_width; -} -NK_API void -nk_layout_row_end(struct nk_context *ctx) -{ - struct nk_window *win; - struct nk_panel *layout; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return; - - win = ctx->current; - layout = win->layout; - NK_ASSERT(layout->row.type == NK_LAYOUT_STATIC_ROW || layout->row.type == NK_LAYOUT_DYNAMIC_ROW); - if (layout->row.type != NK_LAYOUT_STATIC_ROW && layout->row.type != NK_LAYOUT_DYNAMIC_ROW) - return; - layout->row.item_width = 0; - layout->row.item_offset = 0; -} -NK_API void -nk_layout_row(struct nk_context *ctx, enum nk_layout_format fmt, - float height, int cols, const float *ratio) -{ - int i; - int n_undef = 0; - struct nk_window *win; - struct nk_panel *layout; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return; - - win = ctx->current; - layout = win->layout; - nk_panel_layout(ctx, win, height, cols); - if (fmt == NK_DYNAMIC) { - /* calculate width of undefined widget ratios */ - float r = 0; - layout->row.ratio = ratio; - for (i = 0; i < cols; ++i) { - if (ratio[i] < 0.0f) - n_undef++; - else r += ratio[i]; - } - r = NK_SATURATE(1.0f - r); - layout->row.type = NK_LAYOUT_DYNAMIC; - layout->row.item_width = (r > 0 && n_undef > 0) ? (r / (float)n_undef):0; - } else { - layout->row.ratio = ratio; - layout->row.type = NK_LAYOUT_STATIC; - layout->row.item_width = 0; - layout->row.item_offset = 0; - } - layout->row.item_offset = 0; - layout->row.filled = 0; -} -NK_API void -nk_layout_row_template_begin(struct nk_context *ctx, float height) -{ - struct nk_window *win; - struct nk_panel *layout; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return; - - win = ctx->current; - layout = win->layout; - nk_panel_layout(ctx, win, height, 1); - layout->row.type = NK_LAYOUT_TEMPLATE; - layout->row.columns = 0; - layout->row.ratio = 0; - layout->row.item_width = 0; - layout->row.item_height = 0; - layout->row.item_offset = 0; - layout->row.filled = 0; - layout->row.item.x = 0; - layout->row.item.y = 0; - layout->row.item.w = 0; - layout->row.item.h = 0; -} -NK_API void -nk_layout_row_template_push_dynamic(struct nk_context *ctx) -{ - struct nk_window *win; - struct nk_panel *layout; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return; - - win = ctx->current; - layout = win->layout; - NK_ASSERT(layout->row.type == NK_LAYOUT_TEMPLATE); - NK_ASSERT(layout->row.columns < NK_MAX_LAYOUT_ROW_TEMPLATE_COLUMNS); - if (layout->row.type != NK_LAYOUT_TEMPLATE) return; - if (layout->row.columns >= NK_MAX_LAYOUT_ROW_TEMPLATE_COLUMNS) return; - layout->row.templates[layout->row.columns++] = -1.0f; -} -NK_API void -nk_layout_row_template_push_variable(struct nk_context *ctx, float min_width) -{ - struct nk_window *win; - struct nk_panel *layout; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return; - - win = ctx->current; - layout = win->layout; - NK_ASSERT(layout->row.type == NK_LAYOUT_TEMPLATE); - NK_ASSERT(layout->row.columns < NK_MAX_LAYOUT_ROW_TEMPLATE_COLUMNS); - if (layout->row.type != NK_LAYOUT_TEMPLATE) return; - if (layout->row.columns >= NK_MAX_LAYOUT_ROW_TEMPLATE_COLUMNS) return; - layout->row.templates[layout->row.columns++] = -min_width; -} -NK_API void -nk_layout_row_template_push_static(struct nk_context *ctx, float width) -{ - struct nk_window *win; - struct nk_panel *layout; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return; - - win = ctx->current; - layout = win->layout; - NK_ASSERT(layout->row.type == NK_LAYOUT_TEMPLATE); - NK_ASSERT(layout->row.columns < NK_MAX_LAYOUT_ROW_TEMPLATE_COLUMNS); - if (layout->row.type != NK_LAYOUT_TEMPLATE) return; - if (layout->row.columns >= NK_MAX_LAYOUT_ROW_TEMPLATE_COLUMNS) return; - layout->row.templates[layout->row.columns++] = width; -} -NK_API void -nk_layout_row_template_end(struct nk_context *ctx) -{ - struct nk_window *win; - struct nk_panel *layout; - - int i = 0; - int variable_count = 0; - int min_variable_count = 0; - float min_fixed_width = 0.0f; - float total_fixed_width = 0.0f; - float max_variable_width = 0.0f; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return; - - win = ctx->current; - layout = win->layout; - NK_ASSERT(layout->row.type == NK_LAYOUT_TEMPLATE); - if (layout->row.type != NK_LAYOUT_TEMPLATE) return; - for (i = 0; i < layout->row.columns; ++i) { - float width = layout->row.templates[i]; - if (width >= 0.0f) { - total_fixed_width += width; - min_fixed_width += width; - } else if (width < -1.0f) { - width = -width; - total_fixed_width += width; - max_variable_width = NK_MAX(max_variable_width, width); - variable_count++; - } else { - min_variable_count++; - variable_count++; - } - } - if (variable_count) { - float space = nk_layout_row_calculate_usable_space(&ctx->style, layout->type, - layout->bounds.w, layout->row.columns); - float var_width = (NK_MAX(space-min_fixed_width,0.0f)) / (float)variable_count; - int enough_space = var_width >= max_variable_width; - if (!enough_space) - var_width = (NK_MAX(space-total_fixed_width,0)) / (float)min_variable_count; - for (i = 0; i < layout->row.columns; ++i) { - float *width = &layout->row.templates[i]; - *width = (*width >= 0.0f)? *width: (*width < -1.0f && !enough_space)? -(*width): var_width; - } - } -} -NK_API void -nk_layout_space_begin(struct nk_context *ctx, enum nk_layout_format fmt, - float height, int widget_count) -{ - struct nk_window *win; - struct nk_panel *layout; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return; - - win = ctx->current; - layout = win->layout; - nk_panel_layout(ctx, win, height, widget_count); - if (fmt == NK_STATIC) - layout->row.type = NK_LAYOUT_STATIC_FREE; - else layout->row.type = NK_LAYOUT_DYNAMIC_FREE; - - layout->row.ratio = 0; - layout->row.filled = 0; - layout->row.item_width = 0; - layout->row.item_offset = 0; -} -NK_API void -nk_layout_space_end(struct nk_context *ctx) -{ - struct nk_window *win; - struct nk_panel *layout; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return; - - win = ctx->current; - layout = win->layout; - layout->row.item_width = 0; - layout->row.item_height = 0; - layout->row.item_offset = 0; - nk_zero(&layout->row.item, sizeof(layout->row.item)); -} -NK_API void -nk_layout_space_push(struct nk_context *ctx, struct nk_rect rect) -{ - struct nk_window *win; - struct nk_panel *layout; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return; - - win = ctx->current; - layout = win->layout; - layout->row.item = rect; -} -NK_API struct nk_rect -nk_layout_space_bounds(struct nk_context *ctx) -{ - struct nk_rect ret; - struct nk_window *win; - struct nk_panel *layout; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - win = ctx->current; - layout = win->layout; - - ret.x = layout->clip.x; - ret.y = layout->clip.y; - ret.w = layout->clip.w; - ret.h = layout->row.height; - return ret; -} -NK_API struct nk_rect -nk_layout_widget_bounds(struct nk_context *ctx) -{ - struct nk_rect ret; - struct nk_window *win; - struct nk_panel *layout; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - win = ctx->current; - layout = win->layout; - - ret.x = layout->at_x; - ret.y = layout->at_y; - ret.w = layout->bounds.w - NK_MAX(layout->at_x - layout->bounds.x,0); - ret.h = layout->row.height; - return ret; -} -NK_API struct nk_vec2 -nk_layout_space_to_screen(struct nk_context *ctx, struct nk_vec2 ret) -{ - struct nk_window *win; - struct nk_panel *layout; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - win = ctx->current; - layout = win->layout; - - ret.x += layout->at_x - (float)*layout->offset_x; - ret.y += layout->at_y - (float)*layout->offset_y; - return ret; -} -NK_API struct nk_vec2 -nk_layout_space_to_local(struct nk_context *ctx, struct nk_vec2 ret) -{ - struct nk_window *win; - struct nk_panel *layout; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - win = ctx->current; - layout = win->layout; - - ret.x += -layout->at_x + (float)*layout->offset_x; - ret.y += -layout->at_y + (float)*layout->offset_y; - return ret; -} -NK_API struct nk_rect -nk_layout_space_rect_to_screen(struct nk_context *ctx, struct nk_rect ret) -{ - struct nk_window *win; - struct nk_panel *layout; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - win = ctx->current; - layout = win->layout; - - ret.x += layout->at_x - (float)*layout->offset_x; - ret.y += layout->at_y - (float)*layout->offset_y; - return ret; -} -NK_API struct nk_rect -nk_layout_space_rect_to_local(struct nk_context *ctx, struct nk_rect ret) -{ - struct nk_window *win; - struct nk_panel *layout; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - win = ctx->current; - layout = win->layout; - - ret.x += -layout->at_x + (float)*layout->offset_x; - ret.y += -layout->at_y + (float)*layout->offset_y; - return ret; -} -NK_LIB void -nk_panel_alloc_row(const struct nk_context *ctx, struct nk_window *win) -{ - struct nk_panel *layout = win->layout; - struct nk_vec2 spacing = ctx->style.window.spacing; - const float row_height = layout->row.height - spacing.y; - nk_panel_layout(ctx, win, row_height, layout->row.columns); -} -NK_LIB void -nk_layout_widget_space(struct nk_rect *bounds, const struct nk_context *ctx, - struct nk_window *win, int modify) -{ - struct nk_panel *layout; - const struct nk_style *style; - - struct nk_vec2 spacing; - - float item_offset = 0; - float item_width = 0; - float item_spacing = 0; - float panel_space = 0; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return; - - win = ctx->current; - layout = win->layout; - style = &ctx->style; - NK_ASSERT(bounds); - - spacing = style->window.spacing; - panel_space = nk_layout_row_calculate_usable_space(&ctx->style, layout->type, - layout->bounds.w, layout->row.columns); - - #define NK_FRAC(x) (x - (float)(int)x) /* will be used to remove fookin gaps */ - /* calculate the width of one item inside the current layout space */ - switch (layout->row.type) { - case NK_LAYOUT_DYNAMIC_FIXED: { - /* scaling fixed size widgets item width */ - float w = NK_MAX(1.0f,panel_space) / (float)layout->row.columns; - item_offset = (float)layout->row.index * w; - item_width = w + NK_FRAC(item_offset); - item_spacing = (float)layout->row.index * spacing.x; - } break; - case NK_LAYOUT_DYNAMIC_ROW: { - /* scaling single ratio widget width */ - float w = layout->row.item_width * panel_space; - item_offset = layout->row.item_offset; - item_width = w + NK_FRAC(item_offset); - item_spacing = 0; - - if (modify) { - layout->row.item_offset += w + spacing.x; - layout->row.filled += layout->row.item_width; - layout->row.index = 0; - } - } break; - case NK_LAYOUT_DYNAMIC_FREE: { - /* panel width depended free widget placing */ - bounds->x = layout->at_x + (layout->bounds.w * layout->row.item.x); - bounds->x -= (float)*layout->offset_x; - bounds->y = layout->at_y + (layout->row.height * layout->row.item.y); - bounds->y -= (float)*layout->offset_y; - bounds->w = layout->bounds.w * layout->row.item.w + NK_FRAC(bounds->x); - bounds->h = layout->row.height * layout->row.item.h + NK_FRAC(bounds->y); - return; - } - case NK_LAYOUT_DYNAMIC: { - /* scaling arrays of panel width ratios for every widget */ - float ratio, w; - NK_ASSERT(layout->row.ratio); - ratio = (layout->row.ratio[layout->row.index] < 0) ? - layout->row.item_width : layout->row.ratio[layout->row.index]; - - w = (ratio * panel_space); - item_spacing = (float)layout->row.index * spacing.x; - item_offset = layout->row.item_offset; - item_width = w + NK_FRAC(item_offset); - - if (modify) { - layout->row.item_offset += w; - layout->row.filled += ratio; - } - } break; - case NK_LAYOUT_STATIC_FIXED: { - /* non-scaling fixed widgets item width */ - item_width = layout->row.item_width; - item_offset = (float)layout->row.index * item_width; - item_spacing = (float)layout->row.index * spacing.x; - } break; - case NK_LAYOUT_STATIC_ROW: { - /* scaling single ratio widget width */ - item_width = layout->row.item_width; - item_offset = layout->row.item_offset; - item_spacing = (float)layout->row.index * spacing.x; - if (modify) layout->row.item_offset += item_width; - } break; - case NK_LAYOUT_STATIC_FREE: { - /* free widget placing */ - bounds->x = layout->at_x + layout->row.item.x; - bounds->w = layout->row.item.w; - if (((bounds->x + bounds->w) > layout->max_x) && modify) - layout->max_x = (bounds->x + bounds->w); - bounds->x -= (float)*layout->offset_x; - bounds->y = layout->at_y + layout->row.item.y; - bounds->y -= (float)*layout->offset_y; - bounds->h = layout->row.item.h; - return; - } - case NK_LAYOUT_STATIC: { - /* non-scaling array of panel pixel width for every widget */ - item_spacing = (float)layout->row.index * spacing.x; - item_width = layout->row.ratio[layout->row.index]; - item_offset = layout->row.item_offset; - if (modify) layout->row.item_offset += item_width; - } break; - case NK_LAYOUT_TEMPLATE: { - /* stretchy row layout with combined dynamic/static widget width*/ - float w; - NK_ASSERT(layout->row.index < layout->row.columns); - NK_ASSERT(layout->row.index < NK_MAX_LAYOUT_ROW_TEMPLATE_COLUMNS); - w = layout->row.templates[layout->row.index]; - item_offset = layout->row.item_offset; - item_width = w + NK_FRAC(item_offset); - item_spacing = (float)layout->row.index * spacing.x; - if (modify) layout->row.item_offset += w; - } break; - #undef NK_FRAC - default: NK_ASSERT(0); break; - }; - - /* set the bounds of the newly allocated widget */ - bounds->w = item_width; - bounds->h = layout->row.height - spacing.y; - bounds->y = layout->at_y - (float)*layout->offset_y; - bounds->x = layout->at_x + item_offset + item_spacing; - if (((bounds->x + bounds->w) > layout->max_x) && modify) - layout->max_x = bounds->x + bounds->w; - bounds->x -= (float)*layout->offset_x; -} -NK_LIB void -nk_panel_alloc_space(struct nk_rect *bounds, const struct nk_context *ctx) -{ - struct nk_window *win; - struct nk_panel *layout; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return; - - /* check if the end of the row has been hit and begin new row if so */ - win = ctx->current; - layout = win->layout; - if (layout->row.index >= layout->row.columns) - nk_panel_alloc_row(ctx, win); - - /* calculate widget position and size */ - nk_layout_widget_space(bounds, ctx, win, nk_true); - layout->row.index++; -} -NK_LIB void -nk_layout_peek(struct nk_rect *bounds, struct nk_context *ctx) -{ - float y; - int index; - struct nk_window *win; - struct nk_panel *layout; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) { - *bounds = nk_rect(0,0,0,0); - return; - } - - win = ctx->current; - layout = win->layout; - y = layout->at_y; - index = layout->row.index; - if (layout->row.index >= layout->row.columns) { - layout->at_y += layout->row.height; - layout->row.index = 0; - } - nk_layout_widget_space(bounds, ctx, win, nk_false); - if (!layout->row.index) { - bounds->x -= layout->row.item_offset; - } - layout->at_y = y; - layout->row.index = index; -} -NK_API void -nk_spacer(struct nk_context *ctx ) -{ - struct nk_rect dummy_rect = { 0, 0, 0, 0 }; - nk_panel_alloc_space( &dummy_rect, ctx ); -} - - - - -/* =============================================================== - * - * TREE - * - * ===============================================================*/ -NK_INTERN int -nk_tree_state_base(struct nk_context *ctx, enum nk_tree_type type, - struct nk_image *img, const char *title, enum nk_collapse_states *state) -{ - struct nk_window *win; - struct nk_panel *layout; - const struct nk_style *style; - struct nk_command_buffer *out; - const struct nk_input *in; - const struct nk_style_button *button; - enum nk_symbol_type symbol; - float row_height; - - struct nk_vec2 item_spacing; - struct nk_rect header = {0,0,0,0}; - struct nk_rect sym = {0,0,0,0}; - struct nk_text text; - - nk_flags ws = 0; - enum nk_widget_layout_states widget_state; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return 0; - - /* cache some data */ - win = ctx->current; - layout = win->layout; - out = &win->buffer; - style = &ctx->style; - item_spacing = style->window.spacing; - - /* calculate header bounds and draw background */ - row_height = style->font->height + 2 * style->tab.padding.y; - nk_layout_set_min_row_height(ctx, row_height); - nk_layout_row_dynamic(ctx, row_height, 1); - nk_layout_reset_min_row_height(ctx); - - widget_state = nk_widget(&header, ctx); - if (type == NK_TREE_TAB) { - const struct nk_style_item *background = &style->tab.background; - - switch(background->type) { - case NK_STYLE_ITEM_IMAGE: - nk_draw_image(out, header, &background->data.image, nk_white); - break; - case NK_STYLE_ITEM_NINE_SLICE: - nk_draw_nine_slice(out, header, &background->data.slice, nk_white); - break; - case NK_STYLE_ITEM_COLOR: - nk_fill_rect(out, header, 0, style->tab.border_color); - nk_fill_rect(out, nk_shrink_rect(header, style->tab.border), - style->tab.rounding, background->data.color); - break; - } - } else text.background = style->window.background; - - /* update node state */ - in = (!(layout->flags & NK_WINDOW_ROM)) ? &ctx->input: 0; - in = (in && widget_state == NK_WIDGET_VALID) ? &ctx->input : 0; - if (nk_button_behavior(&ws, header, in, NK_BUTTON_DEFAULT)) - *state = (*state == NK_MAXIMIZED) ? NK_MINIMIZED : NK_MAXIMIZED; - - /* select correct button style */ - if (*state == NK_MAXIMIZED) { - symbol = style->tab.sym_maximize; - if (type == NK_TREE_TAB) - button = &style->tab.tab_maximize_button; - else button = &style->tab.node_maximize_button; - } else { - symbol = style->tab.sym_minimize; - if (type == NK_TREE_TAB) - button = &style->tab.tab_minimize_button; - else button = &style->tab.node_minimize_button; - } - - {/* draw triangle button */ - sym.w = sym.h = style->font->height; - sym.y = header.y + style->tab.padding.y; - sym.x = header.x + style->tab.padding.x; - nk_do_button_symbol(&ws, &win->buffer, sym, symbol, NK_BUTTON_DEFAULT, - button, 0, style->font); - - if (img) { - /* draw optional image icon */ - sym.x = sym.x + sym.w + 4 * item_spacing.x; - nk_draw_image(&win->buffer, sym, img, nk_white); - sym.w = style->font->height + style->tab.spacing.x;} - } - - {/* draw label */ - struct nk_rect label; - header.w = NK_MAX(header.w, sym.w + item_spacing.x); - label.x = sym.x + sym.w + item_spacing.x; - label.y = sym.y; - label.w = header.w - (sym.w + item_spacing.y + style->tab.indent); - label.h = style->font->height; - text.text = style->tab.text; - text.padding = nk_vec2(0,0); - nk_widget_text(out, label, title, nk_strlen(title), &text, - NK_TEXT_LEFT, style->font);} - - /* increase x-axis cursor widget position pointer */ - if (*state == NK_MAXIMIZED) { - layout->at_x = header.x + (float)*layout->offset_x + style->tab.indent; - layout->bounds.w = NK_MAX(layout->bounds.w, style->tab.indent); - layout->bounds.w -= (style->tab.indent + style->window.padding.x); - layout->row.tree_depth++; - return nk_true; - } else return nk_false; -} -NK_INTERN int -nk_tree_base(struct nk_context *ctx, enum nk_tree_type type, - struct nk_image *img, const char *title, enum nk_collapse_states initial_state, - const char *hash, int len, int line) -{ - struct nk_window *win = ctx->current; - int title_len = 0; - nk_hash tree_hash = 0; - nk_uint *state = 0; - - /* retrieve tree state from internal widget state tables */ - if (!hash) { - title_len = (int)nk_strlen(title); - tree_hash = nk_murmur_hash(title, (int)title_len, (nk_hash)line); - } else tree_hash = nk_murmur_hash(hash, len, (nk_hash)line); - state = nk_find_value(win, tree_hash); - if (!state) { - state = nk_add_value(ctx, win, tree_hash, 0); - *state = initial_state; - } - return nk_tree_state_base(ctx, type, img, title, (enum nk_collapse_states*)state); -} -NK_API nk_bool -nk_tree_state_push(struct nk_context *ctx, enum nk_tree_type type, - const char *title, enum nk_collapse_states *state) -{ - return nk_tree_state_base(ctx, type, 0, title, state); -} -NK_API nk_bool -nk_tree_state_image_push(struct nk_context *ctx, enum nk_tree_type type, - struct nk_image img, const char *title, enum nk_collapse_states *state) -{ - return nk_tree_state_base(ctx, type, &img, title, state); -} -NK_API void -nk_tree_state_pop(struct nk_context *ctx) -{ - struct nk_window *win = 0; - struct nk_panel *layout = 0; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return; - - win = ctx->current; - layout = win->layout; - layout->at_x -= ctx->style.tab.indent + (float)*layout->offset_x; - layout->bounds.w += ctx->style.tab.indent + ctx->style.window.padding.x; - NK_ASSERT(layout->row.tree_depth); - layout->row.tree_depth--; -} -NK_API nk_bool -nk_tree_push_hashed(struct nk_context *ctx, enum nk_tree_type type, - const char *title, enum nk_collapse_states initial_state, - const char *hash, int len, int line) -{ - return nk_tree_base(ctx, type, 0, title, initial_state, hash, len, line); -} -NK_API nk_bool -nk_tree_image_push_hashed(struct nk_context *ctx, enum nk_tree_type type, - struct nk_image img, const char *title, enum nk_collapse_states initial_state, - const char *hash, int len,int seed) -{ - return nk_tree_base(ctx, type, &img, title, initial_state, hash, len, seed); -} -NK_API void -nk_tree_pop(struct nk_context *ctx) -{ - nk_tree_state_pop(ctx); -} -NK_INTERN int -nk_tree_element_image_push_hashed_base(struct nk_context *ctx, enum nk_tree_type type, - struct nk_image *img, const char *title, int title_len, - enum nk_collapse_states *state, nk_bool *selected) -{ - struct nk_window *win; - struct nk_panel *layout; - const struct nk_style *style; - struct nk_command_buffer *out; - const struct nk_input *in; - const struct nk_style_button *button; - enum nk_symbol_type symbol; - float row_height; - struct nk_vec2 padding; - - int text_len; - float text_width; - - struct nk_vec2 item_spacing; - struct nk_rect header = {0,0,0,0}; - struct nk_rect sym = {0,0,0,0}; - - nk_flags ws = 0; - enum nk_widget_layout_states widget_state; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return 0; - - /* cache some data */ - win = ctx->current; - layout = win->layout; - out = &win->buffer; - style = &ctx->style; - item_spacing = style->window.spacing; - padding = style->selectable.padding; - - /* calculate header bounds and draw background */ - row_height = style->font->height + 2 * style->tab.padding.y; - nk_layout_set_min_row_height(ctx, row_height); - nk_layout_row_dynamic(ctx, row_height, 1); - nk_layout_reset_min_row_height(ctx); - - widget_state = nk_widget(&header, ctx); - if (type == NK_TREE_TAB) { - const struct nk_style_item *background = &style->tab.background; - - switch (background->type) { - case NK_STYLE_ITEM_IMAGE: - nk_draw_image(out, header, &background->data.image, nk_white); - break; - case NK_STYLE_ITEM_NINE_SLICE: - nk_draw_nine_slice(out, header, &background->data.slice, nk_white); - break; - case NK_STYLE_ITEM_COLOR: - nk_fill_rect(out, header, 0, style->tab.border_color); - nk_fill_rect(out, nk_shrink_rect(header, style->tab.border), - style->tab.rounding, background->data.color); - break; - } - } - - in = (!(layout->flags & NK_WINDOW_ROM)) ? &ctx->input: 0; - in = (in && widget_state == NK_WIDGET_VALID) ? &ctx->input : 0; - - /* select correct button style */ - if (*state == NK_MAXIMIZED) { - symbol = style->tab.sym_maximize; - if (type == NK_TREE_TAB) - button = &style->tab.tab_maximize_button; - else button = &style->tab.node_maximize_button; - } else { - symbol = style->tab.sym_minimize; - if (type == NK_TREE_TAB) - button = &style->tab.tab_minimize_button; - else button = &style->tab.node_minimize_button; - } - {/* draw triangle button */ - sym.w = sym.h = style->font->height; - sym.y = header.y + style->tab.padding.y; - sym.x = header.x + style->tab.padding.x; - if (nk_do_button_symbol(&ws, &win->buffer, sym, symbol, NK_BUTTON_DEFAULT, button, in, style->font)) - *state = (*state == NK_MAXIMIZED) ? NK_MINIMIZED : NK_MAXIMIZED;} - - /* draw label */ - {nk_flags dummy = 0; - struct nk_rect label; - /* calculate size of the text and tooltip */ - text_len = nk_strlen(title); - text_width = style->font->width(style->font->userdata, style->font->height, title, text_len); - text_width += (4 * padding.x); - - header.w = NK_MAX(header.w, sym.w + item_spacing.x); - label.x = sym.x + sym.w + item_spacing.x; - label.y = sym.y; - label.w = NK_MIN(header.w - (sym.w + item_spacing.y + style->tab.indent), text_width); - label.h = style->font->height; - - if (img) { - nk_do_selectable_image(&dummy, &win->buffer, label, title, title_len, NK_TEXT_LEFT, - selected, img, &style->selectable, in, style->font); - } else nk_do_selectable(&dummy, &win->buffer, label, title, title_len, NK_TEXT_LEFT, - selected, &style->selectable, in, style->font); - } - /* increase x-axis cursor widget position pointer */ - if (*state == NK_MAXIMIZED) { - layout->at_x = header.x + (float)*layout->offset_x + style->tab.indent; - layout->bounds.w = NK_MAX(layout->bounds.w, style->tab.indent); - layout->bounds.w -= (style->tab.indent + style->window.padding.x); - layout->row.tree_depth++; - return nk_true; - } else return nk_false; -} -NK_INTERN int -nk_tree_element_base(struct nk_context *ctx, enum nk_tree_type type, - struct nk_image *img, const char *title, enum nk_collapse_states initial_state, - nk_bool *selected, const char *hash, int len, int line) -{ - struct nk_window *win = ctx->current; - int title_len = 0; - nk_hash tree_hash = 0; - nk_uint *state = 0; - - /* retrieve tree state from internal widget state tables */ - if (!hash) { - title_len = (int)nk_strlen(title); - tree_hash = nk_murmur_hash(title, (int)title_len, (nk_hash)line); - } else tree_hash = nk_murmur_hash(hash, len, (nk_hash)line); - state = nk_find_value(win, tree_hash); - if (!state) { - state = nk_add_value(ctx, win, tree_hash, 0); - *state = initial_state; - } return nk_tree_element_image_push_hashed_base(ctx, type, img, title, - nk_strlen(title), (enum nk_collapse_states*)state, selected); -} -NK_API nk_bool -nk_tree_element_push_hashed(struct nk_context *ctx, enum nk_tree_type type, - const char *title, enum nk_collapse_states initial_state, - nk_bool *selected, const char *hash, int len, int seed) -{ - return nk_tree_element_base(ctx, type, 0, title, initial_state, selected, hash, len, seed); -} -NK_API nk_bool -nk_tree_element_image_push_hashed(struct nk_context *ctx, enum nk_tree_type type, - struct nk_image img, const char *title, enum nk_collapse_states initial_state, - nk_bool *selected, const char *hash, int len,int seed) -{ - return nk_tree_element_base(ctx, type, &img, title, initial_state, selected, hash, len, seed); -} -NK_API void -nk_tree_element_pop(struct nk_context *ctx) -{ - nk_tree_state_pop(ctx); -} - - - - - -/* =============================================================== - * - * GROUP - * - * ===============================================================*/ -NK_API nk_bool -nk_group_scrolled_offset_begin(struct nk_context *ctx, - nk_uint *x_offset, nk_uint *y_offset, const char *title, nk_flags flags) -{ - struct nk_rect bounds; - struct nk_window panel; - struct nk_window *win; - - win = ctx->current; - nk_panel_alloc_space(&bounds, ctx); - {const struct nk_rect *c = &win->layout->clip; - if (!NK_INTERSECT(c->x, c->y, c->w, c->h, bounds.x, bounds.y, bounds.w, bounds.h) && - !(flags & NK_WINDOW_MOVABLE)) { - return 0; - }} - if (win->flags & NK_WINDOW_ROM) - flags |= NK_WINDOW_ROM; - - /* initialize a fake window to create the panel from */ - nk_zero(&panel, sizeof(panel)); - panel.bounds = bounds; - panel.flags = flags; - panel.scrollbar.x = *x_offset; - panel.scrollbar.y = *y_offset; - panel.buffer = win->buffer; - panel.layout = (struct nk_panel*)nk_create_panel(ctx); - ctx->current = &panel; - nk_panel_begin(ctx, (flags & NK_WINDOW_TITLE) ? title: 0, NK_PANEL_GROUP); - - win->buffer = panel.buffer; - win->buffer.clip = panel.layout->clip; - panel.layout->offset_x = x_offset; - panel.layout->offset_y = y_offset; - panel.layout->parent = win->layout; - win->layout = panel.layout; - - ctx->current = win; - if ((panel.layout->flags & NK_WINDOW_CLOSED) || - (panel.layout->flags & NK_WINDOW_MINIMIZED)) - { - nk_flags f = panel.layout->flags; - nk_group_scrolled_end(ctx); - if (f & NK_WINDOW_CLOSED) - return NK_WINDOW_CLOSED; - if (f & NK_WINDOW_MINIMIZED) - return NK_WINDOW_MINIMIZED; - } - return 1; -} -NK_API void -nk_group_scrolled_end(struct nk_context *ctx) -{ - struct nk_window *win; - struct nk_panel *parent; - struct nk_panel *g; - - struct nk_rect clip; - struct nk_window pan; - struct nk_vec2 panel_padding; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - if (!ctx || !ctx->current) - return; - - /* make sure nk_group_begin was called correctly */ - NK_ASSERT(ctx->current); - win = ctx->current; - NK_ASSERT(win->layout); - g = win->layout; - NK_ASSERT(g->parent); - parent = g->parent; - - /* dummy window */ - nk_zero_struct(pan); - panel_padding = nk_panel_get_padding(&ctx->style, NK_PANEL_GROUP); - pan.bounds.y = g->bounds.y - (g->header_height + g->menu.h); - pan.bounds.x = g->bounds.x - panel_padding.x; - pan.bounds.w = g->bounds.w + 2 * panel_padding.x; - pan.bounds.h = g->bounds.h + g->header_height + g->menu.h; - if (g->flags & NK_WINDOW_BORDER) { - pan.bounds.x -= g->border; - pan.bounds.y -= g->border; - pan.bounds.w += 2*g->border; - pan.bounds.h += 2*g->border; - } - if (!(g->flags & NK_WINDOW_NO_SCROLLBAR)) { - pan.bounds.w += ctx->style.window.scrollbar_size.x; - pan.bounds.h += ctx->style.window.scrollbar_size.y; - } - pan.scrollbar.x = *g->offset_x; - pan.scrollbar.y = *g->offset_y; - pan.flags = g->flags; - pan.buffer = win->buffer; - pan.layout = g; - pan.parent = win; - ctx->current = &pan; - - /* make sure group has correct clipping rectangle */ - nk_unify(&clip, &parent->clip, pan.bounds.x, pan.bounds.y, - pan.bounds.x + pan.bounds.w, pan.bounds.y + pan.bounds.h + panel_padding.x); - nk_push_scissor(&pan.buffer, clip); - nk_end(ctx); - - win->buffer = pan.buffer; - nk_push_scissor(&win->buffer, parent->clip); - ctx->current = win; - win->layout = parent; - g->bounds = pan.bounds; - return; -} -NK_API nk_bool -nk_group_scrolled_begin(struct nk_context *ctx, - struct nk_scroll *scroll, const char *title, nk_flags flags) -{ - return nk_group_scrolled_offset_begin(ctx, &scroll->x, &scroll->y, title, flags); -} -NK_API nk_bool -nk_group_begin_titled(struct nk_context *ctx, const char *id, - const char *title, nk_flags flags) -{ - int id_len; - nk_hash id_hash; - struct nk_window *win; - nk_uint *x_offset; - nk_uint *y_offset; - - NK_ASSERT(ctx); - NK_ASSERT(id); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout || !id) - return 0; - - /* find persistent group scrollbar value */ - win = ctx->current; - id_len = (int)nk_strlen(id); - id_hash = nk_murmur_hash(id, (int)id_len, NK_PANEL_GROUP); - x_offset = nk_find_value(win, id_hash); - if (!x_offset) { - x_offset = nk_add_value(ctx, win, id_hash, 0); - y_offset = nk_add_value(ctx, win, id_hash+1, 0); - - NK_ASSERT(x_offset); - NK_ASSERT(y_offset); - if (!x_offset || !y_offset) return 0; - *x_offset = *y_offset = 0; - } else y_offset = nk_find_value(win, id_hash+1); - return nk_group_scrolled_offset_begin(ctx, x_offset, y_offset, title, flags); -} -NK_API nk_bool -nk_group_begin(struct nk_context *ctx, const char *title, nk_flags flags) -{ - return nk_group_begin_titled(ctx, title, title, flags); -} -NK_API void -nk_group_end(struct nk_context *ctx) -{ - nk_group_scrolled_end(ctx); -} -NK_API void -nk_group_get_scroll(struct nk_context *ctx, const char *id, nk_uint *x_offset, nk_uint *y_offset) -{ - int id_len; - nk_hash id_hash; - struct nk_window *win; - nk_uint *x_offset_ptr; - nk_uint *y_offset_ptr; - - NK_ASSERT(ctx); - NK_ASSERT(id); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout || !id) - return; - - /* find persistent group scrollbar value */ - win = ctx->current; - id_len = (int)nk_strlen(id); - id_hash = nk_murmur_hash(id, (int)id_len, NK_PANEL_GROUP); - x_offset_ptr = nk_find_value(win, id_hash); - if (!x_offset_ptr) { - x_offset_ptr = nk_add_value(ctx, win, id_hash, 0); - y_offset_ptr = nk_add_value(ctx, win, id_hash+1, 0); - - NK_ASSERT(x_offset_ptr); - NK_ASSERT(y_offset_ptr); - if (!x_offset_ptr || !y_offset_ptr) return; - *x_offset_ptr = *y_offset_ptr = 0; - } else y_offset_ptr = nk_find_value(win, id_hash+1); - if (x_offset) - *x_offset = *x_offset_ptr; - if (y_offset) - *y_offset = *y_offset_ptr; -} -NK_API void -nk_group_set_scroll(struct nk_context *ctx, const char *id, nk_uint x_offset, nk_uint y_offset) -{ - int id_len; - nk_hash id_hash; - struct nk_window *win; - nk_uint *x_offset_ptr; - nk_uint *y_offset_ptr; - - NK_ASSERT(ctx); - NK_ASSERT(id); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout || !id) - return; - - /* find persistent group scrollbar value */ - win = ctx->current; - id_len = (int)nk_strlen(id); - id_hash = nk_murmur_hash(id, (int)id_len, NK_PANEL_GROUP); - x_offset_ptr = nk_find_value(win, id_hash); - if (!x_offset_ptr) { - x_offset_ptr = nk_add_value(ctx, win, id_hash, 0); - y_offset_ptr = nk_add_value(ctx, win, id_hash+1, 0); - - NK_ASSERT(x_offset_ptr); - NK_ASSERT(y_offset_ptr); - if (!x_offset_ptr || !y_offset_ptr) return; - *x_offset_ptr = *y_offset_ptr = 0; - } else y_offset_ptr = nk_find_value(win, id_hash+1); - *x_offset_ptr = x_offset; - *y_offset_ptr = y_offset; -} - - - - -/* =============================================================== - * - * LIST VIEW - * - * ===============================================================*/ -NK_API nk_bool -nk_list_view_begin(struct nk_context *ctx, struct nk_list_view *view, - const char *title, nk_flags flags, int row_height, int row_count) -{ - int title_len; - nk_hash title_hash; - nk_uint *x_offset; - nk_uint *y_offset; - - int result; - struct nk_window *win; - struct nk_panel *layout; - const struct nk_style *style; - struct nk_vec2 item_spacing; - - NK_ASSERT(ctx); - NK_ASSERT(view); - NK_ASSERT(title); - if (!ctx || !view || !title) return 0; - - win = ctx->current; - style = &ctx->style; - item_spacing = style->window.spacing; - row_height += NK_MAX(0, (int)item_spacing.y); - - /* find persistent list view scrollbar offset */ - title_len = (int)nk_strlen(title); - title_hash = nk_murmur_hash(title, (int)title_len, NK_PANEL_GROUP); - x_offset = nk_find_value(win, title_hash); - if (!x_offset) { - x_offset = nk_add_value(ctx, win, title_hash, 0); - y_offset = nk_add_value(ctx, win, title_hash+1, 0); - - NK_ASSERT(x_offset); - NK_ASSERT(y_offset); - if (!x_offset || !y_offset) return 0; - *x_offset = *y_offset = 0; - } else y_offset = nk_find_value(win, title_hash+1); - view->scroll_value = *y_offset; - view->scroll_pointer = y_offset; - - *y_offset = 0; - result = nk_group_scrolled_offset_begin(ctx, x_offset, y_offset, title, flags); - win = ctx->current; - layout = win->layout; - - view->total_height = row_height * NK_MAX(row_count,1); - view->begin = (int)NK_MAX(((float)view->scroll_value / (float)row_height), 0.0f); - view->count = (int)NK_MAX(nk_iceilf((layout->clip.h)/(float)row_height),0); - view->count = NK_MIN(view->count, row_count - view->begin); - view->end = view->begin + view->count; - view->ctx = ctx; - return result; -} -NK_API void -nk_list_view_end(struct nk_list_view *view) -{ - struct nk_context *ctx; - struct nk_window *win; - struct nk_panel *layout; - - NK_ASSERT(view); - NK_ASSERT(view->ctx); - NK_ASSERT(view->scroll_pointer); - if (!view || !view->ctx) return; - - ctx = view->ctx; - win = ctx->current; - layout = win->layout; - layout->at_y = layout->bounds.y + (float)view->total_height; - *view->scroll_pointer = *view->scroll_pointer + view->scroll_value; - nk_group_end(view->ctx); -} - - - - - -/* =============================================================== - * - * WIDGET - * - * ===============================================================*/ -NK_API struct nk_rect -nk_widget_bounds(struct nk_context *ctx) -{ - struct nk_rect bounds; - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - if (!ctx || !ctx->current) - return nk_rect(0,0,0,0); - nk_layout_peek(&bounds, ctx); - return bounds; -} -NK_API struct nk_vec2 -nk_widget_position(struct nk_context *ctx) -{ - struct nk_rect bounds; - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - if (!ctx || !ctx->current) - return nk_vec2(0,0); - - nk_layout_peek(&bounds, ctx); - return nk_vec2(bounds.x, bounds.y); -} -NK_API struct nk_vec2 -nk_widget_size(struct nk_context *ctx) -{ - struct nk_rect bounds; - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - if (!ctx || !ctx->current) - return nk_vec2(0,0); - - nk_layout_peek(&bounds, ctx); - return nk_vec2(bounds.w, bounds.h); -} -NK_API float -nk_widget_width(struct nk_context *ctx) -{ - struct nk_rect bounds; - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - if (!ctx || !ctx->current) - return 0; - - nk_layout_peek(&bounds, ctx); - return bounds.w; -} -NK_API float -nk_widget_height(struct nk_context *ctx) -{ - struct nk_rect bounds; - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - if (!ctx || !ctx->current) - return 0; - - nk_layout_peek(&bounds, ctx); - return bounds.h; -} -NK_API nk_bool -nk_widget_is_hovered(struct nk_context *ctx) -{ - struct nk_rect c, v; - struct nk_rect bounds; - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - if (!ctx || !ctx->current || ctx->active != ctx->current) - return 0; - - c = ctx->current->layout->clip; - c.x = (float)((int)c.x); - c.y = (float)((int)c.y); - c.w = (float)((int)c.w); - c.h = (float)((int)c.h); - - nk_layout_peek(&bounds, ctx); - nk_unify(&v, &c, bounds.x, bounds.y, bounds.x + bounds.w, bounds.y + bounds.h); - if (!NK_INTERSECT(c.x, c.y, c.w, c.h, bounds.x, bounds.y, bounds.w, bounds.h)) - return 0; - return nk_input_is_mouse_hovering_rect(&ctx->input, bounds); -} -NK_API nk_bool -nk_widget_is_mouse_clicked(struct nk_context *ctx, enum nk_buttons btn) -{ - struct nk_rect c, v; - struct nk_rect bounds; - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - if (!ctx || !ctx->current || ctx->active != ctx->current) - return 0; - - c = ctx->current->layout->clip; - c.x = (float)((int)c.x); - c.y = (float)((int)c.y); - c.w = (float)((int)c.w); - c.h = (float)((int)c.h); - - nk_layout_peek(&bounds, ctx); - nk_unify(&v, &c, bounds.x, bounds.y, bounds.x + bounds.w, bounds.y + bounds.h); - if (!NK_INTERSECT(c.x, c.y, c.w, c.h, bounds.x, bounds.y, bounds.w, bounds.h)) - return 0; - return nk_input_mouse_clicked(&ctx->input, btn, bounds); -} -NK_API nk_bool -nk_widget_has_mouse_click_down(struct nk_context *ctx, enum nk_buttons btn, nk_bool down) -{ - struct nk_rect c, v; - struct nk_rect bounds; - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - if (!ctx || !ctx->current || ctx->active != ctx->current) - return 0; - - c = ctx->current->layout->clip; - c.x = (float)((int)c.x); - c.y = (float)((int)c.y); - c.w = (float)((int)c.w); - c.h = (float)((int)c.h); - - nk_layout_peek(&bounds, ctx); - nk_unify(&v, &c, bounds.x, bounds.y, bounds.x + bounds.w, bounds.y + bounds.h); - if (!NK_INTERSECT(c.x, c.y, c.w, c.h, bounds.x, bounds.y, bounds.w, bounds.h)) - return 0; - return nk_input_has_mouse_click_down_in_rect(&ctx->input, btn, bounds, down); -} -NK_API enum nk_widget_layout_states -nk_widget(struct nk_rect *bounds, const struct nk_context *ctx) -{ - struct nk_rect c, v; - struct nk_window *win; - struct nk_panel *layout; - const struct nk_input *in; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return NK_WIDGET_INVALID; - - /* allocate space and check if the widget needs to be updated and drawn */ - nk_panel_alloc_space(bounds, ctx); - win = ctx->current; - layout = win->layout; - in = &ctx->input; - c = layout->clip; - - /* if one of these triggers you forgot to add an `if` condition around either - a window, group, popup, combobox or contextual menu `begin` and `end` block. - Example: - if (nk_begin(...) {...} nk_end(...); or - if (nk_group_begin(...) { nk_group_end(...);} */ - NK_ASSERT(!(layout->flags & NK_WINDOW_MINIMIZED)); - NK_ASSERT(!(layout->flags & NK_WINDOW_HIDDEN)); - NK_ASSERT(!(layout->flags & NK_WINDOW_CLOSED)); - - /* need to convert to int here to remove floating point errors */ - bounds->x = (float)((int)bounds->x); - bounds->y = (float)((int)bounds->y); - bounds->w = (float)((int)bounds->w); - bounds->h = (float)((int)bounds->h); - - c.x = (float)((int)c.x); - c.y = (float)((int)c.y); - c.w = (float)((int)c.w); - c.h = (float)((int)c.h); - - nk_unify(&v, &c, bounds->x, bounds->y, bounds->x + bounds->w, bounds->y + bounds->h); - if (!NK_INTERSECT(c.x, c.y, c.w, c.h, bounds->x, bounds->y, bounds->w, bounds->h)) - return NK_WIDGET_INVALID; - if (!NK_INBOX(in->mouse.pos.x, in->mouse.pos.y, v.x, v.y, v.w, v.h)) - return NK_WIDGET_ROM; - return NK_WIDGET_VALID; -} -NK_API enum nk_widget_layout_states -nk_widget_fitting(struct nk_rect *bounds, struct nk_context *ctx, - struct nk_vec2 item_padding) -{ - /* update the bounds to stand without padding */ - enum nk_widget_layout_states state; - NK_UNUSED(item_padding); - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return NK_WIDGET_INVALID; - - state = nk_widget(bounds, ctx); - return state; -} -NK_API void -nk_spacing(struct nk_context *ctx, int cols) -{ - struct nk_window *win; - struct nk_panel *layout; - struct nk_rect none; - int i, index, rows; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return; - - /* spacing over row boundaries */ - win = ctx->current; - layout = win->layout; - index = (layout->row.index + cols) % layout->row.columns; - rows = (layout->row.index + cols) / layout->row.columns; - if (rows) { - for (i = 0; i < rows; ++i) - nk_panel_alloc_row(ctx, win); - cols = index; - } - /* non table layout need to allocate space */ - if (layout->row.type != NK_LAYOUT_DYNAMIC_FIXED && - layout->row.type != NK_LAYOUT_STATIC_FIXED) { - for (i = 0; i < cols; ++i) - nk_panel_alloc_space(&none, ctx); - } layout->row.index = index; -} - - - - - -/* =============================================================== - * - * TEXT - * - * ===============================================================*/ -NK_LIB void -nk_widget_text(struct nk_command_buffer *o, struct nk_rect b, - const char *string, int len, const struct nk_text *t, - nk_flags a, const struct nk_user_font *f) -{ - struct nk_rect label; - float text_width; - - NK_ASSERT(o); - NK_ASSERT(t); - if (!o || !t) return; - - b.h = NK_MAX(b.h, 2 * t->padding.y); - label.x = 0; label.w = 0; - label.y = b.y + t->padding.y; - label.h = NK_MIN(f->height, b.h - 2 * t->padding.y); - - text_width = f->width(f->userdata, f->height, (const char*)string, len); - text_width += (2.0f * t->padding.x); - - /* align in x-axis */ - if (a & NK_TEXT_ALIGN_LEFT) { - label.x = b.x + t->padding.x; - label.w = NK_MAX(0, b.w - 2 * t->padding.x); - } else if (a & NK_TEXT_ALIGN_CENTERED) { - label.w = NK_MAX(1, 2 * t->padding.x + (float)text_width); - label.x = (b.x + t->padding.x + ((b.w - 2 * t->padding.x) - label.w) / 2); - label.x = NK_MAX(b.x + t->padding.x, label.x); - label.w = NK_MIN(b.x + b.w, label.x + label.w); - if (label.w >= label.x) label.w -= label.x; - } else if (a & NK_TEXT_ALIGN_RIGHT) { - label.x = NK_MAX(b.x + t->padding.x, (b.x + b.w) - (2 * t->padding.x + (float)text_width)); - label.w = (float)text_width + 2 * t->padding.x; - } else return; - - /* align in y-axis */ - if (a & NK_TEXT_ALIGN_MIDDLE) { - label.y = b.y + b.h/2.0f - (float)f->height/2.0f; - label.h = NK_MAX(b.h/2.0f, b.h - (b.h/2.0f + f->height/2.0f)); - } else if (a & NK_TEXT_ALIGN_BOTTOM) { - label.y = b.y + b.h - f->height; - label.h = f->height; - } - nk_draw_text(o, label, (const char*)string, len, f, t->background, t->text); -} -NK_LIB void -nk_widget_text_wrap(struct nk_command_buffer *o, struct nk_rect b, - const char *string, int len, const struct nk_text *t, - const struct nk_user_font *f) -{ - float width; - int glyphs = 0; - int fitting = 0; - int done = 0; - struct nk_rect line; - struct nk_text text; - NK_INTERN nk_rune seperator[] = {' '}; - - NK_ASSERT(o); - NK_ASSERT(t); - if (!o || !t) return; - - text.padding = nk_vec2(0,0); - text.background = t->background; - text.text = t->text; - - b.w = NK_MAX(b.w, 2 * t->padding.x); - b.h = NK_MAX(b.h, 2 * t->padding.y); - b.h = b.h - 2 * t->padding.y; - - line.x = b.x + t->padding.x; - line.y = b.y + t->padding.y; - line.w = b.w - 2 * t->padding.x; - line.h = 2 * t->padding.y + f->height; - - fitting = nk_text_clamp(f, string, len, line.w, &glyphs, &width, seperator,NK_LEN(seperator)); - while (done < len) { - if (!fitting || line.y + line.h >= (b.y + b.h)) break; - nk_widget_text(o, line, &string[done], fitting, &text, NK_TEXT_LEFT, f); - done += fitting; - line.y += f->height + 2 * t->padding.y; - fitting = nk_text_clamp(f, &string[done], len - done, line.w, &glyphs, &width, seperator,NK_LEN(seperator)); - } -} -NK_API void -nk_text_colored(struct nk_context *ctx, const char *str, int len, - nk_flags alignment, struct nk_color color) -{ - struct nk_window *win; - const struct nk_style *style; - - struct nk_vec2 item_padding; - struct nk_rect bounds; - struct nk_text text; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) return; - - win = ctx->current; - style = &ctx->style; - nk_panel_alloc_space(&bounds, ctx); - item_padding = style->text.padding; - - text.padding.x = item_padding.x; - text.padding.y = item_padding.y; - text.background = style->window.background; - text.text = color; - nk_widget_text(&win->buffer, bounds, str, len, &text, alignment, style->font); -} -NK_API void -nk_text_wrap_colored(struct nk_context *ctx, const char *str, - int len, struct nk_color color) -{ - struct nk_window *win; - const struct nk_style *style; - - struct nk_vec2 item_padding; - struct nk_rect bounds; - struct nk_text text; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) return; - - win = ctx->current; - style = &ctx->style; - nk_panel_alloc_space(&bounds, ctx); - item_padding = style->text.padding; - - text.padding.x = item_padding.x; - text.padding.y = item_padding.y; - text.background = style->window.background; - text.text = color; - nk_widget_text_wrap(&win->buffer, bounds, str, len, &text, style->font); -} -#ifdef NK_INCLUDE_STANDARD_VARARGS -NK_API void -nk_labelf_colored(struct nk_context *ctx, nk_flags flags, - struct nk_color color, const char *fmt, ...) -{ - va_list args; - va_start(args, fmt); - nk_labelfv_colored(ctx, flags, color, fmt, args); - va_end(args); -} -NK_API void -nk_labelf_colored_wrap(struct nk_context *ctx, struct nk_color color, - const char *fmt, ...) -{ - va_list args; - va_start(args, fmt); - nk_labelfv_colored_wrap(ctx, color, fmt, args); - va_end(args); -} -NK_API void -nk_labelf(struct nk_context *ctx, nk_flags flags, const char *fmt, ...) -{ - va_list args; - va_start(args, fmt); - nk_labelfv(ctx, flags, fmt, args); - va_end(args); -} -NK_API void -nk_labelf_wrap(struct nk_context *ctx, const char *fmt,...) -{ - va_list args; - va_start(args, fmt); - nk_labelfv_wrap(ctx, fmt, args); - va_end(args); -} -NK_API void -nk_labelfv_colored(struct nk_context *ctx, nk_flags flags, - struct nk_color color, const char *fmt, va_list args) -{ - char buf[256]; - nk_strfmt(buf, NK_LEN(buf), fmt, args); - nk_label_colored(ctx, buf, flags, color); -} - -NK_API void -nk_labelfv_colored_wrap(struct nk_context *ctx, struct nk_color color, - const char *fmt, va_list args) -{ - char buf[256]; - nk_strfmt(buf, NK_LEN(buf), fmt, args); - nk_label_colored_wrap(ctx, buf, color); -} - -NK_API void -nk_labelfv(struct nk_context *ctx, nk_flags flags, const char *fmt, va_list args) -{ - char buf[256]; - nk_strfmt(buf, NK_LEN(buf), fmt, args); - nk_label(ctx, buf, flags); -} - -NK_API void -nk_labelfv_wrap(struct nk_context *ctx, const char *fmt, va_list args) -{ - char buf[256]; - nk_strfmt(buf, NK_LEN(buf), fmt, args); - nk_label_wrap(ctx, buf); -} - -NK_API void -nk_value_bool(struct nk_context *ctx, const char *prefix, int value) -{ - nk_labelf(ctx, NK_TEXT_LEFT, "%s: %s", prefix, ((value) ? "true": "false")); -} -NK_API void -nk_value_int(struct nk_context *ctx, const char *prefix, int value) -{ - nk_labelf(ctx, NK_TEXT_LEFT, "%s: %d", prefix, value); -} -NK_API void -nk_value_uint(struct nk_context *ctx, const char *prefix, unsigned int value) -{ - nk_labelf(ctx, NK_TEXT_LEFT, "%s: %u", prefix, value); -} -NK_API void -nk_value_float(struct nk_context *ctx, const char *prefix, float value) -{ - double double_value = (double)value; - nk_labelf(ctx, NK_TEXT_LEFT, "%s: %.3f", prefix, double_value); -} -NK_API void -nk_value_color_byte(struct nk_context *ctx, const char *p, struct nk_color c) -{ - nk_labelf(ctx, NK_TEXT_LEFT, "%s: (%d, %d, %d, %d)", p, c.r, c.g, c.b, c.a); -} -NK_API void -nk_value_color_float(struct nk_context *ctx, const char *p, struct nk_color color) -{ - double c[4]; nk_color_dv(c, color); - nk_labelf(ctx, NK_TEXT_LEFT, "%s: (%.2f, %.2f, %.2f, %.2f)", - p, c[0], c[1], c[2], c[3]); -} -NK_API void -nk_value_color_hex(struct nk_context *ctx, const char *prefix, struct nk_color color) -{ - char hex[16]; - nk_color_hex_rgba(hex, color); - nk_labelf(ctx, NK_TEXT_LEFT, "%s: %s", prefix, hex); -} -#endif -NK_API void -nk_text(struct nk_context *ctx, const char *str, int len, nk_flags alignment) -{ - NK_ASSERT(ctx); - if (!ctx) return; - nk_text_colored(ctx, str, len, alignment, ctx->style.text.color); -} -NK_API void -nk_text_wrap(struct nk_context *ctx, const char *str, int len) -{ - NK_ASSERT(ctx); - if (!ctx) return; - nk_text_wrap_colored(ctx, str, len, ctx->style.text.color); -} -NK_API void -nk_label(struct nk_context *ctx, const char *str, nk_flags alignment) -{ - nk_text(ctx, str, nk_strlen(str), alignment); -} -NK_API void -nk_label_colored(struct nk_context *ctx, const char *str, nk_flags align, - struct nk_color color) -{ - nk_text_colored(ctx, str, nk_strlen(str), align, color); -} -NK_API void -nk_label_wrap(struct nk_context *ctx, const char *str) -{ - nk_text_wrap(ctx, str, nk_strlen(str)); -} -NK_API void -nk_label_colored_wrap(struct nk_context *ctx, const char *str, struct nk_color color) -{ - nk_text_wrap_colored(ctx, str, nk_strlen(str), color); -} - - - - - -/* =============================================================== - * - * IMAGE - * - * ===============================================================*/ -NK_API nk_handle -nk_handle_ptr(void *ptr) -{ - nk_handle handle = {0}; - handle.ptr = ptr; - return handle; -} -NK_API nk_handle -nk_handle_id(int id) -{ - nk_handle handle; - nk_zero_struct(handle); - handle.id = id; - return handle; -} -NK_API struct nk_image -nk_subimage_ptr(void *ptr, nk_ushort w, nk_ushort h, struct nk_rect r) -{ - struct nk_image s; - nk_zero(&s, sizeof(s)); - s.handle.ptr = ptr; - s.w = w; s.h = h; - s.region[0] = (nk_ushort)r.x; - s.region[1] = (nk_ushort)r.y; - s.region[2] = (nk_ushort)r.w; - s.region[3] = (nk_ushort)r.h; - return s; -} -NK_API struct nk_image -nk_subimage_id(int id, nk_ushort w, nk_ushort h, struct nk_rect r) -{ - struct nk_image s; - nk_zero(&s, sizeof(s)); - s.handle.id = id; - s.w = w; s.h = h; - s.region[0] = (nk_ushort)r.x; - s.region[1] = (nk_ushort)r.y; - s.region[2] = (nk_ushort)r.w; - s.region[3] = (nk_ushort)r.h; - return s; -} -NK_API struct nk_image -nk_subimage_handle(nk_handle handle, nk_ushort w, nk_ushort h, struct nk_rect r) -{ - struct nk_image s; - nk_zero(&s, sizeof(s)); - s.handle = handle; - s.w = w; s.h = h; - s.region[0] = (nk_ushort)r.x; - s.region[1] = (nk_ushort)r.y; - s.region[2] = (nk_ushort)r.w; - s.region[3] = (nk_ushort)r.h; - return s; -} -NK_API struct nk_image -nk_image_handle(nk_handle handle) -{ - struct nk_image s; - nk_zero(&s, sizeof(s)); - s.handle = handle; - s.w = 0; s.h = 0; - s.region[0] = 0; - s.region[1] = 0; - s.region[2] = 0; - s.region[3] = 0; - return s; -} -NK_API struct nk_image -nk_image_ptr(void *ptr) -{ - struct nk_image s; - nk_zero(&s, sizeof(s)); - NK_ASSERT(ptr); - s.handle.ptr = ptr; - s.w = 0; s.h = 0; - s.region[0] = 0; - s.region[1] = 0; - s.region[2] = 0; - s.region[3] = 0; - return s; -} -NK_API struct nk_image -nk_image_id(int id) -{ - struct nk_image s; - nk_zero(&s, sizeof(s)); - s.handle.id = id; - s.w = 0; s.h = 0; - s.region[0] = 0; - s.region[1] = 0; - s.region[2] = 0; - s.region[3] = 0; - return s; -} -NK_API nk_bool -nk_image_is_subimage(const struct nk_image* img) -{ - NK_ASSERT(img); - return !(img->w == 0 && img->h == 0); -} -NK_API void -nk_image(struct nk_context *ctx, struct nk_image img) -{ - struct nk_window *win; - struct nk_rect bounds; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) return; - - win = ctx->current; - if (!nk_widget(&bounds, ctx)) return; - nk_draw_image(&win->buffer, bounds, &img, nk_white); -} -NK_API void -nk_image_color(struct nk_context *ctx, struct nk_image img, struct nk_color col) -{ - struct nk_window *win; - struct nk_rect bounds; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) return; - - win = ctx->current; - if (!nk_widget(&bounds, ctx)) return; - nk_draw_image(&win->buffer, bounds, &img, col); -} - - - - - -/* =============================================================== - * - * 9-SLICE - * - * ===============================================================*/ -NK_API struct nk_nine_slice -nk_sub9slice_ptr(void *ptr, nk_ushort w, nk_ushort h, struct nk_rect rgn, nk_ushort l, nk_ushort t, nk_ushort r, nk_ushort b) -{ - struct nk_nine_slice s; - struct nk_image *i = &s.img; - nk_zero(&s, sizeof(s)); - i->handle.ptr = ptr; - i->w = w; i->h = h; - i->region[0] = (nk_ushort)rgn.x; - i->region[1] = (nk_ushort)rgn.y; - i->region[2] = (nk_ushort)rgn.w; - i->region[3] = (nk_ushort)rgn.h; - s.l = l; s.t = t; s.r = r; s.b = b; - return s; -} -NK_API struct nk_nine_slice -nk_sub9slice_id(int id, nk_ushort w, nk_ushort h, struct nk_rect rgn, nk_ushort l, nk_ushort t, nk_ushort r, nk_ushort b) -{ - struct nk_nine_slice s; - struct nk_image *i = &s.img; - nk_zero(&s, sizeof(s)); - i->handle.id = id; - i->w = w; i->h = h; - i->region[0] = (nk_ushort)rgn.x; - i->region[1] = (nk_ushort)rgn.y; - i->region[2] = (nk_ushort)rgn.w; - i->region[3] = (nk_ushort)rgn.h; - s.l = l; s.t = t; s.r = r; s.b = b; - return s; -} -NK_API struct nk_nine_slice -nk_sub9slice_handle(nk_handle handle, nk_ushort w, nk_ushort h, struct nk_rect rgn, nk_ushort l, nk_ushort t, nk_ushort r, nk_ushort b) -{ - struct nk_nine_slice s; - struct nk_image *i = &s.img; - nk_zero(&s, sizeof(s)); - i->handle = handle; - i->w = w; i->h = h; - i->region[0] = (nk_ushort)rgn.x; - i->region[1] = (nk_ushort)rgn.y; - i->region[2] = (nk_ushort)rgn.w; - i->region[3] = (nk_ushort)rgn.h; - s.l = l; s.t = t; s.r = r; s.b = b; - return s; -} -NK_API struct nk_nine_slice -nk_nine_slice_handle(nk_handle handle, nk_ushort l, nk_ushort t, nk_ushort r, nk_ushort b) -{ - struct nk_nine_slice s; - struct nk_image *i = &s.img; - nk_zero(&s, sizeof(s)); - i->handle = handle; - i->w = 0; i->h = 0; - i->region[0] = 0; - i->region[1] = 0; - i->region[2] = 0; - i->region[3] = 0; - s.l = l; s.t = t; s.r = r; s.b = b; - return s; -} -NK_API struct nk_nine_slice -nk_nine_slice_ptr(void *ptr, nk_ushort l, nk_ushort t, nk_ushort r, nk_ushort b) -{ - struct nk_nine_slice s; - struct nk_image *i = &s.img; - nk_zero(&s, sizeof(s)); - NK_ASSERT(ptr); - i->handle.ptr = ptr; - i->w = 0; i->h = 0; - i->region[0] = 0; - i->region[1] = 0; - i->region[2] = 0; - i->region[3] = 0; - s.l = l; s.t = t; s.r = r; s.b = b; - return s; -} -NK_API struct nk_nine_slice -nk_nine_slice_id(int id, nk_ushort l, nk_ushort t, nk_ushort r, nk_ushort b) -{ - struct nk_nine_slice s; - struct nk_image *i = &s.img; - nk_zero(&s, sizeof(s)); - i->handle.id = id; - i->w = 0; i->h = 0; - i->region[0] = 0; - i->region[1] = 0; - i->region[2] = 0; - i->region[3] = 0; - s.l = l; s.t = t; s.r = r; s.b = b; - return s; -} -NK_API int -nk_nine_slice_is_sub9slice(const struct nk_nine_slice* slice) -{ - NK_ASSERT(slice); - return !(slice->img.w == 0 && slice->img.h == 0); -} - - - - - -/* ============================================================== - * - * BUTTON - * - * ===============================================================*/ -NK_LIB void -nk_draw_symbol(struct nk_command_buffer *out, enum nk_symbol_type type, - struct nk_rect content, struct nk_color background, struct nk_color foreground, - float border_width, const struct nk_user_font *font) -{ - switch (type) { - case NK_SYMBOL_X: - case NK_SYMBOL_UNDERSCORE: - case NK_SYMBOL_PLUS: - case NK_SYMBOL_MINUS: { - /* single character text symbol */ - const char *X = (type == NK_SYMBOL_X) ? "x": - (type == NK_SYMBOL_UNDERSCORE) ? "_": - (type == NK_SYMBOL_PLUS) ? "+": "-"; - struct nk_text text; - text.padding = nk_vec2(0,0); - text.background = background; - text.text = foreground; - nk_widget_text(out, content, X, 1, &text, NK_TEXT_CENTERED, font); - } break; - case NK_SYMBOL_CIRCLE_SOLID: - case NK_SYMBOL_CIRCLE_OUTLINE: - case NK_SYMBOL_RECT_SOLID: - case NK_SYMBOL_RECT_OUTLINE: { - /* simple empty/filled shapes */ - if (type == NK_SYMBOL_RECT_SOLID || type == NK_SYMBOL_RECT_OUTLINE) { - nk_fill_rect(out, content, 0, foreground); - if (type == NK_SYMBOL_RECT_OUTLINE) - nk_fill_rect(out, nk_shrink_rect(content, border_width), 0, background); - } else { - nk_fill_circle(out, content, foreground); - if (type == NK_SYMBOL_CIRCLE_OUTLINE) - nk_fill_circle(out, nk_shrink_rect(content, 1), background); - } - } break; - case NK_SYMBOL_TRIANGLE_UP: - case NK_SYMBOL_TRIANGLE_DOWN: - case NK_SYMBOL_TRIANGLE_LEFT: - case NK_SYMBOL_TRIANGLE_RIGHT: { - enum nk_heading heading; - struct nk_vec2 points[3]; - heading = (type == NK_SYMBOL_TRIANGLE_RIGHT) ? NK_RIGHT : - (type == NK_SYMBOL_TRIANGLE_LEFT) ? NK_LEFT: - (type == NK_SYMBOL_TRIANGLE_UP) ? NK_UP: NK_DOWN; - nk_triangle_from_direction(points, content, 0, 0, heading); - nk_fill_triangle(out, points[0].x, points[0].y, points[1].x, points[1].y, - points[2].x, points[2].y, foreground); - } break; - default: - case NK_SYMBOL_NONE: - case NK_SYMBOL_MAX: break; - } -} -NK_LIB nk_bool -nk_button_behavior(nk_flags *state, struct nk_rect r, - const struct nk_input *i, enum nk_button_behavior behavior) -{ - int ret = 0; - nk_widget_state_reset(state); - if (!i) return 0; - if (nk_input_is_mouse_hovering_rect(i, r)) { - *state = NK_WIDGET_STATE_HOVERED; - if (nk_input_is_mouse_down(i, NK_BUTTON_LEFT)) - *state = NK_WIDGET_STATE_ACTIVE; - if (nk_input_has_mouse_click_in_button_rect(i, NK_BUTTON_LEFT, r)) { - ret = (behavior != NK_BUTTON_DEFAULT) ? - nk_input_is_mouse_down(i, NK_BUTTON_LEFT): -#ifdef NK_BUTTON_TRIGGER_ON_RELEASE - nk_input_is_mouse_released(i, NK_BUTTON_LEFT); -#else - nk_input_is_mouse_pressed(i, NK_BUTTON_LEFT); -#endif - } - } - if (*state & NK_WIDGET_STATE_HOVER && !nk_input_is_mouse_prev_hovering_rect(i, r)) - *state |= NK_WIDGET_STATE_ENTERED; - else if (nk_input_is_mouse_prev_hovering_rect(i, r)) - *state |= NK_WIDGET_STATE_LEFT; - return ret; -} -NK_LIB const struct nk_style_item* -nk_draw_button(struct nk_command_buffer *out, - const struct nk_rect *bounds, nk_flags state, - const struct nk_style_button *style) -{ - const struct nk_style_item *background; - if (state & NK_WIDGET_STATE_HOVER) - background = &style->hover; - else if (state & NK_WIDGET_STATE_ACTIVED) - background = &style->active; - else background = &style->normal; - - switch(background->type) { - case NK_STYLE_ITEM_IMAGE: - nk_draw_image(out, *bounds, &background->data.image, nk_white); - break; - case NK_STYLE_ITEM_NINE_SLICE: - nk_draw_nine_slice(out, *bounds, &background->data.slice, nk_white); - break; - case NK_STYLE_ITEM_COLOR: - nk_fill_rect(out, *bounds, style->rounding, background->data.color); - nk_stroke_rect(out, *bounds, style->rounding, style->border, style->border_color); - break; - } - return background; -} -NK_LIB nk_bool -nk_do_button(nk_flags *state, struct nk_command_buffer *out, struct nk_rect r, - const struct nk_style_button *style, const struct nk_input *in, - enum nk_button_behavior behavior, struct nk_rect *content) -{ - struct nk_rect bounds; - NK_ASSERT(style); - NK_ASSERT(state); - NK_ASSERT(out); - if (!out || !style) - return nk_false; - - /* calculate button content space */ - content->x = r.x + style->padding.x + style->border + style->rounding; - content->y = r.y + style->padding.y + style->border + style->rounding; - content->w = r.w - (2 * style->padding.x + style->border + style->rounding*2); - content->h = r.h - (2 * style->padding.y + style->border + style->rounding*2); - - /* execute button behavior */ - bounds.x = r.x - style->touch_padding.x; - bounds.y = r.y - style->touch_padding.y; - bounds.w = r.w + 2 * style->touch_padding.x; - bounds.h = r.h + 2 * style->touch_padding.y; - return nk_button_behavior(state, bounds, in, behavior); -} -NK_LIB void -nk_draw_button_text(struct nk_command_buffer *out, - const struct nk_rect *bounds, const struct nk_rect *content, nk_flags state, - const struct nk_style_button *style, const char *txt, int len, - nk_flags text_alignment, const struct nk_user_font *font) -{ - struct nk_text text; - const struct nk_style_item *background; - background = nk_draw_button(out, bounds, state, style); - - /* select correct colors/images */ - if (background->type == NK_STYLE_ITEM_COLOR) - text.background = background->data.color; - else text.background = style->text_background; - if (state & NK_WIDGET_STATE_HOVER) - text.text = style->text_hover; - else if (state & NK_WIDGET_STATE_ACTIVED) - text.text = style->text_active; - else text.text = style->text_normal; - - text.padding = nk_vec2(0,0); - nk_widget_text(out, *content, txt, len, &text, text_alignment, font); -} -NK_LIB nk_bool -nk_do_button_text(nk_flags *state, - struct nk_command_buffer *out, struct nk_rect bounds, - const char *string, int len, nk_flags align, enum nk_button_behavior behavior, - const struct nk_style_button *style, const struct nk_input *in, - const struct nk_user_font *font) -{ - struct nk_rect content; - int ret = nk_false; - - NK_ASSERT(state); - NK_ASSERT(style); - NK_ASSERT(out); - NK_ASSERT(string); - NK_ASSERT(font); - if (!out || !style || !font || !string) - return nk_false; - - ret = nk_do_button(state, out, bounds, style, in, behavior, &content); - if (style->draw_begin) style->draw_begin(out, style->userdata); - nk_draw_button_text(out, &bounds, &content, *state, style, string, len, align, font); - if (style->draw_end) style->draw_end(out, style->userdata); - return ret; -} -NK_LIB void -nk_draw_button_symbol(struct nk_command_buffer *out, - const struct nk_rect *bounds, const struct nk_rect *content, - nk_flags state, const struct nk_style_button *style, - enum nk_symbol_type type, const struct nk_user_font *font) -{ - struct nk_color sym, bg; - const struct nk_style_item *background; - - /* select correct colors/images */ - background = nk_draw_button(out, bounds, state, style); - if (background->type == NK_STYLE_ITEM_COLOR) - bg = background->data.color; - else bg = style->text_background; - - if (state & NK_WIDGET_STATE_HOVER) - sym = style->text_hover; - else if (state & NK_WIDGET_STATE_ACTIVED) - sym = style->text_active; - else sym = style->text_normal; - nk_draw_symbol(out, type, *content, bg, sym, 1, font); -} -NK_LIB nk_bool -nk_do_button_symbol(nk_flags *state, - struct nk_command_buffer *out, struct nk_rect bounds, - enum nk_symbol_type symbol, enum nk_button_behavior behavior, - const struct nk_style_button *style, const struct nk_input *in, - const struct nk_user_font *font) -{ - int ret; - struct nk_rect content; - - NK_ASSERT(state); - NK_ASSERT(style); - NK_ASSERT(font); - NK_ASSERT(out); - if (!out || !style || !font || !state) - return nk_false; - - ret = nk_do_button(state, out, bounds, style, in, behavior, &content); - if (style->draw_begin) style->draw_begin(out, style->userdata); - nk_draw_button_symbol(out, &bounds, &content, *state, style, symbol, font); - if (style->draw_end) style->draw_end(out, style->userdata); - return ret; -} -NK_LIB void -nk_draw_button_image(struct nk_command_buffer *out, - const struct nk_rect *bounds, const struct nk_rect *content, - nk_flags state, const struct nk_style_button *style, const struct nk_image *img) -{ - nk_draw_button(out, bounds, state, style); - nk_draw_image(out, *content, img, nk_white); -} -NK_LIB nk_bool -nk_do_button_image(nk_flags *state, - struct nk_command_buffer *out, struct nk_rect bounds, - struct nk_image img, enum nk_button_behavior b, - const struct nk_style_button *style, const struct nk_input *in) -{ - int ret; - struct nk_rect content; - - NK_ASSERT(state); - NK_ASSERT(style); - NK_ASSERT(out); - if (!out || !style || !state) - return nk_false; - - ret = nk_do_button(state, out, bounds, style, in, b, &content); - content.x += style->image_padding.x; - content.y += style->image_padding.y; - content.w -= 2 * style->image_padding.x; - content.h -= 2 * style->image_padding.y; - - if (style->draw_begin) style->draw_begin(out, style->userdata); - nk_draw_button_image(out, &bounds, &content, *state, style, &img); - if (style->draw_end) style->draw_end(out, style->userdata); - return ret; -} -NK_LIB void -nk_draw_button_text_symbol(struct nk_command_buffer *out, - const struct nk_rect *bounds, const struct nk_rect *label, - const struct nk_rect *symbol, nk_flags state, const struct nk_style_button *style, - const char *str, int len, enum nk_symbol_type type, - const struct nk_user_font *font) -{ - struct nk_color sym; - struct nk_text text; - const struct nk_style_item *background; - - /* select correct background colors/images */ - background = nk_draw_button(out, bounds, state, style); - if (background->type == NK_STYLE_ITEM_COLOR) - text.background = background->data.color; - else text.background = style->text_background; - - /* select correct text colors */ - if (state & NK_WIDGET_STATE_HOVER) { - sym = style->text_hover; - text.text = style->text_hover; - } else if (state & NK_WIDGET_STATE_ACTIVED) { - sym = style->text_active; - text.text = style->text_active; - } else { - sym = style->text_normal; - text.text = style->text_normal; - } - - text.padding = nk_vec2(0,0); - nk_draw_symbol(out, type, *symbol, style->text_background, sym, 0, font); - nk_widget_text(out, *label, str, len, &text, NK_TEXT_CENTERED, font); -} -NK_LIB nk_bool -nk_do_button_text_symbol(nk_flags *state, - struct nk_command_buffer *out, struct nk_rect bounds, - enum nk_symbol_type symbol, const char *str, int len, nk_flags align, - enum nk_button_behavior behavior, const struct nk_style_button *style, - const struct nk_user_font *font, const struct nk_input *in) -{ - int ret; - struct nk_rect tri = {0,0,0,0}; - struct nk_rect content; - - NK_ASSERT(style); - NK_ASSERT(out); - NK_ASSERT(font); - if (!out || !style || !font) - return nk_false; - - ret = nk_do_button(state, out, bounds, style, in, behavior, &content); - tri.y = content.y + (content.h/2) - font->height/2; - tri.w = font->height; tri.h = font->height; - if (align & NK_TEXT_ALIGN_LEFT) { - tri.x = (content.x + content.w) - (2 * style->padding.x + tri.w); - tri.x = NK_MAX(tri.x, 0); - } else tri.x = content.x + 2 * style->padding.x; - - /* draw button */ - if (style->draw_begin) style->draw_begin(out, style->userdata); - nk_draw_button_text_symbol(out, &bounds, &content, &tri, - *state, style, str, len, symbol, font); - if (style->draw_end) style->draw_end(out, style->userdata); - return ret; -} -NK_LIB void -nk_draw_button_text_image(struct nk_command_buffer *out, - const struct nk_rect *bounds, const struct nk_rect *label, - const struct nk_rect *image, nk_flags state, const struct nk_style_button *style, - const char *str, int len, const struct nk_user_font *font, - const struct nk_image *img) -{ - struct nk_text text; - const struct nk_style_item *background; - background = nk_draw_button(out, bounds, state, style); - - /* select correct colors */ - if (background->type == NK_STYLE_ITEM_COLOR) - text.background = background->data.color; - else text.background = style->text_background; - if (state & NK_WIDGET_STATE_HOVER) - text.text = style->text_hover; - else if (state & NK_WIDGET_STATE_ACTIVED) - text.text = style->text_active; - else text.text = style->text_normal; - - text.padding = nk_vec2(0,0); - nk_widget_text(out, *label, str, len, &text, NK_TEXT_CENTERED, font); - nk_draw_image(out, *image, img, nk_white); -} -NK_LIB nk_bool -nk_do_button_text_image(nk_flags *state, - struct nk_command_buffer *out, struct nk_rect bounds, - struct nk_image img, const char* str, int len, nk_flags align, - enum nk_button_behavior behavior, const struct nk_style_button *style, - const struct nk_user_font *font, const struct nk_input *in) -{ - int ret; - struct nk_rect icon; - struct nk_rect content; - - NK_ASSERT(style); - NK_ASSERT(state); - NK_ASSERT(font); - NK_ASSERT(out); - if (!out || !font || !style || !str) - return nk_false; - - ret = nk_do_button(state, out, bounds, style, in, behavior, &content); - icon.y = bounds.y + style->padding.y; - icon.w = icon.h = bounds.h - 2 * style->padding.y; - if (align & NK_TEXT_ALIGN_LEFT) { - icon.x = (bounds.x + bounds.w) - (2 * style->padding.x + icon.w); - icon.x = NK_MAX(icon.x, 0); - } else icon.x = bounds.x + 2 * style->padding.x; - - icon.x += style->image_padding.x; - icon.y += style->image_padding.y; - icon.w -= 2 * style->image_padding.x; - icon.h -= 2 * style->image_padding.y; - - if (style->draw_begin) style->draw_begin(out, style->userdata); - nk_draw_button_text_image(out, &bounds, &content, &icon, *state, style, str, len, font, &img); - if (style->draw_end) style->draw_end(out, style->userdata); - return ret; -} -NK_API void -nk_button_set_behavior(struct nk_context *ctx, enum nk_button_behavior behavior) -{ - NK_ASSERT(ctx); - if (!ctx) return; - ctx->button_behavior = behavior; -} -NK_API nk_bool -nk_button_push_behavior(struct nk_context *ctx, enum nk_button_behavior behavior) -{ - struct nk_config_stack_button_behavior *button_stack; - struct nk_config_stack_button_behavior_element *element; - - NK_ASSERT(ctx); - if (!ctx) return 0; - - button_stack = &ctx->stacks.button_behaviors; - NK_ASSERT(button_stack->head < (int)NK_LEN(button_stack->elements)); - if (button_stack->head >= (int)NK_LEN(button_stack->elements)) - return 0; - - element = &button_stack->elements[button_stack->head++]; - element->address = &ctx->button_behavior; - element->old_value = ctx->button_behavior; - ctx->button_behavior = behavior; - return 1; -} -NK_API nk_bool -nk_button_pop_behavior(struct nk_context *ctx) -{ - struct nk_config_stack_button_behavior *button_stack; - struct nk_config_stack_button_behavior_element *element; - - NK_ASSERT(ctx); - if (!ctx) return 0; - - button_stack = &ctx->stacks.button_behaviors; - NK_ASSERT(button_stack->head > 0); - if (button_stack->head < 1) - return 0; - - element = &button_stack->elements[--button_stack->head]; - *element->address = element->old_value; - return 1; -} -NK_API nk_bool -nk_button_text_styled(struct nk_context *ctx, - const struct nk_style_button *style, const char *title, int len) -{ - struct nk_window *win; - struct nk_panel *layout; - const struct nk_input *in; - - struct nk_rect bounds; - enum nk_widget_layout_states state; - - NK_ASSERT(ctx); - NK_ASSERT(style); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!style || !ctx || !ctx->current || !ctx->current->layout) return 0; - - win = ctx->current; - layout = win->layout; - state = nk_widget(&bounds, ctx); - - if (!state) return 0; - in = (state == NK_WIDGET_ROM || layout->flags & NK_WINDOW_ROM) ? 0 : &ctx->input; - return nk_do_button_text(&ctx->last_widget_state, &win->buffer, bounds, - title, len, style->text_alignment, ctx->button_behavior, - style, in, ctx->style.font); -} -NK_API nk_bool -nk_button_text(struct nk_context *ctx, const char *title, int len) -{ - NK_ASSERT(ctx); - if (!ctx) return 0; - return nk_button_text_styled(ctx, &ctx->style.button, title, len); -} -NK_API nk_bool nk_button_label_styled(struct nk_context *ctx, - const struct nk_style_button *style, const char *title) -{ - return nk_button_text_styled(ctx, style, title, nk_strlen(title)); -} -NK_API nk_bool nk_button_label(struct nk_context *ctx, const char *title) -{ - return nk_button_text(ctx, title, nk_strlen(title)); -} -NK_API nk_bool -nk_button_color(struct nk_context *ctx, struct nk_color color) -{ - struct nk_window *win; - struct nk_panel *layout; - const struct nk_input *in; - struct nk_style_button button; - - int ret = 0; - struct nk_rect bounds; - struct nk_rect content; - enum nk_widget_layout_states state; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return 0; - - win = ctx->current; - layout = win->layout; - - state = nk_widget(&bounds, ctx); - if (!state) return 0; - in = (state == NK_WIDGET_ROM || layout->flags & NK_WINDOW_ROM) ? 0 : &ctx->input; - - button = ctx->style.button; - button.normal = nk_style_item_color(color); - button.hover = nk_style_item_color(color); - button.active = nk_style_item_color(color); - ret = nk_do_button(&ctx->last_widget_state, &win->buffer, bounds, - &button, in, ctx->button_behavior, &content); - nk_draw_button(&win->buffer, &bounds, ctx->last_widget_state, &button); - return ret; -} -NK_API nk_bool -nk_button_symbol_styled(struct nk_context *ctx, - const struct nk_style_button *style, enum nk_symbol_type symbol) -{ - struct nk_window *win; - struct nk_panel *layout; - const struct nk_input *in; - - struct nk_rect bounds; - enum nk_widget_layout_states state; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return 0; - - win = ctx->current; - layout = win->layout; - state = nk_widget(&bounds, ctx); - if (!state) return 0; - in = (state == NK_WIDGET_ROM || layout->flags & NK_WINDOW_ROM) ? 0 : &ctx->input; - return nk_do_button_symbol(&ctx->last_widget_state, &win->buffer, bounds, - symbol, ctx->button_behavior, style, in, ctx->style.font); -} -NK_API nk_bool -nk_button_symbol(struct nk_context *ctx, enum nk_symbol_type symbol) -{ - NK_ASSERT(ctx); - if (!ctx) return 0; - return nk_button_symbol_styled(ctx, &ctx->style.button, symbol); -} -NK_API nk_bool -nk_button_image_styled(struct nk_context *ctx, const struct nk_style_button *style, - struct nk_image img) -{ - struct nk_window *win; - struct nk_panel *layout; - const struct nk_input *in; - - struct nk_rect bounds; - enum nk_widget_layout_states state; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return 0; - - win = ctx->current; - layout = win->layout; - - state = nk_widget(&bounds, ctx); - if (!state) return 0; - in = (state == NK_WIDGET_ROM || layout->flags & NK_WINDOW_ROM) ? 0 : &ctx->input; - return nk_do_button_image(&ctx->last_widget_state, &win->buffer, bounds, - img, ctx->button_behavior, style, in); -} -NK_API nk_bool -nk_button_image(struct nk_context *ctx, struct nk_image img) -{ - NK_ASSERT(ctx); - if (!ctx) return 0; - return nk_button_image_styled(ctx, &ctx->style.button, img); -} -NK_API nk_bool -nk_button_symbol_text_styled(struct nk_context *ctx, - const struct nk_style_button *style, enum nk_symbol_type symbol, - const char *text, int len, nk_flags align) -{ - struct nk_window *win; - struct nk_panel *layout; - const struct nk_input *in; - - struct nk_rect bounds; - enum nk_widget_layout_states state; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return 0; - - win = ctx->current; - layout = win->layout; - - state = nk_widget(&bounds, ctx); - if (!state) return 0; - in = (state == NK_WIDGET_ROM || layout->flags & NK_WINDOW_ROM) ? 0 : &ctx->input; - return nk_do_button_text_symbol(&ctx->last_widget_state, &win->buffer, bounds, - symbol, text, len, align, ctx->button_behavior, - style, ctx->style.font, in); -} -NK_API nk_bool -nk_button_symbol_text(struct nk_context *ctx, enum nk_symbol_type symbol, - const char* text, int len, nk_flags align) -{ - NK_ASSERT(ctx); - if (!ctx) return 0; - return nk_button_symbol_text_styled(ctx, &ctx->style.button, symbol, text, len, align); -} -NK_API nk_bool nk_button_symbol_label(struct nk_context *ctx, enum nk_symbol_type symbol, - const char *label, nk_flags align) -{ - return nk_button_symbol_text(ctx, symbol, label, nk_strlen(label), align); -} -NK_API nk_bool nk_button_symbol_label_styled(struct nk_context *ctx, - const struct nk_style_button *style, enum nk_symbol_type symbol, - const char *title, nk_flags align) -{ - return nk_button_symbol_text_styled(ctx, style, symbol, title, nk_strlen(title), align); -} -NK_API nk_bool -nk_button_image_text_styled(struct nk_context *ctx, - const struct nk_style_button *style, struct nk_image img, const char *text, - int len, nk_flags align) -{ - struct nk_window *win; - struct nk_panel *layout; - const struct nk_input *in; - - struct nk_rect bounds; - enum nk_widget_layout_states state; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return 0; - - win = ctx->current; - layout = win->layout; - - state = nk_widget(&bounds, ctx); - if (!state) return 0; - in = (state == NK_WIDGET_ROM || layout->flags & NK_WINDOW_ROM) ? 0 : &ctx->input; - return nk_do_button_text_image(&ctx->last_widget_state, &win->buffer, - bounds, img, text, len, align, ctx->button_behavior, - style, ctx->style.font, in); -} -NK_API nk_bool -nk_button_image_text(struct nk_context *ctx, struct nk_image img, - const char *text, int len, nk_flags align) -{ - return nk_button_image_text_styled(ctx, &ctx->style.button,img, text, len, align); -} -NK_API nk_bool nk_button_image_label(struct nk_context *ctx, struct nk_image img, - const char *label, nk_flags align) -{ - return nk_button_image_text(ctx, img, label, nk_strlen(label), align); -} -NK_API nk_bool nk_button_image_label_styled(struct nk_context *ctx, - const struct nk_style_button *style, struct nk_image img, - const char *label, nk_flags text_alignment) -{ - return nk_button_image_text_styled(ctx, style, img, label, nk_strlen(label), text_alignment); -} - - - - - -/* =============================================================== - * - * TOGGLE - * - * ===============================================================*/ -NK_LIB nk_bool -nk_toggle_behavior(const struct nk_input *in, struct nk_rect select, - nk_flags *state, nk_bool active) -{ - nk_widget_state_reset(state); - if (nk_button_behavior(state, select, in, NK_BUTTON_DEFAULT)) { - *state = NK_WIDGET_STATE_ACTIVE; - active = !active; - } - if (*state & NK_WIDGET_STATE_HOVER && !nk_input_is_mouse_prev_hovering_rect(in, select)) - *state |= NK_WIDGET_STATE_ENTERED; - else if (nk_input_is_mouse_prev_hovering_rect(in, select)) - *state |= NK_WIDGET_STATE_LEFT; - return active; -} -NK_LIB void -nk_draw_checkbox(struct nk_command_buffer *out, - nk_flags state, const struct nk_style_toggle *style, nk_bool active, - const struct nk_rect *label, const struct nk_rect *selector, - const struct nk_rect *cursors, const char *string, int len, - const struct nk_user_font *font) -{ - const struct nk_style_item *background; - const struct nk_style_item *cursor; - struct nk_text text; - - /* select correct colors/images */ - if (state & NK_WIDGET_STATE_HOVER) { - background = &style->hover; - cursor = &style->cursor_hover; - text.text = style->text_hover; - } else if (state & NK_WIDGET_STATE_ACTIVED) { - background = &style->hover; - cursor = &style->cursor_hover; - text.text = style->text_active; - } else { - background = &style->normal; - cursor = &style->cursor_normal; - text.text = style->text_normal; - } - - /* draw background and cursor */ - if (background->type == NK_STYLE_ITEM_COLOR) { - nk_fill_rect(out, *selector, 0, style->border_color); - nk_fill_rect(out, nk_shrink_rect(*selector, style->border), 0, background->data.color); - } else nk_draw_image(out, *selector, &background->data.image, nk_white); - if (active) { - if (cursor->type == NK_STYLE_ITEM_IMAGE) - nk_draw_image(out, *cursors, &cursor->data.image, nk_white); - else nk_fill_rect(out, *cursors, 0, cursor->data.color); - } - - text.padding.x = 0; - text.padding.y = 0; - text.background = style->text_background; - nk_widget_text(out, *label, string, len, &text, NK_TEXT_LEFT, font); -} -NK_LIB void -nk_draw_option(struct nk_command_buffer *out, - nk_flags state, const struct nk_style_toggle *style, nk_bool active, - const struct nk_rect *label, const struct nk_rect *selector, - const struct nk_rect *cursors, const char *string, int len, - const struct nk_user_font *font) -{ - const struct nk_style_item *background; - const struct nk_style_item *cursor; - struct nk_text text; - - /* select correct colors/images */ - if (state & NK_WIDGET_STATE_HOVER) { - background = &style->hover; - cursor = &style->cursor_hover; - text.text = style->text_hover; - } else if (state & NK_WIDGET_STATE_ACTIVED) { - background = &style->hover; - cursor = &style->cursor_hover; - text.text = style->text_active; - } else { - background = &style->normal; - cursor = &style->cursor_normal; - text.text = style->text_normal; - } - - /* draw background and cursor */ - if (background->type == NK_STYLE_ITEM_COLOR) { - nk_fill_circle(out, *selector, style->border_color); - nk_fill_circle(out, nk_shrink_rect(*selector, style->border), background->data.color); - } else nk_draw_image(out, *selector, &background->data.image, nk_white); - if (active) { - if (cursor->type == NK_STYLE_ITEM_IMAGE) - nk_draw_image(out, *cursors, &cursor->data.image, nk_white); - else nk_fill_circle(out, *cursors, cursor->data.color); - } - - text.padding.x = 0; - text.padding.y = 0; - text.background = style->text_background; - nk_widget_text(out, *label, string, len, &text, NK_TEXT_LEFT, font); -} -NK_LIB nk_bool -nk_do_toggle(nk_flags *state, - struct nk_command_buffer *out, struct nk_rect r, - nk_bool *active, const char *str, int len, enum nk_toggle_type type, - const struct nk_style_toggle *style, const struct nk_input *in, - const struct nk_user_font *font) -{ - int was_active; - struct nk_rect bounds; - struct nk_rect select; - struct nk_rect cursor; - struct nk_rect label; - - NK_ASSERT(style); - NK_ASSERT(out); - NK_ASSERT(font); - if (!out || !style || !font || !active) - return 0; - - r.w = NK_MAX(r.w, font->height + 2 * style->padding.x); - r.h = NK_MAX(r.h, font->height + 2 * style->padding.y); - - /* add additional touch padding for touch screen devices */ - bounds.x = r.x - style->touch_padding.x; - bounds.y = r.y - style->touch_padding.y; - bounds.w = r.w + 2 * style->touch_padding.x; - bounds.h = r.h + 2 * style->touch_padding.y; - - /* calculate the selector space */ - select.w = font->height; - select.h = select.w; - select.y = r.y + r.h/2.0f - select.h/2.0f; - select.x = r.x; - - /* calculate the bounds of the cursor inside the selector */ - cursor.x = select.x + style->padding.x + style->border; - cursor.y = select.y + style->padding.y + style->border; - cursor.w = select.w - (2 * style->padding.x + 2 * style->border); - cursor.h = select.h - (2 * style->padding.y + 2 * style->border); - - /* label behind the selector */ - label.x = select.x + select.w + style->spacing; - label.y = select.y; - label.w = NK_MAX(r.x + r.w, label.x) - label.x; - label.h = select.w; - - /* update selector */ - was_active = *active; - *active = nk_toggle_behavior(in, bounds, state, *active); - - /* draw selector */ - if (style->draw_begin) - style->draw_begin(out, style->userdata); - if (type == NK_TOGGLE_CHECK) { - nk_draw_checkbox(out, *state, style, *active, &label, &select, &cursor, str, len, font); - } else { - nk_draw_option(out, *state, style, *active, &label, &select, &cursor, str, len, font); - } - if (style->draw_end) - style->draw_end(out, style->userdata); - return (was_active != *active); -} -/*---------------------------------------------------------------- - * - * CHECKBOX - * - * --------------------------------------------------------------*/ -NK_API nk_bool -nk_check_text(struct nk_context *ctx, const char *text, int len, nk_bool active) -{ - struct nk_window *win; - struct nk_panel *layout; - const struct nk_input *in; - const struct nk_style *style; - - struct nk_rect bounds; - enum nk_widget_layout_states state; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return active; - - win = ctx->current; - style = &ctx->style; - layout = win->layout; - - state = nk_widget(&bounds, ctx); - if (!state) return active; - in = (state == NK_WIDGET_ROM || layout->flags & NK_WINDOW_ROM) ? 0 : &ctx->input; - nk_do_toggle(&ctx->last_widget_state, &win->buffer, bounds, &active, - text, len, NK_TOGGLE_CHECK, &style->checkbox, in, style->font); - return active; -} -NK_API unsigned int -nk_check_flags_text(struct nk_context *ctx, const char *text, int len, - unsigned int flags, unsigned int value) -{ - int old_active; - NK_ASSERT(ctx); - NK_ASSERT(text); - if (!ctx || !text) return flags; - old_active = (int)((flags & value) & value); - if (nk_check_text(ctx, text, len, old_active)) - flags |= value; - else flags &= ~value; - return flags; -} -NK_API nk_bool -nk_checkbox_text(struct nk_context *ctx, const char *text, int len, nk_bool *active) -{ - int old_val; - NK_ASSERT(ctx); - NK_ASSERT(text); - NK_ASSERT(active); - if (!ctx || !text || !active) return 0; - old_val = *active; - *active = nk_check_text(ctx, text, len, *active); - return old_val != *active; -} -NK_API nk_bool -nk_checkbox_flags_text(struct nk_context *ctx, const char *text, int len, - unsigned int *flags, unsigned int value) -{ - nk_bool active; - NK_ASSERT(ctx); - NK_ASSERT(text); - NK_ASSERT(flags); - if (!ctx || !text || !flags) return 0; - - active = (int)((*flags & value) & value); - if (nk_checkbox_text(ctx, text, len, &active)) { - if (active) *flags |= value; - else *flags &= ~value; - return 1; - } - return 0; -} -NK_API nk_bool nk_check_label(struct nk_context *ctx, const char *label, nk_bool active) -{ - return nk_check_text(ctx, label, nk_strlen(label), active); -} -NK_API unsigned int nk_check_flags_label(struct nk_context *ctx, const char *label, - unsigned int flags, unsigned int value) -{ - return nk_check_flags_text(ctx, label, nk_strlen(label), flags, value); -} -NK_API nk_bool nk_checkbox_label(struct nk_context *ctx, const char *label, nk_bool *active) -{ - return nk_checkbox_text(ctx, label, nk_strlen(label), active); -} -NK_API nk_bool nk_checkbox_flags_label(struct nk_context *ctx, const char *label, - unsigned int *flags, unsigned int value) -{ - return nk_checkbox_flags_text(ctx, label, nk_strlen(label), flags, value); -} -/*---------------------------------------------------------------- - * - * OPTION - * - * --------------------------------------------------------------*/ -NK_API nk_bool -nk_option_text(struct nk_context *ctx, const char *text, int len, nk_bool is_active) -{ - struct nk_window *win; - struct nk_panel *layout; - const struct nk_input *in; - const struct nk_style *style; - - struct nk_rect bounds; - enum nk_widget_layout_states state; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return is_active; - - win = ctx->current; - style = &ctx->style; - layout = win->layout; - - state = nk_widget(&bounds, ctx); - if (!state) return (int)state; - in = (state == NK_WIDGET_ROM || layout->flags & NK_WINDOW_ROM) ? 0 : &ctx->input; - nk_do_toggle(&ctx->last_widget_state, &win->buffer, bounds, &is_active, - text, len, NK_TOGGLE_OPTION, &style->option, in, style->font); - return is_active; -} -NK_API nk_bool -nk_radio_text(struct nk_context *ctx, const char *text, int len, nk_bool *active) -{ - int old_value; - NK_ASSERT(ctx); - NK_ASSERT(text); - NK_ASSERT(active); - if (!ctx || !text || !active) return 0; - old_value = *active; - *active = nk_option_text(ctx, text, len, old_value); - return old_value != *active; -} -NK_API nk_bool -nk_option_label(struct nk_context *ctx, const char *label, nk_bool active) -{ - return nk_option_text(ctx, label, nk_strlen(label), active); -} -NK_API nk_bool -nk_radio_label(struct nk_context *ctx, const char *label, nk_bool *active) -{ - return nk_radio_text(ctx, label, nk_strlen(label), active); -} - - - - - -/* =============================================================== - * - * SELECTABLE - * - * ===============================================================*/ -NK_LIB void -nk_draw_selectable(struct nk_command_buffer *out, - nk_flags state, const struct nk_style_selectable *style, nk_bool active, - const struct nk_rect *bounds, - const struct nk_rect *icon, const struct nk_image *img, enum nk_symbol_type sym, - const char *string, int len, nk_flags align, const struct nk_user_font *font) -{ - const struct nk_style_item *background; - struct nk_text text; - text.padding = style->padding; - - /* select correct colors/images */ - if (!active) { - if (state & NK_WIDGET_STATE_ACTIVED) { - background = &style->pressed; - text.text = style->text_pressed; - } else if (state & NK_WIDGET_STATE_HOVER) { - background = &style->hover; - text.text = style->text_hover; - } else { - background = &style->normal; - text.text = style->text_normal; - } - } else { - if (state & NK_WIDGET_STATE_ACTIVED) { - background = &style->pressed_active; - text.text = style->text_pressed_active; - } else if (state & NK_WIDGET_STATE_HOVER) { - background = &style->hover_active; - text.text = style->text_hover_active; - } else { - background = &style->normal_active; - text.text = style->text_normal_active; - } - } - /* draw selectable background and text */ - switch (background->type) { - case NK_STYLE_ITEM_IMAGE: - text.background = nk_rgba(0, 0, 0, 0); - nk_draw_image(out, *bounds, &background->data.image, nk_white); - break; - case NK_STYLE_ITEM_NINE_SLICE: - text.background = nk_rgba(0, 0, 0, 0); - nk_draw_nine_slice(out, *bounds, &background->data.slice, nk_white); - break; - case NK_STYLE_ITEM_COLOR: - text.background = background->data.color; - nk_fill_rect(out, *bounds, style->rounding, background->data.color); - break; - } - if (icon) { - if (img) nk_draw_image(out, *icon, img, nk_white); - else nk_draw_symbol(out, sym, *icon, text.background, text.text, 1, font); - } - nk_widget_text(out, *bounds, string, len, &text, align, font); -} -NK_LIB nk_bool -nk_do_selectable(nk_flags *state, struct nk_command_buffer *out, - struct nk_rect bounds, const char *str, int len, nk_flags align, nk_bool *value, - const struct nk_style_selectable *style, const struct nk_input *in, - const struct nk_user_font *font) -{ - int old_value; - struct nk_rect touch; - - NK_ASSERT(state); - NK_ASSERT(out); - NK_ASSERT(str); - NK_ASSERT(len); - NK_ASSERT(value); - NK_ASSERT(style); - NK_ASSERT(font); - - if (!state || !out || !str || !len || !value || !style || !font) return 0; - old_value = *value; - - /* remove padding */ - touch.x = bounds.x - style->touch_padding.x; - touch.y = bounds.y - style->touch_padding.y; - touch.w = bounds.w + style->touch_padding.x * 2; - touch.h = bounds.h + style->touch_padding.y * 2; - - /* update button */ - if (nk_button_behavior(state, touch, in, NK_BUTTON_DEFAULT)) - *value = !(*value); - - /* draw selectable */ - if (style->draw_begin) style->draw_begin(out, style->userdata); - nk_draw_selectable(out, *state, style, *value, &bounds, 0,0,NK_SYMBOL_NONE, str, len, align, font); - if (style->draw_end) style->draw_end(out, style->userdata); - return old_value != *value; -} -NK_LIB nk_bool -nk_do_selectable_image(nk_flags *state, struct nk_command_buffer *out, - struct nk_rect bounds, const char *str, int len, nk_flags align, nk_bool *value, - const struct nk_image *img, const struct nk_style_selectable *style, - const struct nk_input *in, const struct nk_user_font *font) -{ - nk_bool old_value; - struct nk_rect touch; - struct nk_rect icon; - - NK_ASSERT(state); - NK_ASSERT(out); - NK_ASSERT(str); - NK_ASSERT(len); - NK_ASSERT(value); - NK_ASSERT(style); - NK_ASSERT(font); - - if (!state || !out || !str || !len || !value || !style || !font) return 0; - old_value = *value; - - /* toggle behavior */ - touch.x = bounds.x - style->touch_padding.x; - touch.y = bounds.y - style->touch_padding.y; - touch.w = bounds.w + style->touch_padding.x * 2; - touch.h = bounds.h + style->touch_padding.y * 2; - if (nk_button_behavior(state, touch, in, NK_BUTTON_DEFAULT)) - *value = !(*value); - - icon.y = bounds.y + style->padding.y; - icon.w = icon.h = bounds.h - 2 * style->padding.y; - if (align & NK_TEXT_ALIGN_LEFT) { - icon.x = (bounds.x + bounds.w) - (2 * style->padding.x + icon.w); - icon.x = NK_MAX(icon.x, 0); - } else icon.x = bounds.x + 2 * style->padding.x; - - icon.x += style->image_padding.x; - icon.y += style->image_padding.y; - icon.w -= 2 * style->image_padding.x; - icon.h -= 2 * style->image_padding.y; - - /* draw selectable */ - if (style->draw_begin) style->draw_begin(out, style->userdata); - nk_draw_selectable(out, *state, style, *value, &bounds, &icon, img, NK_SYMBOL_NONE, str, len, align, font); - if (style->draw_end) style->draw_end(out, style->userdata); - return old_value != *value; -} -NK_LIB nk_bool -nk_do_selectable_symbol(nk_flags *state, struct nk_command_buffer *out, - struct nk_rect bounds, const char *str, int len, nk_flags align, nk_bool *value, - enum nk_symbol_type sym, const struct nk_style_selectable *style, - const struct nk_input *in, const struct nk_user_font *font) -{ - int old_value; - struct nk_rect touch; - struct nk_rect icon; - - NK_ASSERT(state); - NK_ASSERT(out); - NK_ASSERT(str); - NK_ASSERT(len); - NK_ASSERT(value); - NK_ASSERT(style); - NK_ASSERT(font); - - if (!state || !out || !str || !len || !value || !style || !font) return 0; - old_value = *value; - - /* toggle behavior */ - touch.x = bounds.x - style->touch_padding.x; - touch.y = bounds.y - style->touch_padding.y; - touch.w = bounds.w + style->touch_padding.x * 2; - touch.h = bounds.h + style->touch_padding.y * 2; - if (nk_button_behavior(state, touch, in, NK_BUTTON_DEFAULT)) - *value = !(*value); - - icon.y = bounds.y + style->padding.y; - icon.w = icon.h = bounds.h - 2 * style->padding.y; - if (align & NK_TEXT_ALIGN_LEFT) { - icon.x = (bounds.x + bounds.w) - (2 * style->padding.x + icon.w); - icon.x = NK_MAX(icon.x, 0); - } else icon.x = bounds.x + 2 * style->padding.x; - - icon.x += style->image_padding.x; - icon.y += style->image_padding.y; - icon.w -= 2 * style->image_padding.x; - icon.h -= 2 * style->image_padding.y; - - /* draw selectable */ - if (style->draw_begin) style->draw_begin(out, style->userdata); - nk_draw_selectable(out, *state, style, *value, &bounds, &icon, 0, sym, str, len, align, font); - if (style->draw_end) style->draw_end(out, style->userdata); - return old_value != *value; -} - -NK_API nk_bool -nk_selectable_text(struct nk_context *ctx, const char *str, int len, - nk_flags align, nk_bool *value) -{ - struct nk_window *win; - struct nk_panel *layout; - const struct nk_input *in; - const struct nk_style *style; - - enum nk_widget_layout_states state; - struct nk_rect bounds; - - NK_ASSERT(ctx); - NK_ASSERT(value); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout || !value) - return 0; - - win = ctx->current; - layout = win->layout; - style = &ctx->style; - - state = nk_widget(&bounds, ctx); - if (!state) return 0; - in = (state == NK_WIDGET_ROM || layout->flags & NK_WINDOW_ROM) ? 0 : &ctx->input; - return nk_do_selectable(&ctx->last_widget_state, &win->buffer, bounds, - str, len, align, value, &style->selectable, in, style->font); -} -NK_API nk_bool -nk_selectable_image_text(struct nk_context *ctx, struct nk_image img, - const char *str, int len, nk_flags align, nk_bool *value) -{ - struct nk_window *win; - struct nk_panel *layout; - const struct nk_input *in; - const struct nk_style *style; - - enum nk_widget_layout_states state; - struct nk_rect bounds; - - NK_ASSERT(ctx); - NK_ASSERT(value); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout || !value) - return 0; - - win = ctx->current; - layout = win->layout; - style = &ctx->style; - - state = nk_widget(&bounds, ctx); - if (!state) return 0; - in = (state == NK_WIDGET_ROM || layout->flags & NK_WINDOW_ROM) ? 0 : &ctx->input; - return nk_do_selectable_image(&ctx->last_widget_state, &win->buffer, bounds, - str, len, align, value, &img, &style->selectable, in, style->font); -} -NK_API nk_bool -nk_selectable_symbol_text(struct nk_context *ctx, enum nk_symbol_type sym, - const char *str, int len, nk_flags align, nk_bool *value) -{ - struct nk_window *win; - struct nk_panel *layout; - const struct nk_input *in; - const struct nk_style *style; - - enum nk_widget_layout_states state; - struct nk_rect bounds; - - NK_ASSERT(ctx); - NK_ASSERT(value); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout || !value) - return 0; - - win = ctx->current; - layout = win->layout; - style = &ctx->style; - - state = nk_widget(&bounds, ctx); - if (!state) return 0; - in = (state == NK_WIDGET_ROM || layout->flags & NK_WINDOW_ROM) ? 0 : &ctx->input; - return nk_do_selectable_symbol(&ctx->last_widget_state, &win->buffer, bounds, - str, len, align, value, sym, &style->selectable, in, style->font); -} -NK_API nk_bool -nk_selectable_symbol_label(struct nk_context *ctx, enum nk_symbol_type sym, - const char *title, nk_flags align, nk_bool *value) -{ - return nk_selectable_symbol_text(ctx, sym, title, nk_strlen(title), align, value); -} -NK_API nk_bool nk_select_text(struct nk_context *ctx, const char *str, int len, - nk_flags align, nk_bool value) -{ - nk_selectable_text(ctx, str, len, align, &value);return value; -} -NK_API nk_bool nk_selectable_label(struct nk_context *ctx, const char *str, nk_flags align, nk_bool *value) -{ - return nk_selectable_text(ctx, str, nk_strlen(str), align, value); -} -NK_API nk_bool nk_selectable_image_label(struct nk_context *ctx,struct nk_image img, - const char *str, nk_flags align, nk_bool *value) -{ - return nk_selectable_image_text(ctx, img, str, nk_strlen(str), align, value); -} -NK_API nk_bool nk_select_label(struct nk_context *ctx, const char *str, nk_flags align, nk_bool value) -{ - nk_selectable_text(ctx, str, nk_strlen(str), align, &value);return value; -} -NK_API nk_bool nk_select_image_label(struct nk_context *ctx, struct nk_image img, - const char *str, nk_flags align, nk_bool value) -{ - nk_selectable_image_text(ctx, img, str, nk_strlen(str), align, &value);return value; -} -NK_API nk_bool nk_select_image_text(struct nk_context *ctx, struct nk_image img, - const char *str, int len, nk_flags align, nk_bool value) -{ - nk_selectable_image_text(ctx, img, str, len, align, &value);return value; -} -NK_API nk_bool -nk_select_symbol_text(struct nk_context *ctx, enum nk_symbol_type sym, - const char *title, int title_len, nk_flags align, nk_bool value) -{ - nk_selectable_symbol_text(ctx, sym, title, title_len, align, &value);return value; -} -NK_API nk_bool -nk_select_symbol_label(struct nk_context *ctx, enum nk_symbol_type sym, - const char *title, nk_flags align, nk_bool value) -{ - return nk_select_symbol_text(ctx, sym, title, nk_strlen(title), align, value); -} - - - - - -/* =============================================================== - * - * SLIDER - * - * ===============================================================*/ -NK_LIB float -nk_slider_behavior(nk_flags *state, struct nk_rect *logical_cursor, - struct nk_rect *visual_cursor, struct nk_input *in, - struct nk_rect bounds, float slider_min, float slider_max, float slider_value, - float slider_step, float slider_steps) -{ - int left_mouse_down; - int left_mouse_click_in_cursor; - - /* check if visual cursor is being dragged */ - nk_widget_state_reset(state); - left_mouse_down = in && in->mouse.buttons[NK_BUTTON_LEFT].down; - left_mouse_click_in_cursor = in && nk_input_has_mouse_click_down_in_rect(in, - NK_BUTTON_LEFT, *visual_cursor, nk_true); - - if (left_mouse_down && left_mouse_click_in_cursor) { - float ratio = 0; - const float d = in->mouse.pos.x - (visual_cursor->x+visual_cursor->w*0.5f); - const float pxstep = bounds.w / slider_steps; - - /* only update value if the next slider step is reached */ - *state = NK_WIDGET_STATE_ACTIVE; - if (NK_ABS(d) >= pxstep) { - const float steps = (float)((int)(NK_ABS(d) / pxstep)); - slider_value += (d > 0) ? (slider_step*steps) : -(slider_step*steps); - slider_value = NK_CLAMP(slider_min, slider_value, slider_max); - ratio = (slider_value - slider_min)/slider_step; - logical_cursor->x = bounds.x + (logical_cursor->w * ratio); - in->mouse.buttons[NK_BUTTON_LEFT].clicked_pos.x = logical_cursor->x; - } - } - - /* slider widget state */ - if (nk_input_is_mouse_hovering_rect(in, bounds)) - *state = NK_WIDGET_STATE_HOVERED; - if (*state & NK_WIDGET_STATE_HOVER && - !nk_input_is_mouse_prev_hovering_rect(in, bounds)) - *state |= NK_WIDGET_STATE_ENTERED; - else if (nk_input_is_mouse_prev_hovering_rect(in, bounds)) - *state |= NK_WIDGET_STATE_LEFT; - return slider_value; -} -NK_LIB void -nk_draw_slider(struct nk_command_buffer *out, nk_flags state, - const struct nk_style_slider *style, const struct nk_rect *bounds, - const struct nk_rect *visual_cursor, float min, float value, float max) -{ - struct nk_rect fill; - struct nk_rect bar; - const struct nk_style_item *background; - - /* select correct slider images/colors */ - struct nk_color bar_color; - const struct nk_style_item *cursor; - - NK_UNUSED(min); - NK_UNUSED(max); - NK_UNUSED(value); - - if (state & NK_WIDGET_STATE_ACTIVED) { - background = &style->active; - bar_color = style->bar_active; - cursor = &style->cursor_active; - } else if (state & NK_WIDGET_STATE_HOVER) { - background = &style->hover; - bar_color = style->bar_hover; - cursor = &style->cursor_hover; - } else { - background = &style->normal; - bar_color = style->bar_normal; - cursor = &style->cursor_normal; - } - /* calculate slider background bar */ - bar.x = bounds->x; - bar.y = (visual_cursor->y + visual_cursor->h/2) - bounds->h/12; - bar.w = bounds->w; - bar.h = bounds->h/6; - - /* filled background bar style */ - fill.w = (visual_cursor->x + (visual_cursor->w/2.0f)) - bar.x; - fill.x = bar.x; - fill.y = bar.y; - fill.h = bar.h; - - /* draw background */ - switch(background->type) { - case NK_STYLE_ITEM_IMAGE: - nk_draw_image(out, *bounds, &background->data.image, nk_white); - break; - case NK_STYLE_ITEM_NINE_SLICE: - nk_draw_nine_slice(out, *bounds, &background->data.slice, nk_white); - break; - case NK_STYLE_ITEM_COLOR: - nk_fill_rect(out, *bounds, style->rounding, background->data.color); - nk_stroke_rect(out, *bounds, style->rounding, style->border, style->border_color); - break; - } - - /* draw slider bar */ - nk_fill_rect(out, bar, style->rounding, bar_color); - nk_fill_rect(out, fill, style->rounding, style->bar_filled); - - /* draw cursor */ - if (cursor->type == NK_STYLE_ITEM_IMAGE) - nk_draw_image(out, *visual_cursor, &cursor->data.image, nk_white); - else - nk_fill_circle(out, *visual_cursor, cursor->data.color); -} -NK_LIB float -nk_do_slider(nk_flags *state, - struct nk_command_buffer *out, struct nk_rect bounds, - float min, float val, float max, float step, - const struct nk_style_slider *style, struct nk_input *in, - const struct nk_user_font *font) -{ - float slider_range; - float slider_min; - float slider_max; - float slider_value; - float slider_steps; - float cursor_offset; - - struct nk_rect visual_cursor; - struct nk_rect logical_cursor; - - NK_ASSERT(style); - NK_ASSERT(out); - if (!out || !style) - return 0; - - /* remove padding from slider bounds */ - bounds.x = bounds.x + style->padding.x; - bounds.y = bounds.y + style->padding.y; - bounds.h = NK_MAX(bounds.h, 2*style->padding.y); - bounds.w = NK_MAX(bounds.w, 2*style->padding.x + style->cursor_size.x); - bounds.w -= 2 * style->padding.x; - bounds.h -= 2 * style->padding.y; - - /* optional buttons */ - if (style->show_buttons) { - nk_flags ws; - struct nk_rect button; - button.y = bounds.y; - button.w = bounds.h; - button.h = bounds.h; - - /* decrement button */ - button.x = bounds.x; - if (nk_do_button_symbol(&ws, out, button, style->dec_symbol, NK_BUTTON_DEFAULT, - &style->dec_button, in, font)) - val -= step; - - /* increment button */ - button.x = (bounds.x + bounds.w) - button.w; - if (nk_do_button_symbol(&ws, out, button, style->inc_symbol, NK_BUTTON_DEFAULT, - &style->inc_button, in, font)) - val += step; - - bounds.x = bounds.x + button.w + style->spacing.x; - bounds.w = bounds.w - (2*button.w + 2*style->spacing.x); - } - - /* remove one cursor size to support visual cursor */ - bounds.x += style->cursor_size.x*0.5f; - bounds.w -= style->cursor_size.x; - - /* make sure the provided values are correct */ - slider_max = NK_MAX(min, max); - slider_min = NK_MIN(min, max); - slider_value = NK_CLAMP(slider_min, val, slider_max); - slider_range = slider_max - slider_min; - slider_steps = slider_range / step; - cursor_offset = (slider_value - slider_min) / step; - - /* calculate cursor - Basically you have two cursors. One for visual representation and interaction - and one for updating the actual cursor value. */ - logical_cursor.h = bounds.h; - logical_cursor.w = bounds.w / slider_steps; - logical_cursor.x = bounds.x + (logical_cursor.w * cursor_offset); - logical_cursor.y = bounds.y; - - visual_cursor.h = style->cursor_size.y; - visual_cursor.w = style->cursor_size.x; - visual_cursor.y = (bounds.y + bounds.h*0.5f) - visual_cursor.h*0.5f; - visual_cursor.x = logical_cursor.x - visual_cursor.w*0.5f; - - slider_value = nk_slider_behavior(state, &logical_cursor, &visual_cursor, - in, bounds, slider_min, slider_max, slider_value, step, slider_steps); - visual_cursor.x = logical_cursor.x - visual_cursor.w*0.5f; - - /* draw slider */ - if (style->draw_begin) style->draw_begin(out, style->userdata); - nk_draw_slider(out, *state, style, &bounds, &visual_cursor, slider_min, slider_value, slider_max); - if (style->draw_end) style->draw_end(out, style->userdata); - return slider_value; -} -NK_API nk_bool -nk_slider_float(struct nk_context *ctx, float min_value, float *value, float max_value, - float value_step) -{ - struct nk_window *win; - struct nk_panel *layout; - struct nk_input *in; - const struct nk_style *style; - - int ret = 0; - float old_value; - struct nk_rect bounds; - enum nk_widget_layout_states state; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - NK_ASSERT(value); - if (!ctx || !ctx->current || !ctx->current->layout || !value) - return ret; - - win = ctx->current; - style = &ctx->style; - layout = win->layout; - - state = nk_widget(&bounds, ctx); - if (!state) return ret; - in = (/*state == NK_WIDGET_ROM || */ layout->flags & NK_WINDOW_ROM) ? 0 : &ctx->input; - - old_value = *value; - *value = nk_do_slider(&ctx->last_widget_state, &win->buffer, bounds, min_value, - old_value, max_value, value_step, &style->slider, in, style->font); - return (old_value > *value || old_value < *value); -} -NK_API float -nk_slide_float(struct nk_context *ctx, float min, float val, float max, float step) -{ - nk_slider_float(ctx, min, &val, max, step); return val; -} -NK_API int -nk_slide_int(struct nk_context *ctx, int min, int val, int max, int step) -{ - float value = (float)val; - nk_slider_float(ctx, (float)min, &value, (float)max, (float)step); - return (int)value; -} -NK_API nk_bool -nk_slider_int(struct nk_context *ctx, int min, int *val, int max, int step) -{ - int ret; - float value = (float)*val; - ret = nk_slider_float(ctx, (float)min, &value, (float)max, (float)step); - *val = (int)value; - return ret; -} - - - - - -/* =============================================================== - * - * PROGRESS - * - * ===============================================================*/ -NK_LIB nk_size -nk_progress_behavior(nk_flags *state, struct nk_input *in, - struct nk_rect r, struct nk_rect cursor, nk_size max, nk_size value, nk_bool modifiable) -{ - int left_mouse_down = 0; - int left_mouse_click_in_cursor = 0; - - nk_widget_state_reset(state); - if (!in || !modifiable) return value; - left_mouse_down = in && in->mouse.buttons[NK_BUTTON_LEFT].down; - left_mouse_click_in_cursor = in && nk_input_has_mouse_click_down_in_rect(in, - NK_BUTTON_LEFT, cursor, nk_true); - if (nk_input_is_mouse_hovering_rect(in, r)) - *state = NK_WIDGET_STATE_HOVERED; - - if (in && left_mouse_down && left_mouse_click_in_cursor) { - if (left_mouse_down && left_mouse_click_in_cursor) { - float ratio = NK_MAX(0, (float)(in->mouse.pos.x - cursor.x)) / (float)cursor.w; - value = (nk_size)NK_CLAMP(0, (float)max * ratio, (float)max); - in->mouse.buttons[NK_BUTTON_LEFT].clicked_pos.x = cursor.x + cursor.w/2.0f; - *state |= NK_WIDGET_STATE_ACTIVE; - } - } - /* set progressbar widget state */ - if (*state & NK_WIDGET_STATE_HOVER && !nk_input_is_mouse_prev_hovering_rect(in, r)) - *state |= NK_WIDGET_STATE_ENTERED; - else if (nk_input_is_mouse_prev_hovering_rect(in, r)) - *state |= NK_WIDGET_STATE_LEFT; - return value; -} -NK_LIB void -nk_draw_progress(struct nk_command_buffer *out, nk_flags state, - const struct nk_style_progress *style, const struct nk_rect *bounds, - const struct nk_rect *scursor, nk_size value, nk_size max) -{ - const struct nk_style_item *background; - const struct nk_style_item *cursor; - - NK_UNUSED(max); - NK_UNUSED(value); - - /* select correct colors/images to draw */ - if (state & NK_WIDGET_STATE_ACTIVED) { - background = &style->active; - cursor = &style->cursor_active; - } else if (state & NK_WIDGET_STATE_HOVER){ - background = &style->hover; - cursor = &style->cursor_hover; - } else { - background = &style->normal; - cursor = &style->cursor_normal; - } - - /* draw background */ - switch(background->type) { - case NK_STYLE_ITEM_IMAGE: - nk_draw_image(out, *bounds, &background->data.image, nk_white); - break; - case NK_STYLE_ITEM_NINE_SLICE: - nk_draw_nine_slice(out, *bounds, &background->data.slice, nk_white); - break; - case NK_STYLE_ITEM_COLOR: - nk_fill_rect(out, *bounds, style->rounding, background->data.color); - nk_stroke_rect(out, *bounds, style->rounding, style->border, style->border_color); - break; - } - - /* draw cursor */ - switch(cursor->type) { - case NK_STYLE_ITEM_IMAGE: - nk_draw_image(out, *scursor, &cursor->data.image, nk_white); - break; - case NK_STYLE_ITEM_NINE_SLICE: - nk_draw_nine_slice(out, *scursor, &cursor->data.slice, nk_white); - break; - case NK_STYLE_ITEM_COLOR: - nk_fill_rect(out, *scursor, style->rounding, cursor->data.color); - nk_stroke_rect(out, *scursor, style->rounding, style->border, style->border_color); - break; - } -} -NK_LIB nk_size -nk_do_progress(nk_flags *state, - struct nk_command_buffer *out, struct nk_rect bounds, - nk_size value, nk_size max, nk_bool modifiable, - const struct nk_style_progress *style, struct nk_input *in) -{ - float prog_scale; - nk_size prog_value; - struct nk_rect cursor; - - NK_ASSERT(style); - NK_ASSERT(out); - if (!out || !style) return 0; - - /* calculate progressbar cursor */ - cursor.w = NK_MAX(bounds.w, 2 * style->padding.x + 2 * style->border); - cursor.h = NK_MAX(bounds.h, 2 * style->padding.y + 2 * style->border); - cursor = nk_pad_rect(bounds, nk_vec2(style->padding.x + style->border, style->padding.y + style->border)); - prog_scale = (float)value / (float)max; - - /* update progressbar */ - prog_value = NK_MIN(value, max); - prog_value = nk_progress_behavior(state, in, bounds, cursor,max, prog_value, modifiable); - cursor.w = cursor.w * prog_scale; - - /* draw progressbar */ - if (style->draw_begin) style->draw_begin(out, style->userdata); - nk_draw_progress(out, *state, style, &bounds, &cursor, value, max); - if (style->draw_end) style->draw_end(out, style->userdata); - return prog_value; -} -NK_API nk_bool -nk_progress(struct nk_context *ctx, nk_size *cur, nk_size max, nk_bool is_modifyable) -{ - struct nk_window *win; - struct nk_panel *layout; - const struct nk_style *style; - struct nk_input *in; - - struct nk_rect bounds; - enum nk_widget_layout_states state; - nk_size old_value; - - NK_ASSERT(ctx); - NK_ASSERT(cur); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout || !cur) - return 0; - - win = ctx->current; - style = &ctx->style; - layout = win->layout; - state = nk_widget(&bounds, ctx); - if (!state) return 0; - - in = (state == NK_WIDGET_ROM || layout->flags & NK_WINDOW_ROM) ? 0 : &ctx->input; - old_value = *cur; - *cur = nk_do_progress(&ctx->last_widget_state, &win->buffer, bounds, - *cur, max, is_modifyable, &style->progress, in); - return (*cur != old_value); -} -NK_API nk_size -nk_prog(struct nk_context *ctx, nk_size cur, nk_size max, nk_bool modifyable) -{ - nk_progress(ctx, &cur, max, modifyable); - return cur; -} - - - - - -/* =============================================================== - * - * SCROLLBAR - * - * ===============================================================*/ -NK_LIB float -nk_scrollbar_behavior(nk_flags *state, struct nk_input *in, - int has_scrolling, const struct nk_rect *scroll, - const struct nk_rect *cursor, const struct nk_rect *empty0, - const struct nk_rect *empty1, float scroll_offset, - float target, float scroll_step, enum nk_orientation o) -{ - nk_flags ws = 0; - int left_mouse_down; - unsigned int left_mouse_clicked; - int left_mouse_click_in_cursor; - float scroll_delta; - - nk_widget_state_reset(state); - if (!in) return scroll_offset; - - left_mouse_down = in->mouse.buttons[NK_BUTTON_LEFT].down; - left_mouse_clicked = in->mouse.buttons[NK_BUTTON_LEFT].clicked; - left_mouse_click_in_cursor = nk_input_has_mouse_click_down_in_rect(in, - NK_BUTTON_LEFT, *cursor, nk_true); - if (nk_input_is_mouse_hovering_rect(in, *scroll)) - *state = NK_WIDGET_STATE_HOVERED; - - scroll_delta = (o == NK_VERTICAL) ? in->mouse.scroll_delta.y: in->mouse.scroll_delta.x; - if (left_mouse_down && left_mouse_click_in_cursor && !left_mouse_clicked) { - /* update cursor by mouse dragging */ - float pixel, delta; - *state = NK_WIDGET_STATE_ACTIVE; - if (o == NK_VERTICAL) { - float cursor_y; - pixel = in->mouse.delta.y; - delta = (pixel / scroll->h) * target; - scroll_offset = NK_CLAMP(0, scroll_offset + delta, target - scroll->h); - cursor_y = scroll->y + ((scroll_offset/target) * scroll->h); - in->mouse.buttons[NK_BUTTON_LEFT].clicked_pos.y = cursor_y + cursor->h/2.0f; - } else { - float cursor_x; - pixel = in->mouse.delta.x; - delta = (pixel / scroll->w) * target; - scroll_offset = NK_CLAMP(0, scroll_offset + delta, target - scroll->w); - cursor_x = scroll->x + ((scroll_offset/target) * scroll->w); - in->mouse.buttons[NK_BUTTON_LEFT].clicked_pos.x = cursor_x + cursor->w/2.0f; - } - } else if ((nk_input_is_key_pressed(in, NK_KEY_SCROLL_UP) && o == NK_VERTICAL && has_scrolling)|| - nk_button_behavior(&ws, *empty0, in, NK_BUTTON_DEFAULT)) { - /* scroll page up by click on empty space or shortcut */ - if (o == NK_VERTICAL) - scroll_offset = NK_MAX(0, scroll_offset - scroll->h); - else scroll_offset = NK_MAX(0, scroll_offset - scroll->w); - } else if ((nk_input_is_key_pressed(in, NK_KEY_SCROLL_DOWN) && o == NK_VERTICAL && has_scrolling) || - nk_button_behavior(&ws, *empty1, in, NK_BUTTON_DEFAULT)) { - /* scroll page down by click on empty space or shortcut */ - if (o == NK_VERTICAL) - scroll_offset = NK_MIN(scroll_offset + scroll->h, target - scroll->h); - else scroll_offset = NK_MIN(scroll_offset + scroll->w, target - scroll->w); - } else if (has_scrolling) { - if ((scroll_delta < 0 || (scroll_delta > 0))) { - /* update cursor by mouse scrolling */ - scroll_offset = scroll_offset + scroll_step * (-scroll_delta); - if (o == NK_VERTICAL) - scroll_offset = NK_CLAMP(0, scroll_offset, target - scroll->h); - else scroll_offset = NK_CLAMP(0, scroll_offset, target - scroll->w); - } else if (nk_input_is_key_pressed(in, NK_KEY_SCROLL_START)) { - /* update cursor to the beginning */ - if (o == NK_VERTICAL) scroll_offset = 0; - } else if (nk_input_is_key_pressed(in, NK_KEY_SCROLL_END)) { - /* update cursor to the end */ - if (o == NK_VERTICAL) scroll_offset = target - scroll->h; - } - } - if (*state & NK_WIDGET_STATE_HOVER && !nk_input_is_mouse_prev_hovering_rect(in, *scroll)) - *state |= NK_WIDGET_STATE_ENTERED; - else if (nk_input_is_mouse_prev_hovering_rect(in, *scroll)) - *state |= NK_WIDGET_STATE_LEFT; - return scroll_offset; -} -NK_LIB void -nk_draw_scrollbar(struct nk_command_buffer *out, nk_flags state, - const struct nk_style_scrollbar *style, const struct nk_rect *bounds, - const struct nk_rect *scroll) -{ - const struct nk_style_item *background; - const struct nk_style_item *cursor; - - /* select correct colors/images to draw */ - if (state & NK_WIDGET_STATE_ACTIVED) { - background = &style->active; - cursor = &style->cursor_active; - } else if (state & NK_WIDGET_STATE_HOVER) { - background = &style->hover; - cursor = &style->cursor_hover; - } else { - background = &style->normal; - cursor = &style->cursor_normal; - } - - /* draw background */ - switch (background->type) { - case NK_STYLE_ITEM_IMAGE: - nk_draw_image(out, *bounds, &background->data.image, nk_white); - break; - case NK_STYLE_ITEM_NINE_SLICE: - nk_draw_nine_slice(out, *bounds, &background->data.slice, nk_white); - break; - case NK_STYLE_ITEM_COLOR: - nk_fill_rect(out, *bounds, style->rounding, background->data.color); - nk_stroke_rect(out, *bounds, style->rounding, style->border, style->border_color); - break; - } - - /* draw cursor */ - switch (cursor->type) { - case NK_STYLE_ITEM_IMAGE: - nk_draw_image(out, *scroll, &cursor->data.image, nk_white); - break; - case NK_STYLE_ITEM_NINE_SLICE: - nk_draw_nine_slice(out, *scroll, &cursor->data.slice, nk_white); - break; - case NK_STYLE_ITEM_COLOR: - nk_fill_rect(out, *scroll, style->rounding_cursor, cursor->data.color); - nk_stroke_rect(out, *scroll, style->rounding_cursor, style->border_cursor, style->cursor_border_color); - break; - } -} -NK_LIB float -nk_do_scrollbarv(nk_flags *state, - struct nk_command_buffer *out, struct nk_rect scroll, int has_scrolling, - float offset, float target, float step, float button_pixel_inc, - const struct nk_style_scrollbar *style, struct nk_input *in, - const struct nk_user_font *font) -{ - struct nk_rect empty_north; - struct nk_rect empty_south; - struct nk_rect cursor; - - float scroll_step; - float scroll_offset; - float scroll_off; - float scroll_ratio; - - NK_ASSERT(out); - NK_ASSERT(style); - NK_ASSERT(state); - if (!out || !style) return 0; - - scroll.w = NK_MAX(scroll.w, 1); - scroll.h = NK_MAX(scroll.h, 0); - if (target <= scroll.h) return 0; - - /* optional scrollbar buttons */ - if (style->show_buttons) { - nk_flags ws; - float scroll_h; - struct nk_rect button; - - button.x = scroll.x; - button.w = scroll.w; - button.h = scroll.w; - - scroll_h = NK_MAX(scroll.h - 2 * button.h,0); - scroll_step = NK_MIN(step, button_pixel_inc); - - /* decrement button */ - button.y = scroll.y; - if (nk_do_button_symbol(&ws, out, button, style->dec_symbol, - NK_BUTTON_REPEATER, &style->dec_button, in, font)) - offset = offset - scroll_step; - - /* increment button */ - button.y = scroll.y + scroll.h - button.h; - if (nk_do_button_symbol(&ws, out, button, style->inc_symbol, - NK_BUTTON_REPEATER, &style->inc_button, in, font)) - offset = offset + scroll_step; - - scroll.y = scroll.y + button.h; - scroll.h = scroll_h; - } - - /* calculate scrollbar constants */ - scroll_step = NK_MIN(step, scroll.h); - scroll_offset = NK_CLAMP(0, offset, target - scroll.h); - scroll_ratio = scroll.h / target; - scroll_off = scroll_offset / target; - - /* calculate scrollbar cursor bounds */ - cursor.h = NK_MAX((scroll_ratio * scroll.h) - (2*style->border + 2*style->padding.y), 0); - cursor.y = scroll.y + (scroll_off * scroll.h) + style->border + style->padding.y; - cursor.w = scroll.w - (2 * style->border + 2 * style->padding.x); - cursor.x = scroll.x + style->border + style->padding.x; - - /* calculate empty space around cursor */ - empty_north.x = scroll.x; - empty_north.y = scroll.y; - empty_north.w = scroll.w; - empty_north.h = NK_MAX(cursor.y - scroll.y, 0); - - empty_south.x = scroll.x; - empty_south.y = cursor.y + cursor.h; - empty_south.w = scroll.w; - empty_south.h = NK_MAX((scroll.y + scroll.h) - (cursor.y + cursor.h), 0); - - /* update scrollbar */ - scroll_offset = nk_scrollbar_behavior(state, in, has_scrolling, &scroll, &cursor, - &empty_north, &empty_south, scroll_offset, target, scroll_step, NK_VERTICAL); - scroll_off = scroll_offset / target; - cursor.y = scroll.y + (scroll_off * scroll.h) + style->border_cursor + style->padding.y; - - /* draw scrollbar */ - if (style->draw_begin) style->draw_begin(out, style->userdata); - nk_draw_scrollbar(out, *state, style, &scroll, &cursor); - if (style->draw_end) style->draw_end(out, style->userdata); - return scroll_offset; -} -NK_LIB float -nk_do_scrollbarh(nk_flags *state, - struct nk_command_buffer *out, struct nk_rect scroll, int has_scrolling, - float offset, float target, float step, float button_pixel_inc, - const struct nk_style_scrollbar *style, struct nk_input *in, - const struct nk_user_font *font) -{ - struct nk_rect cursor; - struct nk_rect empty_west; - struct nk_rect empty_east; - - float scroll_step; - float scroll_offset; - float scroll_off; - float scroll_ratio; - - NK_ASSERT(out); - NK_ASSERT(style); - if (!out || !style) return 0; - - /* scrollbar background */ - scroll.h = NK_MAX(scroll.h, 1); - scroll.w = NK_MAX(scroll.w, 2 * scroll.h); - if (target <= scroll.w) return 0; - - /* optional scrollbar buttons */ - if (style->show_buttons) { - nk_flags ws; - float scroll_w; - struct nk_rect button; - button.y = scroll.y; - button.w = scroll.h; - button.h = scroll.h; - - scroll_w = scroll.w - 2 * button.w; - scroll_step = NK_MIN(step, button_pixel_inc); - - /* decrement button */ - button.x = scroll.x; - if (nk_do_button_symbol(&ws, out, button, style->dec_symbol, - NK_BUTTON_REPEATER, &style->dec_button, in, font)) - offset = offset - scroll_step; - - /* increment button */ - button.x = scroll.x + scroll.w - button.w; - if (nk_do_button_symbol(&ws, out, button, style->inc_symbol, - NK_BUTTON_REPEATER, &style->inc_button, in, font)) - offset = offset + scroll_step; - - scroll.x = scroll.x + button.w; - scroll.w = scroll_w; - } - - /* calculate scrollbar constants */ - scroll_step = NK_MIN(step, scroll.w); - scroll_offset = NK_CLAMP(0, offset, target - scroll.w); - scroll_ratio = scroll.w / target; - scroll_off = scroll_offset / target; - - /* calculate cursor bounds */ - cursor.w = (scroll_ratio * scroll.w) - (2*style->border + 2*style->padding.x); - cursor.x = scroll.x + (scroll_off * scroll.w) + style->border + style->padding.x; - cursor.h = scroll.h - (2 * style->border + 2 * style->padding.y); - cursor.y = scroll.y + style->border + style->padding.y; - - /* calculate empty space around cursor */ - empty_west.x = scroll.x; - empty_west.y = scroll.y; - empty_west.w = cursor.x - scroll.x; - empty_west.h = scroll.h; - - empty_east.x = cursor.x + cursor.w; - empty_east.y = scroll.y; - empty_east.w = (scroll.x + scroll.w) - (cursor.x + cursor.w); - empty_east.h = scroll.h; - - /* update scrollbar */ - scroll_offset = nk_scrollbar_behavior(state, in, has_scrolling, &scroll, &cursor, - &empty_west, &empty_east, scroll_offset, target, scroll_step, NK_HORIZONTAL); - scroll_off = scroll_offset / target; - cursor.x = scroll.x + (scroll_off * scroll.w); - - /* draw scrollbar */ - if (style->draw_begin) style->draw_begin(out, style->userdata); - nk_draw_scrollbar(out, *state, style, &scroll, &cursor); - if (style->draw_end) style->draw_end(out, style->userdata); - return scroll_offset; -} - - - - - -/* =============================================================== - * - * TEXT EDITOR - * - * ===============================================================*/ -/* stb_textedit.h - v1.8 - public domain - Sean Barrett */ -struct nk_text_find { - float x,y; /* position of n'th character */ - float height; /* height of line */ - int first_char, length; /* first char of row, and length */ - int prev_first; /*_ first char of previous row */ -}; - -struct nk_text_edit_row { - float x0,x1; - /* starting x location, end x location (allows for align=right, etc) */ - float baseline_y_delta; - /* position of baseline relative to previous row's baseline*/ - float ymin,ymax; - /* height of row above and below baseline */ - int num_chars; -}; - -/* forward declarations */ -NK_INTERN void nk_textedit_makeundo_delete(struct nk_text_edit*, int, int); -NK_INTERN void nk_textedit_makeundo_insert(struct nk_text_edit*, int, int); -NK_INTERN void nk_textedit_makeundo_replace(struct nk_text_edit*, int, int, int); -#define NK_TEXT_HAS_SELECTION(s) ((s)->select_start != (s)->select_end) - -NK_INTERN float -nk_textedit_get_width(const struct nk_text_edit *edit, int line_start, int char_id, - const struct nk_user_font *font) -{ - int len = 0; - nk_rune unicode = 0; - const char *str = nk_str_at_const(&edit->string, line_start + char_id, &unicode, &len); - return font->width(font->userdata, font->height, str, len); -} -NK_INTERN void -nk_textedit_layout_row(struct nk_text_edit_row *r, struct nk_text_edit *edit, - int line_start_id, float row_height, const struct nk_user_font *font) -{ - int l; - int glyphs = 0; - nk_rune unicode; - const char *remaining; - int len = nk_str_len_char(&edit->string); - const char *end = nk_str_get_const(&edit->string) + len; - const char *text = nk_str_at_const(&edit->string, line_start_id, &unicode, &l); - const struct nk_vec2 size = nk_text_calculate_text_bounds(font, - text, (int)(end - text), row_height, &remaining, 0, &glyphs, NK_STOP_ON_NEW_LINE); - - r->x0 = 0.0f; - r->x1 = size.x; - r->baseline_y_delta = size.y; - r->ymin = 0.0f; - r->ymax = size.y; - r->num_chars = glyphs; -} -NK_INTERN int -nk_textedit_locate_coord(struct nk_text_edit *edit, float x, float y, - const struct nk_user_font *font, float row_height) -{ - struct nk_text_edit_row r; - int n = edit->string.len; - float base_y = 0, prev_x; - int i=0, k; - - r.x0 = r.x1 = 0; - r.ymin = r.ymax = 0; - r.num_chars = 0; - - /* search rows to find one that straddles 'y' */ - while (i < n) { - nk_textedit_layout_row(&r, edit, i, row_height, font); - if (r.num_chars <= 0) - return n; - - if (i==0 && y < base_y + r.ymin) - return 0; - - if (y < base_y + r.ymax) - break; - - i += r.num_chars; - base_y += r.baseline_y_delta; - } - - /* below all text, return 'after' last character */ - if (i >= n) - return n; - - /* check if it's before the beginning of the line */ - if (x < r.x0) - return i; - - /* check if it's before the end of the line */ - if (x < r.x1) { - /* search characters in row for one that straddles 'x' */ - k = i; - prev_x = r.x0; - for (i=0; i < r.num_chars; ++i) { - float w = nk_textedit_get_width(edit, k, i, font); - if (x < prev_x+w) { - if (x < prev_x+w/2) - return k+i; - else return k+i+1; - } - prev_x += w; - } - /* shouldn't happen, but if it does, fall through to end-of-line case */ - } - - /* if the last character is a newline, return that. - * otherwise return 'after' the last character */ - if (nk_str_rune_at(&edit->string, i+r.num_chars-1) == '\n') - return i+r.num_chars-1; - else return i+r.num_chars; -} -NK_LIB void -nk_textedit_click(struct nk_text_edit *state, float x, float y, - const struct nk_user_font *font, float row_height) -{ - /* API click: on mouse down, move the cursor to the clicked location, - * and reset the selection */ - state->cursor = nk_textedit_locate_coord(state, x, y, font, row_height); - state->select_start = state->cursor; - state->select_end = state->cursor; - state->has_preferred_x = 0; -} -NK_LIB void -nk_textedit_drag(struct nk_text_edit *state, float x, float y, - const struct nk_user_font *font, float row_height) -{ - /* API drag: on mouse drag, move the cursor and selection endpoint - * to the clicked location */ - int p = nk_textedit_locate_coord(state, x, y, font, row_height); - if (state->select_start == state->select_end) - state->select_start = state->cursor; - state->cursor = state->select_end = p; -} -NK_INTERN void -nk_textedit_find_charpos(struct nk_text_find *find, struct nk_text_edit *state, - int n, int single_line, const struct nk_user_font *font, float row_height) -{ - /* find the x/y location of a character, and remember info about the previous - * row in case we get a move-up event (for page up, we'll have to rescan) */ - struct nk_text_edit_row r; - int prev_start = 0; - int z = state->string.len; - int i=0, first; - - nk_zero_struct(r); - if (n == z) { - /* if it's at the end, then find the last line -- simpler than trying to - explicitly handle this case in the regular code */ - nk_textedit_layout_row(&r, state, 0, row_height, font); - if (single_line) { - find->first_char = 0; - find->length = z; - } else { - while (i < z) { - prev_start = i; - i += r.num_chars; - nk_textedit_layout_row(&r, state, i, row_height, font); - } - - find->first_char = i; - find->length = r.num_chars; - } - find->x = r.x1; - find->y = r.ymin; - find->height = r.ymax - r.ymin; - find->prev_first = prev_start; - return; - } - - /* search rows to find the one that straddles character n */ - find->y = 0; - - for(;;) { - nk_textedit_layout_row(&r, state, i, row_height, font); - if (n < i + r.num_chars) break; - prev_start = i; - i += r.num_chars; - find->y += r.baseline_y_delta; - } - - find->first_char = first = i; - find->length = r.num_chars; - find->height = r.ymax - r.ymin; - find->prev_first = prev_start; - - /* now scan to find xpos */ - find->x = r.x0; - for (i=0; first+i < n; ++i) - find->x += nk_textedit_get_width(state, first, i, font); -} -NK_INTERN void -nk_textedit_clamp(struct nk_text_edit *state) -{ - /* make the selection/cursor state valid if client altered the string */ - int n = state->string.len; - if (NK_TEXT_HAS_SELECTION(state)) { - if (state->select_start > n) state->select_start = n; - if (state->select_end > n) state->select_end = n; - /* if clamping forced them to be equal, move the cursor to match */ - if (state->select_start == state->select_end) - state->cursor = state->select_start; - } - if (state->cursor > n) state->cursor = n; -} -NK_API void -nk_textedit_delete(struct nk_text_edit *state, int where, int len) -{ - /* delete characters while updating undo */ - nk_textedit_makeundo_delete(state, where, len); - nk_str_delete_runes(&state->string, where, len); - state->has_preferred_x = 0; -} -NK_API void -nk_textedit_delete_selection(struct nk_text_edit *state) -{ - /* delete the section */ - nk_textedit_clamp(state); - if (NK_TEXT_HAS_SELECTION(state)) { - if (state->select_start < state->select_end) { - nk_textedit_delete(state, state->select_start, - state->select_end - state->select_start); - state->select_end = state->cursor = state->select_start; - } else { - nk_textedit_delete(state, state->select_end, - state->select_start - state->select_end); - state->select_start = state->cursor = state->select_end; - } - state->has_preferred_x = 0; - } -} -NK_INTERN void -nk_textedit_sortselection(struct nk_text_edit *state) -{ - /* canonicalize the selection so start <= end */ - if (state->select_end < state->select_start) { - int temp = state->select_end; - state->select_end = state->select_start; - state->select_start = temp; - } -} -NK_INTERN void -nk_textedit_move_to_first(struct nk_text_edit *state) -{ - /* move cursor to first character of selection */ - if (NK_TEXT_HAS_SELECTION(state)) { - nk_textedit_sortselection(state); - state->cursor = state->select_start; - state->select_end = state->select_start; - state->has_preferred_x = 0; - } -} -NK_INTERN void -nk_textedit_move_to_last(struct nk_text_edit *state) -{ - /* move cursor to last character of selection */ - if (NK_TEXT_HAS_SELECTION(state)) { - nk_textedit_sortselection(state); - nk_textedit_clamp(state); - state->cursor = state->select_end; - state->select_start = state->select_end; - state->has_preferred_x = 0; - } -} -NK_INTERN int -nk_is_word_boundary( struct nk_text_edit *state, int idx) -{ - int len; - nk_rune c; - if (idx <= 0) return 1; - if (!nk_str_at_rune(&state->string, idx, &c, &len)) return 1; - return (c == ' ' || c == '\t' ||c == 0x3000 || c == ',' || c == ';' || - c == '(' || c == ')' || c == '{' || c == '}' || c == '[' || c == ']' || - c == '|'); -} -NK_INTERN int -nk_textedit_move_to_word_previous(struct nk_text_edit *state) -{ - int c = state->cursor - 1; - while( c >= 0 && !nk_is_word_boundary(state, c)) - --c; - - if( c < 0 ) - c = 0; - - return c; -} -NK_INTERN int -nk_textedit_move_to_word_next(struct nk_text_edit *state) -{ - const int len = state->string.len; - int c = state->cursor+1; - while( c < len && !nk_is_word_boundary(state, c)) - ++c; - - if( c > len ) - c = len; - - return c; -} -NK_INTERN void -nk_textedit_prep_selection_at_cursor(struct nk_text_edit *state) -{ - /* update selection and cursor to match each other */ - if (!NK_TEXT_HAS_SELECTION(state)) - state->select_start = state->select_end = state->cursor; - else state->cursor = state->select_end; -} -NK_API nk_bool -nk_textedit_cut(struct nk_text_edit *state) -{ - /* API cut: delete selection */ - if (state->mode == NK_TEXT_EDIT_MODE_VIEW) - return 0; - if (NK_TEXT_HAS_SELECTION(state)) { - nk_textedit_delete_selection(state); /* implicitly clamps */ - state->has_preferred_x = 0; - return 1; - } - return 0; -} -NK_API nk_bool -nk_textedit_paste(struct nk_text_edit *state, char const *ctext, int len) -{ - /* API paste: replace existing selection with passed-in text */ - int glyphs; - const char *text = (const char *) ctext; - if (state->mode == NK_TEXT_EDIT_MODE_VIEW) return 0; - - /* if there's a selection, the paste should delete it */ - nk_textedit_clamp(state); - nk_textedit_delete_selection(state); - - /* try to insert the characters */ - glyphs = nk_utf_len(ctext, len); - if (nk_str_insert_text_char(&state->string, state->cursor, text, len)) { - nk_textedit_makeundo_insert(state, state->cursor, glyphs); - state->cursor += len; - state->has_preferred_x = 0; - return 1; - } - /* remove the undo since we didn't actually insert the characters */ - if (state->undo.undo_point) - --state->undo.undo_point; - return 0; -} -NK_API void -nk_textedit_text(struct nk_text_edit *state, const char *text, int total_len) -{ - nk_rune unicode; - int glyph_len; - int text_len = 0; - - NK_ASSERT(state); - NK_ASSERT(text); - if (!text || !total_len || state->mode == NK_TEXT_EDIT_MODE_VIEW) return; - - glyph_len = nk_utf_decode(text, &unicode, total_len); - while ((text_len < total_len) && glyph_len) - { - /* don't insert a backward delete, just process the event */ - if (unicode == 127) goto next; - /* can't add newline in single-line mode */ - if (unicode == '\n' && state->single_line) goto next; - /* filter incoming text */ - if (state->filter && !state->filter(state, unicode)) goto next; - - if (!NK_TEXT_HAS_SELECTION(state) && - state->cursor < state->string.len) - { - if (state->mode == NK_TEXT_EDIT_MODE_REPLACE) { - nk_textedit_makeundo_replace(state, state->cursor, 1, 1); - nk_str_delete_runes(&state->string, state->cursor, 1); - } - if (nk_str_insert_text_utf8(&state->string, state->cursor, - text+text_len, 1)) - { - ++state->cursor; - state->has_preferred_x = 0; - } - } else { - nk_textedit_delete_selection(state); /* implicitly clamps */ - if (nk_str_insert_text_utf8(&state->string, state->cursor, - text+text_len, 1)) - { - nk_textedit_makeundo_insert(state, state->cursor, 1); - state->cursor = NK_MIN(state->cursor + 1, state->string.len); - state->has_preferred_x = 0; - } - } - next: - text_len += glyph_len; - glyph_len = nk_utf_decode(text + text_len, &unicode, total_len-text_len); - } -} -NK_LIB void -nk_textedit_key(struct nk_text_edit *state, enum nk_keys key, int shift_mod, - const struct nk_user_font *font, float row_height) -{ -retry: - switch (key) - { - case NK_KEY_NONE: - case NK_KEY_CTRL: - case NK_KEY_ENTER: - case NK_KEY_SHIFT: - case NK_KEY_TAB: - case NK_KEY_COPY: - case NK_KEY_CUT: - case NK_KEY_PASTE: - case NK_KEY_MAX: - default: break; - case NK_KEY_TEXT_UNDO: - nk_textedit_undo(state); - state->has_preferred_x = 0; - break; - - case NK_KEY_TEXT_REDO: - nk_textedit_redo(state); - state->has_preferred_x = 0; - break; - - case NK_KEY_TEXT_SELECT_ALL: - nk_textedit_select_all(state); - state->has_preferred_x = 0; - break; - - case NK_KEY_TEXT_INSERT_MODE: - if (state->mode == NK_TEXT_EDIT_MODE_VIEW) - state->mode = NK_TEXT_EDIT_MODE_INSERT; - break; - case NK_KEY_TEXT_REPLACE_MODE: - if (state->mode == NK_TEXT_EDIT_MODE_VIEW) - state->mode = NK_TEXT_EDIT_MODE_REPLACE; - break; - case NK_KEY_TEXT_RESET_MODE: - if (state->mode == NK_TEXT_EDIT_MODE_INSERT || - state->mode == NK_TEXT_EDIT_MODE_REPLACE) - state->mode = NK_TEXT_EDIT_MODE_VIEW; - break; - - case NK_KEY_LEFT: - if (shift_mod) { - nk_textedit_clamp(state); - nk_textedit_prep_selection_at_cursor(state); - /* move selection left */ - if (state->select_end > 0) - --state->select_end; - state->cursor = state->select_end; - state->has_preferred_x = 0; - } else { - /* if currently there's a selection, - * move cursor to start of selection */ - if (NK_TEXT_HAS_SELECTION(state)) - nk_textedit_move_to_first(state); - else if (state->cursor > 0) - --state->cursor; - state->has_preferred_x = 0; - } break; - - case NK_KEY_RIGHT: - if (shift_mod) { - nk_textedit_prep_selection_at_cursor(state); - /* move selection right */ - ++state->select_end; - nk_textedit_clamp(state); - state->cursor = state->select_end; - state->has_preferred_x = 0; - } else { - /* if currently there's a selection, - * move cursor to end of selection */ - if (NK_TEXT_HAS_SELECTION(state)) - nk_textedit_move_to_last(state); - else ++state->cursor; - nk_textedit_clamp(state); - state->has_preferred_x = 0; - } break; - - case NK_KEY_TEXT_WORD_LEFT: - if (shift_mod) { - if( !NK_TEXT_HAS_SELECTION( state ) ) - nk_textedit_prep_selection_at_cursor(state); - state->cursor = nk_textedit_move_to_word_previous(state); - state->select_end = state->cursor; - nk_textedit_clamp(state ); - } else { - if (NK_TEXT_HAS_SELECTION(state)) - nk_textedit_move_to_first(state); - else { - state->cursor = nk_textedit_move_to_word_previous(state); - nk_textedit_clamp(state ); - } - } break; - - case NK_KEY_TEXT_WORD_RIGHT: - if (shift_mod) { - if( !NK_TEXT_HAS_SELECTION( state ) ) - nk_textedit_prep_selection_at_cursor(state); - state->cursor = nk_textedit_move_to_word_next(state); - state->select_end = state->cursor; - nk_textedit_clamp(state); - } else { - if (NK_TEXT_HAS_SELECTION(state)) - nk_textedit_move_to_last(state); - else { - state->cursor = nk_textedit_move_to_word_next(state); - nk_textedit_clamp(state ); - } - } break; - - case NK_KEY_DOWN: { - struct nk_text_find find; - struct nk_text_edit_row row; - int i, sel = shift_mod; - - if (state->single_line) { - /* on windows, up&down in single-line behave like left&right */ - key = NK_KEY_RIGHT; - goto retry; - } - - if (sel) - nk_textedit_prep_selection_at_cursor(state); - else if (NK_TEXT_HAS_SELECTION(state)) - nk_textedit_move_to_last(state); - - /* compute current position of cursor point */ - nk_textedit_clamp(state); - nk_textedit_find_charpos(&find, state, state->cursor, state->single_line, - font, row_height); - - /* now find character position down a row */ - if (find.length) - { - float x; - float goal_x = state->has_preferred_x ? state->preferred_x : find.x; - int start = find.first_char + find.length; - - state->cursor = start; - nk_textedit_layout_row(&row, state, state->cursor, row_height, font); - x = row.x0; - - for (i=0; i < row.num_chars && x < row.x1; ++i) { - float dx = nk_textedit_get_width(state, start, i, font); - x += dx; - if (x > goal_x) - break; - ++state->cursor; - } - nk_textedit_clamp(state); - - state->has_preferred_x = 1; - state->preferred_x = goal_x; - if (sel) - state->select_end = state->cursor; - } - } break; - - case NK_KEY_UP: { - struct nk_text_find find; - struct nk_text_edit_row row; - int i, sel = shift_mod; - - if (state->single_line) { - /* on windows, up&down become left&right */ - key = NK_KEY_LEFT; - goto retry; - } - - if (sel) - nk_textedit_prep_selection_at_cursor(state); - else if (NK_TEXT_HAS_SELECTION(state)) - nk_textedit_move_to_first(state); - - /* compute current position of cursor point */ - nk_textedit_clamp(state); - nk_textedit_find_charpos(&find, state, state->cursor, state->single_line, - font, row_height); - - /* can only go up if there's a previous row */ - if (find.prev_first != find.first_char) { - /* now find character position up a row */ - float x; - float goal_x = state->has_preferred_x ? state->preferred_x : find.x; - - state->cursor = find.prev_first; - nk_textedit_layout_row(&row, state, state->cursor, row_height, font); - x = row.x0; - - for (i=0; i < row.num_chars && x < row.x1; ++i) { - float dx = nk_textedit_get_width(state, find.prev_first, i, font); - x += dx; - if (x > goal_x) - break; - ++state->cursor; - } - nk_textedit_clamp(state); - - state->has_preferred_x = 1; - state->preferred_x = goal_x; - if (sel) state->select_end = state->cursor; - } - } break; - - case NK_KEY_DEL: - if (state->mode == NK_TEXT_EDIT_MODE_VIEW) - break; - if (NK_TEXT_HAS_SELECTION(state)) - nk_textedit_delete_selection(state); - else { - int n = state->string.len; - if (state->cursor < n) - nk_textedit_delete(state, state->cursor, 1); - } - state->has_preferred_x = 0; - break; - - case NK_KEY_BACKSPACE: - if (state->mode == NK_TEXT_EDIT_MODE_VIEW) - break; - if (NK_TEXT_HAS_SELECTION(state)) - nk_textedit_delete_selection(state); - else { - nk_textedit_clamp(state); - if (state->cursor > 0) { - nk_textedit_delete(state, state->cursor-1, 1); - --state->cursor; - } - } - state->has_preferred_x = 0; - break; - - case NK_KEY_TEXT_START: - if (shift_mod) { - nk_textedit_prep_selection_at_cursor(state); - state->cursor = state->select_end = 0; - state->has_preferred_x = 0; - } else { - state->cursor = state->select_start = state->select_end = 0; - state->has_preferred_x = 0; - } - break; - - case NK_KEY_TEXT_END: - if (shift_mod) { - nk_textedit_prep_selection_at_cursor(state); - state->cursor = state->select_end = state->string.len; - state->has_preferred_x = 0; - } else { - state->cursor = state->string.len; - state->select_start = state->select_end = 0; - state->has_preferred_x = 0; - } - break; - - case NK_KEY_TEXT_LINE_START: { - if (shift_mod) { - struct nk_text_find find; - nk_textedit_clamp(state); - nk_textedit_prep_selection_at_cursor(state); - if (state->string.len && state->cursor == state->string.len) - --state->cursor; - nk_textedit_find_charpos(&find, state,state->cursor, state->single_line, - font, row_height); - state->cursor = state->select_end = find.first_char; - state->has_preferred_x = 0; - } else { - struct nk_text_find find; - if (state->string.len && state->cursor == state->string.len) - --state->cursor; - nk_textedit_clamp(state); - nk_textedit_move_to_first(state); - nk_textedit_find_charpos(&find, state, state->cursor, state->single_line, - font, row_height); - state->cursor = find.first_char; - state->has_preferred_x = 0; - } - } break; - - case NK_KEY_TEXT_LINE_END: { - if (shift_mod) { - struct nk_text_find find; - nk_textedit_clamp(state); - nk_textedit_prep_selection_at_cursor(state); - nk_textedit_find_charpos(&find, state, state->cursor, state->single_line, - font, row_height); - state->has_preferred_x = 0; - state->cursor = find.first_char + find.length; - if (find.length > 0 && nk_str_rune_at(&state->string, state->cursor-1) == '\n') - --state->cursor; - state->select_end = state->cursor; - } else { - struct nk_text_find find; - nk_textedit_clamp(state); - nk_textedit_move_to_first(state); - nk_textedit_find_charpos(&find, state, state->cursor, state->single_line, - font, row_height); - - state->has_preferred_x = 0; - state->cursor = find.first_char + find.length; - if (find.length > 0 && nk_str_rune_at(&state->string, state->cursor-1) == '\n') - --state->cursor; - }} break; - } -} -NK_INTERN void -nk_textedit_flush_redo(struct nk_text_undo_state *state) -{ - state->redo_point = NK_TEXTEDIT_UNDOSTATECOUNT; - state->redo_char_point = NK_TEXTEDIT_UNDOCHARCOUNT; -} -NK_INTERN void -nk_textedit_discard_undo(struct nk_text_undo_state *state) -{ - /* discard the oldest entry in the undo list */ - if (state->undo_point > 0) { - /* if the 0th undo state has characters, clean those up */ - if (state->undo_rec[0].char_storage >= 0) { - int n = state->undo_rec[0].insert_length, i; - /* delete n characters from all other records */ - state->undo_char_point = (short)(state->undo_char_point - n); - NK_MEMCPY(state->undo_char, state->undo_char + n, - (nk_size)state->undo_char_point*sizeof(nk_rune)); - for (i=0; i < state->undo_point; ++i) { - if (state->undo_rec[i].char_storage >= 0) - state->undo_rec[i].char_storage = (short) - (state->undo_rec[i].char_storage - n); - } - } - --state->undo_point; - NK_MEMCPY(state->undo_rec, state->undo_rec+1, - (nk_size)((nk_size)state->undo_point * sizeof(state->undo_rec[0]))); - } -} -NK_INTERN void -nk_textedit_discard_redo(struct nk_text_undo_state *state) -{ -/* discard the oldest entry in the redo list--it's bad if this - ever happens, but because undo & redo have to store the actual - characters in different cases, the redo character buffer can - fill up even though the undo buffer didn't */ - nk_size num; - int k = NK_TEXTEDIT_UNDOSTATECOUNT-1; - if (state->redo_point <= k) { - /* if the k'th undo state has characters, clean those up */ - if (state->undo_rec[k].char_storage >= 0) { - int n = state->undo_rec[k].insert_length, i; - /* delete n characters from all other records */ - state->redo_char_point = (short)(state->redo_char_point + n); - num = (nk_size)(NK_TEXTEDIT_UNDOCHARCOUNT - state->redo_char_point); - NK_MEMCPY(state->undo_char + state->redo_char_point, - state->undo_char + state->redo_char_point-n, num * sizeof(char)); - for (i = state->redo_point; i < k; ++i) { - if (state->undo_rec[i].char_storage >= 0) { - state->undo_rec[i].char_storage = (short) - (state->undo_rec[i].char_storage + n); - } - } - } - ++state->redo_point; - num = (nk_size)(NK_TEXTEDIT_UNDOSTATECOUNT - state->redo_point); - if (num) NK_MEMCPY(state->undo_rec + state->redo_point-1, - state->undo_rec + state->redo_point, num * sizeof(state->undo_rec[0])); - } -} -NK_INTERN struct nk_text_undo_record* -nk_textedit_create_undo_record(struct nk_text_undo_state *state, int numchars) -{ - /* any time we create a new undo record, we discard redo*/ - nk_textedit_flush_redo(state); - - /* if we have no free records, we have to make room, - * by sliding the existing records down */ - if (state->undo_point == NK_TEXTEDIT_UNDOSTATECOUNT) - nk_textedit_discard_undo(state); - - /* if the characters to store won't possibly fit in the buffer, - * we can't undo */ - if (numchars > NK_TEXTEDIT_UNDOCHARCOUNT) { - state->undo_point = 0; - state->undo_char_point = 0; - return 0; - } - - /* if we don't have enough free characters in the buffer, - * we have to make room */ - while (state->undo_char_point + numchars > NK_TEXTEDIT_UNDOCHARCOUNT) - nk_textedit_discard_undo(state); - return &state->undo_rec[state->undo_point++]; -} -NK_INTERN nk_rune* -nk_textedit_createundo(struct nk_text_undo_state *state, int pos, - int insert_len, int delete_len) -{ - struct nk_text_undo_record *r = nk_textedit_create_undo_record(state, insert_len); - if (r == 0) - return 0; - - r->where = pos; - r->insert_length = (short) insert_len; - r->delete_length = (short) delete_len; - - if (insert_len == 0) { - r->char_storage = -1; - return 0; - } else { - r->char_storage = state->undo_char_point; - state->undo_char_point = (short)(state->undo_char_point + insert_len); - return &state->undo_char[r->char_storage]; - } -} -NK_API void -nk_textedit_undo(struct nk_text_edit *state) -{ - struct nk_text_undo_state *s = &state->undo; - struct nk_text_undo_record u, *r; - if (s->undo_point == 0) - return; - - /* we need to do two things: apply the undo record, and create a redo record */ - u = s->undo_rec[s->undo_point-1]; - r = &s->undo_rec[s->redo_point-1]; - r->char_storage = -1; - - r->insert_length = u.delete_length; - r->delete_length = u.insert_length; - r->where = u.where; - - if (u.delete_length) - { - /* if the undo record says to delete characters, then the redo record will - need to re-insert the characters that get deleted, so we need to store - them. - there are three cases: - - there's enough room to store the characters - - characters stored for *redoing* don't leave room for redo - - characters stored for *undoing* don't leave room for redo - if the last is true, we have to bail */ - if (s->undo_char_point + u.delete_length >= NK_TEXTEDIT_UNDOCHARCOUNT) { - /* the undo records take up too much character space; there's no space - * to store the redo characters */ - r->insert_length = 0; - } else { - int i; - /* there's definitely room to store the characters eventually */ - while (s->undo_char_point + u.delete_length > s->redo_char_point) { - /* there's currently not enough room, so discard a redo record */ - nk_textedit_discard_redo(s); - /* should never happen: */ - if (s->redo_point == NK_TEXTEDIT_UNDOSTATECOUNT) - return; - } - - r = &s->undo_rec[s->redo_point-1]; - r->char_storage = (short)(s->redo_char_point - u.delete_length); - s->redo_char_point = (short)(s->redo_char_point - u.delete_length); - - /* now save the characters */ - for (i=0; i < u.delete_length; ++i) - s->undo_char[r->char_storage + i] = - nk_str_rune_at(&state->string, u.where + i); - } - /* now we can carry out the deletion */ - nk_str_delete_runes(&state->string, u.where, u.delete_length); - } - - /* check type of recorded action: */ - if (u.insert_length) { - /* easy case: was a deletion, so we need to insert n characters */ - nk_str_insert_text_runes(&state->string, u.where, - &s->undo_char[u.char_storage], u.insert_length); - s->undo_char_point = (short)(s->undo_char_point - u.insert_length); - } - state->cursor = (short)(u.where + u.insert_length); - - s->undo_point--; - s->redo_point--; -} -NK_API void -nk_textedit_redo(struct nk_text_edit *state) -{ - struct nk_text_undo_state *s = &state->undo; - struct nk_text_undo_record *u, r; - if (s->redo_point == NK_TEXTEDIT_UNDOSTATECOUNT) - return; - - /* we need to do two things: apply the redo record, and create an undo record */ - u = &s->undo_rec[s->undo_point]; - r = s->undo_rec[s->redo_point]; - - /* we KNOW there must be room for the undo record, because the redo record - was derived from an undo record */ - u->delete_length = r.insert_length; - u->insert_length = r.delete_length; - u->where = r.where; - u->char_storage = -1; - - if (r.delete_length) { - /* the redo record requires us to delete characters, so the undo record - needs to store the characters */ - if (s->undo_char_point + u->insert_length > s->redo_char_point) { - u->insert_length = 0; - u->delete_length = 0; - } else { - int i; - u->char_storage = s->undo_char_point; - s->undo_char_point = (short)(s->undo_char_point + u->insert_length); - - /* now save the characters */ - for (i=0; i < u->insert_length; ++i) { - s->undo_char[u->char_storage + i] = - nk_str_rune_at(&state->string, u->where + i); - } - } - nk_str_delete_runes(&state->string, r.where, r.delete_length); - } - - if (r.insert_length) { - /* easy case: need to insert n characters */ - nk_str_insert_text_runes(&state->string, r.where, - &s->undo_char[r.char_storage], r.insert_length); - } - state->cursor = r.where + r.insert_length; - - s->undo_point++; - s->redo_point++; -} -NK_INTERN void -nk_textedit_makeundo_insert(struct nk_text_edit *state, int where, int length) -{ - nk_textedit_createundo(&state->undo, where, 0, length); -} -NK_INTERN void -nk_textedit_makeundo_delete(struct nk_text_edit *state, int where, int length) -{ - int i; - nk_rune *p = nk_textedit_createundo(&state->undo, where, length, 0); - if (p) { - for (i=0; i < length; ++i) - p[i] = nk_str_rune_at(&state->string, where+i); - } -} -NK_INTERN void -nk_textedit_makeundo_replace(struct nk_text_edit *state, int where, - int old_length, int new_length) -{ - int i; - nk_rune *p = nk_textedit_createundo(&state->undo, where, old_length, new_length); - if (p) { - for (i=0; i < old_length; ++i) - p[i] = nk_str_rune_at(&state->string, where+i); - } -} -NK_LIB void -nk_textedit_clear_state(struct nk_text_edit *state, enum nk_text_edit_type type, - nk_plugin_filter filter) -{ - /* reset the state to default */ - state->undo.undo_point = 0; - state->undo.undo_char_point = 0; - state->undo.redo_point = NK_TEXTEDIT_UNDOSTATECOUNT; - state->undo.redo_char_point = NK_TEXTEDIT_UNDOCHARCOUNT; - state->select_end = state->select_start = 0; - state->cursor = 0; - state->has_preferred_x = 0; - state->preferred_x = 0; - state->cursor_at_end_of_line = 0; - state->initialized = 1; - state->single_line = (unsigned char)(type == NK_TEXT_EDIT_SINGLE_LINE); - state->mode = NK_TEXT_EDIT_MODE_VIEW; - state->filter = filter; - state->scrollbar = nk_vec2(0,0); -} -NK_API void -nk_textedit_init_fixed(struct nk_text_edit *state, void *memory, nk_size size) -{ - NK_ASSERT(state); - NK_ASSERT(memory); - if (!state || !memory || !size) return; - NK_MEMSET(state, 0, sizeof(struct nk_text_edit)); - nk_textedit_clear_state(state, NK_TEXT_EDIT_SINGLE_LINE, 0); - nk_str_init_fixed(&state->string, memory, size); -} -NK_API void -nk_textedit_init(struct nk_text_edit *state, struct nk_allocator *alloc, nk_size size) -{ - NK_ASSERT(state); - NK_ASSERT(alloc); - if (!state || !alloc) return; - NK_MEMSET(state, 0, sizeof(struct nk_text_edit)); - nk_textedit_clear_state(state, NK_TEXT_EDIT_SINGLE_LINE, 0); - nk_str_init(&state->string, alloc, size); -} -#ifdef NK_INCLUDE_DEFAULT_ALLOCATOR -NK_API void -nk_textedit_init_default(struct nk_text_edit *state) -{ - NK_ASSERT(state); - if (!state) return; - NK_MEMSET(state, 0, sizeof(struct nk_text_edit)); - nk_textedit_clear_state(state, NK_TEXT_EDIT_SINGLE_LINE, 0); - nk_str_init_default(&state->string); -} -#endif -NK_API void -nk_textedit_select_all(struct nk_text_edit *state) -{ - NK_ASSERT(state); - state->select_start = 0; - state->select_end = state->string.len; -} -NK_API void -nk_textedit_free(struct nk_text_edit *state) -{ - NK_ASSERT(state); - if (!state) return; - nk_str_free(&state->string); -} - - - - - -/* =============================================================== - * - * FILTER - * - * ===============================================================*/ -NK_API nk_bool -nk_filter_default(const struct nk_text_edit *box, nk_rune unicode) -{ - NK_UNUSED(unicode); - NK_UNUSED(box); - return nk_true; -} -NK_API nk_bool -nk_filter_ascii(const struct nk_text_edit *box, nk_rune unicode) -{ - NK_UNUSED(box); - if (unicode > 128) return nk_false; - else return nk_true; -} -NK_API nk_bool -nk_filter_float(const struct nk_text_edit *box, nk_rune unicode) -{ - NK_UNUSED(box); - if ((unicode < '0' || unicode > '9') && unicode != '.' && unicode != '-') - return nk_false; - else return nk_true; -} -NK_API nk_bool -nk_filter_decimal(const struct nk_text_edit *box, nk_rune unicode) -{ - NK_UNUSED(box); - if ((unicode < '0' || unicode > '9') && unicode != '-') - return nk_false; - else return nk_true; -} -NK_API nk_bool -nk_filter_hex(const struct nk_text_edit *box, nk_rune unicode) -{ - NK_UNUSED(box); - if ((unicode < '0' || unicode > '9') && - (unicode < 'a' || unicode > 'f') && - (unicode < 'A' || unicode > 'F')) - return nk_false; - else return nk_true; -} -NK_API nk_bool -nk_filter_oct(const struct nk_text_edit *box, nk_rune unicode) -{ - NK_UNUSED(box); - if (unicode < '0' || unicode > '7') - return nk_false; - else return nk_true; -} -NK_API nk_bool -nk_filter_binary(const struct nk_text_edit *box, nk_rune unicode) -{ - NK_UNUSED(box); - if (unicode != '0' && unicode != '1') - return nk_false; - else return nk_true; -} - -/* =============================================================== - * - * EDIT - * - * ===============================================================*/ -NK_LIB void -nk_edit_draw_text(struct nk_command_buffer *out, - const struct nk_style_edit *style, float pos_x, float pos_y, - float x_offset, const char *text, int byte_len, float row_height, - const struct nk_user_font *font, struct nk_color background, - struct nk_color foreground, nk_bool is_selected) -{ - NK_ASSERT(out); - NK_ASSERT(font); - NK_ASSERT(style); - if (!text || !byte_len || !out || !style) return; - - {int glyph_len = 0; - nk_rune unicode = 0; - int text_len = 0; - float line_width = 0; - float glyph_width; - const char *line = text; - float line_offset = 0; - int line_count = 0; - - struct nk_text txt; - txt.padding = nk_vec2(0,0); - txt.background = background; - txt.text = foreground; - - glyph_len = nk_utf_decode(text+text_len, &unicode, byte_len-text_len); - if (!glyph_len) return; - while ((text_len < byte_len) && glyph_len) - { - if (unicode == '\n') { - /* new line separator so draw previous line */ - struct nk_rect label; - label.y = pos_y + line_offset; - label.h = row_height; - label.w = line_width; - label.x = pos_x; - if (!line_count) - label.x += x_offset; - - if (is_selected) /* selection needs to draw different background color */ - nk_fill_rect(out, label, 0, background); - nk_widget_text(out, label, line, (int)((text + text_len) - line), - &txt, NK_TEXT_CENTERED, font); - - text_len++; - line_count++; - line_width = 0; - line = text + text_len; - line_offset += row_height; - glyph_len = nk_utf_decode(text + text_len, &unicode, (int)(byte_len-text_len)); - continue; - } - if (unicode == '\r') { - text_len++; - glyph_len = nk_utf_decode(text + text_len, &unicode, byte_len-text_len); - continue; - } - glyph_width = font->width(font->userdata, font->height, text+text_len, glyph_len); - line_width += (float)glyph_width; - text_len += glyph_len; - glyph_len = nk_utf_decode(text + text_len, &unicode, byte_len-text_len); - continue; - } - if (line_width > 0) { - /* draw last line */ - struct nk_rect label; - label.y = pos_y + line_offset; - label.h = row_height; - label.w = line_width; - label.x = pos_x; - if (!line_count) - label.x += x_offset; - - if (is_selected) - nk_fill_rect(out, label, 0, background); - nk_widget_text(out, label, line, (int)((text + text_len) - line), - &txt, NK_TEXT_LEFT, font); - }} -} -NK_LIB nk_flags -nk_do_edit(nk_flags *state, struct nk_command_buffer *out, - struct nk_rect bounds, nk_flags flags, nk_plugin_filter filter, - struct nk_text_edit *edit, const struct nk_style_edit *style, - struct nk_input *in, const struct nk_user_font *font) -{ - struct nk_rect area; - nk_flags ret = 0; - float row_height; - char prev_state = 0; - char is_hovered = 0; - char select_all = 0; - char cursor_follow = 0; - struct nk_rect old_clip; - struct nk_rect clip; - - NK_ASSERT(state); - NK_ASSERT(out); - NK_ASSERT(style); - if (!state || !out || !style) - return ret; - - /* visible text area calculation */ - area.x = bounds.x + style->padding.x + style->border; - area.y = bounds.y + style->padding.y + style->border; - area.w = bounds.w - (2.0f * style->padding.x + 2 * style->border); - area.h = bounds.h - (2.0f * style->padding.y + 2 * style->border); - if (flags & NK_EDIT_MULTILINE) - area.w = NK_MAX(0, area.w - style->scrollbar_size.x); - row_height = (flags & NK_EDIT_MULTILINE)? font->height + style->row_padding: area.h; - - /* calculate clipping rectangle */ - old_clip = out->clip; - nk_unify(&clip, &old_clip, area.x, area.y, area.x + area.w, area.y + area.h); - - /* update edit state */ - prev_state = (char)edit->active; - is_hovered = (char)nk_input_is_mouse_hovering_rect(in, bounds); - if (in && in->mouse.buttons[NK_BUTTON_LEFT].clicked && in->mouse.buttons[NK_BUTTON_LEFT].down) { - edit->active = NK_INBOX(in->mouse.pos.x, in->mouse.pos.y, - bounds.x, bounds.y, bounds.w, bounds.h); - } - - /* (de)activate text editor */ - if (!prev_state && edit->active) { - const enum nk_text_edit_type type = (flags & NK_EDIT_MULTILINE) ? - NK_TEXT_EDIT_MULTI_LINE: NK_TEXT_EDIT_SINGLE_LINE; - /* keep scroll position when re-activating edit widget */ - struct nk_vec2 oldscrollbar = edit->scrollbar; - nk_textedit_clear_state(edit, type, filter); - edit->scrollbar = oldscrollbar; - if (flags & NK_EDIT_AUTO_SELECT) - select_all = nk_true; - if (flags & NK_EDIT_GOTO_END_ON_ACTIVATE) { - edit->cursor = edit->string.len; - in = 0; - } - } else if (!edit->active) edit->mode = NK_TEXT_EDIT_MODE_VIEW; - if (flags & NK_EDIT_READ_ONLY) - edit->mode = NK_TEXT_EDIT_MODE_VIEW; - else if (flags & NK_EDIT_ALWAYS_INSERT_MODE) - edit->mode = NK_TEXT_EDIT_MODE_INSERT; - - ret = (edit->active) ? NK_EDIT_ACTIVE: NK_EDIT_INACTIVE; - if (prev_state != edit->active) - ret |= (edit->active) ? NK_EDIT_ACTIVATED: NK_EDIT_DEACTIVATED; - - /* handle user input */ - if (edit->active && in) - { - int shift_mod = in->keyboard.keys[NK_KEY_SHIFT].down; - const float mouse_x = (in->mouse.pos.x - area.x) + edit->scrollbar.x; - const float mouse_y = (in->mouse.pos.y - area.y) + edit->scrollbar.y; - - /* mouse click handler */ - is_hovered = (char)nk_input_is_mouse_hovering_rect(in, area); - if (select_all) { - nk_textedit_select_all(edit); - } else if (is_hovered && in->mouse.buttons[NK_BUTTON_LEFT].down && - in->mouse.buttons[NK_BUTTON_LEFT].clicked) { - nk_textedit_click(edit, mouse_x, mouse_y, font, row_height); - } else if (is_hovered && in->mouse.buttons[NK_BUTTON_LEFT].down && - (in->mouse.delta.x != 0.0f || in->mouse.delta.y != 0.0f)) { - nk_textedit_drag(edit, mouse_x, mouse_y, font, row_height); - cursor_follow = nk_true; - } else if (is_hovered && in->mouse.buttons[NK_BUTTON_RIGHT].clicked && - in->mouse.buttons[NK_BUTTON_RIGHT].down) { - nk_textedit_key(edit, NK_KEY_TEXT_WORD_LEFT, nk_false, font, row_height); - nk_textedit_key(edit, NK_KEY_TEXT_WORD_RIGHT, nk_true, font, row_height); - cursor_follow = nk_true; - } - - {int i; /* keyboard input */ - int old_mode = edit->mode; - for (i = 0; i < NK_KEY_MAX; ++i) { - if (i == NK_KEY_ENTER || i == NK_KEY_TAB) continue; /* special case */ - if (nk_input_is_key_pressed(in, (enum nk_keys)i)) { - nk_textedit_key(edit, (enum nk_keys)i, shift_mod, font, row_height); - cursor_follow = nk_true; - } - } - if (old_mode != edit->mode) { - in->keyboard.text_len = 0; - }} - - /* text input */ - edit->filter = filter; - if (in->keyboard.text_len) { - nk_textedit_text(edit, in->keyboard.text, in->keyboard.text_len); - cursor_follow = nk_true; - in->keyboard.text_len = 0; - } - - /* enter key handler */ - if (nk_input_is_key_pressed(in, NK_KEY_ENTER)) { - cursor_follow = nk_true; - if (flags & NK_EDIT_CTRL_ENTER_NEWLINE && shift_mod) - nk_textedit_text(edit, "\n", 1); - else if (flags & NK_EDIT_SIG_ENTER) - ret |= NK_EDIT_COMMITED; - else nk_textedit_text(edit, "\n", 1); - } - - /* cut & copy handler */ - {int copy= nk_input_is_key_pressed(in, NK_KEY_COPY); - int cut = nk_input_is_key_pressed(in, NK_KEY_CUT); - if ((copy || cut) && (flags & NK_EDIT_CLIPBOARD)) - { - int glyph_len; - nk_rune unicode; - const char *text; - int b = edit->select_start; - int e = edit->select_end; - - int begin = NK_MIN(b, e); - int end = NK_MAX(b, e); - text = nk_str_at_const(&edit->string, begin, &unicode, &glyph_len); - if (edit->clip.copy) - edit->clip.copy(edit->clip.userdata, text, end - begin); - if (cut && !(flags & NK_EDIT_READ_ONLY)){ - nk_textedit_cut(edit); - cursor_follow = nk_true; - } - }} - - /* paste handler */ - {int paste = nk_input_is_key_pressed(in, NK_KEY_PASTE); - if (paste && (flags & NK_EDIT_CLIPBOARD) && edit->clip.paste) { - edit->clip.paste(edit->clip.userdata, edit); - cursor_follow = nk_true; - }} - - /* tab handler */ - {int tab = nk_input_is_key_pressed(in, NK_KEY_TAB); - if (tab && (flags & NK_EDIT_ALLOW_TAB)) { - nk_textedit_text(edit, " ", 4); - cursor_follow = nk_true; - }} - } - - /* set widget state */ - if (edit->active) - *state = NK_WIDGET_STATE_ACTIVE; - else nk_widget_state_reset(state); - - if (is_hovered) - *state |= NK_WIDGET_STATE_HOVERED; - - /* DRAW EDIT */ - {const char *text = nk_str_get_const(&edit->string); - int len = nk_str_len_char(&edit->string); - - {/* select background colors/images */ - const struct nk_style_item *background; - if (*state & NK_WIDGET_STATE_ACTIVED) - background = &style->active; - else if (*state & NK_WIDGET_STATE_HOVER) - background = &style->hover; - else background = &style->normal; - - /* draw background frame */ - switch(background->type) { - case NK_STYLE_ITEM_IMAGE: - nk_draw_image(out, bounds, &background->data.image, nk_white); - break; - case NK_STYLE_ITEM_NINE_SLICE: - nk_draw_nine_slice(out, bounds, &background->data.slice, nk_white); - break; - case NK_STYLE_ITEM_COLOR: - nk_fill_rect(out, bounds, style->rounding, background->data.color); - nk_stroke_rect(out, bounds, style->rounding, style->border, style->border_color); - break; - }} - - - area.w = NK_MAX(0, area.w - style->cursor_size); - if (edit->active) - { - int total_lines = 1; - struct nk_vec2 text_size = nk_vec2(0,0); - - /* text pointer positions */ - const char *cursor_ptr = 0; - const char *select_begin_ptr = 0; - const char *select_end_ptr = 0; - - /* 2D pixel positions */ - struct nk_vec2 cursor_pos = nk_vec2(0,0); - struct nk_vec2 selection_offset_start = nk_vec2(0,0); - struct nk_vec2 selection_offset_end = nk_vec2(0,0); - - int selection_begin = NK_MIN(edit->select_start, edit->select_end); - int selection_end = NK_MAX(edit->select_start, edit->select_end); - - /* calculate total line count + total space + cursor/selection position */ - float line_width = 0.0f; - if (text && len) - { - /* utf8 encoding */ - float glyph_width; - int glyph_len = 0; - nk_rune unicode = 0; - int text_len = 0; - int glyphs = 0; - int row_begin = 0; - - glyph_len = nk_utf_decode(text, &unicode, len); - glyph_width = font->width(font->userdata, font->height, text, glyph_len); - line_width = 0; - - /* iterate all lines */ - while ((text_len < len) && glyph_len) - { - /* set cursor 2D position and line */ - if (!cursor_ptr && glyphs == edit->cursor) - { - int glyph_offset; - struct nk_vec2 out_offset; - struct nk_vec2 row_size; - const char *remaining; - - /* calculate 2d position */ - cursor_pos.y = (float)(total_lines-1) * row_height; - row_size = nk_text_calculate_text_bounds(font, text+row_begin, - text_len-row_begin, row_height, &remaining, - &out_offset, &glyph_offset, NK_STOP_ON_NEW_LINE); - cursor_pos.x = row_size.x; - cursor_ptr = text + text_len; - } - - /* set start selection 2D position and line */ - if (!select_begin_ptr && edit->select_start != edit->select_end && - glyphs == selection_begin) - { - int glyph_offset; - struct nk_vec2 out_offset; - struct nk_vec2 row_size; - const char *remaining; - - /* calculate 2d position */ - selection_offset_start.y = (float)(NK_MAX(total_lines-1,0)) * row_height; - row_size = nk_text_calculate_text_bounds(font, text+row_begin, - text_len-row_begin, row_height, &remaining, - &out_offset, &glyph_offset, NK_STOP_ON_NEW_LINE); - selection_offset_start.x = row_size.x; - select_begin_ptr = text + text_len; - } - - /* set end selection 2D position and line */ - if (!select_end_ptr && edit->select_start != edit->select_end && - glyphs == selection_end) - { - int glyph_offset; - struct nk_vec2 out_offset; - struct nk_vec2 row_size; - const char *remaining; - - /* calculate 2d position */ - selection_offset_end.y = (float)(total_lines-1) * row_height; - row_size = nk_text_calculate_text_bounds(font, text+row_begin, - text_len-row_begin, row_height, &remaining, - &out_offset, &glyph_offset, NK_STOP_ON_NEW_LINE); - selection_offset_end.x = row_size.x; - select_end_ptr = text + text_len; - } - if (unicode == '\n') { - text_size.x = NK_MAX(text_size.x, line_width); - total_lines++; - line_width = 0; - text_len++; - glyphs++; - row_begin = text_len; - glyph_len = nk_utf_decode(text + text_len, &unicode, len-text_len); - glyph_width = font->width(font->userdata, font->height, text+text_len, glyph_len); - continue; - } - - glyphs++; - text_len += glyph_len; - line_width += (float)glyph_width; - - glyph_len = nk_utf_decode(text + text_len, &unicode, len-text_len); - glyph_width = font->width(font->userdata, font->height, - text+text_len, glyph_len); - continue; - } - text_size.y = (float)total_lines * row_height; - - /* handle case when cursor is at end of text buffer */ - if (!cursor_ptr && edit->cursor == edit->string.len) { - cursor_pos.x = line_width; - cursor_pos.y = text_size.y - row_height; - } - } - { - /* scrollbar */ - if (cursor_follow) - { - /* update scrollbar to follow cursor */ - if (!(flags & NK_EDIT_NO_HORIZONTAL_SCROLL)) { - /* horizontal scroll */ - const float scroll_increment = area.w * 0.25f; - if (cursor_pos.x < edit->scrollbar.x) - edit->scrollbar.x = (float)(int)NK_MAX(0.0f, cursor_pos.x - scroll_increment); - if (cursor_pos.x >= edit->scrollbar.x + area.w) - edit->scrollbar.x = (float)(int)NK_MAX(0.0f, cursor_pos.x - area.w + scroll_increment); - } else edit->scrollbar.x = 0; - - if (flags & NK_EDIT_MULTILINE) { - /* vertical scroll */ - if (cursor_pos.y < edit->scrollbar.y) - edit->scrollbar.y = NK_MAX(0.0f, cursor_pos.y - row_height); - if (cursor_pos.y >= edit->scrollbar.y + row_height) - edit->scrollbar.y = edit->scrollbar.y + row_height; - } else edit->scrollbar.y = 0; - } - - /* scrollbar widget */ - if (flags & NK_EDIT_MULTILINE) - { - nk_flags ws; - struct nk_rect scroll; - float scroll_target; - float scroll_offset; - float scroll_step; - float scroll_inc; - - scroll = area; - scroll.x = (bounds.x + bounds.w - style->border) - style->scrollbar_size.x; - scroll.w = style->scrollbar_size.x; - - scroll_offset = edit->scrollbar.y; - scroll_step = scroll.h * 0.10f; - scroll_inc = scroll.h * 0.01f; - scroll_target = text_size.y; - edit->scrollbar.y = nk_do_scrollbarv(&ws, out, scroll, 0, - scroll_offset, scroll_target, scroll_step, scroll_inc, - &style->scrollbar, in, font); - } - } - - /* draw text */ - {struct nk_color background_color; - struct nk_color text_color; - struct nk_color sel_background_color; - struct nk_color sel_text_color; - struct nk_color cursor_color; - struct nk_color cursor_text_color; - const struct nk_style_item *background; - nk_push_scissor(out, clip); - - /* select correct colors to draw */ - if (*state & NK_WIDGET_STATE_ACTIVED) { - background = &style->active; - text_color = style->text_active; - sel_text_color = style->selected_text_hover; - sel_background_color = style->selected_hover; - cursor_color = style->cursor_hover; - cursor_text_color = style->cursor_text_hover; - } else if (*state & NK_WIDGET_STATE_HOVER) { - background = &style->hover; - text_color = style->text_hover; - sel_text_color = style->selected_text_hover; - sel_background_color = style->selected_hover; - cursor_text_color = style->cursor_text_hover; - cursor_color = style->cursor_hover; - } else { - background = &style->normal; - text_color = style->text_normal; - sel_text_color = style->selected_text_normal; - sel_background_color = style->selected_normal; - cursor_color = style->cursor_normal; - cursor_text_color = style->cursor_text_normal; - } - if (background->type == NK_STYLE_ITEM_IMAGE) - background_color = nk_rgba(0,0,0,0); - else - background_color = background->data.color; - - - if (edit->select_start == edit->select_end) { - /* no selection so just draw the complete text */ - const char *begin = nk_str_get_const(&edit->string); - int l = nk_str_len_char(&edit->string); - nk_edit_draw_text(out, style, area.x - edit->scrollbar.x, - area.y - edit->scrollbar.y, 0, begin, l, row_height, font, - background_color, text_color, nk_false); - } else { - /* edit has selection so draw 1-3 text chunks */ - if (edit->select_start != edit->select_end && selection_begin > 0){ - /* draw unselected text before selection */ - const char *begin = nk_str_get_const(&edit->string); - NK_ASSERT(select_begin_ptr); - nk_edit_draw_text(out, style, area.x - edit->scrollbar.x, - area.y - edit->scrollbar.y, 0, begin, (int)(select_begin_ptr - begin), - row_height, font, background_color, text_color, nk_false); - } - if (edit->select_start != edit->select_end) { - /* draw selected text */ - NK_ASSERT(select_begin_ptr); - if (!select_end_ptr) { - const char *begin = nk_str_get_const(&edit->string); - select_end_ptr = begin + nk_str_len_char(&edit->string); - } - nk_edit_draw_text(out, style, - area.x - edit->scrollbar.x, - area.y + selection_offset_start.y - edit->scrollbar.y, - selection_offset_start.x, - select_begin_ptr, (int)(select_end_ptr - select_begin_ptr), - row_height, font, sel_background_color, sel_text_color, nk_true); - } - if ((edit->select_start != edit->select_end && - selection_end < edit->string.len)) - { - /* draw unselected text after selected text */ - const char *begin = select_end_ptr; - const char *end = nk_str_get_const(&edit->string) + - nk_str_len_char(&edit->string); - NK_ASSERT(select_end_ptr); - nk_edit_draw_text(out, style, - area.x - edit->scrollbar.x, - area.y + selection_offset_end.y - edit->scrollbar.y, - selection_offset_end.x, - begin, (int)(end - begin), row_height, font, - background_color, text_color, nk_true); - } - } - - /* cursor */ - if (edit->select_start == edit->select_end) - { - if (edit->cursor >= nk_str_len(&edit->string) || - (cursor_ptr && *cursor_ptr == '\n')) { - /* draw cursor at end of line */ - struct nk_rect cursor; - cursor.w = style->cursor_size; - cursor.h = font->height; - cursor.x = area.x + cursor_pos.x - edit->scrollbar.x; - cursor.y = area.y + cursor_pos.y + row_height/2.0f - cursor.h/2.0f; - cursor.y -= edit->scrollbar.y; - nk_fill_rect(out, cursor, 0, cursor_color); - } else { - /* draw cursor inside text */ - int glyph_len; - struct nk_rect label; - struct nk_text txt; - - nk_rune unicode; - NK_ASSERT(cursor_ptr); - glyph_len = nk_utf_decode(cursor_ptr, &unicode, 4); - - label.x = area.x + cursor_pos.x - edit->scrollbar.x; - label.y = area.y + cursor_pos.y - edit->scrollbar.y; - label.w = font->width(font->userdata, font->height, cursor_ptr, glyph_len); - label.h = row_height; - - txt.padding = nk_vec2(0,0); - txt.background = cursor_color;; - txt.text = cursor_text_color; - nk_fill_rect(out, label, 0, cursor_color); - nk_widget_text(out, label, cursor_ptr, glyph_len, &txt, NK_TEXT_LEFT, font); - } - }} - } else { - /* not active so just draw text */ - int l = nk_str_len_char(&edit->string); - const char *begin = nk_str_get_const(&edit->string); - - const struct nk_style_item *background; - struct nk_color background_color; - struct nk_color text_color; - nk_push_scissor(out, clip); - if (*state & NK_WIDGET_STATE_ACTIVED) { - background = &style->active; - text_color = style->text_active; - } else if (*state & NK_WIDGET_STATE_HOVER) { - background = &style->hover; - text_color = style->text_hover; - } else { - background = &style->normal; - text_color = style->text_normal; - } - if (background->type == NK_STYLE_ITEM_IMAGE) - background_color = nk_rgba(0,0,0,0); - else - background_color = background->data.color; - nk_edit_draw_text(out, style, area.x - edit->scrollbar.x, - area.y - edit->scrollbar.y, 0, begin, l, row_height, font, - background_color, text_color, nk_false); - } - nk_push_scissor(out, old_clip);} - return ret; -} -NK_API void -nk_edit_focus(struct nk_context *ctx, nk_flags flags) -{ - nk_hash hash; - struct nk_window *win; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - if (!ctx || !ctx->current) return; - - win = ctx->current; - hash = win->edit.seq; - win->edit.active = nk_true; - win->edit.name = hash; - if (flags & NK_EDIT_ALWAYS_INSERT_MODE) - win->edit.mode = NK_TEXT_EDIT_MODE_INSERT; -} -NK_API void -nk_edit_unfocus(struct nk_context *ctx) -{ - struct nk_window *win; - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - if (!ctx || !ctx->current) return; - - win = ctx->current; - win->edit.active = nk_false; - win->edit.name = 0; -} -NK_API nk_flags -nk_edit_string(struct nk_context *ctx, nk_flags flags, - char *memory, int *len, int max, nk_plugin_filter filter) -{ - nk_hash hash; - nk_flags state; - struct nk_text_edit *edit; - struct nk_window *win; - - NK_ASSERT(ctx); - NK_ASSERT(memory); - NK_ASSERT(len); - if (!ctx || !memory || !len) - return 0; - - filter = (!filter) ? nk_filter_default: filter; - win = ctx->current; - hash = win->edit.seq; - edit = &ctx->text_edit; - nk_textedit_clear_state(&ctx->text_edit, (flags & NK_EDIT_MULTILINE)? - NK_TEXT_EDIT_MULTI_LINE: NK_TEXT_EDIT_SINGLE_LINE, filter); - - if (win->edit.active && hash == win->edit.name) { - if (flags & NK_EDIT_NO_CURSOR) - edit->cursor = nk_utf_len(memory, *len); - else edit->cursor = win->edit.cursor; - if (!(flags & NK_EDIT_SELECTABLE)) { - edit->select_start = win->edit.cursor; - edit->select_end = win->edit.cursor; - } else { - edit->select_start = win->edit.sel_start; - edit->select_end = win->edit.sel_end; - } - edit->mode = win->edit.mode; - edit->scrollbar.x = (float)win->edit.scrollbar.x; - edit->scrollbar.y = (float)win->edit.scrollbar.y; - edit->active = nk_true; - } else edit->active = nk_false; - - max = NK_MAX(1, max); - *len = NK_MIN(*len, max-1); - nk_str_init_fixed(&edit->string, memory, (nk_size)max); - edit->string.buffer.allocated = (nk_size)*len; - edit->string.len = nk_utf_len(memory, *len); - state = nk_edit_buffer(ctx, flags, edit, filter); - *len = (int)edit->string.buffer.allocated; - - if (edit->active) { - win->edit.cursor = edit->cursor; - win->edit.sel_start = edit->select_start; - win->edit.sel_end = edit->select_end; - win->edit.mode = edit->mode; - win->edit.scrollbar.x = (nk_uint)edit->scrollbar.x; - win->edit.scrollbar.y = (nk_uint)edit->scrollbar.y; - } return state; -} -NK_API nk_flags -nk_edit_buffer(struct nk_context *ctx, nk_flags flags, - struct nk_text_edit *edit, nk_plugin_filter filter) -{ - struct nk_window *win; - struct nk_style *style; - struct nk_input *in; - - enum nk_widget_layout_states state; - struct nk_rect bounds; - - nk_flags ret_flags = 0; - unsigned char prev_state; - nk_hash hash; - - /* make sure correct values */ - NK_ASSERT(ctx); - NK_ASSERT(edit); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return 0; - - win = ctx->current; - style = &ctx->style; - state = nk_widget(&bounds, ctx); - if (!state) return state; - in = (win->layout->flags & NK_WINDOW_ROM) ? 0 : &ctx->input; - - /* check if edit is currently hot item */ - hash = win->edit.seq++; - if (win->edit.active && hash == win->edit.name) { - if (flags & NK_EDIT_NO_CURSOR) - edit->cursor = edit->string.len; - if (!(flags & NK_EDIT_SELECTABLE)) { - edit->select_start = edit->cursor; - edit->select_end = edit->cursor; - } - if (flags & NK_EDIT_CLIPBOARD) - edit->clip = ctx->clip; - edit->active = (unsigned char)win->edit.active; - } else edit->active = nk_false; - edit->mode = win->edit.mode; - - filter = (!filter) ? nk_filter_default: filter; - prev_state = (unsigned char)edit->active; - in = (flags & NK_EDIT_READ_ONLY) ? 0: in; - ret_flags = nk_do_edit(&ctx->last_widget_state, &win->buffer, bounds, flags, - filter, edit, &style->edit, in, style->font); - - if (ctx->last_widget_state & NK_WIDGET_STATE_HOVER) - ctx->style.cursor_active = ctx->style.cursors[NK_CURSOR_TEXT]; - if (edit->active && prev_state != edit->active) { - /* current edit is now hot */ - win->edit.active = nk_true; - win->edit.name = hash; - } else if (prev_state && !edit->active) { - /* current edit is now cold */ - win->edit.active = nk_false; - } return ret_flags; -} -NK_API nk_flags -nk_edit_string_zero_terminated(struct nk_context *ctx, nk_flags flags, - char *buffer, int max, nk_plugin_filter filter) -{ - nk_flags result; - int len = nk_strlen(buffer); - result = nk_edit_string(ctx, flags, buffer, &len, max, filter); - buffer[NK_MIN(NK_MAX(max-1,0), len)] = '\0'; - return result; -} - - - - - -/* =============================================================== - * - * PROPERTY - * - * ===============================================================*/ -NK_LIB void -nk_drag_behavior(nk_flags *state, const struct nk_input *in, - struct nk_rect drag, struct nk_property_variant *variant, - float inc_per_pixel) -{ - int left_mouse_down = in && in->mouse.buttons[NK_BUTTON_LEFT].down; - int left_mouse_click_in_cursor = in && - nk_input_has_mouse_click_down_in_rect(in, NK_BUTTON_LEFT, drag, nk_true); - - nk_widget_state_reset(state); - if (nk_input_is_mouse_hovering_rect(in, drag)) - *state = NK_WIDGET_STATE_HOVERED; - - if (left_mouse_down && left_mouse_click_in_cursor) { - float delta, pixels; - pixels = in->mouse.delta.x; - delta = pixels * inc_per_pixel; - switch (variant->kind) { - default: break; - case NK_PROPERTY_INT: - variant->value.i = variant->value.i + (int)delta; - variant->value.i = NK_CLAMP(variant->min_value.i, variant->value.i, variant->max_value.i); - break; - case NK_PROPERTY_FLOAT: - variant->value.f = variant->value.f + (float)delta; - variant->value.f = NK_CLAMP(variant->min_value.f, variant->value.f, variant->max_value.f); - break; - case NK_PROPERTY_DOUBLE: - variant->value.d = variant->value.d + (double)delta; - variant->value.d = NK_CLAMP(variant->min_value.d, variant->value.d, variant->max_value.d); - break; - } - *state = NK_WIDGET_STATE_ACTIVE; - } - if (*state & NK_WIDGET_STATE_HOVER && !nk_input_is_mouse_prev_hovering_rect(in, drag)) - *state |= NK_WIDGET_STATE_ENTERED; - else if (nk_input_is_mouse_prev_hovering_rect(in, drag)) - *state |= NK_WIDGET_STATE_LEFT; -} -NK_LIB void -nk_property_behavior(nk_flags *ws, const struct nk_input *in, - struct nk_rect property, struct nk_rect label, struct nk_rect edit, - struct nk_rect empty, int *state, struct nk_property_variant *variant, - float inc_per_pixel) -{ - nk_widget_state_reset(ws); - if (in && *state == NK_PROPERTY_DEFAULT) { - if (nk_button_behavior(ws, edit, in, NK_BUTTON_DEFAULT)) - *state = NK_PROPERTY_EDIT; - else if (nk_input_is_mouse_click_down_in_rect(in, NK_BUTTON_LEFT, label, nk_true)) - *state = NK_PROPERTY_DRAG; - else if (nk_input_is_mouse_click_down_in_rect(in, NK_BUTTON_LEFT, empty, nk_true)) - *state = NK_PROPERTY_DRAG; - } - if (*state == NK_PROPERTY_DRAG) { - nk_drag_behavior(ws, in, property, variant, inc_per_pixel); - if (!(*ws & NK_WIDGET_STATE_ACTIVED)) *state = NK_PROPERTY_DEFAULT; - } -} -NK_LIB void -nk_draw_property(struct nk_command_buffer *out, const struct nk_style_property *style, - const struct nk_rect *bounds, const struct nk_rect *label, nk_flags state, - const char *name, int len, const struct nk_user_font *font) -{ - struct nk_text text; - const struct nk_style_item *background; - - /* select correct background and text color */ - if (state & NK_WIDGET_STATE_ACTIVED) { - background = &style->active; - text.text = style->label_active; - } else if (state & NK_WIDGET_STATE_HOVER) { - background = &style->hover; - text.text = style->label_hover; - } else { - background = &style->normal; - text.text = style->label_normal; - } - - /* draw background */ - switch(background->type) { - case NK_STYLE_ITEM_IMAGE: - text.background = nk_rgba(0, 0, 0, 0); - nk_draw_image(out, *bounds, &background->data.image, nk_white); - break; - case NK_STYLE_ITEM_NINE_SLICE: - text.background = nk_rgba(0, 0, 0, 0); - nk_draw_nine_slice(out, *bounds, &background->data.slice, nk_white); - break; - case NK_STYLE_ITEM_COLOR: - text.background = background->data.color; - nk_fill_rect(out, *bounds, style->rounding, background->data.color); - nk_stroke_rect(out, *bounds, style->rounding, style->border, background->data.color); - break; - } - - /* draw label */ - text.padding = nk_vec2(0,0); - nk_widget_text(out, *label, name, len, &text, NK_TEXT_CENTERED, font); -} -NK_LIB void -nk_do_property(nk_flags *ws, - struct nk_command_buffer *out, struct nk_rect property, - const char *name, struct nk_property_variant *variant, - float inc_per_pixel, char *buffer, int *len, - int *state, int *cursor, int *select_begin, int *select_end, - const struct nk_style_property *style, - enum nk_property_filter filter, struct nk_input *in, - const struct nk_user_font *font, struct nk_text_edit *text_edit, - enum nk_button_behavior behavior) -{ - const nk_plugin_filter filters[] = { - nk_filter_decimal, - nk_filter_float - }; - nk_bool active, old; - int num_len = 0, name_len; - char string[NK_MAX_NUMBER_BUFFER]; - float size; - - char *dst = 0; - int *length; - - struct nk_rect left; - struct nk_rect right; - struct nk_rect label; - struct nk_rect edit; - struct nk_rect empty; - - /* left decrement button */ - left.h = font->height/2; - left.w = left.h; - left.x = property.x + style->border + style->padding.x; - left.y = property.y + style->border + property.h/2.0f - left.h/2; - - /* text label */ - name_len = nk_strlen(name); - size = font->width(font->userdata, font->height, name, name_len); - label.x = left.x + left.w + style->padding.x; - label.w = (float)size + 2 * style->padding.x; - label.y = property.y + style->border + style->padding.y; - label.h = property.h - (2 * style->border + 2 * style->padding.y); - - /* right increment button */ - right.y = left.y; - right.w = left.w; - right.h = left.h; - right.x = property.x + property.w - (right.w + style->padding.x); - - /* edit */ - if (*state == NK_PROPERTY_EDIT) { - size = font->width(font->userdata, font->height, buffer, *len); - size += style->edit.cursor_size; - length = len; - dst = buffer; - } else { - switch (variant->kind) { - default: break; - case NK_PROPERTY_INT: - nk_itoa(string, variant->value.i); - num_len = nk_strlen(string); - break; - case NK_PROPERTY_FLOAT: - NK_DTOA(string, (double)variant->value.f); - num_len = nk_string_float_limit(string, NK_MAX_FLOAT_PRECISION); - break; - case NK_PROPERTY_DOUBLE: - NK_DTOA(string, variant->value.d); - num_len = nk_string_float_limit(string, NK_MAX_FLOAT_PRECISION); - break; - } - size = font->width(font->userdata, font->height, string, num_len); - dst = string; - length = &num_len; - } - - edit.w = (float)size + 2 * style->padding.x; - edit.w = NK_MIN(edit.w, right.x - (label.x + label.w)); - edit.x = right.x - (edit.w + style->padding.x); - edit.y = property.y + style->border; - edit.h = property.h - (2 * style->border); - - /* empty left space activator */ - empty.w = edit.x - (label.x + label.w); - empty.x = label.x + label.w; - empty.y = property.y; - empty.h = property.h; - - /* update property */ - old = (*state == NK_PROPERTY_EDIT); - nk_property_behavior(ws, in, property, label, edit, empty, state, variant, inc_per_pixel); - - /* draw property */ - if (style->draw_begin) style->draw_begin(out, style->userdata); - nk_draw_property(out, style, &property, &label, *ws, name, name_len, font); - if (style->draw_end) style->draw_end(out, style->userdata); - - /* execute right button */ - if (nk_do_button_symbol(ws, out, left, style->sym_left, behavior, &style->dec_button, in, font)) { - switch (variant->kind) { - default: break; - case NK_PROPERTY_INT: - variant->value.i = NK_CLAMP(variant->min_value.i, variant->value.i - variant->step.i, variant->max_value.i); break; - case NK_PROPERTY_FLOAT: - variant->value.f = NK_CLAMP(variant->min_value.f, variant->value.f - variant->step.f, variant->max_value.f); break; - case NK_PROPERTY_DOUBLE: - variant->value.d = NK_CLAMP(variant->min_value.d, variant->value.d - variant->step.d, variant->max_value.d); break; - } - } - /* execute left button */ - if (nk_do_button_symbol(ws, out, right, style->sym_right, behavior, &style->inc_button, in, font)) { - switch (variant->kind) { - default: break; - case NK_PROPERTY_INT: - variant->value.i = NK_CLAMP(variant->min_value.i, variant->value.i + variant->step.i, variant->max_value.i); break; - case NK_PROPERTY_FLOAT: - variant->value.f = NK_CLAMP(variant->min_value.f, variant->value.f + variant->step.f, variant->max_value.f); break; - case NK_PROPERTY_DOUBLE: - variant->value.d = NK_CLAMP(variant->min_value.d, variant->value.d + variant->step.d, variant->max_value.d); break; - } - } - if (old != NK_PROPERTY_EDIT && (*state == NK_PROPERTY_EDIT)) { - /* property has been activated so setup buffer */ - NK_MEMCPY(buffer, dst, (nk_size)*length); - *cursor = nk_utf_len(buffer, *length); - *len = *length; - length = len; - dst = buffer; - active = 0; - } else active = (*state == NK_PROPERTY_EDIT); - - /* execute and run text edit field */ - nk_textedit_clear_state(text_edit, NK_TEXT_EDIT_SINGLE_LINE, filters[filter]); - text_edit->active = (unsigned char)active; - text_edit->string.len = *length; - text_edit->cursor = NK_CLAMP(0, *cursor, *length); - text_edit->select_start = NK_CLAMP(0,*select_begin, *length); - text_edit->select_end = NK_CLAMP(0,*select_end, *length); - text_edit->string.buffer.allocated = (nk_size)*length; - text_edit->string.buffer.memory.size = NK_MAX_NUMBER_BUFFER; - text_edit->string.buffer.memory.ptr = dst; - text_edit->string.buffer.size = NK_MAX_NUMBER_BUFFER; - text_edit->mode = NK_TEXT_EDIT_MODE_INSERT; - nk_do_edit(ws, out, edit, NK_EDIT_FIELD|NK_EDIT_AUTO_SELECT, - filters[filter], text_edit, &style->edit, (*state == NK_PROPERTY_EDIT) ? in: 0, font); - - *length = text_edit->string.len; - *cursor = text_edit->cursor; - *select_begin = text_edit->select_start; - *select_end = text_edit->select_end; - if (text_edit->active && nk_input_is_key_pressed(in, NK_KEY_ENTER)) - text_edit->active = nk_false; - - if (active && !text_edit->active) { - /* property is now not active so convert edit text to value*/ - *state = NK_PROPERTY_DEFAULT; - buffer[*len] = '\0'; - switch (variant->kind) { - default: break; - case NK_PROPERTY_INT: - variant->value.i = nk_strtoi(buffer, 0); - variant->value.i = NK_CLAMP(variant->min_value.i, variant->value.i, variant->max_value.i); - break; - case NK_PROPERTY_FLOAT: - nk_string_float_limit(buffer, NK_MAX_FLOAT_PRECISION); - variant->value.f = nk_strtof(buffer, 0); - variant->value.f = NK_CLAMP(variant->min_value.f, variant->value.f, variant->max_value.f); - break; - case NK_PROPERTY_DOUBLE: - nk_string_float_limit(buffer, NK_MAX_FLOAT_PRECISION); - variant->value.d = nk_strtod(buffer, 0); - variant->value.d = NK_CLAMP(variant->min_value.d, variant->value.d, variant->max_value.d); - break; - } - } -} -NK_LIB struct nk_property_variant -nk_property_variant_int(int value, int min_value, int max_value, int step) -{ - struct nk_property_variant result; - result.kind = NK_PROPERTY_INT; - result.value.i = value; - result.min_value.i = min_value; - result.max_value.i = max_value; - result.step.i = step; - return result; -} -NK_LIB struct nk_property_variant -nk_property_variant_float(float value, float min_value, float max_value, float step) -{ - struct nk_property_variant result; - result.kind = NK_PROPERTY_FLOAT; - result.value.f = value; - result.min_value.f = min_value; - result.max_value.f = max_value; - result.step.f = step; - return result; -} -NK_LIB struct nk_property_variant -nk_property_variant_double(double value, double min_value, double max_value, - double step) -{ - struct nk_property_variant result; - result.kind = NK_PROPERTY_DOUBLE; - result.value.d = value; - result.min_value.d = min_value; - result.max_value.d = max_value; - result.step.d = step; - return result; -} -NK_LIB void -nk_property(struct nk_context *ctx, const char *name, struct nk_property_variant *variant, - float inc_per_pixel, const enum nk_property_filter filter) -{ - struct nk_window *win; - struct nk_panel *layout; - struct nk_input *in; - const struct nk_style *style; - - struct nk_rect bounds; - enum nk_widget_layout_states s; - - int *state = 0; - nk_hash hash = 0; - char *buffer = 0; - int *len = 0; - int *cursor = 0; - int *select_begin = 0; - int *select_end = 0; - int old_state; - - char dummy_buffer[NK_MAX_NUMBER_BUFFER]; - int dummy_state = NK_PROPERTY_DEFAULT; - int dummy_length = 0; - int dummy_cursor = 0; - int dummy_select_begin = 0; - int dummy_select_end = 0; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return; - - win = ctx->current; - layout = win->layout; - style = &ctx->style; - s = nk_widget(&bounds, ctx); - if (!s) return; - - /* calculate hash from name */ - if (name[0] == '#') { - hash = nk_murmur_hash(name, (int)nk_strlen(name), win->property.seq++); - name++; /* special number hash */ - } else hash = nk_murmur_hash(name, (int)nk_strlen(name), 42); - - /* check if property is currently hot item */ - if (win->property.active && hash == win->property.name) { - buffer = win->property.buffer; - len = &win->property.length; - cursor = &win->property.cursor; - state = &win->property.state; - select_begin = &win->property.select_start; - select_end = &win->property.select_end; - } else { - buffer = dummy_buffer; - len = &dummy_length; - cursor = &dummy_cursor; - state = &dummy_state; - select_begin = &dummy_select_begin; - select_end = &dummy_select_end; - } - - /* execute property widget */ - old_state = *state; - ctx->text_edit.clip = ctx->clip; - in = ((s == NK_WIDGET_ROM && !win->property.active) || - layout->flags & NK_WINDOW_ROM) ? 0 : &ctx->input; - nk_do_property(&ctx->last_widget_state, &win->buffer, bounds, name, - variant, inc_per_pixel, buffer, len, state, cursor, select_begin, - select_end, &style->property, filter, in, style->font, &ctx->text_edit, - ctx->button_behavior); - - if (in && *state != NK_PROPERTY_DEFAULT && !win->property.active) { - /* current property is now hot */ - win->property.active = 1; - NK_MEMCPY(win->property.buffer, buffer, (nk_size)*len); - win->property.length = *len; - win->property.cursor = *cursor; - win->property.state = *state; - win->property.name = hash; - win->property.select_start = *select_begin; - win->property.select_end = *select_end; - if (*state == NK_PROPERTY_DRAG) { - ctx->input.mouse.grab = nk_true; - ctx->input.mouse.grabbed = nk_true; - } - } - /* check if previously active property is now inactive */ - if (*state == NK_PROPERTY_DEFAULT && old_state != NK_PROPERTY_DEFAULT) { - if (old_state == NK_PROPERTY_DRAG) { - ctx->input.mouse.grab = nk_false; - ctx->input.mouse.grabbed = nk_false; - ctx->input.mouse.ungrab = nk_true; - } - win->property.select_start = 0; - win->property.select_end = 0; - win->property.active = 0; - } -} -NK_API void -nk_property_int(struct nk_context *ctx, const char *name, - int min, int *val, int max, int step, float inc_per_pixel) -{ - struct nk_property_variant variant; - NK_ASSERT(ctx); - NK_ASSERT(name); - NK_ASSERT(val); - - if (!ctx || !ctx->current || !name || !val) return; - variant = nk_property_variant_int(*val, min, max, step); - nk_property(ctx, name, &variant, inc_per_pixel, NK_FILTER_INT); - *val = variant.value.i; -} -NK_API void -nk_property_float(struct nk_context *ctx, const char *name, - float min, float *val, float max, float step, float inc_per_pixel) -{ - struct nk_property_variant variant; - NK_ASSERT(ctx); - NK_ASSERT(name); - NK_ASSERT(val); - - if (!ctx || !ctx->current || !name || !val) return; - variant = nk_property_variant_float(*val, min, max, step); - nk_property(ctx, name, &variant, inc_per_pixel, NK_FILTER_FLOAT); - *val = variant.value.f; -} -NK_API void -nk_property_double(struct nk_context *ctx, const char *name, - double min, double *val, double max, double step, float inc_per_pixel) -{ - struct nk_property_variant variant; - NK_ASSERT(ctx); - NK_ASSERT(name); - NK_ASSERT(val); - - if (!ctx || !ctx->current || !name || !val) return; - variant = nk_property_variant_double(*val, min, max, step); - nk_property(ctx, name, &variant, inc_per_pixel, NK_FILTER_FLOAT); - *val = variant.value.d; -} -NK_API int -nk_propertyi(struct nk_context *ctx, const char *name, int min, int val, - int max, int step, float inc_per_pixel) -{ - struct nk_property_variant variant; - NK_ASSERT(ctx); - NK_ASSERT(name); - - if (!ctx || !ctx->current || !name) return val; - variant = nk_property_variant_int(val, min, max, step); - nk_property(ctx, name, &variant, inc_per_pixel, NK_FILTER_INT); - val = variant.value.i; - return val; -} -NK_API float -nk_propertyf(struct nk_context *ctx, const char *name, float min, - float val, float max, float step, float inc_per_pixel) -{ - struct nk_property_variant variant; - NK_ASSERT(ctx); - NK_ASSERT(name); - - if (!ctx || !ctx->current || !name) return val; - variant = nk_property_variant_float(val, min, max, step); - nk_property(ctx, name, &variant, inc_per_pixel, NK_FILTER_FLOAT); - val = variant.value.f; - return val; -} -NK_API double -nk_propertyd(struct nk_context *ctx, const char *name, double min, - double val, double max, double step, float inc_per_pixel) -{ - struct nk_property_variant variant; - NK_ASSERT(ctx); - NK_ASSERT(name); - - if (!ctx || !ctx->current || !name) return val; - variant = nk_property_variant_double(val, min, max, step); - nk_property(ctx, name, &variant, inc_per_pixel, NK_FILTER_FLOAT); - val = variant.value.d; - return val; -} - - - - - -/* ============================================================== - * - * CHART - * - * ===============================================================*/ -NK_API nk_bool -nk_chart_begin_colored(struct nk_context *ctx, enum nk_chart_type type, - struct nk_color color, struct nk_color highlight, - int count, float min_value, float max_value) -{ - struct nk_window *win; - struct nk_chart *chart; - const struct nk_style *config; - const struct nk_style_chart *style; - - const struct nk_style_item *background; - struct nk_rect bounds = {0, 0, 0, 0}; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - - if (!ctx || !ctx->current || !ctx->current->layout) return 0; - if (!nk_widget(&bounds, ctx)) { - chart = &ctx->current->layout->chart; - nk_zero(chart, sizeof(*chart)); - return 0; - } - - win = ctx->current; - config = &ctx->style; - chart = &win->layout->chart; - style = &config->chart; - - /* setup basic generic chart */ - nk_zero(chart, sizeof(*chart)); - chart->x = bounds.x + style->padding.x; - chart->y = bounds.y + style->padding.y; - chart->w = bounds.w - 2 * style->padding.x; - chart->h = bounds.h - 2 * style->padding.y; - chart->w = NK_MAX(chart->w, 2 * style->padding.x); - chart->h = NK_MAX(chart->h, 2 * style->padding.y); - - /* add first slot into chart */ - {struct nk_chart_slot *slot = &chart->slots[chart->slot++]; - slot->type = type; - slot->count = count; - slot->color = color; - slot->highlight = highlight; - slot->min = NK_MIN(min_value, max_value); - slot->max = NK_MAX(min_value, max_value); - slot->range = slot->max - slot->min;} - - /* draw chart background */ - background = &style->background; - - switch(background->type) { - case NK_STYLE_ITEM_IMAGE: - nk_draw_image(&win->buffer, bounds, &background->data.image, nk_white); - break; - case NK_STYLE_ITEM_NINE_SLICE: - nk_draw_nine_slice(&win->buffer, bounds, &background->data.slice, nk_white); - break; - case NK_STYLE_ITEM_COLOR: - nk_fill_rect(&win->buffer, bounds, style->rounding, style->border_color); - nk_fill_rect(&win->buffer, nk_shrink_rect(bounds, style->border), - style->rounding, style->background.data.color); - break; - } - return 1; -} -NK_API nk_bool -nk_chart_begin(struct nk_context *ctx, const enum nk_chart_type type, - int count, float min_value, float max_value) -{ - return nk_chart_begin_colored(ctx, type, ctx->style.chart.color, - ctx->style.chart.selected_color, count, min_value, max_value); -} -NK_API void -nk_chart_add_slot_colored(struct nk_context *ctx, const enum nk_chart_type type, - struct nk_color color, struct nk_color highlight, - int count, float min_value, float max_value) -{ - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - NK_ASSERT(ctx->current->layout->chart.slot < NK_CHART_MAX_SLOT); - if (!ctx || !ctx->current || !ctx->current->layout) return; - if (ctx->current->layout->chart.slot >= NK_CHART_MAX_SLOT) return; - - /* add another slot into the graph */ - {struct nk_chart *chart = &ctx->current->layout->chart; - struct nk_chart_slot *slot = &chart->slots[chart->slot++]; - slot->type = type; - slot->count = count; - slot->color = color; - slot->highlight = highlight; - slot->min = NK_MIN(min_value, max_value); - slot->max = NK_MAX(min_value, max_value); - slot->range = slot->max - slot->min;} -} -NK_API void -nk_chart_add_slot(struct nk_context *ctx, const enum nk_chart_type type, - int count, float min_value, float max_value) -{ - nk_chart_add_slot_colored(ctx, type, ctx->style.chart.color, - ctx->style.chart.selected_color, count, min_value, max_value); -} -NK_INTERN nk_flags -nk_chart_push_line(struct nk_context *ctx, struct nk_window *win, - struct nk_chart *g, float value, int slot) -{ - struct nk_panel *layout = win->layout; - const struct nk_input *i = &ctx->input; - struct nk_command_buffer *out = &win->buffer; - - nk_flags ret = 0; - struct nk_vec2 cur; - struct nk_rect bounds; - struct nk_color color; - float step; - float range; - float ratio; - - NK_ASSERT(slot >= 0 && slot < NK_CHART_MAX_SLOT); - step = g->w / (float)g->slots[slot].count; - range = g->slots[slot].max - g->slots[slot].min; - ratio = (value - g->slots[slot].min) / range; - - if (g->slots[slot].index == 0) { - /* first data point does not have a connection */ - g->slots[slot].last.x = g->x; - g->slots[slot].last.y = (g->y + g->h) - ratio * (float)g->h; - - bounds.x = g->slots[slot].last.x - 2; - bounds.y = g->slots[slot].last.y - 2; - bounds.w = bounds.h = 4; - - color = g->slots[slot].color; - if (!(layout->flags & NK_WINDOW_ROM) && - NK_INBOX(i->mouse.pos.x,i->mouse.pos.y, g->slots[slot].last.x-3, g->slots[slot].last.y-3, 6, 6)){ - ret = nk_input_is_mouse_hovering_rect(i, bounds) ? NK_CHART_HOVERING : 0; - ret |= (i->mouse.buttons[NK_BUTTON_LEFT].down && - i->mouse.buttons[NK_BUTTON_LEFT].clicked) ? NK_CHART_CLICKED: 0; - color = g->slots[slot].highlight; - } - nk_fill_rect(out, bounds, 0, color); - g->slots[slot].index += 1; - return ret; - } - - /* draw a line between the last data point and the new one */ - color = g->slots[slot].color; - cur.x = g->x + (float)(step * (float)g->slots[slot].index); - cur.y = (g->y + g->h) - (ratio * (float)g->h); - nk_stroke_line(out, g->slots[slot].last.x, g->slots[slot].last.y, cur.x, cur.y, 1.0f, color); - - bounds.x = cur.x - 3; - bounds.y = cur.y - 3; - bounds.w = bounds.h = 6; - - /* user selection of current data point */ - if (!(layout->flags & NK_WINDOW_ROM)) { - if (nk_input_is_mouse_hovering_rect(i, bounds)) { - ret = NK_CHART_HOVERING; - ret |= (!i->mouse.buttons[NK_BUTTON_LEFT].down && - i->mouse.buttons[NK_BUTTON_LEFT].clicked) ? NK_CHART_CLICKED: 0; - color = g->slots[slot].highlight; - } - } - nk_fill_rect(out, nk_rect(cur.x - 2, cur.y - 2, 4, 4), 0, color); - - /* save current data point position */ - g->slots[slot].last.x = cur.x; - g->slots[slot].last.y = cur.y; - g->slots[slot].index += 1; - return ret; -} -NK_INTERN nk_flags -nk_chart_push_column(const struct nk_context *ctx, struct nk_window *win, - struct nk_chart *chart, float value, int slot) -{ - struct nk_command_buffer *out = &win->buffer; - const struct nk_input *in = &ctx->input; - struct nk_panel *layout = win->layout; - - float ratio; - nk_flags ret = 0; - struct nk_color color; - struct nk_rect item = {0,0,0,0}; - - NK_ASSERT(slot >= 0 && slot < NK_CHART_MAX_SLOT); - if (chart->slots[slot].index >= chart->slots[slot].count) - return nk_false; - if (chart->slots[slot].count) { - float padding = (float)(chart->slots[slot].count-1); - item.w = (chart->w - padding) / (float)(chart->slots[slot].count); - } - - /* calculate bounds of current bar chart entry */ - color = chart->slots[slot].color;; - item.h = chart->h * NK_ABS((value/chart->slots[slot].range)); - if (value >= 0) { - ratio = (value + NK_ABS(chart->slots[slot].min)) / NK_ABS(chart->slots[slot].range); - item.y = (chart->y + chart->h) - chart->h * ratio; - } else { - ratio = (value - chart->slots[slot].max) / chart->slots[slot].range; - item.y = chart->y + (chart->h * NK_ABS(ratio)) - item.h; - } - item.x = chart->x + ((float)chart->slots[slot].index * item.w); - item.x = item.x + ((float)chart->slots[slot].index); - - /* user chart bar selection */ - if (!(layout->flags & NK_WINDOW_ROM) && - NK_INBOX(in->mouse.pos.x,in->mouse.pos.y,item.x,item.y,item.w,item.h)) { - ret = NK_CHART_HOVERING; - ret |= (!in->mouse.buttons[NK_BUTTON_LEFT].down && - in->mouse.buttons[NK_BUTTON_LEFT].clicked) ? NK_CHART_CLICKED: 0; - color = chart->slots[slot].highlight; - } - nk_fill_rect(out, item, 0, color); - chart->slots[slot].index += 1; - return ret; -} -NK_API nk_flags -nk_chart_push_slot(struct nk_context *ctx, float value, int slot) -{ - nk_flags flags; - struct nk_window *win; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(slot >= 0 && slot < NK_CHART_MAX_SLOT); - NK_ASSERT(slot < ctx->current->layout->chart.slot); - if (!ctx || !ctx->current || slot >= NK_CHART_MAX_SLOT) return nk_false; - if (slot >= ctx->current->layout->chart.slot) return nk_false; - - win = ctx->current; - if (win->layout->chart.slot < slot) return nk_false; - switch (win->layout->chart.slots[slot].type) { - case NK_CHART_LINES: - flags = nk_chart_push_line(ctx, win, &win->layout->chart, value, slot); break; - case NK_CHART_COLUMN: - flags = nk_chart_push_column(ctx, win, &win->layout->chart, value, slot); break; - default: - case NK_CHART_MAX: - flags = 0; - } - return flags; -} -NK_API nk_flags -nk_chart_push(struct nk_context *ctx, float value) -{ - return nk_chart_push_slot(ctx, value, 0); -} -NK_API void -nk_chart_end(struct nk_context *ctx) -{ - struct nk_window *win; - struct nk_chart *chart; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - if (!ctx || !ctx->current) - return; - - win = ctx->current; - chart = &win->layout->chart; - NK_MEMSET(chart, 0, sizeof(*chart)); - return; -} -NK_API void -nk_plot(struct nk_context *ctx, enum nk_chart_type type, const float *values, - int count, int offset) -{ - int i = 0; - float min_value; - float max_value; - - NK_ASSERT(ctx); - NK_ASSERT(values); - if (!ctx || !values || !count) return; - - min_value = values[offset]; - max_value = values[offset]; - for (i = 0; i < count; ++i) { - min_value = NK_MIN(values[i + offset], min_value); - max_value = NK_MAX(values[i + offset], max_value); - } - - if (nk_chart_begin(ctx, type, count, min_value, max_value)) { - for (i = 0; i < count; ++i) - nk_chart_push(ctx, values[i + offset]); - nk_chart_end(ctx); - } -} -NK_API void -nk_plot_function(struct nk_context *ctx, enum nk_chart_type type, void *userdata, - float(*value_getter)(void* user, int index), int count, int offset) -{ - int i = 0; - float min_value; - float max_value; - - NK_ASSERT(ctx); - NK_ASSERT(value_getter); - if (!ctx || !value_getter || !count) return; - - max_value = min_value = value_getter(userdata, offset); - for (i = 0; i < count; ++i) { - float value = value_getter(userdata, i + offset); - min_value = NK_MIN(value, min_value); - max_value = NK_MAX(value, max_value); - } - - if (nk_chart_begin(ctx, type, count, min_value, max_value)) { - for (i = 0; i < count; ++i) - nk_chart_push(ctx, value_getter(userdata, i + offset)); - nk_chart_end(ctx); - } -} - - - - - -/* ============================================================== - * - * COLOR PICKER - * - * ===============================================================*/ -NK_LIB nk_bool -nk_color_picker_behavior(nk_flags *state, - const struct nk_rect *bounds, const struct nk_rect *matrix, - const struct nk_rect *hue_bar, const struct nk_rect *alpha_bar, - struct nk_colorf *color, const struct nk_input *in) -{ - float hsva[4]; - nk_bool value_changed = 0; - nk_bool hsv_changed = 0; - - NK_ASSERT(state); - NK_ASSERT(matrix); - NK_ASSERT(hue_bar); - NK_ASSERT(color); - - /* color matrix */ - nk_colorf_hsva_fv(hsva, *color); - if (nk_button_behavior(state, *matrix, in, NK_BUTTON_REPEATER)) { - hsva[1] = NK_SATURATE((in->mouse.pos.x - matrix->x) / (matrix->w-1)); - hsva[2] = 1.0f - NK_SATURATE((in->mouse.pos.y - matrix->y) / (matrix->h-1)); - value_changed = hsv_changed = 1; - } - /* hue bar */ - if (nk_button_behavior(state, *hue_bar, in, NK_BUTTON_REPEATER)) { - hsva[0] = NK_SATURATE((in->mouse.pos.y - hue_bar->y) / (hue_bar->h-1)); - value_changed = hsv_changed = 1; - } - /* alpha bar */ - if (alpha_bar) { - if (nk_button_behavior(state, *alpha_bar, in, NK_BUTTON_REPEATER)) { - hsva[3] = 1.0f - NK_SATURATE((in->mouse.pos.y - alpha_bar->y) / (alpha_bar->h-1)); - value_changed = 1; - } - } - nk_widget_state_reset(state); - if (hsv_changed) { - *color = nk_hsva_colorfv(hsva); - *state = NK_WIDGET_STATE_ACTIVE; - } - if (value_changed) { - color->a = hsva[3]; - *state = NK_WIDGET_STATE_ACTIVE; - } - /* set color picker widget state */ - if (nk_input_is_mouse_hovering_rect(in, *bounds)) - *state = NK_WIDGET_STATE_HOVERED; - if (*state & NK_WIDGET_STATE_HOVER && !nk_input_is_mouse_prev_hovering_rect(in, *bounds)) - *state |= NK_WIDGET_STATE_ENTERED; - else if (nk_input_is_mouse_prev_hovering_rect(in, *bounds)) - *state |= NK_WIDGET_STATE_LEFT; - return value_changed; -} -NK_LIB void -nk_draw_color_picker(struct nk_command_buffer *o, const struct nk_rect *matrix, - const struct nk_rect *hue_bar, const struct nk_rect *alpha_bar, - struct nk_colorf col) -{ - NK_STORAGE const struct nk_color black = {0,0,0,255}; - NK_STORAGE const struct nk_color white = {255, 255, 255, 255}; - NK_STORAGE const struct nk_color black_trans = {0,0,0,0}; - - const float crosshair_size = 7.0f; - struct nk_color temp; - float hsva[4]; - float line_y; - int i; - - NK_ASSERT(o); - NK_ASSERT(matrix); - NK_ASSERT(hue_bar); - - /* draw hue bar */ - nk_colorf_hsva_fv(hsva, col); - for (i = 0; i < 6; ++i) { - NK_GLOBAL const struct nk_color hue_colors[] = { - {255, 0, 0, 255}, {255,255,0,255}, {0,255,0,255}, {0, 255,255,255}, - {0,0,255,255}, {255, 0, 255, 255}, {255, 0, 0, 255} - }; - nk_fill_rect_multi_color(o, - nk_rect(hue_bar->x, hue_bar->y + (float)i * (hue_bar->h/6.0f) + 0.5f, - hue_bar->w, (hue_bar->h/6.0f) + 0.5f), hue_colors[i], hue_colors[i], - hue_colors[i+1], hue_colors[i+1]); - } - line_y = (float)(int)(hue_bar->y + hsva[0] * matrix->h + 0.5f); - nk_stroke_line(o, hue_bar->x-1, line_y, hue_bar->x + hue_bar->w + 2, - line_y, 1, nk_rgb(255,255,255)); - - /* draw alpha bar */ - if (alpha_bar) { - float alpha = NK_SATURATE(col.a); - line_y = (float)(int)(alpha_bar->y + (1.0f - alpha) * matrix->h + 0.5f); - - nk_fill_rect_multi_color(o, *alpha_bar, white, white, black, black); - nk_stroke_line(o, alpha_bar->x-1, line_y, alpha_bar->x + alpha_bar->w + 2, - line_y, 1, nk_rgb(255,255,255)); - } - - /* draw color matrix */ - temp = nk_hsv_f(hsva[0], 1.0f, 1.0f); - nk_fill_rect_multi_color(o, *matrix, white, temp, temp, white); - nk_fill_rect_multi_color(o, *matrix, black_trans, black_trans, black, black); - - /* draw cross-hair */ - {struct nk_vec2 p; float S = hsva[1]; float V = hsva[2]; - p.x = (float)(int)(matrix->x + S * matrix->w); - p.y = (float)(int)(matrix->y + (1.0f - V) * matrix->h); - nk_stroke_line(o, p.x - crosshair_size, p.y, p.x-2, p.y, 1.0f, white); - nk_stroke_line(o, p.x + crosshair_size + 1, p.y, p.x+3, p.y, 1.0f, white); - nk_stroke_line(o, p.x, p.y + crosshair_size + 1, p.x, p.y+3, 1.0f, white); - nk_stroke_line(o, p.x, p.y - crosshair_size, p.x, p.y-2, 1.0f, white);} -} -NK_LIB nk_bool -nk_do_color_picker(nk_flags *state, - struct nk_command_buffer *out, struct nk_colorf *col, - enum nk_color_format fmt, struct nk_rect bounds, - struct nk_vec2 padding, const struct nk_input *in, - const struct nk_user_font *font) -{ - int ret = 0; - struct nk_rect matrix; - struct nk_rect hue_bar; - struct nk_rect alpha_bar; - float bar_w; - - NK_ASSERT(out); - NK_ASSERT(col); - NK_ASSERT(state); - NK_ASSERT(font); - if (!out || !col || !state || !font) - return ret; - - bar_w = font->height; - bounds.x += padding.x; - bounds.y += padding.x; - bounds.w -= 2 * padding.x; - bounds.h -= 2 * padding.y; - - matrix.x = bounds.x; - matrix.y = bounds.y; - matrix.h = bounds.h; - matrix.w = bounds.w - (3 * padding.x + 2 * bar_w); - - hue_bar.w = bar_w; - hue_bar.y = bounds.y; - hue_bar.h = matrix.h; - hue_bar.x = matrix.x + matrix.w + padding.x; - - alpha_bar.x = hue_bar.x + hue_bar.w + padding.x; - alpha_bar.y = bounds.y; - alpha_bar.w = bar_w; - alpha_bar.h = matrix.h; - - ret = nk_color_picker_behavior(state, &bounds, &matrix, &hue_bar, - (fmt == NK_RGBA) ? &alpha_bar:0, col, in); - nk_draw_color_picker(out, &matrix, &hue_bar, (fmt == NK_RGBA) ? &alpha_bar:0, *col); - return ret; -} -NK_API nk_bool -nk_color_pick(struct nk_context * ctx, struct nk_colorf *color, - enum nk_color_format fmt) -{ - struct nk_window *win; - struct nk_panel *layout; - const struct nk_style *config; - const struct nk_input *in; - - enum nk_widget_layout_states state; - struct nk_rect bounds; - - NK_ASSERT(ctx); - NK_ASSERT(color); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout || !color) - return 0; - - win = ctx->current; - config = &ctx->style; - layout = win->layout; - state = nk_widget(&bounds, ctx); - if (!state) return 0; - in = (state == NK_WIDGET_ROM || layout->flags & NK_WINDOW_ROM) ? 0 : &ctx->input; - return nk_do_color_picker(&ctx->last_widget_state, &win->buffer, color, fmt, bounds, - nk_vec2(0,0), in, config->font); -} -NK_API struct nk_colorf -nk_color_picker(struct nk_context *ctx, struct nk_colorf color, - enum nk_color_format fmt) -{ - nk_color_pick(ctx, &color, fmt); - return color; -} - - - - - -/* ============================================================== - * - * COMBO - * - * ===============================================================*/ -NK_INTERN nk_bool -nk_combo_begin(struct nk_context *ctx, struct nk_window *win, - struct nk_vec2 size, nk_bool is_clicked, struct nk_rect header) -{ - struct nk_window *popup; - int is_open = 0; - int is_active = 0; - struct nk_rect body; - nk_hash hash; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return 0; - - popup = win->popup.win; - body.x = header.x; - body.w = size.x; - body.y = header.y + header.h-ctx->style.window.combo_border; - body.h = size.y; - - hash = win->popup.combo_count++; - is_open = (popup) ? nk_true:nk_false; - is_active = (popup && (win->popup.name == hash) && win->popup.type == NK_PANEL_COMBO); - if ((is_clicked && is_open && !is_active) || (is_open && !is_active) || - (!is_open && !is_active && !is_clicked)) return 0; - if (!nk_nonblock_begin(ctx, 0, body, - (is_clicked && is_open)?nk_rect(0,0,0,0):header, NK_PANEL_COMBO)) return 0; - - win->popup.type = NK_PANEL_COMBO; - win->popup.name = hash; - return 1; -} -NK_API nk_bool -nk_combo_begin_text(struct nk_context *ctx, const char *selected, int len, - struct nk_vec2 size) -{ - const struct nk_input *in; - struct nk_window *win; - struct nk_style *style; - - enum nk_widget_layout_states s; - int is_clicked = nk_false; - struct nk_rect header; - const struct nk_style_item *background; - struct nk_text text; - - NK_ASSERT(ctx); - NK_ASSERT(selected); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout || !selected) - return 0; - - win = ctx->current; - style = &ctx->style; - s = nk_widget(&header, ctx); - if (s == NK_WIDGET_INVALID) - return 0; - - in = (win->layout->flags & NK_WINDOW_ROM || s == NK_WIDGET_ROM)? 0: &ctx->input; - if (nk_button_behavior(&ctx->last_widget_state, header, in, NK_BUTTON_DEFAULT)) - is_clicked = nk_true; - - /* draw combo box header background and border */ - if (ctx->last_widget_state & NK_WIDGET_STATE_ACTIVED) { - background = &style->combo.active; - text.text = style->combo.label_active; - } else if (ctx->last_widget_state & NK_WIDGET_STATE_HOVER) { - background = &style->combo.hover; - text.text = style->combo.label_hover; - } else { - background = &style->combo.normal; - text.text = style->combo.label_normal; - } - - switch(background->type) { - case NK_STYLE_ITEM_IMAGE: - text.background = nk_rgba(0, 0, 0, 0); - nk_draw_image(&win->buffer, header, &background->data.image, nk_white); - break; - case NK_STYLE_ITEM_NINE_SLICE: - text.background = nk_rgba(0, 0, 0, 0); - nk_draw_nine_slice(&win->buffer, header, &background->data.slice, nk_white); - break; - case NK_STYLE_ITEM_COLOR: - text.background = background->data.color; - nk_fill_rect(&win->buffer, header, style->combo.rounding, background->data.color); - nk_stroke_rect(&win->buffer, header, style->combo.rounding, style->combo.border, style->combo.border_color); - break; - } - { - /* print currently selected text item */ - struct nk_rect label; - struct nk_rect button; - struct nk_rect content; - int draw_button_symbol; - - enum nk_symbol_type sym; - if (ctx->last_widget_state & NK_WIDGET_STATE_HOVER) - sym = style->combo.sym_hover; - else if (is_clicked) - sym = style->combo.sym_active; - else - sym = style->combo.sym_normal; - - /* represents whether or not the combo's button symbol should be drawn */ - draw_button_symbol = sym != NK_SYMBOL_NONE; - - /* calculate button */ - button.w = header.h - 2 * style->combo.button_padding.y; - button.x = (header.x + header.w - header.h) - style->combo.button_padding.x; - button.y = header.y + style->combo.button_padding.y; - button.h = button.w; - - content.x = button.x + style->combo.button.padding.x; - content.y = button.y + style->combo.button.padding.y; - content.w = button.w - 2 * style->combo.button.padding.x; - content.h = button.h - 2 * style->combo.button.padding.y; - - /* draw selected label */ - text.padding = nk_vec2(0,0); - label.x = header.x + style->combo.content_padding.x; - label.y = header.y + style->combo.content_padding.y; - label.h = header.h - 2 * style->combo.content_padding.y; - if (draw_button_symbol) - label.w = button.x - (style->combo.content_padding.x + style->combo.spacing.x) - label.x; - else - label.w = header.w - 2 * style->combo.content_padding.x; - nk_widget_text(&win->buffer, label, selected, len, &text, - NK_TEXT_LEFT, ctx->style.font); - - /* draw open/close button */ - if (draw_button_symbol) - nk_draw_button_symbol(&win->buffer, &button, &content, ctx->last_widget_state, - &ctx->style.combo.button, sym, style->font); - } - return nk_combo_begin(ctx, win, size, is_clicked, header); -} -NK_API nk_bool -nk_combo_begin_label(struct nk_context *ctx, const char *selected, struct nk_vec2 size) -{ - return nk_combo_begin_text(ctx, selected, nk_strlen(selected), size); -} -NK_API nk_bool -nk_combo_begin_color(struct nk_context *ctx, struct nk_color color, struct nk_vec2 size) -{ - struct nk_window *win; - struct nk_style *style; - const struct nk_input *in; - - struct nk_rect header; - int is_clicked = nk_false; - enum nk_widget_layout_states s; - const struct nk_style_item *background; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return 0; - - win = ctx->current; - style = &ctx->style; - s = nk_widget(&header, ctx); - if (s == NK_WIDGET_INVALID) - return 0; - - in = (win->layout->flags & NK_WINDOW_ROM || s == NK_WIDGET_ROM)? 0: &ctx->input; - if (nk_button_behavior(&ctx->last_widget_state, header, in, NK_BUTTON_DEFAULT)) - is_clicked = nk_true; - - /* draw combo box header background and border */ - if (ctx->last_widget_state & NK_WIDGET_STATE_ACTIVED) - background = &style->combo.active; - else if (ctx->last_widget_state & NK_WIDGET_STATE_HOVER) - background = &style->combo.hover; - else background = &style->combo.normal; - - switch(background->type) { - case NK_STYLE_ITEM_IMAGE: - nk_draw_image(&win->buffer, header, &background->data.image, nk_white); - break; - case NK_STYLE_ITEM_NINE_SLICE: - nk_draw_nine_slice(&win->buffer, header, &background->data.slice, nk_white); - break; - case NK_STYLE_ITEM_COLOR: - nk_fill_rect(&win->buffer, header, style->combo.rounding, background->data.color); - nk_stroke_rect(&win->buffer, header, style->combo.rounding, style->combo.border, style->combo.border_color); - break; - } - { - struct nk_rect content; - struct nk_rect button; - struct nk_rect bounds; - int draw_button_symbol; - - enum nk_symbol_type sym; - if (ctx->last_widget_state & NK_WIDGET_STATE_HOVER) - sym = style->combo.sym_hover; - else if (is_clicked) - sym = style->combo.sym_active; - else sym = style->combo.sym_normal; - - /* represents whether or not the combo's button symbol should be drawn */ - draw_button_symbol = sym != NK_SYMBOL_NONE; - - /* calculate button */ - button.w = header.h - 2 * style->combo.button_padding.y; - button.x = (header.x + header.w - header.h) - style->combo.button_padding.x; - button.y = header.y + style->combo.button_padding.y; - button.h = button.w; - - content.x = button.x + style->combo.button.padding.x; - content.y = button.y + style->combo.button.padding.y; - content.w = button.w - 2 * style->combo.button.padding.x; - content.h = button.h - 2 * style->combo.button.padding.y; - - /* draw color */ - bounds.h = header.h - 4 * style->combo.content_padding.y; - bounds.y = header.y + 2 * style->combo.content_padding.y; - bounds.x = header.x + 2 * style->combo.content_padding.x; - if (draw_button_symbol) - bounds.w = (button.x - (style->combo.content_padding.x + style->combo.spacing.x)) - bounds.x; - else - bounds.w = header.w - 4 * style->combo.content_padding.x; - nk_fill_rect(&win->buffer, bounds, 0, color); - - /* draw open/close button */ - if (draw_button_symbol) - nk_draw_button_symbol(&win->buffer, &button, &content, ctx->last_widget_state, - &ctx->style.combo.button, sym, style->font); - } - return nk_combo_begin(ctx, win, size, is_clicked, header); -} -NK_API nk_bool -nk_combo_begin_symbol(struct nk_context *ctx, enum nk_symbol_type symbol, struct nk_vec2 size) -{ - struct nk_window *win; - struct nk_style *style; - const struct nk_input *in; - - struct nk_rect header; - int is_clicked = nk_false; - enum nk_widget_layout_states s; - const struct nk_style_item *background; - struct nk_color sym_background; - struct nk_color symbol_color; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return 0; - - win = ctx->current; - style = &ctx->style; - s = nk_widget(&header, ctx); - if (s == NK_WIDGET_INVALID) - return 0; - - in = (win->layout->flags & NK_WINDOW_ROM || s == NK_WIDGET_ROM)? 0: &ctx->input; - if (nk_button_behavior(&ctx->last_widget_state, header, in, NK_BUTTON_DEFAULT)) - is_clicked = nk_true; - - /* draw combo box header background and border */ - if (ctx->last_widget_state & NK_WIDGET_STATE_ACTIVED) { - background = &style->combo.active; - symbol_color = style->combo.symbol_active; - } else if (ctx->last_widget_state & NK_WIDGET_STATE_HOVER) { - background = &style->combo.hover; - symbol_color = style->combo.symbol_hover; - } else { - background = &style->combo.normal; - symbol_color = style->combo.symbol_hover; - } - - switch(background->type) { - case NK_STYLE_ITEM_IMAGE: - sym_background = nk_rgba(0, 0, 0, 0); - nk_draw_image(&win->buffer, header, &background->data.image, nk_white); - break; - case NK_STYLE_ITEM_NINE_SLICE: - sym_background = nk_rgba(0, 0, 0, 0); - nk_draw_nine_slice(&win->buffer, header, &background->data.slice, nk_white); - break; - case NK_STYLE_ITEM_COLOR: - sym_background = background->data.color; - nk_fill_rect(&win->buffer, header, style->combo.rounding, background->data.color); - nk_stroke_rect(&win->buffer, header, style->combo.rounding, style->combo.border, style->combo.border_color); - break; - } - { - struct nk_rect bounds = {0,0,0,0}; - struct nk_rect content; - struct nk_rect button; - - enum nk_symbol_type sym; - if (ctx->last_widget_state & NK_WIDGET_STATE_HOVER) - sym = style->combo.sym_hover; - else if (is_clicked) - sym = style->combo.sym_active; - else sym = style->combo.sym_normal; - - /* calculate button */ - button.w = header.h - 2 * style->combo.button_padding.y; - button.x = (header.x + header.w - header.h) - style->combo.button_padding.y; - button.y = header.y + style->combo.button_padding.y; - button.h = button.w; - - content.x = button.x + style->combo.button.padding.x; - content.y = button.y + style->combo.button.padding.y; - content.w = button.w - 2 * style->combo.button.padding.x; - content.h = button.h - 2 * style->combo.button.padding.y; - - /* draw symbol */ - bounds.h = header.h - 2 * style->combo.content_padding.y; - bounds.y = header.y + style->combo.content_padding.y; - bounds.x = header.x + style->combo.content_padding.x; - bounds.w = (button.x - style->combo.content_padding.y) - bounds.x; - nk_draw_symbol(&win->buffer, symbol, bounds, sym_background, symbol_color, - 1.0f, style->font); - - /* draw open/close button */ - nk_draw_button_symbol(&win->buffer, &bounds, &content, ctx->last_widget_state, - &ctx->style.combo.button, sym, style->font); - } - return nk_combo_begin(ctx, win, size, is_clicked, header); -} -NK_API nk_bool -nk_combo_begin_symbol_text(struct nk_context *ctx, const char *selected, int len, - enum nk_symbol_type symbol, struct nk_vec2 size) -{ - struct nk_window *win; - struct nk_style *style; - struct nk_input *in; - - struct nk_rect header; - int is_clicked = nk_false; - enum nk_widget_layout_states s; - const struct nk_style_item *background; - struct nk_color symbol_color; - struct nk_text text; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return 0; - - win = ctx->current; - style = &ctx->style; - s = nk_widget(&header, ctx); - if (!s) return 0; - - in = (win->layout->flags & NK_WINDOW_ROM || s == NK_WIDGET_ROM)? 0: &ctx->input; - if (nk_button_behavior(&ctx->last_widget_state, header, in, NK_BUTTON_DEFAULT)) - is_clicked = nk_true; - - /* draw combo box header background and border */ - if (ctx->last_widget_state & NK_WIDGET_STATE_ACTIVED) { - background = &style->combo.active; - symbol_color = style->combo.symbol_active; - text.text = style->combo.label_active; - } else if (ctx->last_widget_state & NK_WIDGET_STATE_HOVER) { - background = &style->combo.hover; - symbol_color = style->combo.symbol_hover; - text.text = style->combo.label_hover; - } else { - background = &style->combo.normal; - symbol_color = style->combo.symbol_normal; - text.text = style->combo.label_normal; - } - - switch(background->type) { - case NK_STYLE_ITEM_IMAGE: - text.background = nk_rgba(0, 0, 0, 0); - nk_draw_image(&win->buffer, header, &background->data.image, nk_white); - break; - case NK_STYLE_ITEM_NINE_SLICE: - text.background = nk_rgba(0, 0, 0, 0); - nk_draw_nine_slice(&win->buffer, header, &background->data.slice, nk_white); - break; - case NK_STYLE_ITEM_COLOR: - text.background = background->data.color; - nk_fill_rect(&win->buffer, header, style->combo.rounding, background->data.color); - nk_stroke_rect(&win->buffer, header, style->combo.rounding, style->combo.border, style->combo.border_color); - break; - } - { - struct nk_rect content; - struct nk_rect button; - struct nk_rect label; - struct nk_rect image; - - enum nk_symbol_type sym; - if (ctx->last_widget_state & NK_WIDGET_STATE_HOVER) - sym = style->combo.sym_hover; - else if (is_clicked) - sym = style->combo.sym_active; - else sym = style->combo.sym_normal; - - /* calculate button */ - button.w = header.h - 2 * style->combo.button_padding.y; - button.x = (header.x + header.w - header.h) - style->combo.button_padding.x; - button.y = header.y + style->combo.button_padding.y; - button.h = button.w; - - content.x = button.x + style->combo.button.padding.x; - content.y = button.y + style->combo.button.padding.y; - content.w = button.w - 2 * style->combo.button.padding.x; - content.h = button.h - 2 * style->combo.button.padding.y; - nk_draw_button_symbol(&win->buffer, &button, &content, ctx->last_widget_state, - &ctx->style.combo.button, sym, style->font); - - /* draw symbol */ - image.x = header.x + style->combo.content_padding.x; - image.y = header.y + style->combo.content_padding.y; - image.h = header.h - 2 * style->combo.content_padding.y; - image.w = image.h; - nk_draw_symbol(&win->buffer, symbol, image, text.background, symbol_color, - 1.0f, style->font); - - /* draw label */ - text.padding = nk_vec2(0,0); - label.x = image.x + image.w + style->combo.spacing.x + style->combo.content_padding.x; - label.y = header.y + style->combo.content_padding.y; - label.w = (button.x - style->combo.content_padding.x) - label.x; - label.h = header.h - 2 * style->combo.content_padding.y; - nk_widget_text(&win->buffer, label, selected, len, &text, NK_TEXT_LEFT, style->font); - } - return nk_combo_begin(ctx, win, size, is_clicked, header); -} -NK_API nk_bool -nk_combo_begin_image(struct nk_context *ctx, struct nk_image img, struct nk_vec2 size) -{ - struct nk_window *win; - struct nk_style *style; - const struct nk_input *in; - - struct nk_rect header; - int is_clicked = nk_false; - enum nk_widget_layout_states s; - const struct nk_style_item *background; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return 0; - - win = ctx->current; - style = &ctx->style; - s = nk_widget(&header, ctx); - if (s == NK_WIDGET_INVALID) - return 0; - - in = (win->layout->flags & NK_WINDOW_ROM || s == NK_WIDGET_ROM)? 0: &ctx->input; - if (nk_button_behavior(&ctx->last_widget_state, header, in, NK_BUTTON_DEFAULT)) - is_clicked = nk_true; - - /* draw combo box header background and border */ - if (ctx->last_widget_state & NK_WIDGET_STATE_ACTIVED) - background = &style->combo.active; - else if (ctx->last_widget_state & NK_WIDGET_STATE_HOVER) - background = &style->combo.hover; - else background = &style->combo.normal; - - switch (background->type) { - case NK_STYLE_ITEM_IMAGE: - nk_draw_image(&win->buffer, header, &background->data.image, nk_white); - break; - case NK_STYLE_ITEM_NINE_SLICE: - nk_draw_nine_slice(&win->buffer, header, &background->data.slice, nk_white); - break; - case NK_STYLE_ITEM_COLOR: - nk_fill_rect(&win->buffer, header, style->combo.rounding, background->data.color); - nk_stroke_rect(&win->buffer, header, style->combo.rounding, style->combo.border, style->combo.border_color); - break; - } - { - struct nk_rect bounds = {0,0,0,0}; - struct nk_rect content; - struct nk_rect button; - int draw_button_symbol; - - enum nk_symbol_type sym; - if (ctx->last_widget_state & NK_WIDGET_STATE_HOVER) - sym = style->combo.sym_hover; - else if (is_clicked) - sym = style->combo.sym_active; - else sym = style->combo.sym_normal; - - /* represents whether or not the combo's button symbol should be drawn */ - draw_button_symbol = sym != NK_SYMBOL_NONE; - - /* calculate button */ - button.w = header.h - 2 * style->combo.button_padding.y; - button.x = (header.x + header.w - header.h) - style->combo.button_padding.y; - button.y = header.y + style->combo.button_padding.y; - button.h = button.w; - - content.x = button.x + style->combo.button.padding.x; - content.y = button.y + style->combo.button.padding.y; - content.w = button.w - 2 * style->combo.button.padding.x; - content.h = button.h - 2 * style->combo.button.padding.y; - - /* draw image */ - bounds.h = header.h - 2 * style->combo.content_padding.y; - bounds.y = header.y + style->combo.content_padding.y; - bounds.x = header.x + style->combo.content_padding.x; - if (draw_button_symbol) - bounds.w = (button.x - style->combo.content_padding.y) - bounds.x; - else - bounds.w = header.w - 2 * style->combo.content_padding.x; - nk_draw_image(&win->buffer, bounds, &img, nk_white); - - /* draw open/close button */ - if (draw_button_symbol) - nk_draw_button_symbol(&win->buffer, &bounds, &content, ctx->last_widget_state, - &ctx->style.combo.button, sym, style->font); - } - return nk_combo_begin(ctx, win, size, is_clicked, header); -} -NK_API nk_bool -nk_combo_begin_image_text(struct nk_context *ctx, const char *selected, int len, - struct nk_image img, struct nk_vec2 size) -{ - struct nk_window *win; - struct nk_style *style; - struct nk_input *in; - - struct nk_rect header; - int is_clicked = nk_false; - enum nk_widget_layout_states s; - const struct nk_style_item *background; - struct nk_text text; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return 0; - - win = ctx->current; - style = &ctx->style; - s = nk_widget(&header, ctx); - if (!s) return 0; - - in = (win->layout->flags & NK_WINDOW_ROM || s == NK_WIDGET_ROM)? 0: &ctx->input; - if (nk_button_behavior(&ctx->last_widget_state, header, in, NK_BUTTON_DEFAULT)) - is_clicked = nk_true; - - /* draw combo box header background and border */ - if (ctx->last_widget_state & NK_WIDGET_STATE_ACTIVED) { - background = &style->combo.active; - text.text = style->combo.label_active; - } else if (ctx->last_widget_state & NK_WIDGET_STATE_HOVER) { - background = &style->combo.hover; - text.text = style->combo.label_hover; - } else { - background = &style->combo.normal; - text.text = style->combo.label_normal; - } - - switch(background->type) { - case NK_STYLE_ITEM_IMAGE: - text.background = nk_rgba(0, 0, 0, 0); - nk_draw_image(&win->buffer, header, &background->data.image, nk_white); - break; - case NK_STYLE_ITEM_NINE_SLICE: - text.background = nk_rgba(0, 0, 0, 0); - nk_draw_nine_slice(&win->buffer, header, &background->data.slice, nk_white); - break; - case NK_STYLE_ITEM_COLOR: - text.background = background->data.color; - nk_fill_rect(&win->buffer, header, style->combo.rounding, background->data.color); - nk_stroke_rect(&win->buffer, header, style->combo.rounding, style->combo.border, style->combo.border_color); - break; - } - { - struct nk_rect content; - struct nk_rect button; - struct nk_rect label; - struct nk_rect image; - int draw_button_symbol; - - enum nk_symbol_type sym; - if (ctx->last_widget_state & NK_WIDGET_STATE_HOVER) - sym = style->combo.sym_hover; - else if (is_clicked) - sym = style->combo.sym_active; - else sym = style->combo.sym_normal; - - /* represents whether or not the combo's button symbol should be drawn */ - draw_button_symbol = sym != NK_SYMBOL_NONE; - - /* calculate button */ - button.w = header.h - 2 * style->combo.button_padding.y; - button.x = (header.x + header.w - header.h) - style->combo.button_padding.x; - button.y = header.y + style->combo.button_padding.y; - button.h = button.w; - - content.x = button.x + style->combo.button.padding.x; - content.y = button.y + style->combo.button.padding.y; - content.w = button.w - 2 * style->combo.button.padding.x; - content.h = button.h - 2 * style->combo.button.padding.y; - if (draw_button_symbol) - nk_draw_button_symbol(&win->buffer, &button, &content, ctx->last_widget_state, - &ctx->style.combo.button, sym, style->font); - - /* draw image */ - image.x = header.x + style->combo.content_padding.x; - image.y = header.y + style->combo.content_padding.y; - image.h = header.h - 2 * style->combo.content_padding.y; - image.w = image.h; - nk_draw_image(&win->buffer, image, &img, nk_white); - - /* draw label */ - text.padding = nk_vec2(0,0); - label.x = image.x + image.w + style->combo.spacing.x + style->combo.content_padding.x; - label.y = header.y + style->combo.content_padding.y; - label.h = header.h - 2 * style->combo.content_padding.y; - if (draw_button_symbol) - label.w = (button.x - style->combo.content_padding.x) - label.x; - else - label.w = (header.x + header.w - style->combo.content_padding.x) - label.x; - nk_widget_text(&win->buffer, label, selected, len, &text, NK_TEXT_LEFT, style->font); - } - return nk_combo_begin(ctx, win, size, is_clicked, header); -} -NK_API nk_bool -nk_combo_begin_symbol_label(struct nk_context *ctx, - const char *selected, enum nk_symbol_type type, struct nk_vec2 size) -{ - return nk_combo_begin_symbol_text(ctx, selected, nk_strlen(selected), type, size); -} -NK_API nk_bool -nk_combo_begin_image_label(struct nk_context *ctx, - const char *selected, struct nk_image img, struct nk_vec2 size) -{ - return nk_combo_begin_image_text(ctx, selected, nk_strlen(selected), img, size); -} -NK_API nk_bool -nk_combo_item_text(struct nk_context *ctx, const char *text, int len,nk_flags align) -{ - return nk_contextual_item_text(ctx, text, len, align); -} -NK_API nk_bool -nk_combo_item_label(struct nk_context *ctx, const char *label, nk_flags align) -{ - return nk_contextual_item_label(ctx, label, align); -} -NK_API nk_bool -nk_combo_item_image_text(struct nk_context *ctx, struct nk_image img, const char *text, - int len, nk_flags alignment) -{ - return nk_contextual_item_image_text(ctx, img, text, len, alignment); -} -NK_API nk_bool -nk_combo_item_image_label(struct nk_context *ctx, struct nk_image img, - const char *text, nk_flags alignment) -{ - return nk_contextual_item_image_label(ctx, img, text, alignment); -} -NK_API nk_bool -nk_combo_item_symbol_text(struct nk_context *ctx, enum nk_symbol_type sym, - const char *text, int len, nk_flags alignment) -{ - return nk_contextual_item_symbol_text(ctx, sym, text, len, alignment); -} -NK_API nk_bool -nk_combo_item_symbol_label(struct nk_context *ctx, enum nk_symbol_type sym, - const char *label, nk_flags alignment) -{ - return nk_contextual_item_symbol_label(ctx, sym, label, alignment); -} -NK_API void nk_combo_end(struct nk_context *ctx) -{ - nk_contextual_end(ctx); -} -NK_API void nk_combo_close(struct nk_context *ctx) -{ - nk_contextual_close(ctx); -} -NK_API int -nk_combo(struct nk_context *ctx, const char **items, int count, - int selected, int item_height, struct nk_vec2 size) -{ - int i = 0; - int max_height; - struct nk_vec2 item_spacing; - struct nk_vec2 window_padding; - - NK_ASSERT(ctx); - NK_ASSERT(items); - NK_ASSERT(ctx->current); - if (!ctx || !items ||!count) - return selected; - - item_spacing = ctx->style.window.spacing; - window_padding = nk_panel_get_padding(&ctx->style, ctx->current->layout->type); - max_height = count * item_height + count * (int)item_spacing.y; - max_height += (int)item_spacing.y * 2 + (int)window_padding.y * 2; - size.y = NK_MIN(size.y, (float)max_height); - if (nk_combo_begin_label(ctx, items[selected], size)) { - nk_layout_row_dynamic(ctx, (float)item_height, 1); - for (i = 0; i < count; ++i) { - if (nk_combo_item_label(ctx, items[i], NK_TEXT_LEFT)) - selected = i; - } - nk_combo_end(ctx); - } - return selected; -} -NK_API int -nk_combo_separator(struct nk_context *ctx, const char *items_separated_by_separator, - int separator, int selected, int count, int item_height, struct nk_vec2 size) -{ - int i; - int max_height; - struct nk_vec2 item_spacing; - struct nk_vec2 window_padding; - const char *current_item; - const char *iter; - int length = 0; - - NK_ASSERT(ctx); - NK_ASSERT(items_separated_by_separator); - if (!ctx || !items_separated_by_separator) - return selected; - - /* calculate popup window */ - item_spacing = ctx->style.window.spacing; - window_padding = nk_panel_get_padding(&ctx->style, ctx->current->layout->type); - max_height = count * item_height + count * (int)item_spacing.y; - max_height += (int)item_spacing.y * 2 + (int)window_padding.y * 2; - size.y = NK_MIN(size.y, (float)max_height); - - /* find selected item */ - current_item = items_separated_by_separator; - for (i = 0; i < count; ++i) { - iter = current_item; - while (*iter && *iter != separator) iter++; - length = (int)(iter - current_item); - if (i == selected) break; - current_item = iter + 1; - } - - if (nk_combo_begin_text(ctx, current_item, length, size)) { - current_item = items_separated_by_separator; - nk_layout_row_dynamic(ctx, (float)item_height, 1); - for (i = 0; i < count; ++i) { - iter = current_item; - while (*iter && *iter != separator) iter++; - length = (int)(iter - current_item); - if (nk_combo_item_text(ctx, current_item, length, NK_TEXT_LEFT)) - selected = i; - current_item = current_item + length + 1; - } - nk_combo_end(ctx); - } - return selected; -} -NK_API int -nk_combo_string(struct nk_context *ctx, const char *items_separated_by_zeros, - int selected, int count, int item_height, struct nk_vec2 size) -{ - return nk_combo_separator(ctx, items_separated_by_zeros, '\0', selected, count, item_height, size); -} -NK_API int -nk_combo_callback(struct nk_context *ctx, void(*item_getter)(void*, int, const char**), - void *userdata, int selected, int count, int item_height, struct nk_vec2 size) -{ - int i; - int max_height; - struct nk_vec2 item_spacing; - struct nk_vec2 window_padding; - const char *item; - - NK_ASSERT(ctx); - NK_ASSERT(item_getter); - if (!ctx || !item_getter) - return selected; - - /* calculate popup window */ - item_spacing = ctx->style.window.spacing; - window_padding = nk_panel_get_padding(&ctx->style, ctx->current->layout->type); - max_height = count * item_height + count * (int)item_spacing.y; - max_height += (int)item_spacing.y * 2 + (int)window_padding.y * 2; - size.y = NK_MIN(size.y, (float)max_height); - - item_getter(userdata, selected, &item); - if (nk_combo_begin_label(ctx, item, size)) { - nk_layout_row_dynamic(ctx, (float)item_height, 1); - for (i = 0; i < count; ++i) { - item_getter(userdata, i, &item); - if (nk_combo_item_label(ctx, item, NK_TEXT_LEFT)) - selected = i; - } - nk_combo_end(ctx); - } return selected; -} -NK_API void -nk_combobox(struct nk_context *ctx, const char **items, int count, - int *selected, int item_height, struct nk_vec2 size) -{ - *selected = nk_combo(ctx, items, count, *selected, item_height, size); -} -NK_API void -nk_combobox_string(struct nk_context *ctx, const char *items_separated_by_zeros, - int *selected, int count, int item_height, struct nk_vec2 size) -{ - *selected = nk_combo_string(ctx, items_separated_by_zeros, *selected, count, item_height, size); -} -NK_API void -nk_combobox_separator(struct nk_context *ctx, const char *items_separated_by_separator, - int separator, int *selected, int count, int item_height, struct nk_vec2 size) -{ - *selected = nk_combo_separator(ctx, items_separated_by_separator, separator, - *selected, count, item_height, size); -} -NK_API void -nk_combobox_callback(struct nk_context *ctx, - void(*item_getter)(void* data, int id, const char **out_text), - void *userdata, int *selected, int count, int item_height, struct nk_vec2 size) -{ - *selected = nk_combo_callback(ctx, item_getter, userdata, *selected, count, item_height, size); -} - - - - - -/* =============================================================== - * - * TOOLTIP - * - * ===============================================================*/ -NK_API nk_bool -nk_tooltip_begin(struct nk_context *ctx, float width) -{ - int x,y,w,h; - struct nk_window *win; - const struct nk_input *in; - struct nk_rect bounds; - int ret; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - if (!ctx || !ctx->current || !ctx->current->layout) - return 0; - - /* make sure that no nonblocking popup is currently active */ - win = ctx->current; - in = &ctx->input; - if (win->popup.win && (win->popup.type & NK_PANEL_SET_NONBLOCK)) - return 0; - - w = nk_iceilf(width); - h = nk_iceilf(nk_null_rect.h); - x = nk_ifloorf(in->mouse.pos.x + 1) - (int)win->layout->clip.x; - y = nk_ifloorf(in->mouse.pos.y + 1) - (int)win->layout->clip.y; - - bounds.x = (float)x; - bounds.y = (float)y; - bounds.w = (float)w; - bounds.h = (float)h; - - ret = nk_popup_begin(ctx, NK_POPUP_DYNAMIC, - "__##Tooltip##__", NK_WINDOW_NO_SCROLLBAR|NK_WINDOW_BORDER, bounds); - if (ret) win->layout->flags &= ~(nk_flags)NK_WINDOW_ROM; - win->popup.type = NK_PANEL_TOOLTIP; - ctx->current->layout->type = NK_PANEL_TOOLTIP; - return ret; -} - -NK_API void -nk_tooltip_end(struct nk_context *ctx) -{ - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - if (!ctx || !ctx->current) return; - ctx->current->seq--; - nk_popup_close(ctx); - nk_popup_end(ctx); -} -NK_API void -nk_tooltip(struct nk_context *ctx, const char *text) -{ - const struct nk_style *style; - struct nk_vec2 padding; - - int text_len; - float text_width; - float text_height; - - NK_ASSERT(ctx); - NK_ASSERT(ctx->current); - NK_ASSERT(ctx->current->layout); - NK_ASSERT(text); - if (!ctx || !ctx->current || !ctx->current->layout || !text) - return; - - /* fetch configuration data */ - style = &ctx->style; - padding = style->window.padding; - - /* calculate size of the text and tooltip */ - text_len = nk_strlen(text); - text_width = style->font->width(style->font->userdata, - style->font->height, text, text_len); - text_width += (4 * padding.x); - text_height = (style->font->height + 2 * padding.y); - - /* execute tooltip and fill with text */ - if (nk_tooltip_begin(ctx, (float)text_width)) { - nk_layout_row_dynamic(ctx, (float)text_height, 1); - nk_text(ctx, text, text_len, NK_TEXT_LEFT); - nk_tooltip_end(ctx); - } -} -#ifdef NK_INCLUDE_STANDARD_VARARGS -NK_API void -nk_tooltipf(struct nk_context *ctx, const char *fmt, ...) -{ - va_list args; - va_start(args, fmt); - nk_tooltipfv(ctx, fmt, args); - va_end(args); -} -NK_API void -nk_tooltipfv(struct nk_context *ctx, const char *fmt, va_list args) -{ - char buf[256]; - nk_strfmt(buf, NK_LEN(buf), fmt, args); - nk_tooltip(ctx, buf); -} -#endif - - - -#endif /* NK_IMPLEMENTATION */ - -/* -/// ## License -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~none -/// ------------------------------------------------------------------------------ -/// This software is available under 2 licenses -- choose whichever you prefer. -/// ------------------------------------------------------------------------------ -/// ALTERNATIVE A - MIT License -/// Copyright (c) 2016-2018 Micha Mettke -/// 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. -/// ------------------------------------------------------------------------------ -/// ALTERNATIVE B - Public Domain (www.unlicense.org) -/// This is free and unencumbered software released into the public domain. -/// Anyone is free to copy, modify, publish, use, compile, sell, or distribute this -/// software, either in source code form or as a compiled binary, for any purpose, -/// commercial or non-commercial, and by any means. -/// In jurisdictions that recognize copyright laws, the author or authors of this -/// software dedicate any and all copyright interest in the software to the public -/// domain. We make this dedication for the benefit of the public at large and to -/// the detriment of our heirs and successors. We intend this dedication to be an -/// overt act of relinquishment in perpetuity of all present and future rights to -/// this software under copyright law. -/// 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 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. -/// ------------------------------------------------------------------------------ -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ - -/// ## Changelog -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~none -/// [date] ([x.y.z]) - [description] -/// - [date]: date on which the change has been pushed -/// - [x.y.z]: Version string, represented in Semantic Versioning format -/// - [x]: Major version with API and library breaking changes -/// - [y]: Minor version with non-breaking API and library changes -/// - [z]: Patch version with no direct changes to the API -/// -/// - 2022/12/23 (4.10.6) - Fix incorrect glyph index in nk_font_bake() -/// - 2022/12/17 (4.10.5) - Fix nk_font_bake_pack() using TTC font offset incorrectly -/// - 2022/10/24 (4.10.4) - Fix nk_str_{append,insert}_str_utf8 always returning 0 -/// - 2022/09/03 (4.10.3) - Renamed the `null` texture variable to `tex_null` -/// - 2022/08/01 (4.10.2) - Fix Apple Silicon with incorrect NK_SITE_TYPE and NK_POINTER_TYPE -/// - 2022/08/01 (4.10.1) - Fix cursor jumping back to beginning of text when typing more than -/// nk_edit_xxx limit -/// - 2022/05/27 (4.10.0) - Add nk_input_has_mouse_click_in_button_rect() to fix window move bug -/// - 2022/04/18 (4.9.7) - Change button behavior when NK_BUTTON_TRIGGER_ON_RELEASE is defined to -/// only trigger when the mouse position was inside the same button on down -/// - 2022/02/03 (4.9.6) - Allow overriding the NK_INV_SQRT function, similar to NK_SIN and NK_COS -/// - 2021/12/22 (4.9.5) - Revert layout bounds not accounting for padding due to regressions -/// - 2021/12/22 (4.9.4) - Fix checking hovering when window is minimized -/// - 2021/12/22 (4.09.3) - Fix layout bounds not accounting for padding -/// - 2021/12/19 (4.09.2) - Update to stb_rect_pack.h v1.01 and stb_truetype.h v1.26 -/// - 2021/12/16 (4.09.1) - Fix the majority of GCC warnings -/// - 2021/10/16 (4.09.0) - Added nk_spacer() widget -/// - 2021/09/22 (4.08.6) - Fix "may be used uninitialized" warnings in nk_widget -/// - 2021/09/22 (4.08.5) - GCC __builtin_offsetof only exists in version 4 and later -/// - 2021/09/15 (4.08.4) - Fix "'num_len' may be used uninitialized" in nk_do_property -/// - 2021/09/15 (4.08.3) - Fix "Templates cannot be declared to have 'C' Linkage" -/// - 2021/09/08 (4.08.2) - Fix warnings in C89 builds -/// - 2021/09/08 (4.08.1) - Use compiler builtins for NK_OFFSETOF when possible -/// - 2021/08/17 (4.08.0) - Implemented 9-slice scaling support for widget styles -/// - 2021/08/16 (4.07.5) - Replace usage of memset in nk_font_atlas_bake with NK_MEMSET -/// - 2021/08/15 (4.07.4) - Fix conversion and sign conversion warnings -/// - 2021/08/08 (4.07.3) - Fix crash when baking merged fonts -/// - 2021/08/08 (4.07.2) - Fix Multiline Edit wrong offset -/// - 2021/03/17 (4.07.1) - Fix warning about unused parameter -/// - 2021/03/17 (4.07.0) - Fix nk_property hover bug -/// - 2021/03/15 (4.06.4) - Change nk_propertyi back to int -/// - 2021/03/15 (4.06.3) - Update documentation for functions that now return nk_bool -/// - 2020/12/19 (4.06.2) - Fix additional C++ style comments which are not allowed in ISO C90. -/// - 2020/10/11 (4.06.1) - Fix C++ style comments which are not allowed in ISO C90. -/// - 2020/10/07 (4.06.0) - Fix nk_combo return type wrongly changed to nk_bool -/// - 2020/09/05 (4.05.0) - Use the nk_font_atlas allocator for stb_truetype memory management. -/// - 2020/09/04 (4.04.1) - Replace every boolean int by nk_bool -/// - 2020/09/04 (4.04.0) - Add nk_bool with NK_INCLUDE_STANDARD_BOOL -/// - 2020/06/13 (4.03.1) - Fix nk_pool allocation sizes. -/// - 2020/06/04 (4.03.0) - Made nk_combo header symbols optional. -/// - 2020/05/27 (4.02.5) - Fix nk_do_edit: Keep scroll position when re-activating edit widget. -/// - 2020/05/09 (4.02.4) - Fix nk_menubar height calculation bug -/// - 2020/05/08 (4.02.3) - Fix missing stdarg.h with NK_INCLUDE_STANDARD_VARARGS -/// - 2020/04/30 (4.02.2) - Fix nk_edit border drawing bug -/// - 2020/04/09 (4.02.1) - Removed unused nk_sqrt function to fix compiler warnings -/// - Fixed compiler warnings if you bring your own methods for -/// nk_cos/nk_sin/nk_strtod/nk_memset/nk_memcopy/nk_dtoa -/// - 2020/04/06 (4.01.10) - Fix bug: Do not use pool before checking for NULL -/// - 2020/03/22 (4.01.9) - Fix bug where layout state wasn't restored correctly after -/// popping a tree. -/// - 2020/03/11 (4.01.8) - Fix bug where padding is subtracted from widget -/// - 2020/03/06 (4.01.7) - Fix bug where width padding was applied twice -/// - 2020/02/06 (4.01.6) - Update stb_truetype.h and stb_rect_pack.h and separate them -/// - 2019/12/10 (4.01.5) - Fix off-by-one error in NK_INTERSECT -/// - 2019/10/09 (4.01.4) - Fix bug for autoscrolling in nk_do_edit -/// - 2019/09/20 (4.01.3) - Fixed a bug wherein combobox cannot be closed by clicking the header -/// when NK_BUTTON_TRIGGER_ON_RELEASE is defined. -/// - 2019/09/10 (4.01.2) - Fixed the nk_cos function, which deviated significantly. -/// - 2019/09/08 (4.01.1) - Fixed a bug wherein re-baking of fonts caused a segmentation -/// fault due to dst_font->glyph_count not being zeroed on subsequent -/// bakes of the same set of fonts. -/// - 2019/06/23 (4.01.0) - Added nk_***_get_scroll and nk_***_set_scroll for groups, windows, and popups. -/// - 2019/06/12 (4.00.3) - Fix panel background drawing bug. -/// - 2018/10/31 (4.00.2) - Added NK_KEYSTATE_BASED_INPUT to "fix" state based backends -/// like GLFW without breaking key repeat behavior on event based. -/// - 2018/04/01 (4.00.1) - Fixed calling `nk_convert` multiple time per single frame. -/// - 2018/04/01 (4.00.0) - BREAKING CHANGE: nk_draw_list_clear no longer tries to -/// clear provided buffers. So make sure to either free -/// or clear each passed buffer after calling nk_convert. -/// - 2018/02/23 (3.00.6) - Fixed slider dragging behavior. -/// - 2018/01/31 (3.00.5) - Fixed overcalculation of cursor data in font baking process. -/// - 2018/01/31 (3.00.4) - Removed name collision with stb_truetype. -/// - 2018/01/28 (3.00.3) - Fixed panel window border drawing bug. -/// - 2018/01/12 (3.00.2) - Added `nk_group_begin_titled` for separated group identifier and title. -/// - 2018/01/07 (3.00.1) - Started to change documentation style. -/// - 2018/01/05 (3.00.0) - BREAKING CHANGE: The previous color picker API was broken -/// because of conversions between float and byte color representation. -/// Color pickers now use floating point values to represent -/// HSV values. To get back the old behavior I added some additional -/// color conversion functions to cast between nk_color and -/// nk_colorf. -/// - 2017/12/23 (2.00.7) - Fixed small warning. -/// - 2017/12/23 (2.00.7) - Fixed `nk_edit_buffer` behavior if activated to allow input. -/// - 2017/12/23 (2.00.7) - Fixed modifyable progressbar dragging visuals and input behavior. -/// - 2017/12/04 (2.00.6) - Added formatted string tooltip widget. -/// - 2017/11/18 (2.00.5) - Fixed window becoming hidden with flag `NK_WINDOW_NO_INPUT`. -/// - 2017/11/15 (2.00.4) - Fixed font merging. -/// - 2017/11/07 (2.00.3) - Fixed window size and position modifier functions. -/// - 2017/09/14 (2.00.2) - Fixed `nk_edit_buffer` and `nk_edit_focus` behavior. -/// - 2017/09/14 (2.00.1) - Fixed window closing behavior. -/// - 2017/09/14 (2.00.0) - BREAKING CHANGE: Modifying window position and size functions now -/// require the name of the window and must happen outside the window -/// building process (between function call nk_begin and nk_end). -/// - 2017/09/11 (1.40.9) - Fixed window background flag if background window is declared last. -/// - 2017/08/27 (1.40.8) - Fixed `nk_item_is_any_active` for hidden windows. -/// - 2017/08/27 (1.40.7) - Fixed window background flag. -/// - 2017/07/07 (1.40.6) - Fixed missing clipping rect check for hovering/clicked -/// query for widgets. -/// - 2017/07/07 (1.40.5) - Fixed drawing bug for vertex output for lines and stroked -/// and filled rectangles. -/// - 2017/07/07 (1.40.4) - Fixed bug in nk_convert trying to add windows that are in -/// process of being destroyed. -/// - 2017/07/07 (1.40.3) - Fixed table internal bug caused by storing table size in -/// window instead of directly in table. -/// - 2017/06/30 (1.40.2) - Removed unneeded semicolon in C++ NK_ALIGNOF macro. -/// - 2017/06/30 (1.40.1) - Fixed drawing lines smaller or equal zero. -/// - 2017/06/08 (1.40.0) - Removed the breaking part of last commit. Auto layout now only -/// comes in effect if you pass in zero was row height argument. -/// - 2017/06/08 (1.40.0) - BREAKING CHANGE: while not directly API breaking it will change -/// how layouting works. From now there will be an internal minimum -/// row height derived from font height. If you need a row smaller than -/// that you can directly set it by `nk_layout_set_min_row_height` and -/// reset the value back by calling `nk_layout_reset_min_row_height. -/// - 2017/06/08 (1.39.1) - Fixed property text edit handling bug caused by past `nk_widget` fix. -/// - 2017/06/08 (1.39.0) - Added function to retrieve window space without calling a `nk_layout_xxx` function. -/// - 2017/06/06 (1.38.5) - Fixed `nk_convert` return flag for command buffer. -/// - 2017/05/23 (1.38.4) - Fixed activation behavior for widgets partially clipped. -/// - 2017/05/10 (1.38.3) - Fixed wrong min window size mouse scaling over boundaries. -/// - 2017/05/09 (1.38.2) - Fixed vertical scrollbar drawing with not enough space. -/// - 2017/05/09 (1.38.1) - Fixed scaler dragging behavior if window size hits minimum size. -/// - 2017/05/06 (1.38.0) - Added platform double-click support. -/// - 2017/04/20 (1.37.1) - Fixed key repeat found inside glfw demo backends. -/// - 2017/04/20 (1.37.0) - Extended properties with selection and clipboard support. -/// - 2017/04/20 (1.36.2) - Fixed #405 overlapping rows with zero padding and spacing. -/// - 2017/04/09 (1.36.1) - Fixed #403 with another widget float error. -/// - 2017/04/09 (1.36.0) - Added window `NK_WINDOW_NO_INPUT` and `NK_WINDOW_NOT_INTERACTIVE` flags. -/// - 2017/04/09 (1.35.3) - Fixed buffer heap corruption. -/// - 2017/03/25 (1.35.2) - Fixed popup overlapping for `NK_WINDOW_BACKGROUND` windows. -/// - 2017/03/25 (1.35.1) - Fixed windows closing behavior. -/// - 2017/03/18 (1.35.0) - Added horizontal scroll requested in #377. -/// - 2017/03/18 (1.34.3) - Fixed long window header titles. -/// - 2017/03/04 (1.34.2) - Fixed text edit filtering. -/// - 2017/03/04 (1.34.1) - Fixed group closable flag. -/// - 2017/02/25 (1.34.0) - Added custom draw command for better language binding support. -/// - 2017/01/24 (1.33.0) - Added programmatic way to remove edit focus. -/// - 2017/01/24 (1.32.3) - Fixed wrong define for basic type definitions for windows. -/// - 2017/01/21 (1.32.2) - Fixed input capture from hidden or closed windows. -/// - 2017/01/21 (1.32.1) - Fixed slider behavior and drawing. -/// - 2017/01/13 (1.32.0) - Added flag to put scaler into the bottom left corner. -/// - 2017/01/13 (1.31.0) - Added additional row layouting method to combine both -/// dynamic and static widgets. -/// - 2016/12/31 (1.30.0) - Extended scrollbar offset from 16-bit to 32-bit. -/// - 2016/12/31 (1.29.2) - Fixed closing window bug of minimized windows. -/// - 2016/12/03 (1.29.1) - Fixed wrapped text with no seperator and C89 error. -/// - 2016/12/03 (1.29.0) - Changed text wrapping to process words not characters. -/// - 2016/11/22 (1.28.6) - Fixed window minimized closing bug. -/// - 2016/11/19 (1.28.5) - Fixed abstract combo box closing behavior. -/// - 2016/11/19 (1.28.4) - Fixed tooltip flickering. -/// - 2016/11/19 (1.28.3) - Fixed memory leak caused by popup repeated closing. -/// - 2016/11/18 (1.28.2) - Fixed memory leak caused by popup panel allocation. -/// - 2016/11/10 (1.28.1) - Fixed some warnings and C++ error. -/// - 2016/11/10 (1.28.0) - Added additional `nk_button` versions which allows to directly -/// pass in a style struct to change buttons visual. -/// - 2016/11/10 (1.27.0) - Added additional `nk_tree` versions to support external state -/// storage. Just like last the `nk_group` commit the main -/// advantage is that you optionally can minimize nuklears runtime -/// memory consumption or handle hash collisions. -/// - 2016/11/09 (1.26.0) - Added additional `nk_group` version to support external scrollbar -/// offset storage. Main advantage is that you can externalize -/// the memory management for the offset. It could also be helpful -/// if you have a hash collision in `nk_group_begin` but really -/// want the name. In addition I added `nk_list_view` which allows -/// to draw big lists inside a group without actually having to -/// commit the whole list to nuklear (issue #269). -/// - 2016/10/30 (1.25.1) - Fixed clipping rectangle bug inside `nk_draw_list`. -/// - 2016/10/29 (1.25.0) - Pulled `nk_panel` memory management into nuklear and out of -/// the hands of the user. From now on users don't have to care -/// about panels unless they care about some information. If you -/// still need the panel just call `nk_window_get_panel`. -/// - 2016/10/21 (1.24.0) - Changed widget border drawing to stroked rectangle from filled -/// rectangle for less overdraw and widget background transparency. -/// - 2016/10/18 (1.23.0) - Added `nk_edit_focus` for manually edit widget focus control. -/// - 2016/09/29 (1.22.7) - Fixed deduction of basic type in non `` compilation. -/// - 2016/09/29 (1.22.6) - Fixed edit widget UTF-8 text cursor drawing bug. -/// - 2016/09/28 (1.22.5) - Fixed edit widget UTF-8 text appending/inserting/removing. -/// - 2016/09/28 (1.22.4) - Fixed drawing bug inside edit widgets which offset all text -/// text in every edit widget if one of them is scrolled. -/// - 2016/09/28 (1.22.3) - Fixed small bug in edit widgets if not active. The wrong -/// text length is passed. It should have been in bytes but -/// was passed as glyphs. -/// - 2016/09/20 (1.22.2) - Fixed color button size calculation. -/// - 2016/09/20 (1.22.1) - Fixed some `nk_vsnprintf` behavior bugs and removed `` -/// again from `NK_INCLUDE_STANDARD_VARARGS`. -/// - 2016/09/18 (1.22.0) - C89 does not support vsnprintf only C99 and newer as well -/// as C++11 and newer. In addition to use vsnprintf you have -/// to include . So just defining `NK_INCLUDE_STD_VAR_ARGS` -/// is not enough. That behavior is now fixed. By default if -/// both varargs as well as stdio is selected I try to use -/// vsnprintf if not possible I will revert to vsprintf. If -/// varargs but not stdio was defined I will use my own function. -/// - 2016/09/15 (1.21.2) - Fixed panel `close` behavior for deeper panel levels. -/// - 2016/09/15 (1.21.1) - Fixed C++ errors and wrong argument to `nk_panel_get_xxxx`. -/// - 2016/09/13 (1.21.0) - !BREAKING! Fixed nonblocking popup behavior in menu, combo, -/// and contextual which prevented closing in y-direction if -/// popup did not reach max height. -/// In addition the height parameter was changed into vec2 -/// for width and height to have more control over the popup size. -/// - 2016/09/13 (1.20.3) - Cleaned up and extended type selection. -/// - 2016/09/13 (1.20.2) - Fixed slider behavior hopefully for the last time. This time -/// all calculation are correct so no more hackery. -/// - 2016/09/13 (1.20.1) - Internal change to divide window/panel flags into panel flags and types. -/// Suprisinly spend years in C and still happened to confuse types -/// with flags. Probably something to take note. -/// - 2016/09/08 (1.20.0) - Added additional helper function to make it easier to just -/// take the produced buffers from `nk_convert` and unplug the -/// iteration process from `nk_context`. So now you can -/// just use the vertex,element and command buffer + two pointer -/// inside the command buffer retrieved by calls `nk__draw_begin` -/// and `nk__draw_end` and macro `nk_draw_foreach_bounded`. -/// - 2016/09/08 (1.19.0) - Added additional asserts to make sure every `nk_xxx_begin` call -/// for windows, popups, combobox, menu and contextual is guarded by -/// `if` condition and does not produce false drawing output. -/// - 2016/09/08 (1.18.0) - Changed confusing name for `NK_SYMBOL_RECT_FILLED`, `NK_SYMBOL_RECT` -/// to hopefully easier to understand `NK_SYMBOL_RECT_FILLED` and -/// `NK_SYMBOL_RECT_OUTLINE`. -/// - 2016/09/08 (1.17.0) - Changed confusing name for `NK_SYMBOL_CIRLCE_FILLED`, `NK_SYMBOL_CIRCLE` -/// to hopefully easier to understand `NK_SYMBOL_CIRCLE_FILLED` and -/// `NK_SYMBOL_CIRCLE_OUTLINE`. -/// - 2016/09/08 (1.16.0) - Added additional checks to select correct types if `NK_INCLUDE_FIXED_TYPES` -/// is not defined by supporting the biggest compiler GCC, clang and MSVC. -/// - 2016/09/07 (1.15.3) - Fixed `NK_INCLUDE_COMMAND_USERDATA` define to not cause an error. -/// - 2016/09/04 (1.15.2) - Fixed wrong combobox height calculation. -/// - 2016/09/03 (1.15.1) - Fixed gaps inside combo boxes in OpenGL. -/// - 2016/09/02 (1.15.0) - Changed nuklear to not have any default vertex layout and -/// instead made it user provided. The range of types to convert -/// to is quite limited at the moment, but I would be more than -/// happy to accept PRs to add additional. -/// - 2016/08/30 (1.14.2) - Removed unused variables. -/// - 2016/08/30 (1.14.1) - Fixed C++ build errors. -/// - 2016/08/30 (1.14.0) - Removed mouse dragging from SDL demo since it does not work correctly. -/// - 2016/08/30 (1.13.4) - Tweaked some default styling variables. -/// - 2016/08/30 (1.13.3) - Hopefully fixed drawing bug in slider, in general I would -/// refrain from using slider with a big number of steps. -/// - 2016/08/30 (1.13.2) - Fixed close and minimize button which would fire even if the -/// window was in Read Only Mode. -/// - 2016/08/30 (1.13.1) - Fixed popup panel padding handling which was previously just -/// a hack for combo box and menu. -/// - 2016/08/30 (1.13.0) - Removed `NK_WINDOW_DYNAMIC` flag from public API since -/// it is bugged and causes issues in window selection. -/// - 2016/08/30 (1.12.0) - Removed scaler size. The size of the scaler is now -/// determined by the scrollbar size. -/// - 2016/08/30 (1.11.2) - Fixed some drawing bugs caused by changes from 1.11.0. -/// - 2016/08/30 (1.11.1) - Fixed overlapping minimized window selection. -/// - 2016/08/30 (1.11.0) - Removed some internal complexity and overly complex code -/// handling panel padding and panel border. -/// - 2016/08/29 (1.10.0) - Added additional height parameter to `nk_combobox_xxx`. -/// - 2016/08/29 (1.10.0) - Fixed drawing bug in dynamic popups. -/// - 2016/08/29 (1.10.0) - Added experimental mouse scrolling to popups, menus and comboboxes. -/// - 2016/08/26 (1.10.0) - Added window name string prepresentation to account for -/// hash collisions. Currently limited to `NK_WINDOW_MAX_NAME` -/// which in term can be redefined if not big enough. -/// - 2016/08/26 (1.10.0) - Added stacks for temporary style/UI changes in code. -/// - 2016/08/25 (1.10.0) - Changed `nk_input_is_key_pressed` and 'nk_input_is_key_released' -/// to account for key press and release happening in one frame. -/// - 2016/08/25 (1.10.0) - Added additional nk_edit flag to directly jump to the end on activate. -/// - 2016/08/17 (1.09.6) - Removed invalid check for value zero in `nk_propertyx`. -/// - 2016/08/16 (1.09.5) - Fixed ROM mode for deeper levels of popup windows parents. -/// - 2016/08/15 (1.09.4) - Editbox are now still active if enter was pressed with flag -/// `NK_EDIT_SIG_ENTER`. Main reasoning is to be able to keep -/// typing after committing. -/// - 2016/08/15 (1.09.4) - Removed redundant code. -/// - 2016/08/15 (1.09.4) - Fixed negative numbers in `nk_strtoi` and remove unused variable. -/// - 2016/08/15 (1.09.3) - Fixed `NK_WINDOW_BACKGROUND` flag behavior to select a background -/// window only as selected by hovering and not by clicking. -/// - 2016/08/14 (1.09.2) - Fixed a bug in font atlas which caused wrong loading -/// of glyphs for font with multiple ranges. -/// - 2016/08/12 (1.09.1) - Added additional function to check if window is currently -/// hidden and therefore not visible. -/// - 2016/08/12 (1.09.1) - nk_window_is_closed now queries the correct flag `NK_WINDOW_CLOSED` -/// instead of the old flag `NK_WINDOW_HIDDEN`. -/// - 2016/08/09 (1.09.0) - Added additional double version to nk_property and changed -/// the underlying implementation to not cast to float and instead -/// work directly on the given values. -/// - 2016/08/09 (1.08.0) - Added additional define to overwrite library internal -/// floating pointer number to string conversion for additional -/// precision. -/// - 2016/08/09 (1.08.0) - Added additional define to overwrite library internal -/// string to floating point number conversion for additional -/// precision. -/// - 2016/08/08 (1.07.2) - Fixed compiling error without define `NK_INCLUDE_FIXED_TYPE`. -/// - 2016/08/08 (1.07.1) - Fixed possible floating point error inside `nk_widget` leading -/// to wrong wiget width calculation which results in widgets falsely -/// becoming tagged as not inside window and cannot be accessed. -/// - 2016/08/08 (1.07.0) - Nuklear now differentiates between hiding a window (NK_WINDOW_HIDDEN) and -/// closing a window (NK_WINDOW_CLOSED). A window can be hidden/shown -/// by using `nk_window_show` and closed by either clicking the close -/// icon in a window or by calling `nk_window_close`. Only closed -/// windows get removed at the end of the frame while hidden windows -/// remain. -/// - 2016/08/08 (1.06.0) - Added `nk_edit_string_zero_terminated` as a second option to -/// `nk_edit_string` which takes, edits and outputs a '\0' terminated string. -/// - 2016/08/08 (1.05.4) - Fixed scrollbar auto hiding behavior. -/// - 2016/08/08 (1.05.3) - Fixed wrong panel padding selection in `nk_layout_widget_space`. -/// - 2016/08/07 (1.05.2) - Fixed old bug in dynamic immediate mode layout API, calculating -/// wrong item spacing and panel width. -/// - 2016/08/07 (1.05.1) - Hopefully finally fixed combobox popup drawing bug. -/// - 2016/08/07 (1.05.0) - Split varargs away from `NK_INCLUDE_STANDARD_IO` into own -/// define `NK_INCLUDE_STANDARD_VARARGS` to allow more fine -/// grained controlled over library includes. -/// - 2016/08/06 (1.04.5) - Changed memset calls to `NK_MEMSET`. -/// - 2016/08/04 (1.04.4) - Fixed fast window scaling behavior. -/// - 2016/08/04 (1.04.3) - Fixed window scaling, movement bug which appears if you -/// move/scale a window and another window is behind it. -/// If you are fast enough then the window behind gets activated -/// and the operation is blocked. I now require activating -/// by hovering only if mouse is not pressed. -/// - 2016/08/04 (1.04.2) - Fixed changing fonts. -/// - 2016/08/03 (1.04.1) - Fixed `NK_WINDOW_BACKGROUND` behavior. -/// - 2016/08/03 (1.04.0) - Added color parameter to `nk_draw_image`. -/// - 2016/08/03 (1.04.0) - Added additional window padding style attributes for -/// sub windows (combo, menu, ...). -/// - 2016/08/03 (1.04.0) - Added functions to show/hide software cursor. -/// - 2016/08/03 (1.04.0) - Added `NK_WINDOW_BACKGROUND` flag to force a window -/// to be always in the background of the screen. -/// - 2016/08/03 (1.03.2) - Removed invalid assert macro for NK_RGB color picker. -/// - 2016/08/01 (1.03.1) - Added helper macros into header include guard. -/// - 2016/07/29 (1.03.0) - Moved the window/table pool into the header part to -/// simplify memory management by removing the need to -/// allocate the pool. -/// - 2016/07/29 (1.02.0) - Added auto scrollbar hiding window flag which if enabled -/// will hide the window scrollbar after NK_SCROLLBAR_HIDING_TIMEOUT -/// seconds without window interaction. To make it work -/// you have to also set a delta time inside the `nk_context`. -/// - 2016/07/25 (1.01.1) - Fixed small panel and panel border drawing bugs. -/// - 2016/07/15 (1.01.0) - Added software cursor to `nk_style` and `nk_context`. -/// - 2016/07/15 (1.01.0) - Added const correctness to `nk_buffer_push' data argument. -/// - 2016/07/15 (1.01.0) - Removed internal font baking API and simplified -/// font atlas memory management by converting pointer -/// arrays for fonts and font configurations to lists. -/// - 2016/07/15 (1.00.0) - Changed button API to use context dependent button -/// behavior instead of passing it for every function call. -/// ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ -/// ## Gallery -/// ![Figure [blue]: Feature overview with blue color styling](https://cloud.githubusercontent.com/assets/8057201/13538240/acd96876-e249-11e5-9547-5ac0b19667a0.png) -/// ![Figure [red]: Feature overview with red color styling](https://cloud.githubusercontent.com/assets/8057201/13538243/b04acd4c-e249-11e5-8fd2-ad7744a5b446.png) -/// ![Figure [widgets]: Widget overview](https://cloud.githubusercontent.com/assets/8057201/11282359/3325e3c6-8eff-11e5-86cb-cf02b0596087.png) -/// ![Figure [blackwhite]: Black and white](https://cloud.githubusercontent.com/assets/8057201/11033668/59ab5d04-86e5-11e5-8091-c56f16411565.png) -/// ![Figure [filexp]: File explorer](https://cloud.githubusercontent.com/assets/8057201/10718115/02a9ba08-7b6b-11e5-950f-adacdd637739.png) -/// ![Figure [opengl]: OpenGL Editor](https://cloud.githubusercontent.com/assets/8057201/12779619/2a20d72c-ca69-11e5-95fe-4edecf820d5c.png) -/// ![Figure [nodedit]: Node Editor](https://cloud.githubusercontent.com/assets/8057201/9976995/e81ac04a-5ef7-11e5-872b-acd54fbeee03.gif) -/// ![Figure [skinning]: Using skinning in Nuklear](https://cloud.githubusercontent.com/assets/8057201/15991632/76494854-30b8-11e6-9555-a69840d0d50b.png) -/// ![Figure [bf]: Heavy modified version](https://cloud.githubusercontent.com/assets/8057201/14902576/339926a8-0d9c-11e6-9fee-a8b73af04473.png) -/// -/// ## Credits -/// Developed by Micha Mettke and every direct or indirect github contributor.

    -/// -/// Embeds [stb_texedit](https://github.com/nothings/stb/blob/master/stb_textedit.h), [stb_truetype](https://github.com/nothings/stb/blob/master/stb_truetype.h) and [stb_rectpack](https://github.com/nothings/stb/blob/master/stb_rect_pack.h) by Sean Barret (public domain)
    -/// Uses [stddoc.c](https://github.com/r-lyeh/stddoc.c) from r-lyeh@github.com for documentation generation

    -/// Embeds ProggyClean.ttf font by Tristan Grimmer (MIT license).
    -/// -/// Big thank you to Omar Cornut (ocornut@github) for his [imgui library](https://github.com/ocornut/imgui) and -/// giving me the inspiration for this library, Casey Muratori for handmade hero -/// and his original immediate mode graphical user interface idea and Sean -/// Barret for his amazing single header libraries which restored my faith -/// in libraries and brought me to create some of my own. Finally Apoorva Joshi -/// for his single header file packer. -*/ - diff --git a/nuklear/nuklear_xlib_gl3.h b/nuklear/nuklear_xlib_gl3.h deleted file mode 100644 index 3c13d09..0000000 --- a/nuklear/nuklear_xlib_gl3.h +++ /dev/null @@ -1,758 +0,0 @@ -/* - * Nuklear - v1.17 - public domain - * no warrenty implied; use at your own risk. - * authored from 2015-2016 by Micha Mettke - */ -/* - * ============================================================== - * - * API - * - * =============================================================== - */ -#ifndef NK_XLIB_GL3_H_ -#define NK_XLIB_GL3_H_ - -#include -NK_API struct nk_context* nk_x11_init(Display *dpy, Window win); -NK_API void nk_x11_font_stash_begin(struct nk_font_atlas **atlas); -NK_API void nk_x11_font_stash_end(void); -NK_API int nk_x11_handle_event(XEvent *evt); -NK_API void nk_x11_render(enum nk_anti_aliasing, int max_vertex_buffer, int max_element_buffer); -NK_API void nk_x11_shutdown(void); -NK_API int nk_x11_device_create(void); -NK_API void nk_x11_device_destroy(void); - -#endif -/* - * ============================================================== - * - * IMPLEMENTATION - * - * =============================================================== - */ -#ifdef NK_XLIB_GL3_IMPLEMENTATION -#include -#include -#include - -#include -#include -#include - -#include -#include -#include -#include - -#include -#include - -#ifndef NK_X11_DOUBLE_CLICK_LO -#define NK_X11_DOUBLE_CLICK_LO 20 -#endif -#ifndef NK_X11_DOUBLE_CLICK_HI -#define NK_X11_DOUBLE_CLICK_HI 200 -#endif - -#ifdef NK_XLIB_LOAD_OPENGL_EXTENSIONS -#include - -/* GL_ARB_vertex_buffer_object */ -typedef void(*nkglGenBuffers)(GLsizei, GLuint*); -typedef void(*nkglBindBuffer)(GLenum, GLuint); -typedef void(*nkglBufferData)(GLenum, GLsizeiptr, const GLvoid*, GLenum); -typedef void(*nkglBufferSubData)(GLenum, GLintptr, GLsizeiptr, const GLvoid*); -typedef void*(*nkglMapBuffer)(GLenum, GLenum); -typedef GLboolean(*nkglUnmapBuffer)(GLenum); -typedef void(*nkglDeleteBuffers)(GLsizei, GLuint*); -/* GL_ARB_vertex_array_object */ -typedef void (*nkglGenVertexArrays)(GLsizei, GLuint*); -typedef void (*nkglBindVertexArray)(GLuint); -typedef void (*nkglDeleteVertexArrays)(GLsizei, const GLuint*); -/* GL_ARB_vertex_program / GL_ARB_fragment_program */ -typedef void(*nkglVertexAttribPointer)(GLuint, GLint, GLenum, GLboolean, GLsizei, const GLvoid*); -typedef void(*nkglEnableVertexAttribArray)(GLuint); -typedef void(*nkglDisableVertexAttribArray)(GLuint); -/* GL_ARB_framebuffer_object */ -typedef void(*nkglGenerateMipmap)(GLenum target); -/* GLSL/OpenGL 2.0 core */ -typedef GLuint(*nkglCreateShader)(GLenum); -typedef void(*nkglShaderSource)(GLuint, GLsizei, const GLchar**, const GLint*); -typedef void(*nkglCompileShader)(GLuint); -typedef void(*nkglGetShaderiv)(GLuint, GLenum, GLint*); -typedef void(*nkglGetShaderInfoLog)(GLuint, GLsizei, GLsizei*, GLchar*); -typedef void(*nkglDeleteShader)(GLuint); -typedef GLuint(*nkglCreateProgram)(void); -typedef void(*nkglAttachShader)(GLuint, GLuint); -typedef void(*nkglDetachShader)(GLuint, GLuint); -typedef void(*nkglLinkProgram)(GLuint); -typedef void(*nkglUseProgram)(GLuint); -typedef void(*nkglGetProgramiv)(GLuint, GLenum, GLint*); -typedef void(*nkglGetProgramInfoLog)(GLuint, GLsizei, GLsizei*, GLchar*); -typedef void(*nkglDeleteProgram)(GLuint); -typedef GLint(*nkglGetUniformLocation)(GLuint, const GLchar*); -typedef GLint(*nkglGetAttribLocation)(GLuint, const GLchar*); -typedef void(*nkglUniform1i)(GLint, GLint); -typedef void(*nkglUniform1f)(GLint, GLfloat); -typedef void(*nkglUniformMatrix3fv)(GLint, GLsizei, GLboolean, const GLfloat*); -typedef void(*nkglUniformMatrix4fv)(GLint, GLsizei, GLboolean, const GLfloat*); - -static nkglGenBuffers glGenBuffers; -static nkglBindBuffer glBindBuffer; -static nkglBufferData glBufferData; -static nkglBufferSubData glBufferSubData; -static nkglMapBuffer glMapBuffer; -static nkglUnmapBuffer glUnmapBuffer; -static nkglDeleteBuffers glDeleteBuffers; -static nkglGenVertexArrays glGenVertexArrays; -static nkglBindVertexArray glBindVertexArray; -static nkglDeleteVertexArrays glDeleteVertexArrays; -static nkglVertexAttribPointer glVertexAttribPointer; -static nkglEnableVertexAttribArray glEnableVertexAttribArray; -static nkglDisableVertexAttribArray glDisableVertexAttribArray; -static nkglGenerateMipmap glGenerateMipmap; -static nkglCreateShader glCreateShader; -static nkglShaderSource glShaderSource; -static nkglCompileShader glCompileShader; -static nkglGetShaderiv glGetShaderiv; -static nkglGetShaderInfoLog glGetShaderInfoLog; -static nkglDeleteShader glDeleteShader; -static nkglCreateProgram glCreateProgram; -static nkglAttachShader glAttachShader; -static nkglDetachShader glDetachShader; -static nkglLinkProgram glLinkProgram; -static nkglUseProgram glUseProgram; -static nkglGetProgramiv glGetProgramiv; -static nkglGetProgramInfoLog glGetProgramInfoLog; -static nkglDeleteProgram glDeleteProgram; -static nkglGetUniformLocation glGetUniformLocation; -static nkglGetAttribLocation glGetAttribLocation; -static nkglUniform1i glUniform1i; -static nkglUniform1f glUniform1f; -static nkglUniformMatrix3fv glUniformMatrix3fv; -static nkglUniformMatrix4fv glUniformMatrix4fv; - -enum graphics_card_vendors { - VENDOR_UNKNOWN, - VENDOR_NVIDIA, - VENDOR_AMD, - VENDOR_INTEL -}; - -struct opengl_info { - /* info */ - const char *vendor_str; - const char *version_str; - const char *extensions_str; - const char *renderer_str; - const char *glsl_version_str; - enum graphics_card_vendors vendor; - /* version */ - float version; - int major_version; - int minor_version; - /* extensions */ - int glsl_available; - int vertex_buffer_obj_available; - int vertex_array_obj_available; - int map_buffer_range_available; - int fragment_program_available; - int frame_buffer_object_available; -}; -#endif - -struct nk_x11_vertex { - float position[2]; - float uv[2]; - nk_byte col[4]; -}; - -struct nk_x11_device { -#ifdef NK_XLIB_LOAD_OPENGL_EXTENSIONS - struct opengl_info info; -#endif - struct nk_buffer cmds; - struct nk_draw_null_texture tex_null; - GLuint vbo, vao, ebo; - GLuint prog; - GLuint vert_shdr; - GLuint frag_shdr; - GLint attrib_pos; - GLint attrib_uv; - GLint attrib_col; - GLint uniform_tex; - GLint uniform_proj; - GLuint font_tex; -}; - -static struct nk_x11 { - struct nk_x11_device ogl; - struct nk_context ctx; - struct nk_font_atlas atlas; - Cursor cursor; - Display *dpy; - Window win; - long last_button_click; -} x11; - -#ifdef __APPLE__ - #define NK_SHADER_VERSION "#version 150\n" -#else - #define NK_SHADER_VERSION "#version 300 es\n" -#endif - -#ifdef NK_XLIB_LOAD_OPENGL_EXTENSIONS -#include - -NK_INTERN long -nk_timestamp(void) -{ - struct timeval tv; - if (gettimeofday(&tv, NULL) < 0) return 0; - return (long)((long)tv.tv_sec * 1000 + (long)tv.tv_usec/1000); -} - -NK_INTERN int -nk_x11_stricmpn(const char *a, const char *b, int len) -{ - int i = 0; - for (i = 0; i < len && a[i] && b[i]; ++i) - if (a[i] != b[i]) return 1; - if (i != len) return 1; - return 0; -} - -NK_INTERN int -nk_x11_check_extension(struct opengl_info *GL, const char *ext) -{ - const char *start, *where, *term; - where = strchr(ext, ' '); - if (where || *ext == '\0') - return nk_false; - - for (start = GL->extensions_str;;) { - where = strstr((const char*)start, ext); - if (!where) break; - term = where + strlen(ext); - if (where == start || *(where - 1) == ' ') { - if (*term == ' ' || *term == '\0') - return nk_true; - } - start = term; - } - return nk_false; -} - -#define GL_EXT(name) (nk##name)nk_gl_ext(#name) -NK_INTERN __GLXextFuncPtr -nk_gl_ext(const char *name) -{ - __GLXextFuncPtr func; - func = glXGetProcAddress((const GLubyte*)name); - if (!func) { - fprintf(stdout, "[GL]: failed to load extension: %s", name); - return NULL; - } - return func; -} - -NK_INTERN int -nk_load_opengl(struct opengl_info *gl) -{ - int failed = nk_false; - gl->version_str = (const char*)glGetString(GL_VERSION); - glGetIntegerv(GL_MAJOR_VERSION, &gl->major_version); - glGetIntegerv(GL_MINOR_VERSION, &gl->minor_version); - if (gl->major_version < 2) { - fprintf(stderr, "[GL]: Graphics card does not fulfill minimum OpenGL 2.0 support\n"); - return 0; - } - gl->version = (float)gl->major_version + (float)gl->minor_version * 0.1f; - gl->renderer_str = (const char*)glGetString(GL_RENDERER); - gl->extensions_str = (const char*)glGetString(GL_EXTENSIONS); - gl->glsl_version_str = (const char*)glGetString(GL_SHADING_LANGUAGE_VERSION); - gl->vendor_str = (const char*)glGetString(GL_VENDOR); - if (!nk_x11_stricmpn(gl->vendor_str, "ATI", 4) || - !nk_x11_stricmpn(gl->vendor_str, "AMD", 4)) - gl->vendor = VENDOR_AMD; - else if (!nk_x11_stricmpn(gl->vendor_str, "NVIDIA", 6)) - gl->vendor = VENDOR_NVIDIA; - else if (!nk_x11_stricmpn(gl->vendor_str, "Intel", 5)) - gl->vendor = VENDOR_INTEL; - else gl->vendor = VENDOR_UNKNOWN; - - /* Extensions */ - gl->glsl_available = (gl->version >= 2.0f); - if (gl->glsl_available) { - /* GLSL core in OpenGL > 2 */ - glCreateShader = GL_EXT(glCreateShader); - glShaderSource = GL_EXT(glShaderSource); - glCompileShader = GL_EXT(glCompileShader); - glGetShaderiv = GL_EXT(glGetShaderiv); - glGetShaderInfoLog = GL_EXT(glGetShaderInfoLog); - glDeleteShader = GL_EXT(glDeleteShader); - glCreateProgram = GL_EXT(glCreateProgram); - glAttachShader = GL_EXT(glAttachShader); - glDetachShader = GL_EXT(glDetachShader); - glLinkProgram = GL_EXT(glLinkProgram); - glUseProgram = GL_EXT(glUseProgram); - glGetProgramiv = GL_EXT(glGetProgramiv); - glGetProgramInfoLog = GL_EXT(glGetProgramInfoLog); - glDeleteProgram = GL_EXT(glDeleteProgram); - glGetUniformLocation = GL_EXT(glGetUniformLocation); - glGetAttribLocation = GL_EXT(glGetAttribLocation); - glUniform1i = GL_EXT(glUniform1i); - glUniform1f = GL_EXT(glUniform1f); - glUniformMatrix3fv = GL_EXT(glUniformMatrix3fv); - glUniformMatrix4fv = GL_EXT(glUniformMatrix4fv); - } - gl->vertex_buffer_obj_available = nk_x11_check_extension(gl, "GL_ARB_vertex_buffer_object"); - if (gl->vertex_buffer_obj_available) { - /* GL_ARB_vertex_buffer_object */ - glGenBuffers = GL_EXT(glGenBuffers); - glBindBuffer = GL_EXT(glBindBuffer); - glBufferData = GL_EXT(glBufferData); - glBufferSubData = GL_EXT(glBufferSubData); - glMapBuffer = GL_EXT(glMapBuffer); - glUnmapBuffer = GL_EXT(glUnmapBuffer); - glDeleteBuffers = GL_EXT(glDeleteBuffers); - } - gl->fragment_program_available = nk_x11_check_extension(gl, "GL_ARB_fragment_program"); - if (gl->fragment_program_available) { - /* GL_ARB_vertex_program / GL_ARB_fragment_program */ - glVertexAttribPointer = GL_EXT(glVertexAttribPointer); - glEnableVertexAttribArray = GL_EXT(glEnableVertexAttribArray); - glDisableVertexAttribArray = GL_EXT(glDisableVertexAttribArray); - } - gl->vertex_array_obj_available = nk_x11_check_extension(gl, "GL_ARB_vertex_array_object"); - if (gl->vertex_array_obj_available) { - /* GL_ARB_vertex_array_object */ - glGenVertexArrays = GL_EXT(glGenVertexArrays); - glBindVertexArray = GL_EXT(glBindVertexArray); - glDeleteVertexArrays = GL_EXT(glDeleteVertexArrays); - } - gl->frame_buffer_object_available = nk_x11_check_extension(gl, "GL_ARB_framebuffer_object"); - if (gl->frame_buffer_object_available) { - /* GL_ARB_framebuffer_object */ - glGenerateMipmap = GL_EXT(glGenerateMipmap); - } - if (!gl->vertex_buffer_obj_available) { - fprintf(stdout, "[GL] Error: GL_ARB_vertex_buffer_object is not available!\n"); - failed = nk_true; - } - if (!gl->fragment_program_available) { - fprintf(stdout, "[GL] Error: GL_ARB_fragment_program is not available!\n"); - failed = nk_true; - } - if (!gl->vertex_array_obj_available) { - fprintf(stdout, "[GL] Error: GL_ARB_vertex_array_object is not available!\n"); - failed = nk_true; - } - if (!gl->frame_buffer_object_available) { - fprintf(stdout, "[GL] Error: GL_ARB_framebuffer_object is not available!\n"); - failed = nk_true; - } - return !failed; -} -#endif - -NK_API int -nk_x11_device_create(void) -{ - GLint status; - static const GLchar *vertex_shader = - NK_SHADER_VERSION - "uniform mat4 ProjMtx;\n" - "in vec2 Position;\n" - "in vec2 TexCoord;\n" - "in vec4 Color;\n" - "out vec2 Frag_UV;\n" - "out vec4 Frag_Color;\n" - "void main() {\n" - " Frag_UV = TexCoord;\n" - " Frag_Color = Color;\n" - " gl_Position = ProjMtx * vec4(Position.xy, 0, 1);\n" - "}\n"; - static const GLchar *fragment_shader = - NK_SHADER_VERSION - "precision mediump float;\n" - "uniform sampler2D Texture;\n" - "in vec2 Frag_UV;\n" - "in vec4 Frag_Color;\n" - "out vec4 Out_Color;\n" - "void main(){\n" - " Out_Color = Frag_Color * texture(Texture, Frag_UV.st);\n" - "}\n"; - - struct nk_x11_device *dev = &x11.ogl; -#ifdef NK_XLIB_LOAD_OPENGL_EXTENSIONS - if (!nk_load_opengl(&dev->info)) return 0; -#endif - nk_buffer_init_default(&dev->cmds); - - dev->prog = glCreateProgram(); - dev->vert_shdr = glCreateShader(GL_VERTEX_SHADER); - dev->frag_shdr = glCreateShader(GL_FRAGMENT_SHADER); - glShaderSource(dev->vert_shdr, 1, &vertex_shader, 0); - glShaderSource(dev->frag_shdr, 1, &fragment_shader, 0); - glCompileShader(dev->vert_shdr); - glCompileShader(dev->frag_shdr); - glGetShaderiv(dev->vert_shdr, GL_COMPILE_STATUS, &status); - assert(status == GL_TRUE); - glGetShaderiv(dev->frag_shdr, GL_COMPILE_STATUS, &status); - assert(status == GL_TRUE); - glAttachShader(dev->prog, dev->vert_shdr); - glAttachShader(dev->prog, dev->frag_shdr); - glLinkProgram(dev->prog); - glGetProgramiv(dev->prog, GL_LINK_STATUS, &status); - assert(status == GL_TRUE); - - dev->uniform_tex = glGetUniformLocation(dev->prog, "Texture"); - dev->uniform_proj = glGetUniformLocation(dev->prog, "ProjMtx"); - dev->attrib_pos = glGetAttribLocation(dev->prog, "Position"); - dev->attrib_uv = glGetAttribLocation(dev->prog, "TexCoord"); - dev->attrib_col = glGetAttribLocation(dev->prog, "Color"); - - { - /* buffer setup */ - GLsizei vs = sizeof(struct nk_x11_vertex); - size_t vp = offsetof(struct nk_x11_vertex, position); - size_t vt = offsetof(struct nk_x11_vertex, uv); - size_t vc = offsetof(struct nk_x11_vertex, col); - - glGenBuffers(1, &dev->vbo); - glGenBuffers(1, &dev->ebo); - glGenVertexArrays(1, &dev->vao); - - glBindVertexArray(dev->vao); - glBindBuffer(GL_ARRAY_BUFFER, dev->vbo); - glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, dev->ebo); - - glEnableVertexAttribArray((GLuint)dev->attrib_pos); - glEnableVertexAttribArray((GLuint)dev->attrib_uv); - glEnableVertexAttribArray((GLuint)dev->attrib_col); - - glVertexAttribPointer((GLuint)dev->attrib_pos, 2, GL_FLOAT, GL_FALSE, vs, (void*)vp); - glVertexAttribPointer((GLuint)dev->attrib_uv, 2, GL_FLOAT, GL_FALSE, vs, (void*)vt); - glVertexAttribPointer((GLuint)dev->attrib_col, 4, GL_UNSIGNED_BYTE, GL_TRUE, vs, (void*)vc); - } - - glBindTexture(GL_TEXTURE_2D, 0); - glBindBuffer(GL_ARRAY_BUFFER, 0); - glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, 0); - glBindVertexArray(0); - return 1; -} - -NK_INTERN void -nk_x11_device_upload_atlas(const void *image, int width, int height) -{ - struct nk_x11_device *dev = &x11.ogl; - glGenTextures(1, &dev->font_tex); - glBindTexture(GL_TEXTURE_2D, dev->font_tex); - glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR); - glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR); - glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, (GLsizei)width, (GLsizei)height, 0, - GL_RGBA, GL_UNSIGNED_BYTE, image); -} - -NK_API void -nk_x11_device_destroy(void) -{ - struct nk_x11_device *dev = &x11.ogl; - glDetachShader(dev->prog, dev->vert_shdr); - glDetachShader(dev->prog, dev->frag_shdr); - glDeleteShader(dev->vert_shdr); - glDeleteShader(dev->frag_shdr); - glDeleteProgram(dev->prog); - glDeleteTextures(1, &dev->font_tex); - glDeleteBuffers(1, &dev->vbo); - glDeleteBuffers(1, &dev->ebo); - nk_buffer_free(&dev->cmds); -} - -NK_API void -nk_x11_render(enum nk_anti_aliasing AA, int max_vertex_buffer, int max_element_buffer) -{ - int width, height; - XWindowAttributes attr; - struct nk_x11_device *dev = &x11.ogl; - GLfloat ortho[4][4] = { - {2.0f, 0.0f, 0.0f, 0.0f}, - {0.0f,-2.0f, 0.0f, 0.0f}, - {0.0f, 0.0f,-1.0f, 0.0f}, - {-1.0f,1.0f, 0.0f, 1.0f}, - }; - XGetWindowAttributes(x11.dpy, x11.win, &attr); - width = attr.width; - height = attr.height; - - ortho[0][0] /= (GLfloat)width; - ortho[1][1] /= (GLfloat)height; - - /* setup global state */ - glEnable(GL_BLEND); - glBlendEquation(GL_FUNC_ADD); - glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); - glDisable(GL_CULL_FACE); - glDisable(GL_DEPTH_TEST); - glEnable(GL_SCISSOR_TEST); - glActiveTexture(GL_TEXTURE0); - - /* setup program */ - glUseProgram(dev->prog); - glUniform1i(dev->uniform_tex, 0); - glUniformMatrix4fv(dev->uniform_proj, 1, GL_FALSE, &ortho[0][0]); - glViewport(0,0,(GLsizei)width,(GLsizei)height); - { - /* convert from command queue into draw list and draw to screen */ - const struct nk_draw_command *cmd; - void *vertices, *elements; - const nk_draw_index *offset = NULL; - struct nk_buffer vbuf, ebuf; - - /* allocate vertex and element buffer */ - glBindVertexArray(dev->vao); - glBindBuffer(GL_ARRAY_BUFFER, dev->vbo); - glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, dev->ebo); - - glBufferData(GL_ARRAY_BUFFER, max_vertex_buffer, NULL, GL_STREAM_DRAW); - glBufferData(GL_ELEMENT_ARRAY_BUFFER, max_element_buffer, NULL, GL_STREAM_DRAW); - - /* load draw vertices & elements directly into vertex + element buffer */ - vertices = glMapBuffer(GL_ARRAY_BUFFER, GL_WRITE_ONLY); - elements = glMapBuffer(GL_ELEMENT_ARRAY_BUFFER, GL_WRITE_ONLY); - { - /* fill convert configuration */ - struct nk_convert_config config; - static const struct nk_draw_vertex_layout_element vertex_layout[] = { - {NK_VERTEX_POSITION, NK_FORMAT_FLOAT, NK_OFFSETOF(struct nk_x11_vertex, position)}, - {NK_VERTEX_TEXCOORD, NK_FORMAT_FLOAT, NK_OFFSETOF(struct nk_x11_vertex, uv)}, - {NK_VERTEX_COLOR, NK_FORMAT_R8G8B8A8, NK_OFFSETOF(struct nk_x11_vertex, col)}, - {NK_VERTEX_LAYOUT_END} - }; - memset(&config, 0, sizeof(config)); - config.vertex_layout = vertex_layout; - config.vertex_size = sizeof(struct nk_x11_vertex); - config.vertex_alignment = NK_ALIGNOF(struct nk_x11_vertex); - config.tex_null = dev->tex_null; - config.circle_segment_count = 22; - config.curve_segment_count = 22; - config.arc_segment_count = 22; - config.global_alpha = 1.0f; - config.shape_AA = AA; - config.line_AA = AA; - - /* setup buffers to load vertices and elements */ - nk_buffer_init_fixed(&vbuf, vertices, (size_t)max_vertex_buffer); - nk_buffer_init_fixed(&ebuf, elements, (size_t)max_element_buffer); - nk_convert(&x11.ctx, &dev->cmds, &vbuf, &ebuf, &config); - } - glUnmapBuffer(GL_ARRAY_BUFFER); - glUnmapBuffer(GL_ELEMENT_ARRAY_BUFFER); - - /* iterate over and execute each draw command */ - nk_draw_foreach(cmd, &x11.ctx, &dev->cmds) - { - if (!cmd->elem_count) continue; -// GLint unit = cmd->texture.id - glBindTexture(GL_TEXTURE_2D, (GLuint)cmd->texture.id); - glScissor( - (GLint)(cmd->clip_rect.x), - (GLint)((height - (GLint)(cmd->clip_rect.y + cmd->clip_rect.h))), - (GLint)(cmd->clip_rect.w), - (GLint)(cmd->clip_rect.h)); - glDrawElements(GL_TRIANGLES, (GLsizei)cmd->elem_count, GL_UNSIGNED_SHORT, offset); - offset += cmd->elem_count; - } - nk_clear(&x11.ctx); - nk_buffer_clear(&dev->cmds); - } - - /* default OpenGL state */ - glUseProgram(0); - glBindBuffer(GL_ARRAY_BUFFER, 0); - glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, 0); - glBindVertexArray(0); - glDisable(GL_BLEND); - glDisable(GL_SCISSOR_TEST); -} - -NK_API void -nk_x11_font_stash_begin(struct nk_font_atlas **atlas) -{ - nk_font_atlas_init_default(&x11.atlas); - nk_font_atlas_begin(&x11.atlas); - *atlas = &x11.atlas; -} - -NK_API void -nk_x11_font_stash_end(void) -{ - const void *image; int w, h; - image = nk_font_atlas_bake(&x11.atlas, &w, &h, NK_FONT_ATLAS_RGBA32); - nk_x11_device_upload_atlas(image, w, h); - nk_font_atlas_end(&x11.atlas, nk_handle_id((int)x11.ogl.font_tex), &x11.ogl.tex_null); - if (x11.atlas.default_font) - nk_style_set_font(&x11.ctx, &x11.atlas.default_font->handle); -} - -NK_API int -nk_x11_handle_event(XEvent *evt) -{ - struct nk_context *ctx = &x11.ctx; - - /* optional grabbing behavior */ - if (ctx->input.mouse.grab) { - XDefineCursor(x11.dpy, x11.win, x11.cursor); - ctx->input.mouse.grab = 0; - } else if (ctx->input.mouse.ungrab) { - XWarpPointer(x11.dpy, None, x11.win, 0, 0, 0, 0, - (int)ctx->input.mouse.pos.x, (int)ctx->input.mouse.pos.y); - XUndefineCursor(x11.dpy, x11.win); - ctx->input.mouse.ungrab = 0; - } - - if (evt->type == KeyPress || evt->type == KeyRelease) - { - /* Key handler */ - int ret, down = (evt->type == KeyPress); - KeySym *code = XGetKeyboardMapping(x11.dpy, (KeyCode)evt->xkey.keycode, 1, &ret); - if (*code == XK_Shift_L || *code == XK_Shift_R) nk_input_key(ctx, NK_KEY_SHIFT, down); - else if (*code == XK_Control_L || *code == XK_Control_R) nk_input_key(ctx, NK_KEY_CTRL, down); - else if (*code == XK_Alt_L || *code == XK_Alt_R) nk_input_key(ctx, NK_KEY_ALT, down); - else if (*code == XK_Delete) nk_input_key(ctx, NK_KEY_DEL, down); - else if (*code == XK_Return) nk_input_key(ctx, NK_KEY_ENTER, down); - else if (*code == XK_Tab) nk_input_key(ctx, NK_KEY_TAB, down); - else if (*code == XK_Left) nk_input_key(ctx, NK_KEY_LEFT, down); - else if (*code == XK_Right) nk_input_key(ctx, NK_KEY_RIGHT, down); - else if (*code == XK_Up) nk_input_key(ctx, NK_KEY_UP, down); - else if (*code == XK_Down) nk_input_key(ctx, NK_KEY_DOWN, down); - else if (*code == XK_BackSpace) nk_input_key(ctx, NK_KEY_BACKSPACE, down); - else if (*code == XK_Escape) nk_input_key(ctx, NK_KEY_ESCAPE, down); - // else if (*code == XK_space && !down) nk_input_char(ctx, ' '); - else if (*code == XK_Page_Up) nk_input_key(ctx, NK_KEY_SCROLL_UP, down); - else if (*code == XK_Page_Down) nk_input_key(ctx, NK_KEY_SCROLL_DOWN, down); - else if (*code == XK_Home) { - nk_input_key(ctx, NK_KEY_TEXT_START, down); - nk_input_key(ctx, NK_KEY_SCROLL_START, down); - } else if (*code == XK_End) { - nk_input_key(ctx, NK_KEY_TEXT_END, down); - nk_input_key(ctx, NK_KEY_SCROLL_END, down); - } else { -#if 0 // MII wants these!! - if (*code == 'c' && (evt->xkey.state & ControlMask)) - nk_input_key(ctx, NK_KEY_COPY, down); - else if (*code == 'v' && (evt->xkey.state & ControlMask)) - nk_input_key(ctx, NK_KEY_PASTE, down); - else if (*code == 'x' && (evt->xkey.state & ControlMask)) - nk_input_key(ctx, NK_KEY_CUT, down); - else if (*code == 'z' && (evt->xkey.state & ControlMask)) - nk_input_key(ctx, NK_KEY_TEXT_UNDO, down); - else if (*code == 'r' && (evt->xkey.state & ControlMask)) - nk_input_key(ctx, NK_KEY_TEXT_REDO, down); - else if (*code == XK_Left && (evt->xkey.state & ControlMask)) - nk_input_key(ctx, NK_KEY_TEXT_WORD_LEFT, down); - else if (*code == XK_Right && (evt->xkey.state & ControlMask)) - nk_input_key(ctx, NK_KEY_TEXT_WORD_RIGHT, down); - else if (*code == 'b' && (evt->xkey.state & ControlMask)) - nk_input_key(ctx, NK_KEY_TEXT_LINE_START, down); - else if (*code == 'e' && (evt->xkey.state & ControlMask)) - nk_input_key(ctx, NK_KEY_TEXT_LINE_END, down); - else { - if (*code == 'i') - nk_input_key(ctx, NK_KEY_TEXT_INSERT_MODE, down); - else if (*code == 'r') - nk_input_key(ctx, NK_KEY_TEXT_REPLACE_MODE, down); -#endif - { - char buf[32]; - KeySym keysym = 0; - if (XLookupString((XKeyEvent*)evt, buf, 32, &keysym, NULL) != NoSymbol) { - if (down) - nk_input_glyph(ctx, buf); - } else { -// printf("Unknown key %lx, pass on RAW %d\n", *code, -// ctx->input.keyboard.text_len); - uint32_t *d = (uint32_t *) ctx->input.keyboard.text; - d[ctx->input.keyboard.text_len / 4] = - (*code << 16) | !!down; - ctx->input.keyboard.text_len += 4; - } - } - } - XFree(code); - return 1; - } else if (evt->type == ButtonPress || evt->type == ButtonRelease) { - /* Button handler */ - int down = (evt->type == ButtonPress); - const int x = evt->xbutton.x, y = evt->xbutton.y; - if (evt->xbutton.button == Button1) { - if (down) { /* Double-Click Button handler */ - long dt = nk_timestamp() - x11.last_button_click; - if (dt > NK_X11_DOUBLE_CLICK_LO && dt < NK_X11_DOUBLE_CLICK_HI) - nk_input_button(ctx, NK_BUTTON_DOUBLE, x, y, nk_true); - x11.last_button_click = nk_timestamp(); - } else nk_input_button(ctx, NK_BUTTON_DOUBLE, x, y, nk_false); - nk_input_button(ctx, NK_BUTTON_LEFT, x, y, down); - } else if (evt->xbutton.button == Button2) - nk_input_button(ctx, NK_BUTTON_MIDDLE, x, y, down); - else if (evt->xbutton.button == Button3) - nk_input_button(ctx, NK_BUTTON_RIGHT, x, y, down); - else if (evt->xbutton.button == Button4) - nk_input_scroll(ctx, nk_vec2(0,1.0f)); - else if (evt->xbutton.button == Button5) - nk_input_scroll(ctx, nk_vec2(0,-1.0f)); - else return 0; - return 1; - } else if (evt->type == MotionNotify) { - /* Mouse motion handler */ - const int x = evt->xmotion.x, y = evt->xmotion.y; - nk_input_motion(ctx, x, y); - if (ctx->input.mouse.grabbed) { - ctx->input.mouse.pos.x = ctx->input.mouse.prev.x; - ctx->input.mouse.pos.y = ctx->input.mouse.prev.y; - XWarpPointer(x11.dpy, None, x11.win, 0, 0, 0, 0, (int)ctx->input.mouse.pos.x, (int)ctx->input.mouse.pos.y); - } - return 1; - } else if (evt->type == KeymapNotify) { - XRefreshKeyboardMapping(&evt->xmapping); - return 1; - } - return 0; -} - -NK_API struct nk_context* -nk_x11_init(Display *dpy, Window win) -{ - if (!setlocale(LC_ALL,"")) return 0; - if (!XSupportsLocale()) return 0; - if (!XSetLocaleModifiers("@im=none")) return 0; - if (!nk_x11_device_create()) return 0; - - x11.dpy = dpy; - x11.win = win; - - /* create invisible cursor */ - {static XColor dummy; char data[1] = {0}; - Pixmap blank = XCreateBitmapFromData(dpy, win, data, 1, 1); - if (blank == None) return 0; - x11.cursor = XCreatePixmapCursor(dpy, blank, blank, &dummy, &dummy, 0, 0); - XFreePixmap(dpy, blank);} - - nk_init_default(&x11.ctx, 0); - return &x11.ctx; -} - -NK_API void -nk_x11_shutdown(void) -{ - nk_font_atlas_clear(&x11.atlas); - nk_free(&x11.ctx); - nk_x11_device_destroy(); - XFreeCursor(x11.dpy, x11.cursor); - memset(&x11, 0, sizeof(x11)); -} - -#endif diff --git a/src/drivers/mii_disk2.c b/src/drivers/mii_disk2.c index 8cf522b..36328db 100644 --- a/src/drivers/mii_disk2.c +++ b/src/drivers/mii_disk2.c @@ -352,17 +352,15 @@ _mii_disk2_command( *(int *)param = 2; break; case MII_SLOT_DRIVE_LOAD ... MII_SLOT_DRIVE_LOAD + 2 - 1: - if (param) { - int drive = cmd - MII_SLOT_DRIVE_LOAD; - const char *filename = param; - if (c->disk[drive].privdat) { - c->disk[drive].eject(&c->disk[drive]); - } - printf("%s: drive %d loading %s\n", __func__, drive, - filename); - c->disk[drive] = disk_format_load(filename); - return 0; + int drive = cmd - MII_SLOT_DRIVE_LOAD; + if (c->disk[drive].privdat) { + c->disk[drive].eject(&c->disk[drive]); } + const char *filename = param; + printf("%s: drive %d loading %s\n", __func__, drive, + filename); + c->disk[drive] = disk_format_load( + filename && *filename ? filename : NULL); break; } return 0; diff --git a/src/drivers/mii_epromcard.c b/src/drivers/mii_epromcard.c index 855d6c5..8bf663b 100644 --- a/src/drivers/mii_epromcard.c +++ b/src/drivers/mii_epromcard.c @@ -8,6 +8,7 @@ * This is a driver for these eprom/flash cards from * Terence J. Boldt and the likes */ +#define _GNU_SOURCE // for asprintf #include #include #include @@ -21,37 +22,55 @@ #include "mii.h" #include "mii_bank.h" +#define INCBIN_STYLE INCBIN_STYLE_SNAKE +#define INCBIN_PREFIX mii_ +#include "incbin.h" + +INCBIN(1mb_rom, "disks/GamesWithFirmware.po"); + typedef struct mii_card_ee_t { - uint8_t * file; - uint16_t latch; + mii_dd_t drive[1]; + uint8_t * file; + uint16_t latch; } mii_card_ee_t; -int mmapfile(const char *fname, uint8_t **buf, size_t *sz, int flags); static int _mii_ee_init( mii_t * mii, struct mii_slot_t *slot ) { - uint8_t * file; - size_t sz; - const char *fname = "disks/Apple IIe Diagnostic 2.1.po"; - //const char *fname = "disks/GamesWithFirmware.po"; - - if (mmapfile(fname, &file, &sz, O_RDONLY) != 0) { - printf("Failed to load %s\n", fname); - return -1; - } mii_card_ee_t *c = calloc(1, sizeof(*c)); slot->drv_priv = c; - c->file = file; - printf("%s loading in slot %d\n", __func__, slot->id); - uint16_t addr = 0xc100 + (slot->id * 0x100); - mii_bank_write( - &mii->bank[MII_BANK_CARD_ROM], - addr, c->file + 0x300, 256); + printf("%s loading in slot %d\n", __func__, slot->id + 1); + for (int i = 0; i < 1; i++) { + mii_dd_t *dd = &c->drive[i]; + dd->slot_id = slot->id + 1; + dd->drive = i + 1; + dd->slot = slot; + dd->ro = 1; dd->wp = 1; + asprintf((char **)&dd->name, "EE1MB S:%d D:%d", + dd->slot_id, dd->drive); + } + mii_dd_register_drives(&mii->dd, c->drive, 1); + +#if 1 + c->file = (uint8_t*)mii_1mb_rom_data; +#else + const char *fname = "disks/GamesWithFirmware.po"; + + mii_dd_file_t *file = mii_dd_file_load(&mii->dd, fname, 0); + mii_dd_drive_load(&c->drive[0], file); + c->file = file->map; +#endif + if (c->file) { + uint16_t addr = 0xc100 + (slot->id * 0x100); + mii_bank_write( + &mii->bank[MII_BANK_CARD_ROM], + addr, c->file + 0x300, 256); + } return 0; } @@ -75,7 +94,35 @@ _mii_ee_access( break; } } else { - return c->file[(c->latch << 4) + psw]; + return c->file ? c->file[(c->latch << 4) + psw] : 0xff; + } + return 0; +} + +static int +_mii_ee_command( + mii_t * mii, + struct mii_slot_t *slot, + uint8_t cmd, + void * param) +{ + mii_card_ee_t *c = slot->drv_priv; + switch (cmd) { + case MII_SLOT_DRIVE_COUNT: + if (param) + *(int *)param = 1; + break; + case MII_SLOT_DRIVE_LOAD: + const char *filename = param; + mii_dd_file_t *file = NULL; + if (filename && *filename) { + file = mii_dd_file_load(&mii->dd, filename, 0); + if (!file) + return -1; + } + mii_dd_drive_load(&c->drive[0], file); + c->file = file ? file->map : (uint8_t*)mii_1mb_rom_data; + break; } return 0; } @@ -85,5 +132,6 @@ static mii_slot_drv_t _driver = { .desc = "EEPROM 1MB card", .init = _mii_ee_init, .access = _mii_ee_access, + .command = _mii_ee_command, }; MI_DRIVER_REGISTER(_driver); diff --git a/src/drivers/mii_mouse.c b/src/drivers/mii_mouse.c index 1a85f84..dccda4e 100644 --- a/src/drivers/mii_mouse.c +++ b/src/drivers/mii_mouse.c @@ -1,4 +1,11 @@ - +/* + * mii_mouse.c + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + * + */ #include "mii.h" @@ -16,6 +23,48 @@ #include "mii.h" #include "mii_bank.h" +/* + * Coded against this information + * http://stason.org/TULARC/pc/apple2/programmer/012-How-do-I-write-programs-which-use-the-mouse.html + */ + +/* + * Screen holes + * $0478 + slot Low byte of absolute X position + * $04F8 + slot Low byte of absolute Y position + * $0578 + slot High byte of absolute X position + * $05F8 + slot High byte of absolute Y position + * $0678 + slot Reserved and used by the firmware + * $06F8 + slot Reserved and used by the firmware + * $0778 + slot Button 0/1 interrupt status byte + * $07F8 + slot Mode byte + * + * Interrupt status byte: + * Set by READMOUSE + * Bit 7 6 5 4 3 2 1 0 + * | | | | | | | | + * | | | | | | | `--- Previously, button 1 was up (0) or down (1) + * | | | | | | `----- Movement interrupt + * | | | | | `------- Button 0/1 interrupt + * | | | | `--------- VBL interrupt + * | | | `----------- Currently, button 1 is up (0) or down (1) + * | | `------------- X/Y moved since last READMOUSE + * | `--------------- Previously, button 0 was up (0) or down (1) + * `----------------- Currently, button 0 is up (0) or down (1) + * + * Mode byte + * Valid after calling SERVEMOUSE, cleared with READMOUSE + * Bit 7 6 5 4 3 2 1 0 + * | | | | | | | | + * | | | | | | | `--- Mouse off (0) or on (1) + * | | | | | | `----- Interrupt if mouse is moved + * | | | | | `------- Interrupt if button is pressed + * | | | | `--------- Interrupt on VBL + * | | | `----------- Reserved + * | | `------------- Reserved + * | `--------------- Reserved + * `----------------- Reserved + */ enum { CLAMP_MIN_LO = 0x478, @@ -42,6 +91,8 @@ enum { typedef struct mii_card_mouse_t { struct mii_slot_t * slot; + mii_t * mii; + uint8_t timer_id; // 60hz timer uint8_t slot_offset; uint8_t mode; // cached mode byte struct { @@ -50,6 +101,44 @@ typedef struct mii_card_mouse_t { } last; } mii_card_mouse_t; +static uint64_t +_mii_mouse_vbl_handler( + mii_t * mii, + void *param) +{ + mii_card_mouse_t *c = param; + /* this is not exact, the VBL interrupt should still work when + * the mouse is disabled, but it's not really important -- for the moment + */ + if (!mii->mouse.enabled) + return 1000000 / 60; + + mii_bank_t * main = &mii->bank[MII_BANK_MAIN]; + uint8_t status = mii_bank_peek(main, MOUSE_STATUS + c->slot_offset); + uint8_t old = status; + + if (c->mode & mouseIntMoveEnabled) { + if ((mii->mouse.x != c->last.x) || (mii->mouse.y != c->last.y)) { + mii->cpu_state.irq = 1; + status |= 1 << 1; + } + } + if (c->mode & mouseIntButtonEnabled) { + if (mii->mouse.button && !c->last.button) { + mii->cpu_state.irq = 1; + status |= 1 << 2; + } + } + if (c->mode & mouseIntVBlankEnabled) { + mii->cpu_state.irq = 1; + status |= 1 << 3; + } +// if (mii->cpu_state.irq) mii->trace_cpu = true; + if (status != old) + mii_bank_poke(main, MOUSE_STATUS + c->slot_offset, status); + return 1000000 / 60; +} + static int _mii_mouse_init( mii_t * mii, @@ -58,12 +147,16 @@ _mii_mouse_init( mii_card_mouse_t *c = calloc(1, sizeof(*c)); c->slot = slot; slot->drv_priv = c; + c->mii = mii; printf("%s loading in slot %d\n", __func__, slot->id + 1); c->slot_offset = slot->id + 1 + 0xc0; uint8_t data[256] = {}; + c->timer_id = mii_timer_register(mii, + _mii_mouse_vbl_handler, c, + 1000000 / 60, __func__); // Identification as a mouse card // From Technical Note Misc #8, "Pascal 1.1 Firmware Protocol ID Bytes": data[0x05] = 0x38; @@ -84,12 +177,10 @@ _mii_mouse_init( data[base+3] = 0x18; // CLC ;no error data[base+4] = 0x60; // RTS } - uint16_t addr = 0xc100 + (slot->id * 0x100); mii_bank_write( &mii->bank[MII_BANK_CARD_ROM], addr, data, 256); - return 0; } @@ -127,21 +218,19 @@ _mii_mouse_access( printf(" Button: %s\n", byte & mouseIntButtonEnabled ? "enabled" : "disabled"); printf(" VBlank: %s\n", byte & mouseIntVBlankEnabled ? "enabled" : "disabled"); c->mode = byte; - mii->video.vbl_irq = !!(byte & mouseIntVBlankEnabled); } } break; + case 3: // service mouse + // no need to handle that, the VBL handler does it + break; case 4: {// read mouse if (!mii->mouse.enabled) break; uint8_t status = 0; - if (mii->mouse.button) - status |= 1 << 7; - if (c->last.button) { - status |= 1 << 6; - } - if ((mii->mouse.x != c->last.x) || (mii->mouse.y != c->last.y)) { + if ((mii->mouse.x != c->last.x) || (mii->mouse.y != c->last.y)) status |= 1 << 5; - } + status |= c->last.button ? 1 << 6 : 0; + status |= mii->mouse.button ? 1 << 7 : 0; mii_bank_poke(main, MOUSE_X_HI + c->slot_offset, mii->mouse.x >> 8); mii_bank_poke(main, MOUSE_Y_HI + c->slot_offset, mii->mouse.y >> 8); mii_bank_poke(main, MOUSE_X_LO + c->slot_offset, mii->mouse.x); diff --git a/src/drivers/mii_noslotclock.c b/src/drivers/mii_noslotclock.c index ca45ca2..cdcd421 100644 --- a/src/drivers/mii_noslotclock.c +++ b/src/drivers/mii_noslotclock.c @@ -105,7 +105,7 @@ _mii_nsc_access( bool write) { if (!bank) { - printf("%s: disposing of NSC\n", __func__); + // printf("%s: disposing of NSC\n", __func__); free(param); return false; } @@ -158,7 +158,7 @@ _mii_nsc_probe( mii_t *mii, uint32_t flags) { - printf("%s %s\n", __func__, flags & MII_INIT_NSC ? "enabled" : "disabled"); +// printf("%s %s\n", __func__, flags & MII_INIT_NSC ? "enabled" : "disabled"); if (!(flags & MII_INIT_NSC)) return 0; mii_nsc_t * nsc = calloc(1, sizeof(*nsc)); diff --git a/src/drivers/mii_smartport.c b/src/drivers/mii_smartport.c index cf96051..e280c57 100644 --- a/src/drivers/mii_smartport.c +++ b/src/drivers/mii_smartport.c @@ -36,7 +36,7 @@ extern const unsigned char mii_smartport_rom_data[]; #define MII_SM_DRIVE_COUNT 2 typedef struct mii_card_sm_t { - struct mii_card_sm_t *next; +// struct mii_card_sm_t *next; mii_dd_t drive[MII_SM_DRIVE_COUNT]; struct mii_slot_t *slot; } mii_card_sm_t; @@ -244,7 +244,7 @@ _mii_sm_init( c->slot = slot; slot->drv_priv = c; - printf("%s loading in slot %d\n", __func__, slot->id); + printf("%s loading in slot %d\n", __func__, slot->id + 1); uint16_t addr = 0xc100 + (slot->id * 0x100); mii_bank_write( &mii->bank[MII_BANK_CARD_ROM], @@ -252,7 +252,7 @@ _mii_sm_init( uint8_t trap_hd = mii_register_trap(mii, _mii_hd_callback); uint8_t trap_sm = mii_register_trap(mii, _mii_sm_callback); - printf("%s: traps %02x %02x\n", __func__, trap_hd, trap_sm); +// printf("%s: traps %02x %02x\n", __func__, trap_hd, trap_sm); mii_bank_write( &mii->bank[MII_BANK_CARD_ROM], addr + 0xd2, &trap_hd, 1); @@ -302,18 +302,15 @@ _mii_sm_command( *(int *)param = MII_SM_DRIVE_COUNT; break; case MII_SLOT_DRIVE_LOAD ... MII_SLOT_DRIVE_LOAD + MII_SM_DRIVE_COUNT - 1: - if (param) { - int drive = cmd - MII_SLOT_DRIVE_LOAD; - const char *filename = param; - mii_dd_file_t *file = NULL; - if (filename) { - file = mii_dd_file_load(&mii->dd, filename, 0); - if (!file) - return -1; - } - mii_dd_drive_load(&c->drive[drive], file); - return 0; + int drive = cmd - MII_SLOT_DRIVE_LOAD; + const char *filename = param; + mii_dd_file_t *file = NULL; + if (filename && *filename) { + file = mii_dd_file_load(&mii->dd, filename, 0); + if (!file) + return -1; } + mii_dd_drive_load(&c->drive[drive], file); break; } return 0; @@ -324,25 +321,6 @@ _mii_sm_access( mii_t * mii, struct mii_slot_t *slot, uint16_t addr, uint8_t byte, bool write) { - #if 0 - mii_card_sm_t *c = slot->drv_priv; - - printf("%s PC:%04x addr %04x %02x wr:%d\n", __func__, - mii->cpu.PC, addr, byte, write); - int psw = addr & 0x0F; - if (write) { - switch (psw) { - case 0: - // c->latch = (c->latch & 0xff00) | byte; - break; - case 1: - // c->latch = (c->latch & 0x00ff) | (byte << 8); - break; - } - } else { - // return c->file[(c->latch << 4) + psw]; - } - #endif return 0; } diff --git a/src/drivers/mii_ssc.c b/src/drivers/mii_ssc.c new file mode 100644 index 0000000..c3e5bee --- /dev/null +++ b/src/drivers/mii_ssc.c @@ -0,0 +1,143 @@ +/* + * mii_ssc.c + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + * + */ + +/* + THIS IS A PLACEHOLDER DO NOT USE + */ +#include "mii.h" + +#include +#include +#include +#include +#include +#include + +#include "mii.h" +#include "mii_bank.h" +#include "mii_sw.h" + +#define INCBIN_STYLE INCBIN_STYLE_SNAKE +#define INCBIN_PREFIX mii_ +#include "incbin.h" + +INCBIN(ssc_rom, "roms/mii_rom_scc_3410065a.bin"); + +typedef struct mii_card_ssc_t { + struct mii_slot_t * slot; + struct mii_bank_t * rom; + mii_t * mii; + uint8_t slot_offset; +} mii_card_ssc_t; + +static bool +_mii_ssc_select( + struct mii_bank_t *bank, + void *param, + uint16_t addr, + uint8_t * byte, + bool write) +{ + mii_card_ssc_t *c = param; + printf("%s selected:%d\n", __func__, c->slot->aux_rom_selected); + if (c->slot->aux_rom_selected) + return false; + mii_bank_write(c->rom, 0xc800, mii_ssc_rom_data, 2048); + c->slot->aux_rom_selected = true; + SW_SETSTATE(c->mii, SLOTAUXROM, 1); + c->mii->mem_dirty = true; + return false; +} + +static int +_mii_ssc_init( + mii_t * mii, + struct mii_slot_t *slot ) +{ + mii_card_ssc_t *c = calloc(1, sizeof(*c)); + c->slot = slot; + slot->drv_priv = c; + c->mii = mii; + + printf("%s: THIS IS A PLACEHOLDER DO NOT USE\n", __func__); + printf("%s loading in slot %d\n", __func__, slot->id + 1); + + c->slot_offset = slot->id + 1 + 0xc0; + + uint16_t addr = 0xc100 + (slot->id * 0x100); + c->rom = &mii->bank[MII_BANK_CARD_ROM]; + mii_bank_write(c->rom, addr, mii_ssc_rom_data + 7*256, 256); + /* + * install a callback that will be called for every access to the + * ROM area, we need this to re-install the secondary part of the ROM + * when the card 'slot' rom is accessed. + */ + mii_bank_install_access_cb(c->rom, + _mii_ssc_select, c, addr >> 8, addr >> 8); + + return 0; +} + +static void +_mii_ssc_dispose( + mii_t * mii, + struct mii_slot_t *slot ) +{ + mii_card_ssc_t *c = slot->drv_priv; + free(c); + slot->drv_priv = NULL; +} + +static uint8_t +_mii_ssc_access( + mii_t * mii, + struct mii_slot_t *slot, + uint16_t addr, + uint8_t byte, + bool write) +{ +// mii_card_ssc_t *c = slot->drv_priv; + + int psw = addr & 0x0F; +// mii_bank_t * main = &mii->bank[MII_BANK_MAIN]; + switch (psw) { + default: + printf("%s PC:%04x addr %04x %02x wr:%d\n", __func__, + mii->cpu.PC, addr, byte, write); + break; + } + return 0; +} + +static int +_mii_ssc_command( + mii_t * mii, + struct mii_slot_t *slot, + uint8_t cmd, + void * param) +{ +// mii_card_ssc_t *c = slot->drv_priv; + switch (cmd) { + case MII_SLOT_SSC_SET_TTY: { + const char * tty = param; + printf("%s: set tty %s\n", __func__, tty); + } break; + } + return -1; +} + +static mii_slot_drv_t _driver = { + .name = "ssc", + .desc = "Super Serial card", + .init = _mii_ssc_init, + .dispose = _mii_ssc_dispose, + .access = _mii_ssc_access, + .command = _mii_ssc_command, +}; +MI_DRIVER_REGISTER(_driver); diff --git a/src/drivers/mii_titan_iie.c b/src/drivers/mii_titan_iie.c index ca54a27..45bf564 100644 --- a/src/drivers/mii_titan_iie.c +++ b/src/drivers/mii_titan_iie.c @@ -60,7 +60,7 @@ _mii_titan_probe( mii_t *mii, uint32_t flags) { - printf("%s %s\n", __func__, flags & MII_INIT_TITAN ? "enabled" : "disabled"); +// printf("%s %s\n", __func__, flags & MII_INIT_TITAN ? "enabled" : "disabled"); if (!(flags & MII_INIT_TITAN)) return 0; // this override a read-only soft switch, but we only handle writes diff --git a/src/format/dsk.c b/src/format/dsk.c index 38de340..95089f5 100644 --- a/src/format/dsk.c +++ b/src/format/dsk.c @@ -196,7 +196,7 @@ header: continue; } - const int data_field_sz = 0x15A; //counts prologue, not epilogue + // const int data_field_sz = 0x15A; //counts prologue, not epilogue for (;;) { if (rd >= (end - 0x15A)) goto bail; if (rd[0] == 0xD5 && rd[1] == 0xAA) { diff --git a/src/format/mii_dd.c b/src/format/mii_dd.c index 45c190a..17f22e0 100644 --- a/src/format/mii_dd.c +++ b/src/format/mii_dd.c @@ -36,6 +36,7 @@ mii_dd_system_init( struct mii_t *mii, mii_dd_system_t *dd ) { +// printf("*** %s: %p\n", __func__, dd); dd->drive = NULL; dd->file = NULL; } @@ -44,6 +45,7 @@ void mii_dd_system_dispose( mii_dd_system_t *dd ) { +// printf("*** %s: %p\n", __func__, dd); while (dd->file) mii_dd_file_dispose(dd, dd->file); dd->file = NULL; @@ -56,6 +58,7 @@ mii_dd_register_drives( mii_dd_t * drives, uint8_t count ) { +// printf("%s: registering %d drives\n", __func__, count); for (int i = 0; i < count; i++) { mii_dd_t *d = &drives[i]; d->dd = dd; @@ -121,6 +124,8 @@ mii_dd_drive_load( dd->file = file; printf("%s: %s loading %s\n", __func__, dd->name, file->pathname); + if (dd->ro || dd->wp) + return 0; if (mii_dd_overlay_load(dd) < 0) { printf("%s: No overlay to load, we're fine for now\n", __func__); // no overlay.. what to do? @@ -371,6 +376,8 @@ mii_dd_write( { if (!dd || !dd->file || !dd->file->map) return -1; + if (dd->ro || dd->wp) + return -1; // printf("%s: %s write %d blocks at %d\n", // __func__, dd->name, blockcount, blk); mii_dd_overlay_prepare(dd); diff --git a/src/format/mii_dd.h b/src/format/mii_dd.h index 855ac6d..f49e9dc 100644 --- a/src/format/mii_dd.h +++ b/src/format/mii_dd.h @@ -13,8 +13,9 @@ struct mii_dd_t; enum { - MII_DD_FILE_OVERLAY = 1, +// MII_DD_FILE_OVERLAY = 1, MII_DD_FILE_RAM, + MII_DD_FILE_ROM, MII_DD_FILE_2MG = 5, }; @@ -52,9 +53,9 @@ typedef union mii_dd_overlay_header_t { typedef struct mii_dd_overlay_t { mii_dd_overlay_header_t *header; // points to the file mapped in memory - uint64_t * bitmap; // usage bitmap - uint8_t * blocks; // raw block data - mii_dd_file_t *file; // overlay file mapping + uint64_t * bitmap; // usage bitmap + uint8_t * blocks; // raw block data + mii_dd_file_t * file; // overlay file mapping } mii_dd_overlay_t; struct mii_slot_t; @@ -62,19 +63,19 @@ struct mii_dd_system_t; // a disk drive, with a slot, a drive number, and a file typedef struct mii_dd_t { - struct mii_dd_t *next; + struct mii_dd_t * next; struct mii_dd_system_t *dd; - const char * name; // ie "Disk ][ D:2" - uint8_t slot_id : 4, drive : 4; - struct mii_slot_t *slot; - unsigned int ro : 1, wp : 1, can_eject : 1; - mii_dd_file_t * file; - mii_dd_overlay_t overlay; + const char * name; // ie "Disk ][ D:2" + uint8_t slot_id : 4, drive : 4; + struct mii_slot_t * slot; + unsigned int ro : 1, wp : 1, can_eject : 1; + mii_dd_file_t * file; + mii_dd_overlay_t overlay; } mii_dd_t; typedef struct mii_dd_system_t { - mii_dd_t * drive; // list of all drives on all slots - mii_dd_file_t * file; // list of all open files (inc overlays) + mii_dd_t * drive; // list of all drives on all slots + mii_dd_file_t * file; // list of all open files (inc overlays) } mii_dd_system_t; struct mii_t; diff --git a/src/mii.c b/src/mii.c index 8ea25c1..db1f4f9 100644 --- a/src/mii.c +++ b/src/mii.c @@ -83,7 +83,7 @@ mii_dump_trace_state( // display the S flags static const char *s_flags = "CZIDBRVN"; for (int i = 0; i < 8; i++) - printf("%c", cpu->P.P[7-i] ? s_flags[7-i] : tolower(s_flags[7-i])); + printf("%c", MII_GET_P_BIT(cpu, i) ? s_flags[i] : tolower(s_flags[i])); if (s.sync) { uint8_t op[16]; for (int i = 0; i < 4; i++) { @@ -96,7 +96,7 @@ mii_dump_trace_state( MII_DUMP_DIS_DUMP_HEX); printf(": %s", dis); if (d.desc.branch) { - if (cpu->P.P[d.desc.s_bit] == d.desc.s_bit_value) + if (MII_GET_P_BIT(cpu, d.desc.s_bit) == d.desc.s_bit_value) printf(" ; taken"); } printf("\n"); @@ -109,7 +109,7 @@ mii_dump_run_trace( mii_t *mii) { // walk all the previous PC values in mii->trace, and display a line - // of disassebly for all of them + // of disassembly for all of them for (int li = 0; li < MII_PC_LOG_SIZE; li++) { int idx = (mii->trace.idx + li) & (MII_PC_LOG_SIZE - 1); uint16_t pc = mii->trace.log[idx]; @@ -154,7 +154,7 @@ static void mii_page_table_update( mii_t *mii) { - if (!mii->mem_dirty) + if (likely(!mii->mem_dirty)) return; mii->mem_dirty = 0; bool altzp = SW_GETSTATE(mii, SWALTPZ); @@ -165,9 +165,10 @@ mii_page_table_update( bool ramwrt = SW_GETSTATE(mii, SWRAMWRT); bool intcxrom = SW_GETSTATE(mii, SWINTCXROM); bool slotc3rom = SW_GETSTATE(mii, SWSLOTC3ROM); + bool slotauxrom = SW_GETSTATE(mii, SLOTAUXROM); - if (mii->trace_cpu) - printf("%04x: page table update altzp:%d page2:%d store80:%d " + if (unlikely(mii->trace_cpu)) + printf("%04x: MEM update altzp:%d page2:%d store80:%d " "hires:%d ramrd:%d ramwrt:%d intcxrom:%d " "slotc3rom:%d\n", mii->cpu.PC, altzp, page2, store80, hires, ramrd, ramwrt, intcxrom, slotc3rom); @@ -188,8 +189,11 @@ mii_page_table_update( page2 ? MII_BANK_AUX : MII_BANK_MAIN, page2 ? MII_BANK_AUX : MII_BANK_MAIN, 0x20, 0x3f); } - if (!intcxrom) + if (!intcxrom) { mii_page_set(mii, MII_BANK_CARD_ROM, _SAME, 0xc1, 0xc7); + if (slotauxrom) + mii_page_set(mii, MII_BANK_CARD_ROM, _SAME, 0xc8, 0xcf); + } mii_page_set(mii, slotc3rom ? MII_BANK_CARD_ROM : MII_BANK_ROM, _SAME, 0xc3, 0xc3); bool bsrread = SW_GETSTATE(mii, BSRREAD); @@ -229,6 +233,55 @@ mii_set_sw_override( mii->soft_switches_override[sw_addr].param = param; } +/* + * This watches for any write to 0xcfff -- if a card had it's aux rom + * selected, it will deselect it. + */ +static bool +_mii_deselect_auxrom( + struct mii_bank_t *bank, + void *param, + uint16_t addr, + uint8_t * byte, + bool write) +{ + if (addr != 0xcfff) + return false; + mii_t * mii = param; +// printf("%s AUXROM:%d\n", __func__, !!(mii->sw_state & M_SLOTAUXROM)); + if (!(mii->sw_state & M_SLOTAUXROM)) + return false; + for (int i = 0; i < 7; i++) { + mii_slot_t * slot = &mii->slot[i]; + if (slot->aux_rom_selected) { + printf("%s %d: %s\n", __func__, + i, slot->drv ? slot->drv->name : "(none?)"); + slot->aux_rom_selected = false; + } + } + mii->sw_state &= ~M_SLOTAUXROM; + mii->mem_dirty = true; + return false; +} + +static bool +_mii_select_c3rom( + struct mii_bank_t *bank, + void *param, + uint16_t addr, + uint8_t * byte, + bool write) +{ + mii_t * mii = param; + printf("%s\n", __func__); + if (mii->sw_state & M_SLOTAUXROM) { + // printf("%s: C3 aux rom re-selected\n", __func__); + mii->sw_state &= ~M_SLOTAUXROM; + } + mii->mem_dirty = true; + return false; +} + static bool mii_access_soft_switches( mii_t *mii, @@ -236,7 +289,7 @@ mii_access_soft_switches( uint8_t * byte, bool write) { - if (!(addr >= 0xc000 && addr <= 0xc0ff) || addr == 0xcfff) + if (!(addr >= 0xc000 && addr <= 0xc0ff)) return false; bool res = false; uint8_t on = 0; @@ -280,7 +333,9 @@ mii_access_soft_switches( SW_SETSTATE(mii, BSRREAD, rd); SW_SETSTATE(mii, BSRPAGE2, !(mode & 0x08)); mii->mem_dirty = 1; - if (mii->trace_cpu) +// mii->trace_cpu = 1; +// mii->state = MII_STOPPED; + if (unlikely(mii->trace_cpu)) printf("%04x: BSR mode addr %04x:%02x read:%s write:%s %s altzp:%02x\n", mii->cpu.PC, addr, mode, @@ -289,11 +344,6 @@ mii_access_soft_switches( SW_GETSTATE(mii, BSRPAGE2) ? "page2" : "page1", mii_sw(mii, SWALTPZ)); } break; - case 0xcfff: - res = true; - mii->mem_dirty = 1; - printf("%s TODO reset SLOT roms\n", __func__); - break; case SWPAGE2OFF: case SWPAGE2ON: res = true; @@ -390,15 +440,22 @@ mii_access_soft_switches( *byte = mii_bank_peek(main, addr); break; case 0xc020: // toggle TAPE output ?!?! - case 0xc064: // Analog Input 0 (paddle 0) - case 0xc065: // Analog Input 1 (paddle 1) - case 0xc079: // Analog Input Reset res = true; break; case 0xc068: res = true; // IIgs register, read by prodos tho break; + default: + res = true; + /* + * this is moderately important, return some random value + * as it is supposed to represent what's on the bus at the time, + * typically video being decoded etc. + */ + *byte = mii->random[mii->random_index++]; + mii->random_index &= 0xff; + break; } } if (!res) { @@ -460,15 +517,35 @@ mii_init( mii_t *mii ) { memset(mii, 0, sizeof(*mii)); - mii->speed = 1.0; + mii->speed = 1.023; + mii->timer.map = 0; for (int i = 0; i < MII_BANK_COUNT; i++) mii->bank[i] = _mii_banks_init[i]; mii->bank[MII_BANK_ROM].mem = (uint8_t*)&iie_enhanced_rom_bin[0]; + for (int i = 0; i < MII_BANK_COUNT; i++) + mii_bank_init(&mii->bank[i]); mii->cpu.trap = MII_TRAP; + // these are called once, regardless of reset + mii_dd_system_init(mii, &mii->dd); + mii_analog_init(mii, &mii->analog); + mii_video_init(mii); + mii_speaker_init(mii, &mii->speaker); + mii_reset(mii, true); mii->cpu_state = mii_cpu_init(&mii->cpu); for (int i = 0; i < 7; i++) mii->slot[i].id = i; + +// srandom(time(NULL)); + for (int i = 0; i < 256; i++) + mii->random[i] = random(); + + mii_bank_install_access_cb(&mii->bank[MII_BANK_CARD_ROM], + _mii_deselect_auxrom, mii, 0xcf, 0xcf); + mii_bank_install_access_cb(&mii->bank[MII_BANK_ROM], + _mii_deselect_auxrom, mii, 0xcf, 0xcf); + mii_bank_install_access_cb(&mii->bank[MII_BANK_ROM], + _mii_select_c3rom, mii, 0xc3, 0xc3); } void @@ -476,14 +553,12 @@ mii_prepare( mii_t *mii, uint32_t flags ) { - mii_dd_system_init(mii, &mii->dd); - mii_speaker_init(mii, &mii->speaker); - printf("%s driver table\n", __func__); +// printf("%s driver table\n", __func__); mii_slot_drv_t * drv = mii_slot_drv_list; while (drv) { printf("%s driver: %s\n", __func__, drv->name); if (drv->probe && drv->probe(mii, flags)) { - printf("%s %s probe done\n", __func__, drv->name); + // printf("%s %s probe done\n", __func__, drv->name); } drv = drv->next; } @@ -501,6 +576,7 @@ mii_dispose( mii_bank_dispose(&mii->bank[i]); mii_speaker_dispose(&mii->speaker); mii_dd_system_dispose(&mii->dd); + mii->state = MII_INIT; } void @@ -509,6 +585,7 @@ mii_reset( bool cold) { // printf("%s cold %d\n", __func__, cold); + mii->state = MII_RUNNING; mii->cpu_state.reset = 1; mii_bank_t * main = &mii->bank[MII_BANK_MAIN]; mii->sw_state = M_BSRWRITE | M_BSRPAGE2; @@ -547,27 +624,27 @@ mii_mem_access( mii_access_keyboard(mii, addr, d, wr) || mii_access_video(mii, addr, d, wr) || mii_access_soft_switches(mii, addr, d, wr); - if (!done) { - uint8_t page = addr >> 8; - if (wr) { - uint8_t m = mii->mem[page].write; - mii_bank_t * b = &mii->bank[m]; - if (b->ro) { - // printf("%s write to RO bank %s %04x:%02x\n", - // __func__, b->name, addr, *d); - } else - mii_bank_write(b, addr, d, 1); - } else { - uint8_t m = mii->mem[page].read; - mii_bank_t * b = &mii->bank[m]; - *d = mii_bank_peek(b, addr); - } + if (done) + return; + uint8_t page = addr >> 8; + if (wr) { + uint8_t m = mii->mem[page].write; + mii_bank_t * b = &mii->bank[m]; + if (b->ro) { + // printf("%s write to RO bank %s %04x:%02x\n", + // __func__, b->name, addr, *d); + } else + mii_bank_write(b, addr, d, 1); + } else { + uint8_t m = mii->mem[page].read; + mii_bank_t * b = &mii->bank[m]; + *d = mii_bank_peek(b, addr); } } static void _mii_handle_trap( - mii_t *mii) + mii_t *mii) { // printf("%s TRAP hit PC: %04x\n", __func__, mii->cpu.PC); mii->cpu_state.sync = 1; @@ -588,8 +665,8 @@ _mii_handle_trap( uint8_t mii_register_trap( - mii_t *mii, - mii_trap_handler_cb cb) + mii_t *mii, + mii_trap_handler_cb cb) { if (mii->trap.map == 0xffff) { printf("%s no more traps!!\n", __func__); @@ -605,23 +682,91 @@ mii_register_trap( return 0xff; } +uint8_t +mii_timer_register( + mii_t *mii, + mii_timer_p cb, + void *param, + int64_t when, + const char *name) +{ + if (mii->timer.map == (uint64_t)-1) { + printf("%s no more timers!!\n", __func__); + return 0xff; + } + int i = ffsll(~mii->timer.map) - 1; + mii->timer.map |= 1ull << i; + mii->timer.timers[i].cb = cb; + mii->timer.timers[i].param = param; + mii->timer.timers[i].when = when; + mii->timer.timers[i].name = name; + return i; +} + +int64_t +mii_timer_get( + mii_t *mii, + uint8_t timer_id) +{ + if (timer_id >= (int)sizeof(mii->timer.map) * 8) + return 0; + return mii->timer.timers[timer_id].when; +} + +int +mii_timer_set( + mii_t *mii, + uint8_t timer_id, + int64_t when) +{ + if (timer_id >= (int)sizeof(mii->timer.map) * 8) + return -1; + mii->timer.timers[timer_id].when = when; + return 0; +} + +static void +mii_timer_run( + mii_t *mii, + uint64_t cycles) +{ + uint64_t timer = mii->timer.map; + while (timer) { + int i = ffsll(timer) - 1; + timer &= ~(1ull << i); + if (mii->timer.timers[i].when > 0) { + mii->timer.timers[i].when -= cycles; + if (mii->timer.timers[i].when <= 0) { + if (mii->timer.timers[i].cb) + mii->timer.timers[i].when += mii->timer.timers[i].cb(mii, + mii->timer.timers[i].param); + } + } + } +} + void mii_run( mii_t *mii) { /* this runs all cycles for one instruction */ + uint16_t cycle = mii->cpu.cycle; do { - if (mii->trace_cpu) + if (unlikely(mii->trace_cpu > 1)) mii_dump_trace_state(mii); mii->cpu_state = mii_cpu_run(&mii->cpu, mii->cpu_state); - mii_video_run(mii); - mii_speaker_run(&mii->speaker); + + mii_timer_run(mii, + mii->cpu.cycle > cycle ? mii->cpu.cycle - cycle : + mii->cpu.cycle); + cycle = mii->cpu.cycle; + // extract 16-bit address from pin mask const uint16_t addr = mii->cpu_state.addr; - const uint8_t data = mii->cpu_state.data; +// const uint8_t data = mii->cpu_state.data; int wr = mii->cpu_state.w; - uint8_t d = data; - if (mii->debug.bp_map) { +// uint8_t d = data; + if (unlikely(mii->debug.bp_map)) { for (int i = 0; i < (int)sizeof(mii->debug.bp_map) * 8; i++) { if (!(mii->debug.bp_map & (1 << i))) continue; @@ -644,14 +789,12 @@ mii_run( } } } - mii_mem_access(mii, addr, &d, wr, true); - if (!wr) - mii->cpu_state.data = d; - if (mii->cpu_state.trap) { + mii_mem_access(mii, addr, &mii->cpu_state.data, wr, true); + if (unlikely(mii->cpu_state.trap)) { _mii_handle_trap(mii); } } while (!(mii->cpu_state.sync)); - mii->cycles += mii->cpu.cycle; + // log PC for the running disassembler display mii->trace.log[mii->trace.idx] = mii->cpu.PC; mii->trace.idx = (mii->trace.idx + 1) & (MII_PC_LOG_SIZE - 1); diff --git a/src/mii.h b/src/mii.h index ba7ff41..6ff276e 100644 --- a/src/mii.h +++ b/src/mii.h @@ -10,7 +10,6 @@ #include #include -#include "mii_types.h" #include "mii_65c02.h" #include "mii_dd.h" #include "mii_bank.h" @@ -20,6 +19,9 @@ #include "mii_mouse.h" #include "mii_analog.h" +#define likely(x) __builtin_expect(!!(x), 1) +#define unlikely(x) __builtin_expect(!!(x), 0) + enum { MII_BANK_MAIN = 0, // main 48K address space MII_BANK_BSR, // 0xd000 - 0xffff bank switched RAM 16KB @@ -50,7 +52,8 @@ typedef struct mii_trap_t { // state of the emulator enum { - MII_RUNNING = 0, // default + MII_INIT = 0, + MII_RUNNING, // default MII_STOPPED, MII_STEP, MII_TERMINATE, @@ -78,16 +81,32 @@ typedef struct mii_trace_t { uint32_t step_inst; } mii_trace_t; - +typedef uint64_t (*mii_timer_p)( + mii_t * mii, + void * param ); /* * principal emulator state, for a faceless emulation */ typedef struct mii_t { unsigned int state; - mii_cycles_t cycles; + /* + * These are 'cycle timers' -- they count down from a set value, + * and stop at 0 (or possiblu -1 or -2, depending on the instructions) + * and call the callback (if present). + * The callback returns the number of cycles to wait until the next + * call. + */ + struct { + uint64_t map; + struct { + mii_timer_p cb; + void * param; + int64_t when; + const char * name; // debug + } timers[64]; + } timer; /* this is the video frame/VBL rate vs 60hz, default to 1.0 */ float speed; - float speed_current; // calculated speed mii_cpu_t cpu; mii_cpu_state_t cpu_state; /* @@ -125,6 +144,9 @@ typedef struct mii_t { mii_mouse_t mouse; mii_dd_system_t dd; mii_analog_t analog; + + uint8_t random[256]; + uint8_t random_index; } mii_t; enum { @@ -235,7 +257,6 @@ mii_write_word( mii_t *mii, uint16_t addr, uint16_t w); - /* lower level call to access memory -- this one can trigger softswitches * if specified. Otherwise behaves as the previous ones, one byte at a time */ @@ -256,12 +277,33 @@ mii_set_sw_override( mii_bank_access_cb cb, void *param); +/* register a cycle timer. cb will be called when (at least) when + * cycles have been spent -- the callback returns how many it should + * spend until the next call */ +uint8_t +mii_timer_register( + mii_t *mii, + mii_timer_p cb, // this is optional, can be NULL + void *param, + int64_t when, + const char *name); +/* return the cycles left for timer_id (can be negative !)*/ +int64_t +mii_timer_get( + mii_t *mii, + uint8_t timer_id); +int +mii_timer_set( + mii_t *mii, + uint8_t timer_id, + int64_t when); + void mii_dump_trace_state( - mii_t *mii); + mii_t *mii); void mii_dump_run_trace( - mii_t *mii); + mii_t *mii); extern mii_slot_drv_t * mii_slot_drv_list; diff --git a/src/mii_65c02.c b/src/mii_65c02.c index 5ab5587..6eadea0 100644 --- a/src/mii_65c02.c +++ b/src/mii_65c02.c @@ -8,6 +8,9 @@ #define MII_CPU_65C02_IMPL #include "mii_65c02_ops.h" +#define likely(x) __builtin_expect(!!(x), 1) +#define unlikely(x) __builtin_expect(!!(x), 0) + mii_cpu_state_t mii_cpu_init( mii_cpu_t *cpu ) @@ -23,31 +26,27 @@ mii_cpu_init( #define _FETCH(_val) { \ s.addr = _val; s.w = 0; cpu->cycle++; \ pt_yield(cpu->state); \ - } while (0); + } #define _STORE(_addr, _val) { \ s.addr = _addr; s.data = _val; s.w = 1; cpu->cycle++; \ pt_yield(cpu->state); \ - } while (0); + } + -#define _SET_P(_byte) { \ - for (int _pi = 0; _pi < 8; _pi++) \ - cpu->P.P[_pi] = _pi == B_B || _pi == B_X || \ - ((_byte) & (1 << _pi)); \ - } #define _NZC(_val) { \ uint16_t v = (_val); \ cpu->P.N = !!(v & 0x80); \ cpu->P.Z = (v & 0xff) == 0; \ cpu->P.C = !!(v & 0xff00); \ - } while (0) + } #define _NZ(_val) { \ uint16_t v = (_val); \ cpu->P.N = !!(v & 0x80); \ cpu->P.Z = (v & 0xff) == 0; \ - } while (0) + } #define _C(_val) { \ cpu->P.C = !!(_val); \ - } while (0) + } mii_cpu_state_t mii_cpu_run( @@ -57,28 +56,30 @@ mii_cpu_run( mii_op_desc_t d = mii_cpu_op[cpu->IR].desc; pt_start(cpu->state); next_instruction: - if (s.reset) { + if (unlikely(s.reset)) { s.reset = 0; _FETCH(0xfffc); cpu->_P = s.data; _FETCH(0xfffd); cpu->_P |= s.data << 8; cpu->PC = cpu->_P; cpu->S = 0xFF; - _SET_P(0); + MII_SET_P(cpu, 0); } - if (s.irq) { + if (unlikely(s.irq) && cpu->P.I == 0) { + if (!cpu->IRQ) + cpu->IRQ = 1; + } + if (unlikely(cpu->IRQ)) { s.irq = 0; - cpu->P.P[B_B] = cpu->IRQ == 2; - cpu->IRQ = 1; - cpu->_D = cpu->PC + 1; + cpu->P.B = cpu->IRQ == 2; + cpu->_D = cpu->PC; _STORE(0x0100 | cpu->S--, cpu->_D >> 8); _STORE(0x0100 | cpu->S--, cpu->_D & 0xff); uint8_t p = 0; - for (int i = 0; i < 8; i++) - p |= (i == B_X || cpu->P.P[i]) << i; + MII_GET_P(cpu, p); _STORE(0x0100 | cpu->S--, p); - cpu->P.P[B_I] = 1; - } - if (cpu->IRQ) { + cpu->P.I = 1; + if (cpu->IRQ == 2) + cpu->P.D = 0; cpu->IRQ = 0; _FETCH(0xfffe); cpu->_P = s.data; _FETCH(0xffff); cpu->_P |= s.data << 8; @@ -95,7 +96,7 @@ next_instruction: d = mii_cpu_op[cpu->IR].desc; cpu->ir_log = (cpu->ir_log << 8) | cpu->IR; s.trap = cpu->trap && (cpu->ir_log & 0xffff) == cpu->trap; - if (s.trap) + if (unlikely(s.trap)) cpu->ir_log = 0; switch (d.mode) { case IMM: @@ -173,17 +174,20 @@ next_instruction: case 0x79: case 0x61: case 0x71: case 0x72: { // ADC // Handle adding in BCD with bit D - if (cpu->P.D) { - uint8_t lo = (cpu->A & 0x0f) + (cpu->_D & 0x0f) + !!cpu->P.C; + if (unlikely(cpu->P.D)) { + uint8_t D = cpu->_D; + uint8_t lo = (cpu->A & 0x0f) + (D & 0x0f) + !!cpu->P.C; if (lo > 9) lo += 6; - uint8_t hi = (cpu->A >> 4) + (cpu->_D >> 4) + (lo > 0x0f); - cpu->P.Z = ((uint8_t)(cpu->A + cpu->_D + cpu->P.C)) == 0; + uint8_t hi = (cpu->A >> 4) + (D >> 4) + (lo > 0x0f); + cpu->P.Z = ((uint8_t)(cpu->A + D + cpu->P.C)) == 0; // that is 6502 behaviour // cpu->P.N = !!(hi & 0xf8); - cpu->P.V = !!((!((cpu->A ^ cpu->_D) & 0x80) && + cpu->P.V = !!((!((cpu->A ^ D) & 0x80) && ((cpu->A ^ (hi << 4))) & 0x80)); if (hi > 9) hi += 6; cpu->P.C = hi > 15; + // printf("ADC %02x %02x C:%d %x%x\n", + // cpu->A, D, !!cpu->P.C, hi & 0xf, lo & 0xf); cpu->A = (hi << 4) | (lo & 0x0f); // THAT is 65c02 behaviour cpu->P.N = !!(cpu->A & 0x80); @@ -219,9 +223,11 @@ next_instruction: case 0xaf: case 0xbf: case 0xcf: case 0xdf: case 0xef: case 0xff: { // BBR/BBS +// printf(" BB%c%d vs %02x\n", d.s_bit_value ? 'S' : 'R', +// d.s_bit, cpu->_D); _FETCH(cpu->PC++); // relative branch if (((cpu->_D >> d.s_bit) & 1) == d.s_bit_value) { - cpu->_P = cpu->PC + (int8_t)cpu->_P; + cpu->_P = cpu->PC + (int8_t)s.data; cpu->cycle++; if ((cpu->_P & 0xff00) != (cpu->PC & 0xff00)) cpu->cycle++; @@ -231,7 +237,7 @@ next_instruction: case 0x90: case 0xB0: case 0xF0: case 0x30: case 0xD0: case 0x10: case 0x50: case 0x70: { // BCC, BCS, BEQ, BMI, BNE, BPL, BVC, BVS - if (d.s_bit_value == cpu->P.P[d.s_bit]) { + if (d.s_bit_value == MII_GET_P_BIT(cpu, d.s_bit)) { cpu->_P = cpu->PC + (int8_t)cpu->_P; cpu->cycle++; if ((cpu->_P & 0xff00) != (cpu->PC & 0xff00)) @@ -259,14 +265,16 @@ next_instruction: } break; case 0x00: { // BRK + // Turns out BRK is a 2 byte opcode, who knew? well that guy did: + // https://www.nesdev.org/the%20'B'%20flag%20&%20BRK%20instruction.txt#:~:text=A%20note%20on%20the%20BRK,opcode%2C%20and%20not%20just%201. + _FETCH(cpu->PC++); // cpu->cycle++; s.irq = 1; - cpu->IRQ = 2; // IRQ interrupt - // cpu->P.P[B_D] = 0; // 65c02 clears this + cpu->IRQ = 2; // BRK sort of IRQ interrupt } break; case 0x18: case 0xD8: case 0x58: case 0xB8: { // CLC, CLD, CLI, CLV _FETCH(cpu->PC); - cpu->P.P[d.s_bit] = 0; + MII_SET_P_BIT(cpu, d.s_bit, 0); } break; case 0xC9: case 0xC5: case 0xD5: case 0xCD: case 0xDD: case 0xD9: case 0xC1: case 0xD1: case 0xD2: @@ -338,10 +346,13 @@ next_instruction: cpu->PC = cpu->_P; } break; case 0x20: + // https://github.com/AppleWin/AppleWin/issues/1257 { // JSR - cpu->_D = cpu->PC - 1; - _STORE(0x0100 | cpu->S--, cpu->_D >> 8); - _STORE(0x0100 | cpu->S--, cpu->_D & 0xff); + _FETCH(cpu->PC++); cpu->_P = s.data; + _FETCH(0x0100 | cpu->S); + _STORE(0x0100 | cpu->S--, cpu->PC >> 8); + _STORE(0x0100 | cpu->S--, cpu->PC & 0xff); + _FETCH(cpu->PC++); cpu->_P |= s.data << 8; cpu->PC = cpu->_P; } break; case 0xA9: case 0xA5: case 0xB5: case 0xAD: case 0xBD: @@ -389,9 +400,8 @@ next_instruction: case 0x08: { // PHP uint8_t p = 0; - for (int i = 0; i < 8; i++) - p |= (i == B_X || cpu->P.P[i]) << i; - p |= (1 << B_B); + MII_GET_P(cpu, p); + p |= (1 << B_B) | (1 << B_X); _STORE(0x0100 | cpu->S--, p); cpu->cycle++; } break; case 0xDA: @@ -411,7 +421,7 @@ next_instruction: case 0x28: { // PLP _FETCH(0x0100 | ++cpu->S);cpu->cycle++; - _SET_P(s.data);cpu->cycle++; + MII_SET_P(cpu, s.data);cpu->cycle++; } break; case 0xFA: { // PLX @@ -459,14 +469,15 @@ next_instruction: } break; case 0x40: { // RTI - _FETCH(0x0100 | ((++cpu->S) & 0xff)); + _FETCH(cpu->PC); // dummy write + cpu->S++; _FETCH(0x0100 | cpu->S); for (int i = 0; i < 8; i++) - cpu->P.P[i] = i == B_B || (s.data & (1 << i)); + MII_SET_P_BIT(cpu, i, i == B_B || (s.data & (1 << i))); cpu->P._R = 1; - _FETCH(0x0100 | ((++cpu->S) & 0xff)); + cpu->S++; _FETCH(0x0100 | cpu->S); cpu->_P = s.data; - _FETCH(0x0100 | ((++cpu->S) & 0xff)); - cpu->_P = (s.data << 8) | (cpu->_P ); + cpu->S++; _FETCH(0x0100 | cpu->S); + cpu->_P |= s.data << 8; cpu->PC = cpu->_P; } break; case 0x60: @@ -481,17 +492,24 @@ next_instruction: case 0xF9: case 0xE1: case 0xF1: case 0xF2: { // SBC // Handle subbing in BCD with bit D - if (cpu->P.D) { - uint8_t lo = (cpu->A & 0x0f) - (cpu->_D & 0x0f) - !cpu->P.C; - if (lo & 0x10) lo -= 6; - uint8_t hi = (cpu->A >> 4) - (cpu->_D >> 4) - (lo & 0x10); - if (hi & 0x10) hi -= 6; - cpu->P.Z = ((uint8_t)(cpu->A - cpu->_D - !cpu->P.C)) == 0; - cpu->P.N = !!(hi & 0x8); - cpu->P.V = !!(((cpu->A ^ cpu->_D) & + if (unlikely(cpu->P.D)) { + uint8_t D = 0x99 - cpu->_D; + // verbatim ADC code here + uint8_t lo = (cpu->A & 0x0f) + (D & 0x0f) + !!cpu->P.C; + if (lo > 9) lo += 6; + uint8_t hi = (cpu->A >> 4) + (D >> 4) + (lo > 0x0f); + cpu->P.Z = ((uint8_t)(cpu->A + D + cpu->P.C)) == 0; + // that is 6502 behaviour +// cpu->P.N = !!(hi & 0xf8); + cpu->P.V = !!((!((cpu->A ^ D) & 0x80) && ((cpu->A ^ (hi << 4))) & 0x80)); - cpu->P.C = !(hi & 0x10); + if (hi > 9) hi += 6; + cpu->P.C = hi > 15; + // printf("SBC %02x %02x C:%d %x%x\n", + // cpu->A, D, !!cpu->P.C, hi & 0xf, lo & 0xf); cpu->A = (hi << 4) | (lo & 0x0f); + // THAT is 65c02 behaviour + cpu->P.N = !!(cpu->A & 0x80); } else { cpu->_D = (~cpu->_D) & 0xff; uint16_t sum = cpu->A + cpu->_D + !!cpu->P.C; @@ -504,7 +522,7 @@ next_instruction: } break; case 0x38: case 0xF8: case 0x78: { // SEC, SED, SEI - cpu->P.P[d.s_bit] = 1; + MII_SET_P_BIT(cpu, d.s_bit, 1); } break; case 0x85: case 0x95: case 0x8D: case 0x9D: case 0x99: case 0x81: case 0x91: case 0x92: diff --git a/src/mii_65c02.h b/src/mii_65c02.h index 7d370b7..1db6bfc 100644 --- a/src/mii_65c02.h +++ b/src/mii_65c02.h @@ -2,6 +2,13 @@ #include +/* + * This is pretty heavily dependant on the way bitfields are packed in + * bytes, so it's not technically portable; if you have problems using a + * strange compiler, undefine this and use the 'discrete' version below. + */ +#define MII_PACK_P + /* * State structure used to 'talk' to the CPU emulator. * It works like this: @@ -27,9 +34,9 @@ */ typedef union mii_cpu_state_t { struct { - uint32_t addr : 16, - data : 8, - w : 1, + uint16_t addr; + uint8_t data; + uint8_t w : 1, sync : 1, reset : 1, irq : 1, @@ -49,20 +56,30 @@ typedef struct mii_cpu_t { * are 'indirect' and load from memory will set _P and read in _D * so the opcode doesn't have to handle s.data at all */ uint16_t _D, _P; + +#ifdef MII_PACK_P + union { + struct { + uint8_t C:1, Z:1, I:1, D:1, B:1, _R:1, V:1, N:1; + }; + uint8_t P; + } P; +#else /* My experience with simavr shows that maintaining a 8 bits bitfield * for a status register is a lot slower than having discrete flags - * and 'constructing' the matching 8 biots register when needed */ + * and 'constructing' the matching 8 bits register when needed */ union { struct { uint8_t C, Z, I, D, B, _R, V, N; }; uint8_t P[8]; } P; +#endif uint16_t PC; uint8_t IR; uint8_t IRQ; // IRQ (0) or NMI (1) or BRK (2) uint8_t cycle; // for current instruction - /* State of the CPU state machine */ + /* State of the protothread for the CPU state machine (minipt.h) */ void * state; /* sequence of instruction that will trigger a trap flag. @@ -73,8 +90,7 @@ typedef struct mii_cpu_t { // last 4 instructions, as a shift register, used for traps or debug uint32_t ir_log; - /* Debug only; the callback is called every cycles, with the current - * state of the cpu. */ + /* Debug only; Only used by the test units. */ uint8_t * ram; // DEBUG } mii_cpu_t; @@ -86,3 +102,35 @@ mii_cpu_state_t mii_cpu_run( mii_cpu_t *cpu, mii_cpu_state_t s); + + +#ifdef MII_PACK_P +#define MII_SET_P(_cpu, _byte) { \ + (_cpu)->P.P = _byte | 0x30; \ + } +#define MII_GET_P(_cpu, _res) \ + (_res) = (_cpu)->P.P +#define MII_SET_P_BIT(_cpu, _bit, _val) { \ + const int __bit = _bit; \ + (_cpu)->P.P = ((_cpu)->P.P & ~(1 << __bit)) | (!!(_val) << __bit); \ + } +#define MII_GET_P_BIT(_cpu, _bit) \ + !!(((_cpu)->P.P & (1 << (_bit)))) +#else +#define MII_SET_P(_cpu, _byte) { \ + const int __byte = _byte; \ + for (int _pi = 0; _pi < 8; _pi++) \ + (_cpu)->P.P[_pi] = _pi == B_B || _pi == B_X || \ + ((__byte) & (1 << _pi)); \ + } +#define MII_GET_P(_cpu, _res) { \ + (_res) = 0; \ + for (int _pi = 0; _pi < 8; _pi++) \ + (_res) |= (_cpu)->P.P[_pi] << _pi; \ + } +#define MII_SET_P_BIT(_cpu, _bit, _val) { \ + (_cpu)->P.P[_bit] = _val; \ + } +#define MII_GET_P_BIT(_cpu, _bit) \ + ((_cpu)->P.P[_bit]) +#endif diff --git a/src/mii_65c02_asm.c b/src/mii_65c02_asm.c index c31cf61..c5fe50d 100644 --- a/src/mii_65c02_asm.c +++ b/src/mii_65c02_asm.c @@ -150,7 +150,7 @@ mii_cpu_asm_load( * that can be resolved immediately */ char *kw = position; while (*kw == ' ' || *kw == '\t') kw++; - char *cur = kw; + char *cur = NULL; while ((cur = strsep(&kw, ",")) != NULL) { while (*cur == ' ' || *cur == '\t') cur++; char *ke = cur + strlen(cur); @@ -339,7 +339,7 @@ int mii_cpu_asm_assemble( mii_cpu_asm_program_t *p ) { - mii_cpu_asm_line_t *l = p->prog; + mii_cpu_asm_line_t *l = NULL; int error = 0; // fix symbols @@ -386,6 +386,12 @@ mii_cpu_asm_assemble( } else if (l->mode == mii_cpu_op[i].desc.mode) { found = i; break; + } else if (mii_cpu_op[i].desc.op == 0x20 && + l->mode == ABS) { + // this is JSR -- TECHNICALLY it's ABS mode, but + // it has to do a 'special' fetch so it's marked as + // implied in the table + found = i; } } } @@ -433,7 +439,7 @@ mii_cpu_asm_assemble( while (l2) { int32_t value = 0; if (!strcasecmp(l->op_name, l2->label)) { - value = l2->op_value; + // value = l2->op_value; if (!l2->symbol) value = l2->addr; else diff --git a/src/mii_65c02_disasm.c b/src/mii_65c02_disasm.c index 703c70f..bcf71cb 100644 --- a/src/mii_65c02_disasm.c +++ b/src/mii_65c02_disasm.c @@ -21,6 +21,10 @@ mii_cpu_disasm_one( mii_op_desc_t d = mii_cpu_op[op].desc; if (!d.pc) d.pc = 1; + // special case for JSR, it is marked as IMPLIED for execution, but is + // in fact ABSOLUTE for PC calculation + if (op == 0x20) + d.mode = ABS; *out = 0; int len = out_len; if (flags & MII_DUMP_DIS_PC) diff --git a/src/mii_65c02_ops.h b/src/mii_65c02_ops.h index a66a418..73ca68d 100644 --- a/src/mii_65c02_ops.h +++ b/src/mii_65c02_ops.h @@ -189,7 +189,7 @@ const mii_op_t mii_cpu_op[256] = { PCODE_PC(JMP, ABS, 0x4C, 3) PCODE_PC(JMP, IND, 0x6C, 3) PCODE_PC(JMP, IND_AX, 0x7C, 3) - PCODE_PC(JSR, ABS, 0x20, 3) + PCODE___(JSR, IMPLIED, 0x20, 3) PCODE___(LDA, IMM, 0xA9, 2) PCODE_R_(LDA, ZP_REL, 0xA5, 2) PCODE_R_(LDA, ZP_X, 0xB5, 2) diff --git a/src/mii_analog.c b/src/mii_analog.c index 335aa8b..7b87b05 100644 --- a/src/mii_analog.c +++ b/src/mii_analog.c @@ -14,6 +14,15 @@ #include "mii.h" #include "mii_analog.h" +/* + * Analog joystick + * This is fairly easy, as long as the 65c02 respects the proper cycle + * count for all the instruction involved in reading, as it's very cycle + * sensitive. + * the UI fills up the analog values in mii_t, and here we just simulate + * the capacitor decay. + */ + void mii_analog_init( struct mii_t *mii, @@ -37,17 +46,35 @@ mii_analog_access( return; switch (addr) { case 0xc070: { - // multiplying by mii->speed allows reading joystick in 'fast' mode, - // this basically simulate slowing down just for the joystick reading + /* + * No need starting the cycle timers when nobody cares about + * the analog values aka joysticks + */ + if (!a->enabled) { + a->enabled = true; + /* + * No need for a function pointer here for the timer, the + * decrementing value is just what we need, and we're quite + * happy to stop at ~0 as well. + */ + for (int i = 0; i < 4; i++) + a->v[i].timer_id = mii_timer_register(mii, + NULL, NULL, 0, __func__); + } + /* + * Multiplying by mii->speed allows reading joystick in + * 'fast' emulation mode, this basically simulate slowing down + * just for the joystick reading + */ for (int i = 0; i < 4; i++) { - a->v[i].decay = mii->cycles + - ((a->v[i].value * 11) * mii->speed); + mii_timer_set(mii, a->v[i].timer_id, + ((a->v[i].value * 11) * mii->speed)); // printf("joystick %d: %d\n", i, a->v[i].value); } } break; case 0xc064 ... 0xc067: { addr -= 0xc064; - *byte = mii->cycles <= a->v[addr].decay ? 0x80 : 0x00; + *byte = mii_timer_get(mii, a->v[addr].timer_id) > 0 ? 0x80 : 0x00; } break; } } diff --git a/src/mii_analog.h b/src/mii_analog.h index 4f4aab8..3fedb44 100644 --- a/src/mii_analog.h +++ b/src/mii_analog.h @@ -7,14 +7,15 @@ */ #pragma once -#include "mii_types.h" - +#include typedef struct mii_analog_t { struct { uint8_t value; - mii_cycles_t decay; +// mii_cycles_t decay; + uint8_t timer_id; } v[4]; + bool enabled; } mii_analog_t; struct mii_t; diff --git a/src/mii_argv.c b/src/mii_argv.c index c9e59b8..db85c99 100644 --- a/src/mii_argv.c +++ b/src/mii_argv.c @@ -150,7 +150,7 @@ mii_argv_parse( } else if (!strcmp(arg, "-speed") || !strcmp(arg, "--speed")) { if (i < argc-1) { mii->speed = atof(argv[++i]); - if (mii->speed < 0.0) + if (mii->speed <= 0.0) mii->speed = 1.0; } else { printf("mii: missing speed value\n"); @@ -201,4 +201,4 @@ mii_argv_parse( } *index = argc; return 1; -} +} \ No newline at end of file diff --git a/src/mii_bank.c b/src/mii_bank.c index be21ccc..3a3894f 100644 --- a/src/mii_bank.c +++ b/src/mii_bank.c @@ -14,6 +14,16 @@ #include "mii.h" #include "mii_bank.h" +void +mii_bank_init( + mii_bank_t *bank) +{ + if (bank->mem) + return; + bank->mem = calloc(1, bank->size * 256); + bank->alloc = 1; +} + void mii_bank_dispose( mii_bank_t *bank) @@ -39,8 +49,8 @@ mii_bank_write( const uint8_t *data, uint16_t len) { - if ((addr < bank->base) || - ((addr + len) > (uint32_t)(bank->base + (bank->size * 256)))) { + uint32_t end = bank->base + (bank->size << 8); + if (unlikely(addr < bank->base) || unlikely((addr + len) > end)) { printf("%s %s INVALID write addr %04x len %d %04x:%04x\n", __func__, bank->name, addr, (int)len, bank->base, bank->base + (bank->size * 256)); @@ -52,14 +62,10 @@ mii_bank_write( addr, (uint8_t *)data, true)) return; } - if (!bank->mem) { - bank->alloc = 1; - bank->mem = calloc(1, bank->size * 256); - } addr -= bank->base; - for (uint16_t i = 0; i < len; i++, addr++) { - bank->mem[addr] = data[i]; - } + do { + bank->mem[addr++] = *data++; + } while (unlikely(--len)); } void @@ -69,27 +75,25 @@ mii_bank_read( uint8_t *data, uint16_t len) { - if (addr < bank->base || - (addr + len) > (uint32_t)(bank->base + (bank->size * 256))) { + #if 0 // rather expensive test when profiling! + uint32_t end = bank->base + (bank->size << 8); + if (unlikely(addr < bank->base) || unlikely((addr + len) > end)) { printf("%s %s INVALID read addr %04x len %d %04x-%04x\n", __func__, bank->name, addr, (int)len, bank->base, bank->base + (bank->size * 256)); return; } + #endif uint8_t page_index = (addr - bank->base) >> 8; if (bank->access && bank->access[page_index].cb) { if (bank->access[page_index].cb(bank, bank->access[page_index].param, addr, data, false)) return; } - if (!bank->mem) { - bank->alloc = 1; - bank->mem = calloc(1, bank->size * 256); - } addr -= bank->base; - for (uint16_t i = 0; i < len; i++, addr++) { - data[i] = bank->mem[addr]; - } + do { + *data++ = bank->mem[addr++]; + } while (unlikely(--len)); } @@ -113,8 +117,8 @@ mii_bank_install_access_cb( if (!bank->access) { bank->access = calloc(1, bank->size * sizeof(bank->access[0])); } - printf("%s %s install access cb page %02x:%02x\n", - __func__, bank->name, page, end); +// printf("%s %s install access cb page %02x:%02x\n", +// __func__, bank->name, page, end); for (int i = page; i <= end; i++) { bank->access[i].cb = cb; bank->access[i].param = param; diff --git a/src/mii_bank.h b/src/mii_bank.h index 8daa928..0603ee5 100644 --- a/src/mii_bank.h +++ b/src/mii_bank.h @@ -45,6 +45,9 @@ typedef struct mii_bank_t { uint8_t *mem; } mii_bank_t; +void +mii_bank_init( + mii_bank_t *bank); void mii_bank_dispose( mii_bank_t *bank); diff --git a/nuklear/mii_mish.c b/src/mii_mish.c similarity index 90% rename from nuklear/mii_mish.c rename to src/mii_mish.c index 0a9063f..efb03f3 100644 --- a/nuklear/mii_mish.c +++ b/src/mii_mish.c @@ -55,21 +55,18 @@ _mii_mish_cmd( mii_t * mii = param; if (!argv[1]) { show_state: - printf("mii: %s Target speed: %.3fMHz Current: %.3fMHz\n", - state[mii->state], mii->speed, mii->speed_current); + printf("mii: %s Target speed: %.3fMHz\n", + state[mii->state], mii->speed); mii_dump_run_trace(mii); mii_dump_trace_state(mii); - mii_bank_t * main = &mii->bank[MII_BANK_MAIN]; - bool text = !!mii_bank_peek(main, SWTEXT); - bool page2 = !!mii_bank_peek(main, SWPAGE2); - bool col80 = !!mii_bank_peek(main, SW80COL); - bool mixed = !!mii_bank_peek(main, SWMIXED); - bool hires = !!mii_bank_peek(main, SWHIRES); - bool dhires = !!mii_bank_peek(main, SWRDDHIRES); - - printf("text:%d page2:%d col80:%d mixed:%d hires:%d dhires:%d\n", - text, page2, col80, mixed, hires, dhires); + for (int i = 0; mii_sw_names[i]; i++) { + char buf[32]; + sprintf(buf, "%s:%d", mii_sw_names[i], + !!(mii->sw_state & (1 << i))); + printf("%-13.13s%s", buf, !(i % 6) ? "\n" : " "); + } + printf("\n"); return; } if (!strcmp(argv[1], "reset")) { @@ -77,6 +74,10 @@ show_state: return; } if (!strcmp(argv[1], "mem")) { + printf("mii: memory map: "); + for (int i = 0; i < MII_BANK_COUNT; i++) + printf("%0d:%s ", i, mii->bank[i].name); + printf("\n"); for (int i = 0; i < 16; i++) { printf("%02x: ", i * 16); for (int j = 0; j < 16; j++) @@ -104,7 +105,7 @@ show_state: } uint16_t addr = strtol(argv[2], NULL, 16); uint8_t val = strtol(argv[3], NULL, 16); - mii_mem_access(mii, addr, &val, false, true); + mii_mem_access(mii, addr, &val, true, true); return; } if (!strcmp(argv[1], "peek")) { @@ -114,7 +115,7 @@ show_state: } uint16_t addr = strtol(argv[2], NULL, 16); uint8_t val; - mii_mem_access(mii, addr, &val, false, false); + mii_mem_access(mii, addr, &val, false, true); printf("%04x: %02x\n", addr, val); return; } @@ -141,6 +142,17 @@ show_state: printf("mii: terminating\n"); return; } + if (!strcmp(argv[1], "timers")) { + uint64_t timer = mii->timer.map; + printf("mii: %d cycle timers\n", __builtin_popcountll(timer)); + while (timer) { + int i = ffsll(timer) - 1; + timer &= ~(1ull << i); + printf("%2d: %8ld %s\n", i, mii->timer.timers[i].when, + mii->timer.timers[i].name); + } + return; + } printf("mii: unknown command %s\n", argv[1]); } @@ -277,6 +289,10 @@ _mii_mish_step( { if (argv[0][0] == 's') { mii_t * mii = param; + if (mii->state != MII_STOPPED) { + printf("mii: can't step/next, not stopped\n"); + return; + } if (argv[1]) { int n = strtol(argv[1], NULL, 10); mii->trace.step_inst = n; @@ -287,13 +303,17 @@ _mii_mish_step( } if (argv[0][0] == 'n') { mii_t * mii = param; + if (mii->state != MII_STOPPED) { + printf("mii: can't step/next, not stopped\n"); + return; + } // read current opcode, find how how many bytes it take, // then put a temporary breakpoint to the next PC. // all of that if this is not a relative branch of course, in // which case we use a normal 'step' behaviour uint8_t op; mii_mem_access(mii, mii->cpu.PC, &op, false, false); - if (op == 0x20) { + if (op == 0x20) { // JSR here? // set a temp breakpoint on reading 3 bytes from PC for (int i = 0; i < (int)sizeof(mii->debug.bp_map) * 8; i++) { if ((mii->debug.bp_map & (1 << i))) @@ -337,7 +357,9 @@ _mii_mish_audio( { mii_t * mii = param; if (argc < 2) { - printf("audio: missing argument\n"); + printf("audio volume: %.3f multiplier:%.3f muted:%d\n", + mii->speaker.volume, mii->speaker.vol_multiplier, + mii->speaker.muted); return; } if (!strcmp(argv[1], "record")) { diff --git a/nuklear/mii_mish_dd.c b/src/mii_mish_dd.c similarity index 90% rename from nuklear/mii_mish_dd.c rename to src/mii_mish_dd.c index c605f92..1593cf2 100644 --- a/nuklear/mii_mish_dd.c +++ b/src/mii_mish_dd.c @@ -23,6 +23,7 @@ _mii_mish_dd( mii_t * mii = param; if (!argv[1] || !strcmp(argv[1], "list")) { mii_dd_t *d = mii->dd.drive; + printf("dd %p %p drives\n", &mii->dd, mii->dd.drive); printf(" ID %-16s %-20s\n", "Card", "Name"); while (d) { printf("%d:%d %-16s %-20s : %s\n", @@ -42,4 +43,4 @@ MISH_CMD_HELP(dd, "mii: disk commands", " |list: list all disk drives" ); -MII_MISH(dd, _mii_mish_dd); +MII_MISH(dd, _mii_mish_dd); \ No newline at end of file diff --git a/src/mii_slot.h b/src/mii_slot.h index b083b71..cda9ba5 100644 --- a/src/mii_slot.h +++ b/src/mii_slot.h @@ -12,8 +12,8 @@ typedef struct mii_slot_drv_t mii_slot_drv_t; typedef struct mii_slot_t { - uint8_t id; - void * drv_priv; // for driver use + uint8_t aux_rom_selected: 1, id; + void * drv_priv; // for driver use const mii_slot_drv_t * drv; } mii_slot_t; @@ -27,11 +27,24 @@ typedef struct mii_slot_drv_t { uint32_t enable_flag; // if there is a MII_INIT_xxx flag const char * name; const char * desc; - int (*probe)(mii_t * mii, uint32_t flags); - int (*init)(mii_t * mii, struct mii_slot_t *slot); - void (*dispose)(mii_t * mii, struct mii_slot_t *slot); /* optional */ - void (*reset)(mii_t * mii, struct mii_slot_t *slot); /* optional */ - void (*run)(mii_t * mii, struct mii_slot_t *slot); /* optional */ + int (*probe)( + mii_t * mii, + uint32_t flags); + int (*init)( + mii_t * mii, + struct mii_slot_t *slot); + /* optional */ + void (*dispose)( + mii_t * mii, + struct mii_slot_t *slot); + /* optional */ + void (*reset)( + mii_t * mii, + struct mii_slot_t *slot); + void (*run)( + mii_t * mii, + struct mii_slot_t *slot); + // access to the slot's soft switches. uint8_t (*access)( mii_t * mii, struct mii_slot_t *slot, @@ -64,6 +77,8 @@ mii_slot_drv_find( enum { MII_SLOT_DRIVE_COUNT = 0x01, MII_SLOT_DRIVE_LOAD = 0x20, // + drive index 0...n + + MII_SLOT_SSC_SET_TTY = 0x10, // param is a pathname, or NULL for a pty }; // send a command to a slot/driver. Return >=0 if ok, -1 if error diff --git a/src/mii_speaker.c b/src/mii_speaker.c index 73f9016..bc37d41 100644 --- a/src/mii_speaker.c +++ b/src/mii_speaker.c @@ -81,6 +81,11 @@ _alsa_init( } #endif +static uint64_t +_mii_speaker_timer_cb( + mii_t * mii, + void * param ); + // Initialize the speaker with the frame size in samples void mii_speaker_init( @@ -90,32 +95,86 @@ mii_speaker_init( s->mii = mii; s->debug_fd = -1; s->fsize = MII_SPEAKER_FRAME_SIZE; + // disabled at start... + s->timer_id = mii_timer_register(mii, + _mii_speaker_timer_cb, s, 0, __func__); #ifdef HAS_ALSA if (!s->off) _alsa_init(s); // this can/will change fsize #endif - s->vol_multiplier = 0.2; + mii_speaker_volume(s, 1); s->sample = 0x8000; s->findex = 0; for (int i = 0; i < MII_SPEAKER_FRAME_COUNT; i++) s->frame[i].audio = calloc(sizeof(s->frame[i].audio[0]), s->fsize); - s->frame[0].start = mii->cycles; +// s->frame[0].start = mii->cycles; } void mii_speaker_dispose( - mii_speaker_t *speaker) + mii_speaker_t *s) { + s->fsize = 0; + mii_timer_set(s->mii, s->timer_id, 0); #ifdef HAS_ALSA - if (speaker->alsa_pcm) - snd_pcm_close(speaker->alsa_pcm); + if (s->alsa_pcm) + snd_pcm_close(s->alsa_pcm); #endif for (int i = 0; i < MII_SPEAKER_FRAME_COUNT; i++) { - free(speaker->frame[i].audio); - speaker->frame[i].audio = NULL; + free(s->frame[i].audio); + s->frame[i].audio = NULL; } } +// Check to see if there's a new frame to send, send it +// this timer is always running; it keeps checking for non-empty frames +static uint64_t +_mii_speaker_timer_cb( + mii_t * mii, + void * param ) +{ + mii_speaker_t *s = (mii_speaker_t *)param; + + if (s->muted || s->off) + goto done; + mii_audio_frame_t *f = &s->frame[s->fplay]; + // if the frame is empty, we mark the fact we are in underrun, + // so we can restart the audio later on. + if (!f->fill) { + if (s->under < 10) + s->under++; + goto done; + } + s->under = 0; + // Here we got a frame to play, so we play it, and move on to the next + // There's also the case were we stopped playing and the last frame + // wasn't complete, in which case we pad it, and flush it as well +// printf("%s: fplay %d findex %d fsize %d fill %d\n", +// __func__, s->fplay, s->findex, s->fsize, f->fill); + uint16_t sample = f->audio[f->fill - 1] ^ 0xffff; + while (f->fill < s->fsize) + f->audio[f->fill++] = sample; + s->fplay = (s->fplay + 1) % MII_SPEAKER_FRAME_COUNT; + s->frame[s->fplay].fill = 0; + if (!s->muted) { + if (s->debug_fd != -1) + write(s->debug_fd, f->audio, + f->fill * sizeof(s->frame[0].audio[0])); +#ifdef HAS_ALSA + if (s->alsa_pcm) { + int pcm; + if ((pcm = snd_pcm_writei(s->alsa_pcm, + f->audio, f->fill)) == -EPIPE) { + printf("%s Underrun.\n", __func__); + snd_pcm_recover(s->alsa_pcm, pcm, 1); + } + } +#endif + } +done: + return s->fsize * s->clk_per_sample; +} + // Called when $c030 is touched, place a sample at the 'appropriate' time void mii_speaker_click( @@ -126,18 +185,20 @@ mii_speaker_click( s->cpu_speed = s->mii->speed; s->clk_per_sample = ((1000000.0 /* / s->mii->speed */) / (float)MII_SPEAKER_FREQ) + 0.5f; - printf("%s: %d cycles per sample\n", __func__, s->clk_per_sample); + printf("%s: %.2f cycles per sample\n", __func__, s->clk_per_sample); + mii_timer_set(s->mii, s->timer_id, s->fsize * s->clk_per_sample); } + int64_t remains = mii_timer_get(s->mii, s->timer_id); + mii_audio_frame_t *f = &s->frame[s->findex]; // if we had stopped playing for 2 frames, restart - if (f->start == 0 || - (s->mii->cycles - f->start) > (2 * s->fsize * s->clk_per_sample)) { + if (s->under > 1) { + s->under = 0; // printf("Restarting playback\n"); #ifdef HAS_ALSA if (s->alsa_pcm) snd_pcm_prepare(s->alsa_pcm); #endif - f->start = s->mii->cycles - (s->clk_per_sample * 8); f->fill = 0; // add a small attack to the start of the frame to soften the beeps // we are going to flip the sample, so we need to preemptively @@ -146,20 +207,24 @@ mii_speaker_click( for (int i = 8; i >= 1; i--) f->audio[f->fill++] = (attack / i) * s->vol_multiplier; s->fplay = s->findex; // restart here + mii_timer_set(s->mii, s->timer_id, (s->fsize - 16) * s->clk_per_sample); + remains = mii_timer_get(s->mii, s->timer_id); } - - long sample_index = (s->mii->cycles - f->start) / s->clk_per_sample; + // calculate the sample index we are going to fill -- this is relative + // to the frame we are waiting to play + long sample_index = // (s->fsize / 2) + + (((s->fsize * s->clk_per_sample) - remains) / + s->clk_per_sample); // fill from last sample to here with the current sample for (; f->fill < sample_index && f->fill < s->fsize; f->fill++) f->audio[f->fill] = s->sample * s->vol_multiplier; - +// printf("%s: findex %d fsize %d fill %d sample_index %ld\n", +// __func__, s->findex, s->fsize, f->fill, sample_index); // if we've gone past the end of the frame, switch to the next one - if (sample_index >= s->fsize) { + if (sample_index >= s->fsize || sample_index < f->fill) { sample_index = sample_index % s->fsize; - mii_cycles_t newstart = s->mii->cycles - (sample_index * s->clk_per_sample); s->findex = (s->findex + 1) % MII_SPEAKER_FRAME_COUNT; f = &s->frame[s->findex]; - f->start = newstart; f->fill = 0; // fill from start of this frame to newly calculated sample_index for (; f->fill < sample_index && f->fill < s->fsize; f->fill++) @@ -168,50 +233,22 @@ mii_speaker_click( } s->sample ^= 0xffff; // if we are touching a new sample, make sure the next one is clear too. +#if 1 + int32_t mix = s->sample * s->vol_multiplier; +#else + /* Mixing code; in case theres multiple clicks within a sample period. + * This is not used, because it's not really needed, as 2 clicks + * would cancel each others anyway. */ if (!f->audio[sample_index] && sample_index < s->fsize) f->audio[sample_index + 1] = 0; int32_t mix = f->audio[sample_index] + (s->sample * s->vol_multiplier); if (mix > 0x7fff) mix = 0x7fff; else if (mix < -0x8000) mix = -0x8000; +#endif f->audio[sample_index] = mix; } -// Check to see if there's a new frame to send, send it -void -mii_speaker_run( - mii_speaker_t *s) -{ - mii_audio_frame_t *f = &s->frame[s->fplay]; - - // here we check if the frame we want to play is filled, and if it's - // been long enough since we started filling it to be sure we have - // enough samples to play. - // There's also the case were we stopped playing and the last frame - // wasn't complete, in which case we flush it as well - if (f->fill && ((s->mii->cycles - f->start) > - (s->fsize * s->clk_per_sample * 2))) { - s->fplaying = s->fplay; - s->fplay = (s->fplay + 1) % MII_SPEAKER_FRAME_COUNT; - f = &s->frame[s->fplaying]; - if (!s->muted) { - if (s->debug_fd != -1) - write(s->debug_fd, f->audio, - f->fill * sizeof(s->frame[0].audio[0])); -#ifdef HAS_ALSA - if (s->alsa_pcm) { - int pcm; - if ((pcm = snd_pcm_writei(s->alsa_pcm, - f->audio, f->fill)) == -EPIPE) { - printf("%s Underrun.\n", __func__); - snd_pcm_recover(s->alsa_pcm, pcm, 1); - } - } -#endif - } - f->fill = 0; - } -} // this is here so we dont' have to drag in libm math library. double fastPow(double a, double b) { @@ -234,4 +271,4 @@ mii_speaker_volume( s->volume = volume; // printf("audio: speaker volume set to %.3f (%.4f)\n", volume, mul); -} +} \ No newline at end of file diff --git a/src/mii_speaker.h b/src/mii_speaker.h index 37a2a76..4585701 100644 --- a/src/mii_speaker.h +++ b/src/mii_speaker.h @@ -17,13 +17,14 @@ struct snd_pcm_t; typedef int16_t mii_audio_sample_t; typedef struct mii_audio_frame_t { - mii_cycles_t start; +// mii_cycles_t start; uint16_t fill; mii_audio_sample_t * audio; } mii_audio_frame_t; typedef struct mii_speaker_t { struct mii_t * mii; + uint8_t timer_id; // timer id for the audio thread int debug_fd; // if > 0, dump audio to this fd void * alsa_pcm; // alsa pcm handle bool muted; // if true, don't play anything @@ -31,11 +32,11 @@ typedef struct mii_speaker_t { float volume; // volume, 0.0 to 10.0 float vol_multiplier; // sample multiplier, 0.0 to 1.0 float cpu_speed; // CPU speed in MHz, to calculate clk_per_sample + uint8_t under; // number of frames we've skipped uint8_t fplay; // frame we want to play - uint8_t fplaying; // index of the current playing frame uint16_t fsize; // size in samples of a frame uint8_t findex; // frame we are currently filling - uint16_t clk_per_sample; // number of cycles per sample (at current CPU speed) + float clk_per_sample; // number of cycles per sample (at current CPU speed) mii_audio_sample_t sample; // current value for the speaker output mii_audio_frame_t frame[MII_SPEAKER_FRAME_COUNT]; } mii_speaker_t; @@ -53,9 +54,9 @@ void mii_speaker_click( mii_speaker_t *speaker); // Check to see if there's a new frame to send, send it -void -mii_speaker_run( - mii_speaker_t *speaker); +//void +//mii_speaker_run( +// mii_speaker_t *speaker); // volume from 0 to 10, sets the audio sample multiplier. void mii_speaker_volume( diff --git a/src/mii_sw.h b/src/mii_sw.h index a0727c1..1b56843 100644 --- a/src/mii_sw.h +++ b/src/mii_sw.h @@ -85,27 +85,33 @@ enum { B_BSRREAD = (13), B_BSRPAGE2 = (14), B_SWDHIRES = (15), + // this is no 'real' softwitch, but a bit to mention a card has + // it's secondary rom online in pages c800-cfff + B_SLOTAUXROM = (16), - M_SW80STORE = (1 << B_SW80STORE), - M_SWALTCHARSET = (1 << B_SWALTCHARSET), - M_SW80COL = (1 << B_SW80COL), - M_SWTEXT = (1 << B_SWTEXT), - M_SWMIXED = (1 << B_SWMIXED), - M_SWPAGE2 = (1 << B_SWPAGE2), - M_SWHIRES = (1 << B_SWHIRES), - M_SWRAMRD = (1 << B_SWRAMRD), - M_SWRAMWRT = (1 << B_SWRAMWRT), - M_SWINTCXROM = (1 << B_SWINTCXROM), - M_SWALTPZ = (1 << B_SWALTPZ), - M_SWSLOTC3ROM = (1 << B_SWSLOTC3ROM), - M_BSRWRITE = (1 << B_BSRWRITE), - M_BSRREAD = (1 << B_BSRREAD), - M_BSRPAGE2 = (1 << B_BSRPAGE2), - M_SWDHIRES = (1 << B_SWDHIRES), + M_SW80STORE = (1 << B_SW80STORE), + M_SWALTCHARSET = (1 << B_SWALTCHARSET), + M_SW80COL = (1 << B_SW80COL), + M_SWTEXT = (1 << B_SWTEXT), + M_SWMIXED = (1 << B_SWMIXED), + M_SWPAGE2 = (1 << B_SWPAGE2), + M_SWHIRES = (1 << B_SWHIRES), + M_SWRAMRD = (1 << B_SWRAMRD), + M_SWRAMWRT = (1 << B_SWRAMWRT), + M_SWINTCXROM = (1 << B_SWINTCXROM), + M_SWALTPZ = (1 << B_SWALTPZ), + M_SWSLOTC3ROM = (1 << B_SWSLOTC3ROM), + M_BSRWRITE = (1 << B_BSRWRITE), + M_BSRREAD = (1 << B_BSRREAD), + M_BSRPAGE2 = (1 << B_BSRPAGE2), + M_SWDHIRES = (1 << B_SWDHIRES), + M_SLOTAUXROM = (1 << B_SLOTAUXROM), }; +#define __unused__ __attribute__((unused)) + // unused is to prevent the stupid warnings about unused static stuff -static const char __attribute__((unused)) *mii_sw_names[] = { +static const char __unused__ *mii_sw_names[] = { "80STORE", "ALTCHARSET", "80COL", @@ -122,6 +128,7 @@ static const char __attribute__((unused)) *mii_sw_names[] = { "BSRREAD", "BSRPAGE2", "DHIRES", + "AUXROMON", NULL, } ; diff --git a/src/mii_types.h b/src/mii_types.h deleted file mode 100644 index c6a45b1..0000000 --- a/src/mii_types.h +++ /dev/null @@ -1,14 +0,0 @@ -/* - * mii_types.h - * - * Copyright (C) 2023 Michel Pollet - * - * SPDX-License-Identifier: MIT - */ - -#pragma once - -#include -#include - -typedef __uint128_t mii_cycles_t; diff --git a/src/mii_video.c b/src/mii_video.c index 1e59b10..7b7c223 100644 --- a/src/mii_video.c +++ b/src/mii_video.c @@ -36,6 +36,8 @@ enum { /* * Colors were lifted from * https://comp.sys.apple2.narkive.com/lTSrj2ZI/apple-ii-colour-rgb + * and + * https://www.mrob.com/pub/xapple2/colors.html */ #define HI_RGB(r,g,b) (0xff000000 | ((r) << 16) | ((g) << 8) | (b)) static const uint32_t lores_colors[] = { @@ -106,7 +108,7 @@ static const uint32_t mono[3][2] = { /* this 'dims' the colors for every second line of pixels */ -#define C_SCANLINE_MASK 0xffc0c0c0 +#define C_SCANLINE_MASK HI_RGB(0xc0, 0xc0, 0xc0) static inline uint16_t _mii_line_to_video_addr( @@ -118,6 +120,14 @@ _mii_line_to_video_addr( ((line >> 6) << 5) | ((line >> 6) << 3); return addr; } + +unsigned char reverse(unsigned char b) { + b = (b & 0b11110000) >> 4 | (b & 0b00001111) << 4; + b = (b & 0b11001100) >> 2 | (b & 0b00110011) << 2; + b = (b & 0b10101010) >> 1 | (b & 0b01010101) << 1; + return b; +} + /* * This is the state of the video output * All timings lifted from https://rich12345.tripod.com/aiivideo/vbl.html @@ -132,19 +142,20 @@ _mii_line_to_video_addr( * static or global. * *everything* before the pt_start call is ran every time, so you can use * that to reload some sort of state, as here, were we reload all the - * video mode softwsitches. + * video mode softswitches. + * + * This function is also a 'cycle timer' it returns the number of 6502 + * cycles to wait until being called again, so it mostly returns the + * number of cycles until the next horizontal blanking between each lines, + * but also the number of cycles until the next vertical blanking once + * the last line is drawn. */ -void -mii_video_run( - mii_t *mii) +static uint64_t +mii_video_timer_cb( + mii_t *mii, + void *param) { - // no need to do anything, we're waiting cycles - if (mii->video.wait) { - if (mii->video.wait > mii->cycles) - return; - // extra cycles we waited are kept around for next delay - mii->video.wait = mii->cycles - mii->video.wait; - } + uint64_t res = MII_VIDEO_H_CYCLES * mii->speed; mii_bank_t * main = &mii->bank[MII_BANK_MAIN]; bool text = SW_GETSTATE(mii, SWTEXT); bool page2 = SW_GETSTATE(mii, SWPAGE2); @@ -179,14 +190,15 @@ mii_video_run( mii_bank_t * aux = &mii->bank[MII_BANK_AUX]; if (reg == 0 || mii->video.color_mode != MII_VIDEO_COLOR) { - for (int x = 0; x < 40; x += 1) { + const uint32_t clut[2] = { + mono[mii->video.color_mode][0], + mono[mii->video.color_mode][1] }; + for (int x = 0; x < 40; x++) { uint32_t ext = (mii_bank_peek(aux, a + x) & 0x7f) | ((mii_bank_peek(main, a + x) & 0x7f) << 7); for (int bi = 0; bi < 14; bi++) { uint8_t pixel = (ext >> bi) & 1; - uint32_t col = pixel ? - mono[mii->video.color_mode][1] : - mono[mii->video.color_mode][0]; + uint32_t col = clut[pixel]; *screen++ = col; *l2++ = col & C_SCANLINE_MASK; } @@ -233,8 +245,8 @@ mii_video_run( uint16_t run = ((b0 & 0x60) >> ( 5 )) | ((b1 & 0x7f) << ( 2 )) | ((b2 & 0x03) << ( 9 )); - int odd = (x & 1) << 1; - int offset = (b1 & 0x80) >> 5; + int odd = (x & 1) << 1; + int offset = (b1 & 0x80) >> 5; if (mii->video.color_mode == MII_VIDEO_COLOR) { for (int i = 0; i < 7; i++) { @@ -263,9 +275,7 @@ mii_video_run( } else { for (int i = 0; i < 7; i++) { uint8_t pixel = (run >> (2 + i)) & 1; - uint32_t col = pixel ? - mono[mii->video.color_mode][1] : - mono[mii->video.color_mode][0]; + uint32_t col = mono[mii->video.color_mode][pixel]; *screen++ = col; *screen++ = col; *l2++ = col & C_SCANLINE_MASK; @@ -298,9 +308,7 @@ mii_video_run( uint8_t bits = rom[mii->video.line & 0x07]; for (int pi = 0; pi < 7; pi++) { uint8_t pixel = (bits >> pi) & 1; - uint32_t col = pixel ? - mono[mii->video.color_mode][0] : - mono[mii->video.color_mode][1]; + uint32_t col = mono[mii->video.color_mode][!pixel]; *screen++ = col; *l2++ = col & C_SCANLINE_MASK; if (!col80) { @@ -326,25 +334,20 @@ mii_video_run( mii->video.line++; if (mii->video.line == 192) { mii->video.line = 0; - mii->video.wait = mii->cycles - mii->video.wait + - MII_VIDEO_H_CYCLES * mii->speed; + res = MII_VIDEO_H_CYCLES * mii->speed; pt_yield(mii->video.state); mii_bank_poke(main, SWVBL, 0x00); - if (mii->video.vbl_irq) - mii->cpu_state.irq = 1; - mii->video.wait = mii->cycles - mii->video.wait + - MII_VBL_UP_CYCLES * mii->speed; + res = MII_VBL_UP_CYCLES * mii->speed; mii->video.frame_count++; pt_yield(mii->video.state); } else { - mii->video.wait = mii->cycles - mii->video.wait + - (MII_VIDEO_H_CYCLES + MII_VIDEO_HB_CYCLES) * + res = (MII_VIDEO_H_CYCLES + MII_VIDEO_HB_CYCLES) * mii->speed; pt_yield(mii->video.state); } } while (1); pt_end(mii->video.state); - return; + return res; } bool @@ -418,3 +421,11 @@ mii_access_video( } return res; } + +void +mii_video_init( + mii_t *mii) +{ + mii->video.timer_id = mii_timer_register(mii, + mii_video_timer_cb, NULL, MII_VIDEO_H_CYCLES, __func__); +} \ No newline at end of file diff --git a/src/mii_video.h b/src/mii_video.h index 1be80df..fc6d14a 100644 --- a/src/mii_video.h +++ b/src/mii_video.h @@ -8,18 +8,15 @@ #pragma once -#include "mii_types.h" +#include +#include // TODO move VRAM stuff to somewhere else -/* - These are currently way bigger than needed, because they need to be - a power of two to easily map to an opengl texture. - */ -#define MII_VRAM_WIDTH 1024 -#define MII_VRAM_HEIGHT 512 - #define MII_VIDEO_WIDTH (280 * 2) #define MII_VIDEO_HEIGHT (192 * 2) +// in case padding is needed, these can be changed +#define MII_VRAM_WIDTH (MII_VIDEO_WIDTH * 4) +#define MII_VRAM_HEIGHT MII_VIDEO_HEIGHT enum { MII_VIDEO_COLOR = 0, @@ -30,14 +27,13 @@ enum { struct mii_t; typedef struct mii_video_t { - void * state; // protothread state in mii_video.c - uint8_t line; // current line - bool vbl_irq; // VBL IRQ emabled (set by mouse card) - mii_cycles_t wait; // 'wait until' cycle marker - uint32_t pixels[MII_VRAM_WIDTH * MII_VRAM_HEIGHT]; - uint32_t frame_count; // incremented every frame - uint32_t frame_drawn; - uint8_t color_mode; // color, green, amber + void * state; // protothread state in mii_video.c + uint8_t timer_id; // timer id for the video thread + uint8_t line; // current line + uint32_t pixels[MII_VRAM_WIDTH * MII_VRAM_HEIGHT]; + uint32_t frame_count; // incremented every frame + uint32_t frame_drawn; + uint8_t color_mode; // color, green, amber } mii_video_t; bool @@ -47,7 +43,7 @@ mii_access_video( uint8_t * byte, bool write); void -mii_video_run( - struct mii_t *mii); +mii_video_init( + mii_t *mii); diff --git a/src/roms/mii_rom_scc_3410065a.bin b/src/roms/mii_rom_scc_3410065a.bin new file mode 100644 index 0000000000000000000000000000000000000000..2c94c9b22d7becade3e41d619380cf7ce7c30ea7 GIT binary patch literal 2048 zcmYLJZEPIX6`kFkZ~QUPCfY`7@`N^?oi(bA6gyLdi-O&X#)Y`0w!Z`fzicK8u>HtmuL#bsu_Hji3rC6XpCK_P%318G(c{_1^e zr0$P-Gk4#;@11kcxu_W8tp;XF9$WGfB^{nzN#{!*U;Bk`_vUgAYia9-S9{jaxkAqE z!Z9+K!wENz#~|mPLmQP|4P_A|kI1MP{Bc=;3T6c|wL~?7jS+v;9I^P{ec8ZJ3v-D9O8xMeybWDfG3OX$vK zT8!RB?}T|0plx&RpDU_S*9FZdTPvw(Pbfxgs1R9-20cFcgOr_u`m?D>!(}L^3$n5U zkJ~E17DqZhf@RnP#Q7T`Ivb)Rb9pPYJkeGSWX(au%1+9J11@P9R_Uzkd?Y57LNAN&Tyt?3?H>sO33%;@~4A} znib`U&3;Ox%K$$CCzzq{uB^jK!HCgTIB#0@UnHzFMtJ|%gwsZW2(thL%l2l~k?u0{ z+Wk;A-bCkCozG45kA|}X%$J#C_tVG?livDX95?VNdT)L@^;+6iv9R+1QKnP)$~$Qi zuc*h;*A84e02})xAlJf^wn|&&bifzTj7~T6y$#x$zmL4$Ch%^nT2cL2gxp#ju>8aSqa(d6lIUbiqU(a>&W3>T)tY98n#+yAW|fkVGWDF{r$yDS5{Wn);CTy z8e~oXFh*L?3v;@sHsKo@%H`~SOY06rht&i(Ug_uhBTfRLa+_PUkx*$v-U5)Nv9Lg- zDkTq*#FTC(=_acA0f43n5r=6>@)Pc=4L@>f6jTNEjl)Q67qiJ-W?vXjBHUGB}w;d^t2t)*k)H|g6wnT z6ENjvowaP0s!65-?~ABYp>O$*=HMw;A?PQgj$F?FYQ)2H8`-HPqRwhEioSV?^O>zdmWUjv~*N#IT^z(Gns*Wq4;TZbq z0@~G7R>JY3Os-SqfT48r+}{=#%d<^FE=~GX8S_f?JUbb*V%FM z1#F47l)pg+Fk@!`2Bl=VO^FHaWBmss$$yC4*}i*rWApBqCySUGm)xp-+>HJ^3q}rt zfP5anr=0=E3U$zkLOzBimrG?_R$1zc4_~Te3{3#Z1s41%z-Bm|;(|B81;?3hfZ|3v zbSLQz^Rathc=0zc+3Ius1!Kw(e=@o8HafI%uFk|JmxY;jFM6u!VGZbeLg#>YZ3pms z198FoyU86`m>q~Xdm!nE?|+lr#+&h%GQ7(q%_qICDvJnZ1Vyk%~|n(5UGK5 literal 0 HcmV?d00001 diff --git a/test/mii_cpu_test.c b/test/mii_cpu_test.c index 52f0318..75fa74d 100644 --- a/test/mii_cpu_test.c +++ b/test/mii_cpu_test.c @@ -33,7 +33,7 @@ _run_one_dump_state( // display the S flags static const char *s_flags = "CZIDBRVN"; for (int i = 0; i < 8; i++) - printf("%c", cpu->P.P[i] ? s_flags[i] : tolower(s_flags[i])); + printf("%c", MII_GET_P_BIT(cpu, i) ? s_flags[i] : tolower(s_flags[i])); if (s.sync) { mii_op_t d = mii_cpu_op[cpu->ram[cpu->PC]]; printf(" "); @@ -42,7 +42,7 @@ _run_one_dump_state( MII_DUMP_DIS_PC | MII_DUMP_DIS_DUMP_HEX); printf("%s", dis); if (d.desc.branch) { - if (cpu->P.P[d.desc.s_bit] == d.desc.s_bit_value) + if (MII_GET_P_BIT(cpu, d.desc.s_bit) == d.desc.s_bit_value) printf(" ; taken"); } printf("\n"); @@ -77,7 +77,7 @@ _run_one_test( if (verbose) { // cpu.debug = _run_one_dump_state; } - cpu.P.P[5] = 1; + cpu.P._R = 1; cpu.S = 0xFF; cpu.PC = p.org; cpu.A = 0x01; @@ -145,7 +145,7 @@ _run_this_one( mii_cpu_state_t s = {0}; verbose += p.verbose; cpu.ram = ram; - cpu.P.P[B_X] = 1; + cpu.P._R = 1; expected_flags |= (1 << B_X); cpu.S = 0xFF; cpu.PC = p.org; @@ -162,9 +162,10 @@ _run_this_one( int err = 0; static const char *s_flags = "CZIDBRVN"; for (int i = 0; i < 8; i++) - if (cpu.P.P[i] != ((expected_flags >> i) & 1)) { + if (MII_GET_P_BIT(&cpu, i) != ((expected_flags >> i) & 1)) { printf("** S bit %c mismatch %d want %d\n", - s_flags[i], cpu.P.P[i], (expected_flags >> i) & 1); + s_flags[i], MII_GET_P_BIT(&cpu, i), + (expected_flags >> i) & 1); err++; } if (err) @@ -225,6 +226,12 @@ static char * doSED_ADC(uint8_t a, uint8_t b) { #include int main() { + printf("Sizeof mii_op_desc_t %d\n", (int)sizeof(mii_op_desc_t)); + printf("Sizeof mii_op_t %d\n", (int)sizeof(mii_op_t)); + printf("Sizeof mii_cpu_t %d\n", (int)sizeof(mii_cpu_t)); + printf("Sizeof mii_cpu_state_t %d\n", (int)sizeof(mii_cpu_state_t)); + printf("Sizeof mii_cpu_op %d\n", (int)sizeof(mii_cpu_op)); + glob_t globbuf; glob("test/asm/0*.asm", 0, NULL, &globbuf); for (int i = 0; i < (int)globbuf.gl_pathc; i++) { @@ -249,6 +256,17 @@ int main() V = (1 << B_V), N = (1 << B_N), }; + // https://github.com/AppleWin/AppleWin/issues/1257 + _run_one_test( + "; Test of JSR *in* the stack\n" + " .org $0100\n" + " jmp test\n" + "pass: .org $0155\n" + "test: .org $0178\n" + " LDX #$7D\n" + " TXS\n" + " JSR $1355\n", + 0); _run_this("RMB $12", " lda #$FF\n" " sta $12\n" @@ -348,7 +366,7 @@ int main() 0x3F, C, 0); _run_this("SBC ($12) SED", indirect("SEC\n SED", 0x75, 0x25, "SBC ($12)"), - 0x50, C | D, 0); + 0x50, C | D | V, 0); _run_this("STA ($12)", indirect("", 0xF1, 0xC0, "STA ($12)\n LDA $3001"), 0xF1, N, 0); @@ -439,9 +457,7 @@ int main() _run_this("SED ADC 99", doSED_ADC(0x99, 0), 0x99, N | D, 0); _run_this("SED ADC 99", doSED_ADC(0x99, 1), 0x00, C | D, 0); _run_this("SED ADC BD", doSED_ADC(0xBD, 0), 0x23, C | D, 0); - -#if 0 -// _run_this("SED ADC FF", doSED_ADC(0xFF, 0), 0x65, C | D | N, 0); + _run_this("SED ADC FF", doSED_ADC(0xFF, 0), 0x65, C | D , 0); _run_this("SED ADC 0,1", doSED_ADC(0, 1), 0x01, D, 0); _run_this("SED ADC 0,9", doSED_ADC(0, 9), 0x09, D, 0); @@ -449,15 +465,15 @@ int main() _run_this("SED ADC 0,1D", doSED_ADC(0, 0x1D), 0x23, D, 0); _run_this("SED ADC 0,99", doSED_ADC(0, 0x99), 0x99, N | D, 0); _run_this("SED ADC 0,BD", doSED_ADC(0, 0xBD), 0x23, C | D, 0); -// _run_this("SED ADC 0,FF", doSED_ADC(0, 0xFF), 0x65, C | D, 0); + _run_this("SED ADC 0,FF", doSED_ADC(0, 0xFF), 0x65, C | D, 0); - _run_this("SED ADC 99,1", doSED_ADC(0x99, 1), 0x0, Z | C | D, 0); + _run_this("SED ADC 99,1", doSED_ADC(0x99, 1), 0x0, C | D, 0); _run_this("SED ADC 35,35", doSED_ADC(0x35, 0x35), 0x70, D, 0); _run_this("SED ADC 45,45", doSED_ADC(0x45, 0x45), 0x90, N | V | D, 0); - _run_this("SED ADC 50,50", doSED_ADC(0x50, 0x50), 0x0, V | Z | C | D, 0); + _run_this("SED ADC 50,50", doSED_ADC(0x50, 0x50), 0x0, V | C | D, 0); _run_this("SED ADC 99,99", doSED_ADC(0x99, 0x99), 0x98, N | V | C | D, 0); -// _run_this("SED ADC B1,C1", doSED_ADC(0xB1, 0xC1), 0xD2, N | V | C | D, 0); -#endif + _run_this("SED ADC B1,C1", doSED_ADC(0xB1, 0xC1), 0xD2, N | V | C | D, 0); + // create an emulator, load the binary file 6502_functional_test.bin at $0000 // and run it until we hit a BRK const char *bigtest[] = { @@ -479,8 +495,8 @@ int main() mii_cpu_state_t s = {0}; cpu.ram = ram; // cpu.debug = _run_one_dump_state; - cpu.P.P[B_X] = 1; - cpu.P.P[B_I] = 1; + cpu.P._R = 1; + cpu.P.I = 1; cpu.S = 0xFF; cpu.PC = 0x400; cpu.A = 0x00; @@ -495,11 +511,15 @@ int main() if (cpu.PC == prev_pc) { same_pc_count++; if (same_pc_count > 3) { - printf("TEST %s: FAIL (stuck at %04X)\n", - bigtest[ti], cpu.PC); - _run_one_dump_state(&cpu, s, NULL); - printf(" Failed instruction is %c%c%c\n", - cpu.A, cpu.X, cpu.Y); + if (ram[cpu.PC] == 0x4c) + printf("TEST %s: PASS\n", bigtest[ti]); + else { + printf("TEST %s: FAIL (stuck at %04X)\n", + bigtest[ti], cpu.PC); + _run_one_dump_state(&cpu, s, NULL); + printf(" Failed instruction is %c%c%c\n", + cpu.A, cpu.X, cpu.Y); + } break; } } else { @@ -508,7 +528,7 @@ int main() } } // if (s.sync) - _run_one_dump_state(&cpu, s, NULL); + // _run_one_dump_state(&cpu, s, NULL); } while (count--); printf("TEST run with %d spare\n", count); _run_one_dump_state(&cpu, s, NULL); diff --git a/ui_gl/mii_emu_gl.c b/ui_gl/mii_emu_gl.c new file mode 100644 index 0000000..d541f6f --- /dev/null +++ b/ui_gl/mii_emu_gl.c @@ -0,0 +1,1008 @@ +/* + * mii_emu_gl.c + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ +#define _GNU_SOURCE // for asprintf +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include "mii_mui.h" +#include "mish.h" +#include "mii_thread.h" + +#include "mii_mui.h" +#include "mii-icon-64.h" +#include "minipt.h" + +/* + * Note: This *assumes* that the GL implementation has support for non-power-of-2 + * textures, which is not a given for older implementations. However, I think + * (by 2024) that's a safe assumption. + */ +#define WINDOW_WIDTH 1280 +#define WINDOW_HEIGHT 720 + + +typedef struct mii_gl_tex_t { + c2_rect_t frame; + GLuint id; +} mii_gl_tex_t; + +typedef struct mii_x11_t { + mii_mui_t video; + pthread_t cpu_thread; + + mui_drawable_t dr; // drawable + + union { + struct { + mii_gl_tex_t mii_tex, mui_tex; + }; + mii_gl_tex_t tex[2]; + }; + + c2_rect_t video_frame; // current video frame + float mui_alpha; + void * transision_state; + struct { + mui_time_t start, end; + c2_rect_t from, to; + } transition; + + Cursor cursor; + Display * dpy; + Window win; + long last_button_click; + struct { + int ungrab, grab, grabbed, down; + c2_pt_t pos; + } mouse; + mui_event_t key; + + XVisualInfo * vis; + Colormap cmap; + XSetWindowAttributes swa; + XWindowAttributes attr; + GLXFBConfig fbc; + Atom wm_delete_window; + int width, height; + GLXContext glContext; +} mii_x11_t; + + +static int gl_err = 0; +static int gl_error_handler( + Display *dpy, + XErrorEvent *ev) +{ + gl_err = true; + return 0; +} + +#define die(_w) { fprintf(stderr,"%s\n",_w); exit(1); } + +static int +has_gl_extension( + const char *string, + const char *ext) +{ + if (!string || !ext) + return false; + int l = strlen(ext); + while (*string) { + char * gotit = strstr((const char *)string, ext); + if (!gotit) + return false; + if (gotit && (gotit[l] == ' ' || gotit[l] == '\0')) + return true; + string += l; + } + return false; +} + +c2_rect_t +c2_rect_interpolate( + c2_rect_t *a, + c2_rect_t *b, + float t) +{ + c2_rect_t r = {}; + r.l = 0.5 + a->l + (b->l - a->l) * t; + r.r = 0.5 + a->r + (b->r - a->r) * t; + r.t = 0.5 + a->t + (b->t - a->t) * t; + r.b = 0.5 + a->b + (b->b - a->b) * t; + return r; +} + +static c2_rect_t +_mii_get_video_position( + mii_x11_t * ui, + bool ui_visible ) +{ + c2_rect_t r = C2_RECT(0, 0, MII_VIDEO_WIDTH, MII_VIDEO_HEIGHT); + if (ui_visible) { + float fac = (ui->attr.height - 38) / (float)MII_VIDEO_HEIGHT; + c2_rect_scale(&r, fac); + c2_rect_offset(&r, + (ui->attr.width / 2) - (c2_rect_width(&r) / 2), 36); + } else { + float fac = (ui->attr.height) / (float)MII_VIDEO_HEIGHT; + c2_rect_scale(&r, fac); + c2_rect_offset(&r, + (ui->attr.width / 2) - (c2_rect_width(&r) / 2), + (ui->attr.height / 2) - (c2_rect_height(&r) / 2)); + c2_rect_inset(&r, 10, 10); + } + return r; +} + +static void +_mii_transition( + mii_x11_t * ui ) +{ + pt_start(ui->transision_state); + + while (ui->video.transition == MII_MUI_TRANSITION_NONE) + pt_yield(ui->transision_state); + + ui->transition.start = mui_get_time(); + ui->transition.end = ui->transition.start + (MUI_TIME_SECOND / 2); + ui->transition.from = ui->video_frame; + + switch (ui->video.transition) { + case MII_MUI_TRANSITION_HIDE_UI: + ui->transition.to = _mii_get_video_position(ui, false); + ui->video.mui_visible = true; + break; + case MII_MUI_TRANSITION_SHOW_UI: + ui->transition.to = _mii_get_video_position(ui, true); + ui->video.mui_visible = true; + break; + } + while (1) { + mui_time_t now = mui_get_time(); + float t = (now - ui->transition.start) / + (float)(ui->transition.end - ui->transition.start); + if (t >= 1.0f) + break; + switch (ui->video.transition) { + case MII_MUI_TRANSITION_HIDE_UI: + ui->mui_alpha = 1.0f - t; + break; + case MII_MUI_TRANSITION_SHOW_UI: + ui->mui_alpha = t; + break; + } + ui->video_frame = c2_rect_interpolate( + &ui->transition.from, &ui->transition.to, t); + pt_yield(ui->transision_state); + } + switch (ui->video.transition) { + case MII_MUI_TRANSITION_HIDE_UI: + ui->video.mui_visible = false; + ui->mui_alpha = 0.0f; + break; + case MII_MUI_TRANSITION_SHOW_UI: + ui->mui_alpha = 1.0f; + break; + } + ui->video.transition = MII_MUI_TRANSITION_NONE; + + pt_end(ui->transision_state); +} + +/* + * xmodmap -pke or -pk will print the list of keycodes + */ +static bool +_mii_x11_convert_keycode( + mii_x11_t *ui, + KeySym sym, + mui_event_t *out ) +{ + switch (sym) { + case XK_F1 ... XK_F12: + out->key.key = MUI_KEY_F1 + (sym - XK_F1); + break; + case XK_Escape: out->key.key = MUI_KEY_ESCAPE; break; + case XK_Left: out->key.key = MUI_KEY_LEFT; break; + case XK_Up: out->key.key = MUI_KEY_UP; break; + case XK_Right: out->key.key = MUI_KEY_RIGHT; break; + case XK_Down: out->key.key = MUI_KEY_DOWN; break; + // XK_Begin + case XK_Insert: out->key.key = MUI_KEY_INSERT; break; + case XK_Home: out->key.key = MUI_KEY_HOME; break; + case XK_End: out->key.key = MUI_KEY_END; break; + case XK_Page_Up: out->key.key = MUI_KEY_PAGEUP; break; + case XK_Page_Down: out->key.key = MUI_KEY_PAGEDOWN; break; + + case XK_Shift_R: out->key.key = MUI_KEY_RSHIFT; break; + case XK_Shift_L: out->key.key = MUI_KEY_LSHIFT; break; + case XK_Control_R: out->key.key = MUI_KEY_RCTRL; break; + case XK_Control_L: out->key.key = MUI_KEY_LCTRL; break; + case XK_Alt_L: out->key.key = MUI_KEY_LALT; break; + case XK_Alt_R: out->key.key = MUI_KEY_RALT; break; + case XK_Super_L: out->key.key = MUI_KEY_LSUPER; break; + case XK_Super_R: out->key.key = MUI_KEY_RSUPER; break; + default: + out->key.key = sym & 0xff; + break; + } +// printf("%s %08x to %04x\n", __func__, sym, out->key.key); + return true; +} + +static int +mii_x11_init( + struct mii_x11_t *ui ) +{ + mui_t * mui = &ui->video.mui; + + if (!setlocale(LC_ALL,"") || + !XSupportsLocale() || + !XSetLocaleModifiers("@im=none")) + return -1; + + ui->dpy = XOpenDisplay(NULL); + if (!ui->dpy) + die("Failed to open X display\n"); + { + /* check glx version */ + int glx_major, glx_minor; + if (!glXQueryVersion(ui->dpy, &glx_major, &glx_minor)) + die("[X11]: Error: Failed to query OpenGL version\n"); + if ((glx_major == 1 && glx_minor < 3) || (glx_major < 1)) + die("[X11]: Error: Invalid GLX version!\n"); + } + { + /* find and pick matching framebuffer visual */ + int fb_count; + static const GLint attr[] = { + GLX_X_RENDERABLE, True, + GLX_DRAWABLE_TYPE, GLX_WINDOW_BIT, + GLX_RENDER_TYPE, GLX_RGBA_BIT, + GLX_X_VISUAL_TYPE, GLX_TRUE_COLOR, + GLX_RED_SIZE, 8, + GLX_GREEN_SIZE, 8, + GLX_BLUE_SIZE, 8, + GLX_ALPHA_SIZE, 8, + None + }; + GLXFBConfig *fbc = glXChooseFBConfig(ui->dpy, + DefaultScreen(ui->dpy), attr, &fb_count); + if (!fbc) + die("[X11]: Error: failed to retrieve framebuffer configuration\n"); + { + /* pick framebuffer with most samples per pixel */ + int fb_best = -1, best_num_samples = -1; + for (int i = 0; i < fb_count; ++i) { + XVisualInfo *vi = glXGetVisualFromFBConfig(ui->dpy, fbc[i]); + if (vi) { + int sample_buffer, samples; + glXGetFBConfigAttrib(ui->dpy, fbc[i], + GLX_SAMPLE_BUFFERS, &sample_buffer); + glXGetFBConfigAttrib(ui->dpy, fbc[i], + GLX_SAMPLES, &samples); + if ((fb_best < 0) || + (sample_buffer && samples > best_num_samples)) + fb_best = i, best_num_samples = samples; + XFree(vi); + } + } + ui->fbc = fbc[fb_best]; + XFree(fbc); + ui->vis = glXGetVisualFromFBConfig(ui->dpy, ui->fbc); + } + } + { + /* create window */ + ui->cmap = XCreateColormap(ui->dpy, + RootWindow(ui->dpy, ui->vis->screen), + ui->vis->visual, AllocNone); + ui->swa.colormap = ui->cmap; + ui->swa.background_pixmap = None; + ui->swa.border_pixel = 0; + ui->swa.event_mask = + ExposureMask | KeyPressMask | KeyReleaseMask | + ButtonPress | ButtonReleaseMask | ButtonMotionMask | + Button1MotionMask | Button3MotionMask | + Button4MotionMask | Button5MotionMask | + PointerMotionMask | StructureNotifyMask | FocusChangeMask; + ui->win = XCreateWindow(ui->dpy, + RootWindow(ui->dpy, ui->vis->screen), 0, 0, + WINDOW_WIDTH, WINDOW_HEIGHT, 0, ui->vis->depth, InputOutput, + ui->vis->visual, + CWBorderPixel | CWColormap | CWEventMask, + &ui->swa); + if (!ui->win) + die("[X11]: Failed to create window\n"); + XFree(ui->vis); + { + char title[128]; + sprintf(title, "MII //e Emulator"); + char *telnet = getenv("MISH_TELNET_PORT"); + if (telnet) + sprintf(title + strlen(title), " (telnet port %s)", telnet); + else + sprintf(title + strlen(title), " (telnet disabled)"); + XStoreName(ui->dpy, ui->win, title); + } + { + Atom net_wm_icon_atom = XInternAtom(ui->dpy, "_NET_WM_ICON", False); + XChangeProperty(ui->dpy, ui->win, net_wm_icon_atom, XA_CARDINAL, 32, PropModeReplace, (unsigned char *)mii_icon64, sizeof(mii_icon64) / sizeof(mii_icon64[0])); + XFlush(ui->dpy); + } + XMapWindow(ui->dpy, ui->win); + ui->wm_delete_window = XInternAtom(ui->dpy, "WM_DELETE_WINDOW", False); + XSetWMProtocols(ui->dpy, ui->win, &ui->wm_delete_window, 1); + } + /* create invisible cursor */ + { + static XColor dummy; char data[1] = {0}; + Pixmap blank = XCreateBitmapFromData(ui->dpy, ui->win, data, 1, 1); + if (blank == None) return 0; + ui->cursor = XCreatePixmapCursor(ui->dpy, blank, blank, &dummy, &dummy, 0, 0); + XFreePixmap(ui->dpy, blank); + } + { + /* create opengl context */ + typedef GLXContext (*glxCreateContext)( + Display *, GLXFBConfig, GLXContext, Bool, const int *); + int (*old_handler)(Display *, XErrorEvent *) = + XSetErrorHandler(gl_error_handler); + const char *extensions_str = glXQueryExtensionsString( + ui->dpy, DefaultScreen(ui->dpy)); + glxCreateContext create_context = (glxCreateContext) + glXGetProcAddressARB((const GLubyte *)"glXCreateContextAttribsARB"); + + gl_err = false; + if (!has_gl_extension(extensions_str, "GLX_ARB_create_context") || + !create_context) { + fprintf(stdout, "[X11]: glXCreateContextAttribARB() not found...\n"); + fprintf(stdout, "[X11]: ... using old-style GLX context\n"); + ui->glContext = glXCreateNewContext(ui->dpy, ui->fbc, GLX_RGBA_TYPE, 0, True); + } else { + GLint attr[] = { + GLX_CONTEXT_MAJOR_VERSION_ARB, 2, + GLX_CONTEXT_MINOR_VERSION_ARB, 2, + None + }; + ui->glContext = create_context(ui->dpy, ui->fbc, 0, True, attr); + XSync(ui->dpy, False); + if (gl_err || !ui->glContext) { + attr[1] = 1; + attr[3] = 0; + gl_err = false; + fprintf(stdout, "[X11] Failed to create OpenGL 3.0 context\n"); + fprintf(stdout, "[X11] ... using old-style GLX context!\n"); + ui->glContext = create_context(ui->dpy, ui->fbc, 0, True, attr); + } + } + XSync(ui->dpy, False); + XSetErrorHandler(old_handler); + if (gl_err || !ui->glContext) + die("[X11]: Failed to create an OpenGL context\n"); + glXMakeCurrent(ui->dpy, ui->win, ui->glContext); + } + { // create the MUI 'screen' at the window size + mui_pixmap_t* pix = &ui->dr.pix; + pix->size.y = WINDOW_HEIGHT; + pix->size.x = WINDOW_WIDTH; + pix->row_bytes = WINDOW_WIDTH * 4; + pix->bpp = 32; + pix->pixels = malloc(pix->row_bytes * WINDOW_HEIGHT * (pix->bpp / 8)); + mui->screen_size = pix->size; + } + { + XGetWindowAttributes(ui->dpy, ui->win, &ui->attr); + ui->mui_alpha = 1.0f; + ui->video.mui_visible = true; + ui->video_frame = _mii_get_video_position(ui, ui->video.mui_visible); + } + return 0; +} + +static void +mii_x11_update_mouse_card( + mii_x11_t * ui) +{ + mii_t * mii = &ui->video.mii; + mui_t * mui = &ui->video.mui; + /* + * We can grab the mouse if it is enabled by the driver, it is in the + * video frame, and there is no active MUI windows (or menus). + */ + if (mii->mouse.enabled && + c2_rect_contains_pt(&ui->video_frame, &ui->mouse.pos) && + !(ui->video.mui_visible && mui_has_active_windows(mui))) { + if (!ui->mouse.grabbed) { + ui->mouse.grab = 1; + ui->mouse.grabbed = 1; + // printf("Grab mouse\n"); + } + } else { + if (ui->mouse.grabbed) { + ui->mouse.ungrab = 1; + ui->mouse.grabbed = 0; + // printf("Ungrab mouse\n"); + } + } + if (!ui->mouse.grabbed) + return; + double x = ui->mouse.pos.x - ui->video_frame.l; + double y = ui->mouse.pos.y - ui->video_frame.t; + // get mouse button state + int button = ui->mouse.down; + // clamp coordinates inside bounds + double vw = c2_rect_width(&ui->video_frame); + double vh = c2_rect_height(&ui->video_frame); + double mw = mii->mouse.max_x - mii->mouse.min_x; + double mh = mii->mouse.max_y - mii->mouse.min_y; + // normalize mouse coordinates + mii->mouse.x = mii->mouse.min_x + (x * mw / vw) + 0.5; + mii->mouse.y = mii->mouse.min_y + (y * mh / vh) + 0.5; + mii->mouse.button = button; +} + +static int +mii_x11_handle_event( + mii_x11_t * ui, + XEvent *evt) +{ + int quit = 0; + + /* We don't actually 'grab' as in warp the pointer, we just show/hide it + * dynamically when we enter/leave the video rectangle */ + if (ui->mouse.grab) { + XDefineCursor(ui->dpy, ui->win, ui->cursor); + ui->mouse.grab = 0; + } else if (ui->mouse.ungrab) { + XUndefineCursor(ui->dpy, ui->win); + ui->mouse.ungrab = 0; + } + mui_t * mui = &ui->video.mui; + mii_t * mii = &ui->video.mii; + switch (evt->type) { + case FocusIn: + case FocusOut: + // printf("%s focus %d\n", __func__, evt->type == FocusIn); + /* This prevents 'stale' modifiers keys when the emulator is + * in front and we switches window/desktop with a modifier on.. + * this used to trigger 'phantom' modifiers still being 'on' */ + mui->modifier_keys = 0; + break; + case KeyRelease: + case KeyPress: { + int ret, down = (evt->type == KeyPress); + KeySym *code = XGetKeyboardMapping(ui->dpy, ( + KeyCode)evt->xkey.keycode, 1, &ret); + ui->key.type = down ? MUI_EVENT_KEYDOWN : MUI_EVENT_KEYUP; + ui->key.key.up = 0; + bool handled = false; + bool converted = _mii_x11_convert_keycode(ui, *code, &ui->key); + bool is_modifier = ui->key.key.key >= MUI_KEY_MODIFIERS && + ui->key.key.key <= MUI_KEY_MODIFIERS_LAST; + if (converted) { + // convert keycodes into a bitfields of current modifiers + if (ui->key.key.key >= MUI_KEY_MODIFIERS && + ui->key.key.key <= MUI_KEY_MODIFIERS_LAST) { + if (down) + mui->modifier_keys |= (1 << (ui->key.key.key - MUI_KEY_MODIFIERS)); + else + mui->modifier_keys &= ~(1 << (ui->key.key.key - MUI_KEY_MODIFIERS)); + } + ui->key.modifiers = mui->modifier_keys; + switch (ui->key.key.key) { + case MUI_KEY_RSUPER: + case MUI_KEY_LSUPER: { + int apple = ui->key.key.key - MUI_KEY_RSUPER; + mii_bank_t *bank = &mii->bank[MII_BANK_MAIN]; + uint8_t old = mii_bank_peek(bank, 0xc061 + apple); + mii_bank_poke(bank, 0xc061 + apple, down ? 0x80 : 0); + if (!!down != !!old) { + // printf("Apple %s %s\n", apple ? "Open" : "Close", + // down ? "down" : "up"); + } + } break; + } + handled = mui_handle_event(mui, &ui->key); + // if not handled and theres a window visible, assume + // it's a dialog and it OUGHT to eat the key + if (!handled) + handled = mui_window_front(mui) != NULL; + // printf("%s key handled %d\n", __func__, handled); + } + if (!handled && down && !is_modifier) { + uint16_t mii_key = ui->key.key.key; + char buf[32] = ""; + KeySym keysym = 0; + if (XLookupString((XKeyEvent*)evt, buf, 32, &keysym, NULL) != NoSymbol) { + #if 0 + printf(" lookup sym %04x %d:", (int)keysym, (int)strlen(buf)); + for (int i = 0; i < (int)strlen(buf); i++) + printf(" %02x", buf[i]); + printf("\n"); + #endif + mii_key = buf[0]; + } + switch (mii_key) { + case MUI_KEY_DOWN: mii_key = 'J' - 'A' + 1; break; + case MUI_KEY_UP: mii_key = 'K' - 'A' + 1; break; + case MUI_KEY_RIGHT: mii_key = 'U' - 'A' + 1; break; + case MUI_KEY_LEFT: mii_key = 'H' - 'A' + 1; break; + } + // printf("key %04x %4x\n", mii_key, mui->modifier_keys); + mii_keypress(mii, mii_key); + } + XFree(code); + } break; + case ButtonPress: + case ButtonRelease: { + // printf("%s %s button %d grabbed:%d\n", __func__, + // evt->type == ButtonPress ? "Down":"Up ", + // evt->xbutton.button, ui->mouse.grabbed); + switch (evt->xbutton.button) { + case Button1: { + ui->mouse.down = evt->type == ButtonPress; + ui->mouse.pos.x = evt->xbutton.x; + ui->mouse.pos.y = evt->xbutton.y; + if (ui->video.mui_visible) { + mui_event_t ev = { + .type = ui->mouse.down ? + MUI_EVENT_BUTTONDOWN : + MUI_EVENT_BUTTONUP, + .mouse.where = ui->mouse.pos, + .modifiers = mui->modifier_keys, // | MUI_MODIFIER_EVENT_TRACE, + }; + mui_handle_event(mui, &ev); + } + mii_x11_update_mouse_card(ui); + } break; + case Button4: + case Button5: { + // printf("%s wheel %d %d\n", __func__, + // evt->xbutton.button, ui->video.mui_visible); + if (ui->video.mui_visible) { + mui_event_t ev = { + .type = MUI_EVENT_WHEEL, + .modifiers = mui->modifier_keys,// | MUI_MODIFIER_EVENT_TRACE, + .wheel.where = ui->mouse.pos, + .wheel.delta = evt->xbutton.button == Button4 ? -1 : 1, + }; + mui_handle_event(mui, &ev); + } + } break; + } + } break; + case MotionNotify: { + ui->mouse.pos.x = evt->xmotion.x; + ui->mouse.pos.y = evt->xmotion.y; + mii_x11_update_mouse_card(ui); + if (ui->mouse.grabbed) + break; + if (ui->video.mui_visible) { + mui_event_t ev = { + .type = MUI_EVENT_DRAG, + .mouse.where = ui->mouse.pos, + .modifiers = mui->modifier_keys, + }; + mui_handle_event(mui, &ev); + } + } break; + case ClientMessage: + if ((Atom)evt->xclient.data.l[0] == ui->wm_delete_window) + quit = 1; + break; + case KeymapNotify: + XRefreshKeyboardMapping(&evt->xmapping); + break; + } + return quit; +} + +static void +mii_x11_terminate( + mii_x11_t *ui) +{ + glXMakeCurrent(ui->dpy, 0, 0); + glXDestroyContext(ui->dpy, ui->glContext); + XUnmapWindow(ui->dpy, ui->win); + XFreeColormap(ui->dpy, ui->cmap); + XDestroyWindow(ui->dpy, ui->win); + XCloseDisplay(ui->dpy); +} + +void +mii_x11_prepare_textures( + mii_x11_t *ui) +{ + mii_t * mii = &ui->video.mii; + mui_t * mui = &ui->video.mui; + GLuint tex[2]; + glGenTextures(2, tex); + for (int i = 0; i < 2; i++) { + mii_gl_tex_t * t = &ui->tex[i]; + memset(t, 0, sizeof(*t)); + t->id = tex[i]; + } + glEnable(GL_TEXTURE_2D); + // bind the mii texture using the GL_ARB_texture_rectangle extension + glBindTexture(GL_TEXTURE_2D, ui->mii_tex.id); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR); + // disable the repeat of textures + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE); + + glTexImage2D(GL_TEXTURE_2D, 0, 4, + MII_VRAM_WIDTH, + MII_VRAM_HEIGHT, 0, GL_RGBA, + GL_UNSIGNED_BYTE, + mii->video.pixels); + // bind the mui texture using the GL_ARB_texture_rectangle as well + glEnable(GL_TEXTURE_2D); + glBindTexture(GL_TEXTURE_2D, ui->mui_tex.id); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST); + glTexImage2D(GL_TEXTURE_2D, 0, 4, + ui->dr.pix.row_bytes / 4, + mui->screen_size.y, 0, GL_RGBA, + GL_UNSIGNED_BYTE, + ui->dr.pix.pixels); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE); + +// printf("%s texture created %d\n", __func__, mii_apple_screen_tex); +// display opengl error + GLenum err = glGetError(); + if (err != GL_NO_ERROR) { + printf("Error creating texture: %d\n", err); + } +} + +void +mii_x11_render( + mii_x11_t *ui) +{ + glClearColor( + .6f * ui->mui_alpha, + .6f * ui->mui_alpha, + .6f * ui->mui_alpha, + 1.0f); + glClear(GL_COLOR_BUFFER_BIT); + glPushAttrib(GL_ENABLE_BIT|GL_COLOR_BUFFER_BIT|GL_TRANSFORM_BIT); + glDisable(GL_CULL_FACE); + glDisable(GL_DEPTH_TEST); + glDisable(GL_SCISSOR_TEST); + glEnable(GL_BLEND); + glEnable(GL_TEXTURE_2D); + glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); + + /* setup viewport/project */ + glViewport(0, 0, (GLsizei)ui->attr.width, (GLsizei)ui->attr.height); + glMatrixMode(GL_PROJECTION); + glPushMatrix(); + glLoadIdentity(); + glOrtho(0.0f, ui->attr.width, ui->attr.height, 0.0f, -1.0f, 1.0f); + glMatrixMode(GL_MODELVIEW); + glPushMatrix(); + glLoadIdentity(); + // This (was) the recommended way to handle pixel alignment in glOrtho + // mode, but this seems to have changed -- now it looks like Linear filtering +// glTranslatef(0.375f, 0.375f, 0.0f); + { + /* draw mii texture */ + glColor3f(1.0f, 1.0f, 1.0f); + glBindTexture(GL_TEXTURE_2D, ui->mii_tex.id); + glBegin(GL_QUADS); + c2_rect_t r = ui->video_frame; + glTexCoord2f(0, 0); + glVertex2f(r.l, r.t); + glTexCoord2f(MII_VIDEO_WIDTH / (double)MII_VRAM_WIDTH, 0); + glVertex2f(r.r, r.t); + glTexCoord2f(MII_VIDEO_WIDTH / (double)MII_VRAM_WIDTH, + MII_VIDEO_HEIGHT / (double)MII_VRAM_HEIGHT); + glVertex2f(r.r, r.b); + glTexCoord2f(0, + MII_VIDEO_HEIGHT / (double)MII_VRAM_HEIGHT); + glVertex2f(r.l, r.b); + glEnd(); + /* draw mui texture */ + if (ui->mui_alpha > 0.0f) { + glColor4f(1.0f, 1.0f, 1.0f, ui->mui_alpha); + glBindTexture(GL_TEXTURE_2D, ui->mui_tex.id); + glBegin(GL_QUADS); + glTexCoord2f(0, 0); glVertex2f(0, 0); + glTexCoord2f(ui->attr.width / (double)(ui->dr.pix.row_bytes / 4), 0); + glVertex2f(ui->attr.width, 0); + glTexCoord2f(ui->attr.width / (double)(ui->dr.pix.row_bytes / 4), + ui->attr.height / (double)(ui->dr.pix.size.y)); + glVertex2f(ui->attr.width, ui->attr.height); + glTexCoord2f(0, + ui->attr.height / (double)(ui->dr.pix.size.y)); + glVertex2f(0, ui->attr.height); + glEnd(); + } + } + glDisable(GL_CULL_FACE); + glDisable(GL_DEPTH_TEST); + glDisable(GL_SCISSOR_TEST); + glDisable(GL_BLEND); + glDisable(GL_TEXTURE_2D); + + glBindTexture(GL_TEXTURE_2D, 0); + glMatrixMode(GL_MODELVIEW); + glPopMatrix(); + glMatrixMode(GL_PROJECTION); + glPopMatrix(); + glPopAttrib(); +} + +// TODO factor this into a single table, this is dupped from mii_mui_settings.c! +static const struct { + const char * name; +} mii_slot_driver[MII_SLOT_DRIVER_COUNT] = { + [MII_SLOT_DRIVER_NONE] = { "none", }, + [MII_SLOT_DRIVER_SMARTPORT] = { "smartport", }, + [MII_SLOT_DRIVER_DISK2] = { "disk2", }, + [MII_SLOT_DRIVER_MOUSE] = { "mouse", }, + [MII_SLOT_DRIVER_SUPERSERIAL] = { "ssc", }, + [MII_SLOT_DRIVER_ROM1MB] = { "eecard", }, +}; + +void +mii_ui_reconfigure_slot( + mii_t * mii, + mii_machine_config_t * config, + int slot ) +{ + printf("%s slot %d\n", __func__, slot); + int i = slot-1; + switch (config->slot[i].driver) { + case MII_SLOT_DRIVER_SMARTPORT: { + mii_slot_command(mii, slot, + MII_SLOT_DRIVE_LOAD, + (void*)config->slot[i].conf.smartport.drive[0].disk); + mii_slot_command(mii, slot, + MII_SLOT_DRIVE_LOAD + 1, + (void*)config->slot[i].conf.smartport.drive[1].disk); + } break; + case MII_SLOT_DRIVER_DISK2: { + mii_slot_command(mii, slot, + MII_SLOT_DRIVE_LOAD, + (void*)config->slot[i].conf.disk2.drive[0].disk); + mii_slot_command(mii, slot, + MII_SLOT_DRIVE_LOAD + 1, + (void*)config->slot[i].conf.disk2.drive[1].disk); + } break; + case MII_SLOT_DRIVER_ROM1MB: { + mii_slot_command(mii, slot, + MII_SLOT_DRIVE_LOAD, + config->slot[i].conf.rom1mb.use_default ? + "" : + (void*)config->slot[i].conf.rom1mb.drive.disk); + } break; + } +} + +// load the content of the mii_machine_config_t into the initialized +// mii_t struct, similar to what mii_argv_parse does +// TODO: Change argv to modify that struct instead of the mii_t directly +static void +_mii_ui_load_config( + mii_t * mii, + mii_machine_config_t * config, + uint32_t *ioFlags ) +{ + if (config->no_slot_clock) + *ioFlags |= MII_INIT_NSC; + if (config->titan_accelerator) + *ioFlags |= MII_INIT_TITAN; + mii->speaker.muted = config->audio_muted; + mii->video.color_mode = config->video_mode; + for (int i = 0; i < 7; i++) { + if (config->slot[i].driver == MII_SLOT_DRIVER_NONE) + continue; + if (config->slot[i].driver >= MII_SLOT_DRIVER_COUNT) { + printf("%s invalid driver %d\n", __func__, + config->slot[i].driver); + continue; + } + int slot = i + 1; + if (mii_slot_drv_register(mii, slot, + mii_slot_driver[config->slot[i].driver].name) < 0) { + printf("%s failed to register driver %s\n", __func__, + mii_slot_driver[config->slot[i].driver].name); + } + mii_ui_reconfigure_slot(mii, config, slot); + } +} + +void +mii_x11_reload_config( + struct mii_x11_t *ui ) +{ + mii_t * mii = &ui->video.mii; + mii_machine_config_t * config = &ui->video.config; + uint32_t flags = MII_INIT_DEFAULT; + + if (mii->state != MII_INIT) { + printf("%s mii is running, terminating thread\n", __func__); + mii->state = MII_TERMINATE; + pthread_join(ui->cpu_thread, NULL); + printf("%s mii terminated\n", __func__); + } + mii_mui_menu_slot_menu_update(&ui->video); + printf("%s (re)loading config\n", __func__); + mii_init(mii); + _mii_ui_load_config(mii, config, &flags); + mii_prepare(mii, flags); + mii_reset(mii, true); + + /* start the CPU/emulator thread */ + ui->cpu_thread = mii_thread_start(mii); +} + +int +main( + int argc, + const char *argv[]) +{ + char * conf_path; + asprintf(&conf_path, "%s/.local/share/mii", getenv("HOME")); + mkdir(conf_path, 0755); + + mii_x11_t g_mii = {}; + mii_x11_t * ui = &g_mii; + mii_t * mii = &g_mii.video.mii; + bool no_config_found = false; + + if (mii_settings_load( + &ui->video.cf, getcwd(NULL,0), ".mii_emu_config") < 0 && + mii_settings_load( + &ui->video.cf, conf_path, "mii_emu_gl") < 0) { + asprintf(&ui->video.cf.path, "%s/.local/share/mii/mii_emu_gl", + getenv("HOME")); + printf("%s failed to load settings\n", argv[0]); + ui->video.config.load_defaults = true; + no_config_found = true; + } else { + mii_emu_load(&ui->video.cf, &ui->video.config); + } + { + mii_init(mii); + int idx = 1; + uint32_t flags = MII_INIT_DEFAULT; + + if (!no_config_found) { + // load the config we have in the UI config struct into the + // mii config struct + _mii_ui_load_config(mii, &ui->video.config, &flags); + } + int r = mii_argv_parse(mii, argc, argv, &idx, &flags); + if (r == 0) { + printf("mii: Invalid argument %s, skipped\n", argv[idx]); + } else if (r == -1) + exit(1); + mii_prepare(mii, flags); + } + { + mish_prepare(1); + mish_set_command_parameter(MISH_FCC('m','i','i',' '), mii); + printf("MISH_TELNET_PORT = %s\n", getenv("MISH_TELNET_PORT")); + } + mii_x11_init(ui); + mui_t * mui = &ui->video.mui; // to move to a function later + mui_init(mui); + mui->clear_color.value = 0; + asprintf(&mui->pref_directory, "%s/.local/share/mii", getenv("HOME")); + + mii_mui_menus_init((mii_mui_t*)ui); + ui->video.mui_visible = 1; + mii_mui_menu_slot_menu_update(&ui->video); + + mii_x11_prepare_textures(ui); + + // use a 60fps timerfd here as well + int timerfd = timerfd_create(CLOCK_MONOTONIC, 0); + if (timerfd < 0) { + perror(__func__); + goto cleanup; + } + mii_thread_set_fps(timerfd, 60); + /* + * If it is the first run, theres no config files, open the + * slot dialog to let the user configure some stuff himself + */ + if (no_config_found) { + ui->video.config.reboot_on_load = true; + mii_config_open_slots_dialog(&ui->video); + } + /* start the CPU/emulator thread */ + ui->cpu_thread = mii_thread_start(mii); + + while (mii->state != MII_INIT) { + /* Input */ + XEvent evt; + while (XPending(ui->dpy)) { + XNextEvent(ui->dpy, &evt); + if (evt.type == ClientMessage) + goto cleanup; + if (XFilterEvent(&evt, ui->win)) + continue; + mii_x11_handle_event(ui, &evt); + } + mui_run(mui); + bool draw = false; + if (pixman_region32_not_empty(&mui->inval)) { + draw = true; + mui_draw(mui, &ui->dr, 0); + glBindTexture(GL_TEXTURE_2D, ui->mui_tex.id); + + pixman_region32_intersect_rect(&mui->redraw, &mui->redraw, + 0, 0, ui->dr.pix.size.x, ui->dr.pix.size.y); + int rc = 0; + c2_rect_t *ra = (c2_rect_t*)pixman_region32_rectangles(&mui->redraw, &rc); + // rc = 1; ra = &C2_RECT(0, 0, mui->screen_size.x, mui->screen_size.y); + if (rc) { + // printf("GL: %d rects to redraw\n", rc); + for (int i = 0; i < rc; i++) { + c2_rect_t r = ra[i]; + // printf("GL: %d,%d %dx%d\n", r.l, r.t, c2_rect_width(&r), c2_rect_height(&r)); + glPixelStorei(GL_UNPACK_ROW_LENGTH, ui->dr.pix.row_bytes / 4); + glTexSubImage2D(GL_TEXTURE_2D, 0, r.l, r.t, + c2_rect_width(&r), c2_rect_height(&r), + GL_RGBA, GL_UNSIGNED_BYTE, + ui->dr.pix.pixels + (r.t * ui->dr.pix.row_bytes) + (r.l * 4)); + } + } + glPixelStorei(GL_UNPACK_ROW_LENGTH, 0); + pixman_region32_clear(&mui->redraw); + } + if (mii->video.frame_count != mii->video.frame_drawn) { + draw = true; + mii->video.frame_drawn = mii->video.frame_count; + // update the whole texture + glBindTexture(GL_TEXTURE_2D, ui->mii_tex.id); + glTexSubImage2D(GL_TEXTURE_2D, 0, 0, 0, + MII_VRAM_WIDTH, + MII_VIDEO_HEIGHT, GL_RGBA, + GL_UNSIGNED_BYTE, + mii->video.pixels); + } + /* Draw */ + if (draw) { + XGetWindowAttributes(ui->dpy, ui->win, &ui->attr); + glViewport(0, 0, ui->width, ui->height); + _mii_transition(ui); + mii_x11_render(ui); + glXSwapBuffers(ui->dpy, ui->win); + } + /* Wait for next frame */ + uint64_t timerv; + if (read(timerfd, &timerv, sizeof(timerv)) < 0) { + perror(__func__); + goto cleanup; + } + } +cleanup: + mii_emu_save(&ui->video.cf, &ui->video.config); + // this was already done in the mii_thread +// mii_dispose(mii); + mii_x11_terminate(ui); + return 0; +} diff --git a/ui_gl/mii_mui.h b/ui_gl/mii_mui.h new file mode 100644 index 0000000..6be2a47 --- /dev/null +++ b/ui_gl/mii_mui.h @@ -0,0 +1,53 @@ +/* + * mii_mui.h + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ +#pragma once +/* + This tries to contains a structure that is the MUI interface over the MII + video, but without any attachment to x11 or opengl. Basically hopefully + segregating the relevant logic without tying it to a specific windowing system. + + Hopefully with a bit more work this OUGHT to allow Windows/macOS port + with a native frontend. + */ + +#include +#include +#include "mii.h" +#include "mui.h" +#include "mii_mui_settings.h" + +enum mii_mui_transition_e { + MII_MUI_TRANSITION_NONE, + MII_MUI_TRANSITION_HIDE_UI, + MII_MUI_TRANSITION_SHOW_UI, +}; + +typedef struct mii_mui_t { + mui_t mui; // mui interface + mii_t mii; // apple II emulator + + bool mui_visible; + uint8_t transition; + + mii_machine_config_t config; + mii_loadbin_conf_t loadbin_conf; + + mii_config_file_t cf; +} mii_mui_t; + + +void +mii_mui_menus_init( + mii_mui_t * ui); +void +mii_mui_menu_slot_menu_update( + mii_mui_t * ui); +// slot can be <= 0 to open the machine dialog instead +void +mii_config_open_slots_dialog( + mii_mui_t * ui); diff --git a/ui_gl/mii_mui_1mb.c b/ui_gl/mii_mui_1mb.c new file mode 100644 index 0000000..115cf1a --- /dev/null +++ b/ui_gl/mii_mui_1mb.c @@ -0,0 +1,336 @@ +/* + * mui_mui_1mb.c + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ + +#define _GNU_SOURCE +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "mui.h" +#include "mii_mui_settings.h" + +enum { + MII_1MB_WINDOW_ID = FCC('1','m','b','c'), + MII_1MB_SAVE = FCC('s','a','v','e'), + MII_1MB_CANCEL = FCC('c','a','n','c'), + MII_1MB_SELECT = FCC('s','e','l','e'), + MII_1MB_USE_FILE = FCC('r','o','m','0'), + MII_1MB_USE_BIN = FCC('r','o','m','1'), +}; + +typedef struct mii_mui_1mb_t { + mui_window_t win; + mui_control_t * load, *icon, *fname; + mui_control_t * r0, *r1; + + mii_1mb_conf_t * dst; + mii_1mb_conf_t config; +} mii_mui_1mb_t; + +static void +_size_string( + size_t s, + char *out, + int out_size, + uint16_t flags) +{ + if (s < 1024) { + snprintf(out, out_size, "%ld", s); + return; + } + s /= 1024; + if (s < 1024) + snprintf(out, out_size, "%ldK", s); + else if (s < 1024*1024) { + long r = ((s * 10) / (1024) % 10); + if (r == 0 && (flags & 1)) + snprintf(out, out_size, "%ldM", + s / (1024)); + else + snprintf(out, out_size, "%ld.%ldM", + s / (1024), r); + } else { + long r = ((s * 10) / (1024 * 1024) % 10); + if (r == 0 && (flags & 1)) + snprintf(out, out_size, "%ldG", + s / (1024 * 1024)); + else + snprintf(out, out_size, "%ld.%ldG", + s / (1024 * 1024), r); + } +} + +static int +_mii_check_1mb_file( + struct mui_t *mui, + const char * path) +{ + struct stat st; + if (stat(path, &st) < 0) { + char *msg; + asprintf(&msg, "'%s': %s", path, strerror(errno)); + mui_alert(mui, C2_PT(0,0), + "Could not find file", + msg, MUI_ALERT_FLAG_OK); + free(msg); + return -1; + } + if (st.st_size != 1024*1024) { + char stt[64]; + long delta = 1024*1024 - st.st_size; + _size_string(delta < 0 ? -delta : delta, stt, sizeof(stt)-2, 1); + strcpy(stt + strlen(stt), "B"); + char *msg; + asprintf(&msg, "File '%s' is the wrong size, %s too %s.", + basename((char*)path), + stt, delta < 0 ? "big" : "small"); + mui_alert(mui, C2_PT(0,0), + "Invalid file", + msg, MUI_ALERT_FLAG_OK); + free(msg); + return -1; + } + FILE * f = fopen(path, "rb"); + if (!f) { + char *msg; + asprintf(&msg, "Could not open '%s' for reading: %s", + path, strerror(errno)); + mui_alert(mui, C2_PT(0,0), + "Failed to open!", + msg, MUI_ALERT_FLAG_OK); + free(msg); + return -1; + } + uint8_t buf[4]; + fseek(f, 512, SEEK_SET); + fread(buf, 1, 4, f); + fclose(f); + printf("%s %02x %02x %02x %02x\n", __func__, buf[0], buf[1], buf[2], buf[3]); + if (buf[0] != 0x20 || buf[1] != 0x58 || buf[2] != 0xfc || buf[3] != 0xa0) { + char *msg; + asprintf(&msg, "WARNING: '%s' Lacks the 'bootloader' in block #1, " + "this will likely not boot properly.", path); + mui_alert(mui, C2_PT(0,0), + "No bootloader found", + msg, MUI_ALERT_FLAG_OK); + free(msg); + return 0; + } + return 1; +} + +static int +_mii_1mb_stdfile_cb( + mui_window_t * w, + void * cb_param, // mii_mui_1mb_t + uint32_t what, + void * param) // not used +{ + mii_mui_1mb_t * m = cb_param; + switch (what) { + case MUI_STDF_ACTION_SELECT: { + char * path = mui_stdfile_get_selected_path(w); + printf("%s select %s\n", __func__, path); + if (_mii_check_1mb_file(w->ui, path) < 0) { + mui_control_set_title(m->fname, "Click \"Select\" to pick a file"); + mui_control_set_state(m->fname, MUI_CONTROL_STATE_DISABLED); + mui_control_set_state(m->icon, MUI_CONTROL_STATE_DISABLED); + mui_control_set_state(m->load, MUI_CONTROL_STATE_DISABLED); + break; + } + strncpy(m->config.drive.disk, path, sizeof(m->config.drive.disk)-1); + mui_control_set_state(m->fname, MUI_CONTROL_STATE_NORMAL); + char *dup = strdup(path); + mui_control_set_title(m->fname, basename(dup)); + free(dup); + mui_control_set_state(m->icon, MUI_CONTROL_STATE_NORMAL); + mui_control_set_state(m->load, MUI_CONTROL_STATE_NORMAL); + mui_control_set_value(m->r0, 1); + mui_control_set_value(m->r1, 0); + m->config.use_default = 0; + mui_window_dispose(w); + } break; + case MUI_STDF_ACTION_CANCEL: + printf("%s cancel\n", __func__); + mui_window_dispose(w); + break; + } + return 0; +} + +static int +_mii_1mb_action_cb( + mui_control_t * c, + void * cb_param, // mii_mui_1mb_t + uint32_t what, + void * param) // not used +{ + printf("%s %4.4s\n", __func__, (char*)&what); + mii_mui_1mb_t * m = cb_param; + uint32_t uid = mui_control_get_uid(c); + + switch (what) { + case MUI_CONTROL_ACTION_SELECT: + printf("%s control %4.4s\n", __func__, (char*)&uid); + switch (uid) { + case MII_1MB_SAVE: { + // save the config + printf("%s save\n", __func__); + if (m->dst) + *m->dst = m->config; + mui_window_action(&m->win, MII_MUI_1MB_SAVE, m->dst); + mui_window_dispose(&m->win); + } break; + case MII_1MB_CANCEL: { + // cancel the config + printf("%s cancel\n", __func__); + mui_window_dispose(&m->win); + } break; + case MII_1MB_SELECT: { + // select a file + printf("%s select\n", __func__); + mui_window_t * w = mui_stdfile_get(m->win.ui, + C2_PT(0, 0), + "Select a file (Exactly 1MB in size)", + "\\.(po|hdv|bin|rom)$", + getenv("HOME")); + mui_window_set_action(w, _mii_1mb_stdfile_cb, m); + } break; + case MII_1MB_USE_BIN: + mui_control_set_state(m->load, MUI_CONTROL_STATE_NORMAL); + m->config.use_default = 1; + break; + case MII_1MB_USE_FILE: { + m->config.use_default = 1; + if (m->config.drive.disk[0] == 0) { + mui_control_set_state(m->load, MUI_CONTROL_STATE_DISABLED); + } else { + m->config.use_default = 0; + mui_control_set_state(m->load, MUI_CONTROL_STATE_NORMAL); + } + } break; + } + break; + } + return 0; +} + +struct mui_window_t * +mii_mui_load_1mbrom( + struct mui_t *mui, + mii_1mb_conf_t *config) +{ + mui_t *ui = mui; + float base_size = mui_font_find(ui, "main")->size; + float margin = base_size * 0.7; + mui_window_t *w = mui_window_get_by_id(mui, MII_1MB_WINDOW_ID); + if (w) { + mui_window_select(w); + return w; + } + c2_pt_t where = {}; + c2_rect_t wpos = C2_RECT_WH(where.x, where.y, 560, 240); + if (where.x == 0 && where.y == 0) + c2_rect_offset(&wpos, + (ui->screen_size.x / 2) - (c2_rect_width(&wpos) / 2), + (ui->screen_size.y * 0.4) - (c2_rect_height(&wpos) / 2)); + w = mui_window_create(mui, + wpos, + NULL, MUI_WINDOW_LAYER_MODAL, + "1MB ROM Card File", + sizeof(mii_mui_1mb_t)); + mui_window_set_id(w, MII_1MB_WINDOW_ID); + mii_mui_1mb_t * m = (mii_mui_1mb_t*)w; + m->dst = config; + mui_control_t * c = NULL; + c2_rect_t cf; + cf = C2_RECT_WH(0, 0, base_size * 4, base_size * 1.4); + c2_rect_left_of(&cf, c2_rect_width(&w->content), margin); + c2_rect_top_of(&cf, c2_rect_height(&w->content), margin); + m->load = c = mui_button_new(w, + cf, MUI_BUTTON_STYLE_DEFAULT, + "OK", MII_1MB_SAVE); + c->key_equ = MUI_KEY_EQU(0, 13); + c->state = MUI_CONTROL_STATE_DISABLED; + c2_rect_left_of(&cf, cf.l, margin); + c = mui_button_new(w, + cf, MUI_BUTTON_STYLE_NORMAL, + "Cancel", MII_1MB_CANCEL); + c->key_equ = MUI_KEY_EQU(0, 27); + + c2_rect_right_of(&cf, 0, margin); + cf.r += 15; + c = mui_button_new(w, + cf, MUI_BUTTON_STYLE_NORMAL, + "Select…" , MII_1MB_SELECT); + c->key_equ = MUI_KEY_EQU(MUI_MODIFIER_ALT, 's'); + + c2_rect_set(&cf, margin * 3, (margin/2), + c2_rect_width(&w->frame) - margin, (margin/2) + base_size); + c2_rect_t cp = cf; + cp.l -= margin * 0.2; + cp.b += base_size * 1.3; + c = mui_groupbox_new(w, cp, "File to load:", MUI_CONTROL_TEXTBOX_FRAME); + + float icons_size = mui_font_find(ui, "icon_small")->size; + c2_rect_bottom_of(&cf, cf.b, 0); + cf.b = cf.t + icons_size; + cf.r = cf.l + icons_size; + m->icon = c = mui_textbox_new(w, cf, MUI_ICON_FILE, "icon_small", + MUI_TEXT_ALIGN_MIDDLE | MUI_TEXT_ALIGN_CENTER | 0); + c->state = MUI_CONTROL_STATE_DISABLED; + cf.l = cf.r; + cf.r = c2_rect_width(&w->content) - margin; + m->fname = c = mui_textbox_new(w, cf, + "Click \"Select\" to pick a file", NULL, 0); + c->state = MUI_CONTROL_STATE_DISABLED; + + uint32_t uid_mask = FCC(0xff,0xff,0xff,0); + cf.l = margin; + m->r0 = c = mui_button_new(w, + cf, MUI_BUTTON_STYLE_RADIO, + "", FCC('r','o','m','0')); + c->uid_mask = uid_mask; + c->value = config->use_default == 0; + c2_rect_bottom_of(&cf, cf.b, margin * 0.8); + m->r1 = c = mui_button_new(w, + cf, MUI_BUTTON_STYLE_RADIO, + "Use Built-in image with a few games", + FCC('r','o','m','1')); + c->uid_mask = uid_mask; + c->value = config->use_default != 0; + + c = NULL; + TAILQ_FOREACH(c, &w->controls, self) { + if (mui_control_get_uid(c) == 0) + continue; + mui_control_set_action(c, _mii_1mb_action_cb, m); + } + m->config = *config; + if (m->config.drive.disk[0] && + _mii_check_1mb_file(mui, m->config.drive.disk) >= 0) { + char * path = m->config.drive.disk; + mui_control_set_state(m->fname, MUI_CONTROL_STATE_NORMAL); + char *dup = strdup(path); + mui_control_set_title(m->fname, basename(dup)); + free(dup); + mui_control_set_state(m->icon, MUI_CONTROL_STATE_NORMAL); + mui_control_set_state(m->load, MUI_CONTROL_STATE_NORMAL); + } else { + m->config.drive.disk[0] = 0; + mui_control_set_state(m->load, MUI_CONTROL_STATE_NORMAL); + } + return w; +} + diff --git a/ui_gl/mii_mui_2dsk.c b/ui_gl/mii_mui_2dsk.c new file mode 100644 index 0000000..e596ac0 --- /dev/null +++ b/ui_gl/mii_mui_2dsk.c @@ -0,0 +1,255 @@ +/* + * mui_mui_2disks.c + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ + +#include +#include +#include +#include +#include +#include +#include "mui.h" + +#include "mii_mui_settings.h" + +enum { + MII_2DSK_WINDOW_ID = FCC('2','d','s','k'), + MII_2DSK_SAVE = FCC('s','a','v','e'), + MII_2DSK_CANCEL = FCC('c','a','n','c'), + MII_2DSK_SELECT1 = FCC('s','e','l','1'), + MII_2DSK_SELECT2 = FCC('s','e','l','2'), +}; + +typedef struct mii_mui_2dsk_t { + mui_window_t win; + uint8_t drive_kind; + mui_control_t * load; + uint32_t selecting; + struct { + mui_control_t *icon, *fname, *button; + } drive[2]; + + mii_2dsk_conf_t * dst; + mii_2dsk_conf_t config; +} mii_mui_2dsk_t; + +static void +mii_mui_2dsk_load_conf( + mii_mui_2dsk_t * m, + mii_2dsk_conf_t * config) +{ + int ok = 0; + for (int i = 0; i < 2; i++) { + if (config->drive[i].disk[0]) { + ok = 1; + mui_control_set_state(m->drive[i].fname, MUI_CONTROL_STATE_NORMAL); + char *dup = strdup(config->drive[i].disk); + mui_control_set_title(m->drive[i].fname, basename(dup)); + free(dup); + mui_control_set_state(m->drive[i].icon, MUI_CONTROL_STATE_NORMAL); + mui_control_set_title(m->drive[i].button, "Eject"); + } else { + mui_control_set_state(m->drive[i].fname, MUI_CONTROL_STATE_DISABLED); + mui_control_set_title(m->drive[i].fname, "Click \"Select\" to pick a file"); + mui_control_set_state(m->drive[i].icon, MUI_CONTROL_STATE_DISABLED); + mui_control_set_title(m->drive[i].button, "Select…"); + } + } + if (ok) + mui_control_set_state(m->load, MUI_CONTROL_STATE_NORMAL); +// else +// mui_control_set_state(m->load, MUI_CONTROL_STATE_DISABLED); +} + +static int +_mii_2dsk_stdfile_cb( + mui_window_t * w, + void * cb_param, // mii_mui_2dsk_t + uint32_t what, + void * param) // not used +{ + mii_mui_2dsk_t * m = cb_param; + switch (what) { + case MUI_STDF_ACTION_SELECT: { + int idx = m->selecting == MII_2DSK_SELECT1 ? 0 : 1; + char * path = mui_stdfile_get_selected_path(w); + printf("%s select %s\n", __func__, path); + strncpy(m->config.drive[idx].disk, path, + sizeof(m->config.drive[idx].disk)-1); + mii_mui_2dsk_load_conf(m, &m->config); + mui_window_dispose(w); + } break; + case MUI_STDF_ACTION_CANCEL: + printf("%s cancel\n", __func__); + mui_window_dispose(w); + break; + } + return 0; +} + +static int +_mii_2dsk_action_cb( + mui_control_t * c, + void * cb_param, // mii_mui_2dsk_t + uint32_t what, + void * param) // not used +{ + printf("%s %4.4s\n", __func__, (char*)&what); + mii_mui_2dsk_t * m = cb_param; + uint32_t uid = mui_control_get_uid(c); + + switch (what) { + case MUI_CONTROL_ACTION_SELECT: + printf("%s control %4.4s\n", __func__, (char*)&uid); + switch (uid) { + case MII_2DSK_SAVE: { + // save the config + printf("%s save\n", __func__); + if (m->dst) + *m->dst = m->config; + mui_window_action(&m->win, + m->drive_kind == MII_2DSK_SMARTPORT ? + MII_MUI_SMARTPORT_SAVE : + MII_MUI_DISK2_SAVE, + m->dst); + mui_window_dispose(&m->win); + } break; + case MII_2DSK_CANCEL: { + // cancel the config + printf("%s cancel\n", __func__); + mui_window_dispose(&m->win); + } break; + case MII_2DSK_SELECT1: + case MII_2DSK_SELECT2: { + mii_2dsk_conf_t * config = &m->config; + // select a file + m->selecting = uid; // remember which drive we're selecting + int idx = uid == MII_2DSK_SELECT1 ? 0 : 1; + if (config->drive[idx].disk[0]) { + printf("%s eject %d\n", __func__, idx); + config->drive[idx].disk[0] = 0; + mii_mui_2dsk_load_conf(m, config); + } else { + printf("%s select %d\n", __func__, idx); + mui_window_t * w = mui_stdfile_get(m->win.ui, + C2_PT(0, 0), + m->drive_kind == MII_2DSK_SMARTPORT ? + "Select PO/HDV/2MG file to load" : + "Select DSK file to load", + m->drive_kind == MII_2DSK_SMARTPORT ? + "\\.(po|hdv|2mg)$" : + "\\.(dsk)$", + getenv("HOME")); + mui_window_set_action(w, _mii_2dsk_stdfile_cb, m); + } + } break; + } + break; + } + return 0; +} + +struct mui_window_t * +mii_mui_load_2dsk( + struct mui_t *mui, + mii_2dsk_conf_t *config, + uint8_t drive_kind) +{ + mui_t *ui = mui; + float base_size = mui_font_find(ui, "main")->size; + float margin = base_size * 0.7; + mui_window_t *w = mui_window_get_by_id(mui, MII_2DSK_WINDOW_ID); + if (w) { + mui_window_select(w); + return w; + } + c2_pt_t where = {}; + c2_rect_t wpos = C2_RECT_WH(where.x, where.y, 640, 275); + if (where.x == 0 && where.y == 0) + c2_rect_offset(&wpos, + (ui->screen_size.x / 2) - (c2_rect_width(&wpos) / 2), + (ui->screen_size.y * 0.4) - (c2_rect_height(&wpos) / 2)); + char label[128]; + sprintf(label, "Select Files for %s 1&2 (Slot %d)", + drive_kind == MII_2DSK_SMARTPORT ? "SmartPort" : "Disk II", + config->slot_id + 1); + w = mui_window_create(mui, + wpos, NULL, 0, label, + sizeof(mii_mui_2dsk_t)); + mui_window_set_id(w, MII_2DSK_WINDOW_ID); + mii_mui_2dsk_t * m = (mii_mui_2dsk_t*)w; + m->drive_kind = drive_kind; + + mui_control_t * c = NULL; + c2_rect_t cf; + cf = C2_RECT_WH(0, 0, base_size * 4, base_size * 1.4); + c2_rect_left_of(&cf, c2_rect_width(&w->content), margin); + c2_rect_top_of(&cf, c2_rect_height(&w->content), margin); + m->load = c = mui_button_new(w, + cf, MUI_BUTTON_STYLE_DEFAULT, + "OK", MII_2DSK_SAVE); + c->key_equ = MUI_KEY_EQU(0, 13); + c2_rect_left_of(&cf, cf.l, margin); + c = mui_button_new(w, + cf, MUI_BUTTON_STYLE_NORMAL, + "Cancel", MII_2DSK_CANCEL); + c->key_equ = MUI_KEY_EQU(0, 27); + + c2_rect_set(&cf, margin, (margin / 2), + c2_rect_width(&w->frame) - margin - 120, + (margin/2) + base_size); + c2_rect_t cp = cf; + cp.l -= margin * 0.2; + cp.b += base_size * 1.3; + for (int i = 0; i < 2; i++) { + char buf[32]; + snprintf(buf, sizeof(buf), "Drive %d:", i+1); + c = mui_groupbox_new(w, cp, buf, MUI_CONTROL_TEXTBOX_FRAME); + + float icons_size = mui_font_find(ui, "icon_small")->size; + c2_rect_bottom_of(&cf, cp.t, base_size); + c2_rect_right_of(&cf, cp.l, margin * 0.5); + cf.b = cf.t + icons_size; + cf.r = cf.l + icons_size; + m->drive[i].icon = c = mui_textbox_new(w, cf, + MUI_ICON_FILE, "icon_small", + MUI_TEXT_ALIGN_MIDDLE | MUI_TEXT_ALIGN_CENTER | 0); + c->state = MUI_CONTROL_STATE_DISABLED; + cf.l = cf.r; + cf.r = cp.r - margin * 0.5; + m->drive[i].fname = c = mui_textbox_new(w, cf, + "Click \"Select\" to pick a file", NULL, 0); + c->state = MUI_CONTROL_STATE_DISABLED; + + c2_rect_right_of(&cf, cp.r, margin * 0.8); + cf.r = c2_rect_width(&w->frame) - margin * 1.2; + c2_rect_inset(&cf, -4,-4); + m->drive[i].button = c = mui_button_new(w, + cf, MUI_BUTTON_STYLE_NORMAL, + "Select…" , i == 0 ? + MII_2DSK_SELECT1 : MII_2DSK_SELECT2); + c->key_equ = MUI_KEY_EQU(MUI_MODIFIER_ALT, '1' + i); + + c2_rect_bottom_of(&cp, cp.b, margin * 0.2); + } + c2_rect_bottom_of(&cp, cp.t, margin * 0.8); + cp.l = margin * 4; + cp.r = c2_rect_width(&w->frame) - margin * 4; + c = mui_separator_new(w, cp); + + c = NULL; + TAILQ_FOREACH(c, &w->controls, self) { + if (mui_control_get_uid(c) == 0) + continue; + mui_control_set_action(c, _mii_2dsk_action_cb, m); + } + m->dst = config; + m->config = *config; + mii_mui_2dsk_load_conf(m, config); + return w; +} + diff --git a/ui_gl/mii_mui_loadbin.c b/ui_gl/mii_mui_loadbin.c new file mode 100644 index 0000000..2578579 --- /dev/null +++ b/ui_gl/mii_mui_loadbin.c @@ -0,0 +1,213 @@ +/* + * mui_mui_loadbin.c + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ + +#include +#include +#include +#include +#include +#include +#include "mui.h" + +#include "mii_mui_settings.h" + +enum { + MII_LBIN_WINDOW_ID = FCC('l','b','i','n'), + MII_LBIN_SAVE = FCC('s','a','v','e'), + MII_LBIN_CANCEL = FCC('c','a','n','c'), + MII_LBIN_SELECT = FCC('s','e','l','e'), +}; + + +typedef struct mii_mui_loadbin_t { + mui_window_t win; + mui_control_t * load, *icon, *fname; +} mii_mui_loadbin_t; + +static int +_mii_loadbin_stdfile_cb( + mui_window_t * w, + void * cb_param, // mii_mui_loadbin_t + uint32_t what, + void * param) // not used +{ + mii_mui_loadbin_t * m = cb_param; + switch (what) { + case MUI_STDF_ACTION_SELECT: { + char * path = mui_stdfile_get_selected_path(w); + printf("%s select %s\n", __func__, path); + mui_control_set_state(m->fname, MUI_CONTROL_STATE_NORMAL); + char *dup = strdup(path); + mui_control_set_title(m->fname, basename(dup)); + free(dup); + mui_control_set_state(m->icon, MUI_CONTROL_STATE_NORMAL); + mui_control_set_state(m->load, MUI_CONTROL_STATE_NORMAL); + mui_window_dispose(w); + } break; + case MUI_STDF_ACTION_CANCEL: + printf("%s cancel\n", __func__); + mui_window_dispose(w); + break; + } + return 0; +} + +static int +_mii_loadbin_action_cb( + mui_control_t * c, + void * cb_param, // mii_mui_loadbin_t + uint32_t what, + void * param) // not used +{ + printf("%s %4.4s\n", __func__, (char*)&what); + mii_mui_loadbin_t * m = cb_param; + uint32_t uid = mui_control_get_uid(c); + + switch (what) { + case MUI_CONTROL_ACTION_SELECT: + printf("%s control %4.4s\n", __func__, (char*)&uid); + switch (uid) { + case MII_LBIN_SAVE: { + // save the config + printf("%s save\n", __func__); + mui_window_dispose(&m->win); + } break; + case MII_LBIN_CANCEL: { + // cancel the config + printf("%s cancel\n", __func__); + mui_window_dispose(&m->win); + } break; + case MII_LBIN_SELECT: { + // select a file + printf("%s select\n", __func__); + mui_window_t * w = mui_stdfile_get(m->win.ui, + C2_PT(0, 0), + "Select a .bin file to run", + "\\.(bin|rom)$", + getenv("HOME")); + mui_window_set_action(w, _mii_loadbin_stdfile_cb, m); + } break; + } + break; + } + return 0; +} + +struct mui_window_t * +mii_mui_load_binary( + struct mui_t *mui, + mii_loadbin_conf_t *config) +{ + mui_t *ui = mui; + float base_size = mui_font_find(ui, "main")->size; + float margin = base_size * 0.7; + mui_window_t *w = mui_window_get_by_id(mui, MII_LBIN_WINDOW_ID); + if (w) { + mui_window_select(w); + return w; + } + c2_pt_t where = {}; + c2_rect_t wpos = C2_RECT_WH(where.x, where.y, 480, 294); + if (where.x == 0 && where.y == 0) + c2_rect_offset(&wpos, + (ui->screen_size.x / 2) - (c2_rect_width(&wpos) / 2), + (ui->screen_size.y * 0.4) - (c2_rect_height(&wpos) / 2)); + w = mui_window_create(mui, + wpos, + NULL, 0, + "Load&Run Binary File", + sizeof(mii_mui_loadbin_t)); + mui_window_set_id(w, MII_LBIN_WINDOW_ID); + mii_mui_loadbin_t * m = (mii_mui_loadbin_t*)w; + + mui_control_t * c = NULL; + c2_rect_t cf; + cf = C2_RECT_WH(0, 0, base_size * 4, base_size * 1.4); + c2_rect_left_of(&cf, c2_rect_width(&w->content), margin); + c2_rect_top_of(&cf, c2_rect_height(&w->content), margin); + m->load = c = mui_button_new(w, + cf, MUI_BUTTON_STYLE_DEFAULT, + "Load", MII_LBIN_SAVE); + c->key_equ = MUI_KEY_EQU(0, 13); + c->state = MUI_CONTROL_STATE_DISABLED; + c2_rect_left_of(&cf, cf.l, margin); + c = mui_button_new(w, + cf, MUI_BUTTON_STYLE_NORMAL, + "Cancel", MII_LBIN_CANCEL); + c->key_equ = MUI_KEY_EQU(0, 27); + + c2_rect_right_of(&cf, 0, margin); + c = mui_button_new(w, + cf, MUI_BUTTON_STYLE_NORMAL, + "Select…" , MII_LBIN_SELECT); + c->key_equ = MUI_KEY_EQU(MUI_MODIFIER_ALT, 's'); + + cf.b = cf.t + base_size; + c2_rect_top_of(&cf, cf.t, margin); + c2_rect_right_of(&cf, 0, margin); + cf.r = cf.l + 200; + c = mui_button_new(w, + cf, MUI_BUTTON_STYLE_CHECKBOX, + "Auto Reload", 0); + c->key_equ = MUI_KEY_EQU(MUI_MODIFIER_ALT, 'r'); + // this tell libmui that it can clear the radio values of the 'sister' + // radio buttons when one matching the uid&mask is selected + uint32_t uid_mask = FCC(0xff,0xff,0xff,0); + c2_rect_top_of(&cf, cf.t, 10); + cf.l += margin * 2; + cf.r = cf.l + 110; + c = mui_button_new(w, + cf, MUI_BUTTON_STYLE_RADIO, + "$0300", FCC('a','d','d','0')); + c->uid_mask = uid_mask; + c->value = 1; + c2_rect_right_of(&cf, cf.r, margin); + c = mui_button_new(w, + cf, MUI_BUTTON_STYLE_RADIO, + "$0800", FCC('a','d','d','1')); + c->uid_mask = uid_mask; + c2_rect_right_of(&cf, cf.r, margin); + c = mui_button_new(w, + cf, MUI_BUTTON_STYLE_RADIO, + "$2000", FCC('a','d','d','2')); + c->uid_mask = uid_mask; + + c2_rect_right_of(&cf, 0, margin); + c2_rect_top_of(&cf, cf.t, margin / 2); + cf.r = cf.l + 200; + c = mui_textbox_new(w, cf, "Load address:", NULL, 0); + + c2_rect_set(&cf, margin, (margin/2), + c2_rect_width(&w->frame) - margin, (margin/2) + base_size); + c2_rect_t cp = cf; + cp.l -= margin * 0.2; + cp.b += base_size * 1.3; + c = mui_groupbox_new(w, cp, "File to load:", MUI_CONTROL_TEXTBOX_FRAME); + + float icons_size = mui_font_find(ui, "icon_small")->size; + c2_rect_bottom_of(&cf, cf.b, 0); + cf.b = cf.t + icons_size; + cf.r = cf.l + icons_size; + m->icon = c = mui_textbox_new(w, cf, MUI_ICON_FILE, "icon_small", + MUI_TEXT_ALIGN_MIDDLE | MUI_TEXT_ALIGN_CENTER | 0); + c->state = MUI_CONTROL_STATE_DISABLED; + cf.l = cf.r; + cf.r = c2_rect_width(&w->content) - margin; + m->fname = c = mui_textbox_new(w, cf, "Click \"Select\" to pick a file", NULL, 0); + c->state = MUI_CONTROL_STATE_DISABLED; + + c = NULL; + TAILQ_FOREACH(c, &w->controls, self) { + if (mui_control_get_uid(c) == 0) + continue; + mui_control_set_action(c, _mii_loadbin_action_cb, m); + } + + return w; +} + diff --git a/ui_gl/mii_mui_menus.c b/ui_gl/mii_mui_menus.c new file mode 100644 index 0000000..4e63294 --- /dev/null +++ b/ui_gl/mii_mui_menus.c @@ -0,0 +1,403 @@ +/* + * mii_mui_menus.c + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ +#include +#include +#include +#include +#include + +#include "mii_mui.h" +#include "mii_thread.h" +#define MII_MUI_MENUS_C +#include "mii_mui_menus.h" +#include "mii_mui_settings.h" + +/* this is heavily endian dependend, as is the FCC macro */ +#define FCC_INDEX(_fcc) (isdigit(_fcc>>24) ? ((_fcc >> 24) - '0') : 0) + +struct mii_x11_t; +void +mii_x11_reload_config( + struct mii_x11_t *ui ); +void +mii_ui_reconfigure_slot( + mii_t * mii, + mii_machine_config_t * config, + int slot ); + + +static int +mii_quit_confirm_cb( + mui_window_t *win, + void * cb_param, + uint32_t what, + void * param) +{ + mii_mui_t * ui = cb_param; + printf("%s %4.4s\n", __func__, (char*)&what); + if (what == MUI_CONTROL_ACTION_SELECT) { + mui_control_t * c = param; + printf("%s %4.4s\n", __func__, (char*)&c->uid); + if (c->uid == MUI_ALERT_BUTTON_OK) { + printf("%s Quit\n", __func__); + mii_t * mii = &ui->mii; + mii->state = MII_TERMINATE; + } + } + return 0; +} + +static int +mii_config_save_cb( + mui_window_t *win, + void * cb_param, + uint32_t what, + void * param) +{ + mii_mui_t * ui = cb_param; + printf("%s %4.4s\n", __func__, (char*)&what); + switch (what) { + case MII_MUI_SLOTS_SAVE: + printf("%s *** Rebooting\n", __func__); + mii_x11_reload_config((void*)ui); + break; + case MII_MUI_DISK2_SAVE: + case MII_MUI_SMARTPORT_SAVE: { + mii_2dsk_conf_t * conf = param; + mii_t * mii = &ui->mii; + mii_machine_config_t * config = &ui->config; + mii_ui_reconfigure_slot(mii, config, conf->slot_id + 1); + } break; + case MII_MUI_1MB_SAVE: { + mii_1mb_conf_t * conf = param; + mii_t * mii = &ui->mii; + mii_machine_config_t * config = &ui->config; + mii_ui_reconfigure_slot(mii, config, conf->slot_id + 1); + } break; + } + return 0; +} + +void +mii_config_open_slots_dialog( + mii_mui_t * ui) +{ + mui_window_set_action( + mii_mui_configure_slots(&ui->mui, &ui->config), + mii_config_save_cb, ui); +} + + +static void +_mii_show_about( + mii_mui_t * ui) +{ + mui_t * mui = &ui->mui; + mui_window_t *w = mui_window_get_by_id(mui, FCC('a','b','o','t')); + if (w) { + mui_window_select(w); + return; + } + w = mui_alert(mui, C2_PT(0,0), + "About MII", + "Version " MII_VERSION "\n" + "Build " __DATE__ " " __TIME__, + MUI_ALERT_INFO); + mui_window_set_id(w, FCC('a','b','o','t')); +} + +static int +mii_menubar_action( + mui_window_t *win, // window (menubar) + void * cb_param, + uint32_t what, + void * param) +{ + mii_mui_t * ui = cb_param; + mui_t * mui = &ui->mui; + mii_t * mii = &ui->mii; + +// printf("%s %4.4s\n", __func__, (char*)&what); + + switch (what) { + case MUI_MENUBAR_ACTION_PREPARE: { + mui_menu_item_t * items = param; + // printf("%s prepare (%s)\n", __func__, items[0].title); + for (int i = 0; items[i].title; i++) { + switch (items[i].uid) { + case FCC('v','d','c','0'): + case FCC('v','d','c','1'): + case FCC('v','d','c','2'): { + int idx = FCC_INDEX(items[i].uid); + if (mii->video.color_mode == idx) + strcpy(items[i].mark, MUI_GLYPH_TICK); + else + items[i].mark[0] = 0; + } break; + case FCC('a','u','d','0'): + if (mii->speaker.muted) + strcpy(items[i].mark, MUI_GLYPH_TICK); + else + items[i].mark[0] = 0; + break; + case FCC('a','u','d','+'): + items[i].disabled = mii->speaker.volume >= 10; + break; + case FCC('a','u','d','-'): + items[i].disabled = mii->speaker.volume <= 0.1; + break; + case FCC('m','h','z','1'): + if (mii->speed <= 1.1 && mii->speed >= 0.9) + strcpy(items[i].mark, MUI_GLYPH_TICK); + else + items[i].mark[0] = 0; + break; + case FCC('m','h','z','3'): + if (mii->speed >= 2) + strcpy(items[i].mark, MUI_GLYPH_TICK); + else + items[i].mark[0] = 0; + break; + case FCC('s','t','o','p'): { + if (mii->state == MII_STOPPED) { + items[i].title = "Stopped"; + strcpy(items[i].mark, MUI_GLYPH_TICK); + } else { + items[i].title = "Stop"; + items[i].mark[0] = 0; + } + } break; + case FCC('r','u','n',' '): { + if (mii->state == MII_RUNNING) { + items[i].title = "Running"; + strcpy(items[i].mark, MUI_GLYPH_TICK); + } else { + items[i].title = "Continue"; + items[i].mark[0] = 0; + } + } break; + case FCC('n','e','x','t'): + case FCC('s','t','e','p'): { + items[i].disabled = mii->state == MII_RUNNING; + } break; + /* + * These menu items are disabled when a window/dialog + * is open. + */ + // case FCC('j','o','y','s'): + // case FCC('l','r','u','n'): + case FCC('a','b','o','t'): + case FCC('s','l','o','t'): { + items[i].disabled = mui_window_front(mui) != NULL; + } break; + case FCC('s','h','m','b'): { + items[i].disabled = + (mui_window_front(mui) != NULL) || + (ui->transition != MII_MUI_TRANSITION_NONE); + } break; + } + } + } break; + case MUI_MENUBAR_ACTION_SELECT: { + mui_menu_item_t * item = param; + // printf("%s Selected %4.4s '%s'\n", __func__, + // (char*)&item->uid, item->title); + switch (item->uid) { + case FCC('a','b','o','t'): { + _mii_show_about(ui); + } break; + case FCC('q','u','i','t'): { +// printf("%s Quit?\n", __func__); + if (!ui->mui_visible && + ui->transition == MII_MUI_TRANSITION_NONE) + ui->transition = MII_MUI_TRANSITION_SHOW_UI; + mui_window_t * really = mui_window_get_by_id( + &ui->mui, FCC('q','u','i','t')); + if (really) + mui_window_select(really); + else { + really = mui_alert(mui, C2_PT(0,0), + "Quitting", + "Do you really want to quit the emulator?", + MUI_ALERT_WARN); + really->uid = FCC('q','u','i','t'); + mui_window_set_action(really, mii_quit_confirm_cb, ui); + } + } break; + case FCC('s','h','m','b'): { + if (ui->transition != MII_MUI_TRANSITION_NONE) + break; + if (ui->mui_visible) { + ui->transition = MII_MUI_TRANSITION_HIDE_UI; + } else { + ui->transition = MII_MUI_TRANSITION_SHOW_UI; + } + } break; + case FCC('d','s','k','0'): + case FCC('d','s','k','1'): + case FCC('d','s','k','2'): + case FCC('d','s','k','3'): + case FCC('d','s','k','4'): + case FCC('d','s','k','5'): + case FCC('d','s','k','6'): { + int slot = FCC_INDEX(item->uid); + printf("%s configure slot %d\n", __func__, slot); + mui_window_set_action( + mii_mui_configure_slot(&ui->mui, &ui->config, slot), + mii_config_save_cb, ui); + } break; + case FCC('s','l','o','t'): { + mii_config_open_slots_dialog(ui); + } break; + case FCC('l','r','u','n'): { + mui_window_set_action( + mii_mui_load_binary(&ui->mui, &ui->loadbin_conf), + mii_config_save_cb, ui); + } break; + case FCC('c','r','e','s'): + case FCC('r','e','s','t'): { + mii_th_signal_t sig = { + .cmd = SIGNAL_RESET, + .data = item->uid == FCC('c','r','e','s'), + }; + mii_th_fifo_write(mii_thread_get_fifo(&ui->mii), sig); + } break; + case FCC('a','u','d','0'): + mii->speaker.muted = !mii->speaker.muted; + ui->config.audio_muted = mii->speaker.muted; + break; + case FCC('a','u','d','+'): + mii_speaker_volume(&mii->speaker, mii->speaker.volume + 1); + break; + case FCC('a','u','d','-'): + mii_speaker_volume(&mii->speaker, mii->speaker.volume - 1); + break; + case FCC('v','d','C','l'): { +// printf("%s Cycle video\n", __func__); + mii->video.color_mode = (mii->video.color_mode + 1) % 3; + ui->config.video_mode = mii->video.color_mode; + } break; + case FCC('v','d','c','0'): + case FCC('v','d','c','1'): + case FCC('v','d','c','2'): + mii->video.color_mode = FCC_INDEX(item->uid); + ui->config.video_mode = mii->video.color_mode; + break; + case FCC('m','h','z','1'): + mii->speed = 1; + break; + case FCC('m','h','z','3'): + mii->speed = 3.58; + break; + case FCC('s','t','o','p'): { + mii_th_signal_t sig = { + .cmd = SIGNAL_STOP, + }; + mii_th_fifo_write(mii_thread_get_fifo(&ui->mii), sig); + } break; + case FCC('s','t','e','p'): { + mii_th_signal_t sig = { + .cmd = SIGNAL_STEP, + }; + mii_th_fifo_write(mii_thread_get_fifo(&ui->mii), sig); + } break; + case FCC('r','u','n',' '): { + mii_th_signal_t sig = { + .cmd = SIGNAL_RUN, + }; + mii_th_fifo_write(mii_thread_get_fifo(&ui->mii), sig); + } break; + default: + printf("%s menu item %4.4s %s IGNORED\n", + __func__, (char*)&item->uid, item->title); + break; + } + } break; + default: + printf("%s %4.4s IGNORED?\n", __func__, (char*)&what); + break; + } + return 0; +} + +void +mii_mui_menu_slot_menu_update( + mii_mui_t * ui) +{ + mui_window_t * mbar = mui_menubar_get(&ui->mui); + mui_control_t * m = mui_control_get_by_id(mbar, FCC('f','i','l','e')); + mui_menu_items_t * items = mui_popupmenu_get_items(m); + + printf("%s items %p count %d RO:%d\n", + __func__, items, items? items->count : 0, items? items->read_only : 0); + + mui_menu_items_clear(items); + if (!items->read_only) + mui_menu_items_free(items); + static const mui_menu_items_t zero = {0}; + *items = zero; + + for (int i = 0; i < 7; i++) { + static char static_label[7][64]; + char * label = static_label[i]; + label[0] = 0; + int slot = i + 1; + switch (ui->config.slot[i].driver) { + case MII_SLOT_DRIVER_SMARTPORT: + sprintf(label, "%d: SmartPort…", slot); + break; + case MII_SLOT_DRIVER_DISK2: + sprintf(label, "%d: Disk ][…", slot); + break; + case MII_SLOT_DRIVER_ROM1MB: + sprintf(label, "%d: ROM 1MB…", slot); + break; + default: + break; + } + if (label[0]) { + mui_menu_items_push(items, (mui_menu_item_t){ + .title = label, + .uid = FCC('d','s','k','0' + i), + }); + } + } + int start = items->count ? 1 : 0; + for (int i = start; m_file_menu[i].title; i++) { + mui_menu_items_push(items, m_file_menu[i]); + } + // array is NULL terminated + mui_menu_items_push(items, (mui_menu_item_t){}); + for (int i = 0; i < (int)items->count; i++) + printf(" %d: %s\n", i, items->e[i].title); + + printf("%s items %p count %d RO:%d done\n", + __func__, items, items? items->count : 0, items? items->read_only : 0); +} + +void +mii_mui_menus_init( + mii_mui_t * ui) +{ +// printf("%s\n", __func__); + mui_window_t * mbar = mui_menubar_new(&ui->mui); + mui_window_set_action(mbar, mii_menubar_action, ui); + + mui_menubar_add_simple(mbar, MUI_GLYPH_APPLE, + FCC('a','p','p','l'), + m_apple_menu); + mui_menubar_add_simple(mbar, "File", + FCC('f','i','l','e'), + m_file_menu); + mui_menubar_add_simple(mbar, "Machine", + FCC('m','a','c','h'), + m_machine_menu); + mui_menubar_add_simple(mbar, "CPU", + FCC('c','p','u','m'), + m_cpu_menu); +} \ No newline at end of file diff --git a/ui_gl/mii_mui_menus.h b/ui_gl/mii_mui_menus.h new file mode 100644 index 0000000..2c4ebf2 --- /dev/null +++ b/ui_gl/mii_mui_menus.h @@ -0,0 +1,132 @@ +/* + * mui_mui_menus.h + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ + +#pragma once +#include "mui.h" + +extern mui_menu_item_t m_apple_menu[]; +extern mui_menu_item_t m_file_menu[]; +extern mui_menu_item_t m_video_menu[]; +extern mui_menu_item_t m_audio_menu[]; +extern mui_menu_item_t m_machine_menu[]; +extern mui_menu_item_t m_cpu_menu[]; + +#ifdef MII_MUI_MENUS_C +mui_menu_item_t m_apple_menu[] = { + { .title = "About MII…", + .uid = FCC('a','b','o','t') }, +// { .title = "-", }, + { }, +}; +mui_menu_item_t m_file_menu[] = { + /* these two don't do anything for the moment */ + { .title = "No Drives Installed…", + .disabled = 1, + .uid = FCC('l','d','i','m'), + .kcombo = MUI_GLYPH_SHIFT MUI_GLYPH_COMMAND "N" }, + { .title = "-", }, + { .title = "Load & Run Binary…", + .disabled = 1, + .uid = FCC('l','r','u','n'), }, + { .title = "-" }, + { .title = "Quit", + .uid = FCC('q','u','i','t'), + .key_equ = MUI_KEY_EQU(MUI_MODIFIER_RCTRL|MUI_MODIFIER_RSUPER, 'Q'), + .kcombo = MUI_GLYPH_CONTROL MUI_GLYPH_COMMAND "Q" }, + { }, +}; +mui_menu_item_t m_video_menu[] = { + { .title = "Show/Hide Menus", + .uid = FCC('s','h','m','b'), + .key_equ = MUI_KEY_EQU(MUI_MODIFIER_RCTRL, MUI_KEY_F1), + .kcombo = MUI_GLYPH_CONTROL MUI_GLYPH_F1 }, + { .title = "Cycle Mode", + .uid = FCC('v','d','C','l'), + .key_equ = MUI_KEY_EQU(MUI_MODIFIER_RCTRL, MUI_KEY_F11), + .kcombo = MUI_GLYPH_CONTROL MUI_GLYPH_F11 }, + { .title = "-", }, + { .title = "Color", + .mark = MUI_GLYPH_TICK, + .uid = FCC('v','d','c','0') }, + { .title = "Green", + .uid = FCC('v','d','c','1') }, + { .title = "Amber", + .uid = FCC('v','d','c','2') }, + { }, +}; +mui_menu_item_t m_audio_menu[] = { + { .title = "Mute", + .uid = FCC('a','u','d','0'), + .key_equ = MUI_KEY_EQU(MUI_MODIFIER_RCTRL, MUI_KEY_F10), + .kcombo = MUI_GLYPH_CONTROL MUI_GLYPH_F10 }, + { .title = "-", }, + { .title = "Louder", + .uid = FCC('a','u','d','+'), + .key_equ = MUI_KEY_EQU(MUI_MODIFIER_RCTRL|MUI_MODIFIER_RSHIFT, '+'), + .kcombo = MUI_GLYPH_CONTROL MUI_GLYPH_SHIFT "+" }, + { .title = "Quieter", + .uid = FCC('a','u','d','-'), + // shift-minus is _ (underscore) + .key_equ = MUI_KEY_EQU(MUI_MODIFIER_RCTRL|MUI_MODIFIER_RSHIFT, '_'), + .kcombo = MUI_GLYPH_CONTROL MUI_GLYPH_SHIFT "-" }, + { }, +}; +mui_menu_item_t m_machine_menu[] = { + { .title = MUI_GLYPH_OAPPLE "-Control-Reset", + .uid = FCC('c','r','e','s'), + .key_equ = MUI_KEY_EQU(MUI_MODIFIER_RCTRL|MUI_MODIFIER_RSHIFT, MUI_KEY_F12), + .kcombo = MUI_GLYPH_CONTROL MUI_GLYPH_SHIFT MUI_GLYPH_F12 }, + { .title = "Control-Reset", + .uid = FCC('r','e','s','t'), + .key_equ = MUI_KEY_EQU(MUI_MODIFIER_RCTRL, MUI_KEY_F12), + .kcombo = MUI_GLYPH_CONTROL MUI_GLYPH_F12 }, + { .title = "-", }, + { .title = "Configure Slots…", + .uid = FCC('s','l','o','t') }, + { .title = "Joystick…", + .disabled = 1, + .uid = FCC('j','o','y','s') }, + { .title = "-", }, + { .title = "Video", + .kcombo = MUI_GLYPH_SUBMENU, + .submenu = m_video_menu }, + { .title = "Audio", + .kcombo = MUI_GLYPH_SUBMENU, + .submenu = m_audio_menu }, + { }, +}; +mui_menu_item_t m_cpu_menu[] = { + { .title = "Normal: 1MHz", + .uid = FCC('m','h','z','1'), + .key_equ = MUI_KEY_EQU(MUI_MODIFIER_LSUPER, MUI_KEY_F2), + .kcombo = MUI_GLYPH_COMMAND MUI_GLYPH_F2 }, + { .title = "Fast: 3.5MHz", + .uid = FCC('m','h','z','3'), + .key_equ = MUI_KEY_EQU(MUI_MODIFIER_LSUPER, MUI_KEY_F3), + .kcombo = MUI_GLYPH_COMMAND MUI_GLYPH_F3 }, + { .title = "-", }, + { .title = "Stop", + .kcombo = MUI_GLYPH_COMMAND MUI_GLYPH_F4, + .key_equ = MUI_KEY_EQU(MUI_MODIFIER_LSUPER, MUI_KEY_F4), + .uid = FCC('s','t','o','p') }, + { .title = "Running", + .kcombo = MUI_GLYPH_COMMAND MUI_GLYPH_F5, + .key_equ = MUI_KEY_EQU(MUI_MODIFIER_LSUPER, MUI_KEY_F5), + .uid = FCC('r','u','n',' ') }, + { .title = "Step", + .kcombo = MUI_GLYPH_COMMAND MUI_GLYPH_F6, + .key_equ = MUI_KEY_EQU(MUI_MODIFIER_LSUPER, MUI_KEY_F6), + .uid = FCC('s','t','e','p') }, + { .title = "Next", + .kcombo = MUI_GLYPH_COMMAND MUI_GLYPH_F7, + .key_equ = MUI_KEY_EQU(MUI_MODIFIER_LSUPER, MUI_KEY_F7), + .uid = FCC('n','e','x','t') }, + { }, +}; + +#endif diff --git a/ui_gl/mii_mui_settings.c b/ui_gl/mii_mui_settings.c new file mode 100644 index 0000000..72a558a --- /dev/null +++ b/ui_gl/mii_mui_settings.c @@ -0,0 +1,381 @@ +/* + * mui_mui_settings.c + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ +#define _GNU_SOURCE +#include +#include +#include +#include +#include + +#include "mui.h" +#include "mii_mui_settings.h" + +IMPLEMENT_C_ARRAY(mii_config_array); + +mii_config_line_t * +mii_config_get_section( + mii_config_file_t * cf, + const char * section, + bool add ) +{ + mii_config_line_t * cl; + for (int i = 0; i < (int)cf->line.count; i++) { + cl = cf->line.e[i]; + if (cl->section && !strcmp(cl->key, section)) + return cl; + } + if (!add) + return NULL; + cl = calloc(1, sizeof(*cl) + strlen(section) + 3); + if (!cl) + return NULL; + cl->section = 1; + sprintf(cl->line, "[%s", section); + cl->key = cl->line + 1; + cl->number = cf->line.count; + mii_config_array_push(&cf->line, cl); + return cl; +} + +mii_config_line_t * +mii_config_get( + mii_config_file_t * cf, + mii_config_line_t * section, + const char * key) +{ + if (!cf || !section || !key) + return NULL; + for (int i = section->number + 1; i < (int)cf->line.count; i++) { + mii_config_line_t * cl = cf->line.e[i]; + if (cl->section) + return NULL; + if (!strcmp(cl->key, key)) + return cl; + } + return NULL; +} + +mii_config_line_t * +mii_config_set( + mii_config_file_t * cf, + mii_config_line_t * section, + const char * key, + const char * value) +{ + if (!cf || !section || !key) + return NULL; + mii_config_line_t * cl = mii_config_get(cf, section, key); + + int idx = section->number + 1; + if (cl) { + if (value && cl->value && !strcmp(cl->value, value)) + return cl; + idx = cl->number; + mii_config_array_delete(&cf->line, idx, 1); + free(cl); + } + cl = calloc(1, sizeof(*cl) + strlen(key) + strlen(value) + 3); + strcpy(cl->line, key); + // this wouldnt work if memory was not zeroes by calloc + strcpy(cl->line + strlen(key) + 1, value ? value : ""); + cl->key = cl->line; + cl->value = cl->line + strlen(key) + 1; + mii_config_array_insert(&cf->line, idx, &cl, 1); + for (int i = idx; i < (int)cf->line.count; i++) + cf->line.e[i]->number = i; + + return cl; +} + +int +mii_config_file_load( + mii_config_file_t * cf, + const char * path ) +{ + int res = -1; + mii_config_array_clear(&cf->line); + FILE * f = fopen(path, "r"); + if (!f) + return -1; + cf->path = strdup(path); + char line[512]; + int n = 0; + while (fgets(line, sizeof(line), f)) { + int l = strlen(line); + while (l && (line[l-1] == '\n' || line[l-1] == ' ' || line[l-1] == '\t')) + line[--l] = 0; + mii_config_line_t * cl = calloc(1, sizeof(*cl) + l + 1); + if (!cl) + goto exit; + cl->number = n++; + strcpy(cl->line, line); + mii_config_array_push(&cf->line, cl); + + char * s = cl->line; + while (isspace(*s)) + s++; + if (*s == '#' || *s == ';' || *s == 0) + cl->ignore = 1; // comment or empty line + else if (*s == '[') { + char * d = s + 1; + char * e = strchr(d, ']'); + if (e) + *e = 0; + cl->key = d; + cl->value = NULL; + cl->section = 1; + } else { + char * d = s; + char * k = strsep(&d, " ="); + cl->key = k; + // we want a pointer to zero here if there's no value + cl->value = d ? d : s + strlen(s); + } + } + res = 0; +exit: + fclose(f); + return res; +} + +// same as previous function, but write the file back +int +mii_settings_save( + mii_config_file_t * cf) +{ + if (!cf || !cf->path) + return -1; + FILE * f = fopen(cf->path, "w"); + if (!f) + return -1; + mii_config_line_t * cl; + for (int i = 0; i < (int)cf->line.count; i++) { + cl = cf->line.e[i]; + if (cl->section) + fprintf(f, "[%s]\n", cl->key); + else if (cl->ignore) + fprintf(f, "%s\n", cl->line); + else + fprintf(f, "%s %s\n", cl->key, cl->value ? cl->value : ""); + } + fclose(f); + return 0; +} + +int +mii_settings_load( + mii_config_file_t * cf, + const char * path, + const char * file ) +{ + char * full_path = NULL; + if (path) { + asprintf(&full_path, "%s/%s", path, file); + } else { + full_path = strdup(file); + } + if (!full_path) + return -1; + + int r = mii_config_file_load(cf, full_path); + free(full_path); + return r; +} + +static const struct { + const char * name; +} mii_slot_driver[MII_SLOT_DRIVER_COUNT] = { + [MII_SLOT_DRIVER_NONE] = { "none", }, + [MII_SLOT_DRIVER_SMARTPORT] = { "smartport", }, + [MII_SLOT_DRIVER_DISK2] = { "disk2", }, + [MII_SLOT_DRIVER_MOUSE] = { "mouse", }, + [MII_SLOT_DRIVER_SUPERSERIAL] = { "ssc", }, + [MII_SLOT_DRIVER_ROM1MB] = { "eecard", }, +}; + +int +mii_emu_save( + mii_config_file_t * cf, + mii_machine_config_t * config ) +{ + if (!cf || !config) + return -1; + char label[64]; + mii_config_line_t * section = mii_config_get_section(cf, "emu", true); + + mii_config_set(cf, section, "titan", + config->titan_accelerator ? "1" : "0"); + mii_config_set(cf, section, "no_slot_clock", + config->no_slot_clock ? "1" : "0"); + mii_config_set(cf, section, "audio_muted", + config->audio_muted ? "1" : "0"); + sprintf(label, "%d", config->video_mode); + mii_config_set(cf, section, "video_mode", label); + + section = mii_config_get_section(cf, "joystick", true); + mii_config_set(cf, section, "device", + config->joystick.device); + for (int i = 0; i < 2; i++) { + sprintf(label, "%d", config->joystick.buttons[i]); + char name[32] = "button0"; + name[6] += i; + mii_config_set(cf, section, name, label); + strcpy(name, "axis0"); + name[4] += i; + sprintf(label, "%d", config->joystick.axes[i]); + mii_config_set(cf, section, name, label); + } + + for (int i = 0; i < 7; i++) { + char key[32]; + sprintf(key, "slot_%d", i+1); + section = mii_config_get_section(cf, key, true); + + switch(config->slot[i].driver) { + case MII_SLOT_DRIVER_SMARTPORT: + mii_config_set(cf, section, "image0", + config->slot[i].conf.smartport.drive[0].disk); + sprintf(label, "%lu", config->slot[i].conf.smartport.drive[0].flags); + mii_config_set(cf, section, "flags0", label); + mii_config_set(cf, section, "image1", + config->slot[i].conf.smartport.drive[1].disk); + sprintf(label, "%lu", config->slot[i].conf.smartport.drive[1].flags); + mii_config_set(cf, section, "flags1", label); + break; + case MII_SLOT_DRIVER_DISK2: + mii_config_set(cf, section, "image0", + config->slot[i].conf.disk2.drive[0].disk); + sprintf(label, "%lu", config->slot[i].conf.disk2.drive[0].flags); + mii_config_set(cf, section, "flags0", label); + mii_config_set(cf, section, "image1", + config->slot[i].conf.disk2.drive[1].disk); + sprintf(label, "%lu", config->slot[i].conf.disk2.drive[1].flags); + mii_config_set(cf, section, "flags1", label); + break; + case MII_SLOT_DRIVER_SUPERSERIAL: + mii_config_set(cf, section, "device", + config->slot[i].conf.ssc.device); + break; + case MII_SLOT_DRIVER_ROM1MB: + mii_config_set(cf, section, "use_default", + config->slot[i].conf.rom1mb.use_default ? "1" : "0"); + mii_config_set(cf, section, "image", + config->slot[i].conf.rom1mb.drive.disk); + break; + } + mii_config_set(cf, section, "driver", + mii_slot_driver[config->slot[i].driver].name); + } + mii_settings_save(cf); + return 0; +} + +int +mii_emu_load( + mii_config_file_t * cf, + mii_machine_config_t * config ) +{ + if (!cf || !config) + return -1; + mii_config_line_t * section = mii_config_get_section(cf, "emu", false); + if (section) { + mii_config_line_t * cl = mii_config_get(cf, section, "titan"); + if (cl) + config->titan_accelerator = atoi(cl->value); + cl = mii_config_get(cf, section, "no_slot_clock"); + if (cl) + config->no_slot_clock = atoi(cl->value); + cl = mii_config_get(cf, section, "audio_muted"); + if (cl) + config->audio_muted = !!atoi(cl->value); + cl = mii_config_get(cf, section, "video_mode"); + if (cl) + config->video_mode = atoi(cl->value); + } + + section = mii_config_get_section(cf, "joystick", false); + if (section) { + mii_config_line_t * cl = mii_config_get(cf, section, "device"); + if (cl) + strcpy(config->joystick.device, cl->value); + for (int i = 0; i < 2; i++) { + char name[32] = "button0"; + name[6] += i; + cl = mii_config_get(cf, section, name); + if (cl) + config->joystick.buttons[i] = atoi(cl->value); + strcpy(name, "axis0"); + name[4] += i; + cl = mii_config_get(cf, section, name); + if (cl) + config->joystick.axes[i] = atoi(cl->value); + } + } + + for (int i = 0; i < 7; i++) { + char key[32]; + sprintf(key, "slot_%d", i+1); + section = mii_config_get_section(cf, key, false); + if (!section) + continue; + + mii_config_line_t * cl = mii_config_get(cf, section, "driver"); + if (!cl) + continue; + for (int j = 0; j < MII_SLOT_DRIVER_COUNT; j++) { + if (!strcmp(mii_slot_driver[j].name, cl->value)) { + config->slot[i].driver = j; + break; + } + } + switch (config->slot[i].driver) { + case MII_SLOT_DRIVER_SMARTPORT: + cl = mii_config_get(cf, section, "image0"); + if (cl) + strcpy(config->slot[i].conf.smartport.drive[0].disk, cl->value); + cl = mii_config_get(cf, section, "flags0"); + if (cl) + config->slot[i].conf.smartport.drive[0].flags = strtoul(cl->value, NULL, 0); + cl = mii_config_get(cf, section, "image1"); + if (cl) + strcpy(config->slot[i].conf.smartport.drive[1].disk, cl->value); + cl = mii_config_get(cf, section, "flags1"); + if (cl) + config->slot[i].conf.smartport.drive[1].flags = strtoul(cl->value, NULL, 0); + break; + case MII_SLOT_DRIVER_DISK2: + cl = mii_config_get(cf, section, "image0"); + if (cl) + strcpy(config->slot[i].conf.disk2.drive[0].disk, cl->value); + cl = mii_config_get(cf, section, "flags0"); + if (cl) + config->slot[i].conf.disk2.drive[0].flags = strtoul(cl->value, NULL, 0); + cl = mii_config_get(cf, section, "image1"); + if (cl) + strcpy(config->slot[i].conf.disk2.drive[1].disk, cl->value); + cl = mii_config_get(cf, section, "flags1"); + if (cl) + config->slot[i].conf.disk2.drive[1].flags = strtoul(cl->value, NULL, 0); + break; + case MII_SLOT_DRIVER_SUPERSERIAL: + cl = mii_config_get(cf, section, "device"); + if (cl) + strcpy(config->slot[i].conf.ssc.device, cl->value); + break; + case MII_SLOT_DRIVER_ROM1MB: + cl = mii_config_get(cf, section, "use_default"); + if (cl) + config->slot[i].conf.rom1mb.use_default = atoi(cl->value); + cl = mii_config_get(cf, section, "image"); + if (cl) + strcpy(config->slot[i].conf.rom1mb.drive.disk, cl->value); + break; + } + } + return 0; +} \ No newline at end of file diff --git a/ui_gl/mii_mui_settings.h b/ui_gl/mii_mui_settings.h new file mode 100644 index 0000000..011e7fe --- /dev/null +++ b/ui_gl/mii_mui_settings.h @@ -0,0 +1,190 @@ +/* + * mii_settings.h + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ + +/* + * This file defines the configuration data model for the emulator, it is + * made so it can be included without having to include the emulator's + * own headers. + * This way the dialog/ui can still be tested without having to link + * against the emulator, using the mui_playground test. + */ +#pragma once + +#include + +#include "bsd_queue.h" +#include "c_array.h" + +#define MII_PATH_SIZE_MAX 256 + +typedef struct mii_config_line_t { + unsigned int ignore : 1, + section : 1, + number; // line number in file + char * key; + char * value; + char line[]; +} mii_config_line_t; + +DECLARE_C_ARRAY(mii_config_line_t*, mii_config_array, 16); + +typedef struct mii_config_file_t { + char * path; + mii_config_array_t line; +} mii_config_file_t; + +typedef struct mii_drive_conf_t { + unsigned long flags; + char disk[MII_PATH_SIZE_MAX]; +} mii_drive_conf_t; + +typedef struct mii_1mb_conf_t { + unsigned long slot_id : 3, + use_default : 1, + reserved; + mii_drive_conf_t drive; +} mii_1mb_conf_t; + +typedef struct mii_2dsk_conf_t { + unsigned long slot_id : 3, reserved; + mii_drive_conf_t drive[2]; +} mii_2dsk_conf_t; + +typedef struct mii_ssc_conf_t { + uint8_t kind; // device, pty, socket + int socket_port; + char device[MII_PATH_SIZE_MAX]; + // rom/card configuration + unsigned int baud, bits, parity, stop; +} mii_ssc_conf_t; + +typedef struct mii_joystick_conf_t { + // /dev/input/js0 if empty + char device[MII_PATH_SIZE_MAX]; + unsigned int buttons[2]; + unsigned int axes[2]; +} mii_joystick_conf_t; + +typedef struct mii_loadbin_conf_t { + uint16_t active: 1, audo_reload : 1; + uint16_t bank; // unused for now + uint16_t addr; // address in bank + char path[MII_PATH_SIZE_MAX]; +} mii_loadbin_conf_t; + +enum mii_mui_driver_e { + MII_SLOT_DRIVER_NONE = 0, + MII_SLOT_DRIVER_SMARTPORT, + MII_SLOT_DRIVER_DISK2, + MII_SLOT_DRIVER_MOUSE, + MII_SLOT_DRIVER_SUPERSERIAL, + MII_SLOT_DRIVER_ROM1MB, + MII_SLOT_DRIVER_COUNT +}; + +// This is obviously not made to be terribly compact. +typedef struct mii_machine_config_t { + uint32_t reboot_on_load : 1, + load_defaults : 1, + audio_muted : 1, + no_slot_clock : 1, + titan_accelerator : 1; + uint32_t video_mode; + mii_joystick_conf_t joystick; + struct { + uint16_t driver; + union { + mii_2dsk_conf_t smartport; + mii_2dsk_conf_t disk2; + mii_1mb_conf_t rom1mb; + mii_ssc_conf_t ssc; + } conf; + } slot[7]; +} mii_machine_config_t; + +struct mui_window_t; +struct mui_t; + +/* + * These are passed to the window action proc when the dialogs have + * been validated and closed. The matching config structures have been + * updated and can be used to update the emulator's configuration. + */ +enum mii_mui_dialog_e { + MII_MUI_SLOTS_SAVE = FCC('S','L','O','T'), + MII_MUI_LOADBIN_SAVE = FCC('B','I','N',' '), + MII_MUI_1MB_SAVE = FCC('1','M','B',' '), + MII_MUI_DISK2_SAVE = FCC('2','D','S','K'), + MII_MUI_SMARTPORT_SAVE = FCC('S','M','P','T'), +}; + +struct mui_window_t * +mii_mui_configure_slots( + struct mui_t *mui, + mii_machine_config_t *config); +mui_window_t * +mii_mui_configure_slot( + struct mui_t *mui, + mii_machine_config_t *config, + int slot_id); + +struct mui_window_t * +mii_mui_load_binary( + struct mui_t *mui, + mii_loadbin_conf_t *config); +struct mui_window_t * +mii_mui_load_1mbrom( + struct mui_t *mui, + mii_1mb_conf_t *config); + +enum mii_mui_2dsk_e { + MII_2DSK_DISKII = 0, + MII_2DSK_SMARTPORT, +}; + +struct mui_window_t * +mii_mui_load_2dsk( + struct mui_t *mui, + mii_2dsk_conf_t *config, + uint8_t drive_kind); + +/* + * Config file related + */ +mii_config_line_t * +mii_config_get_section( + mii_config_file_t * cf, + const char * section, + bool add ); +mii_config_line_t * +mii_config_get( + mii_config_file_t * cf, + mii_config_line_t * section, + const char * key); +mii_config_line_t * +mii_config_set( + mii_config_file_t * cf, + mii_config_line_t * section, + const char * key, + const char * value); +int +mii_settings_save( + mii_config_file_t * cf); +int +mii_settings_load( + mii_config_file_t * cf, + const char * path, + const char * file ); +int +mii_emu_save( + mii_config_file_t * cf, + mii_machine_config_t * config ); +int +mii_emu_load( + mii_config_file_t * cf, + mii_machine_config_t * config ); diff --git a/ui_gl/mii_mui_slots.c b/ui_gl/mii_mui_slots.c new file mode 100644 index 0000000..27baa31 --- /dev/null +++ b/ui_gl/mii_mui_slots.c @@ -0,0 +1,439 @@ +/* + * mui_mui_slots.c + * + * Copyright (C) 2023 Michel Pollet + * + * SPDX-License-Identifier: MIT + */ +#define _GNU_SOURCE +#include +#include +#include +#include +#include + +#include "mui.h" +#include "mii_mui_settings.h" + + +typedef struct mii_ui_machine_config_t { + mui_window_t win; + + mui_control_t * slot[8]; + + // for sub-windows + unsigned slot_id; + mii_machine_config_t * dst; + mii_machine_config_t config; +} mii_ui_machine_config_t; + +static const mii_machine_config_t _default_config = { + .no_slot_clock = 1, + .titan_accelerator = 0, + .slot = { + [0] = { + // .driver = MII_SLOT_DRIVER_SUPERSERIAL, + .conf.ssc = { + .kind = 0, + .socket_port = 1969, + .device = "/dev/ttyS0", + .baud = 9600, + .bits = 8, + .parity = 0, + .stop = 1, + } + }, + [1] = { .driver = 0, }, + [2] = { .driver = 0, }, + [3] = { + .driver = MII_SLOT_DRIVER_MOUSE, + }, + [4] = { .driver = 0, }, + [5] = { + .driver = MII_SLOT_DRIVER_DISK2, + }, + [6] = { + .driver = MII_SLOT_DRIVER_SMARTPORT, + }, + }, +}; + +enum { + MII_SLOT_WINDOW_ID = FCC('s','l','o','t'), + MII_SLOT_SAVE = FCC('s','a','v','e'), + MII_SLOT_CANCEL = FCC('c','a','n','c'), + MII_SLOT_DEFAULT = FCC('d','e','f','a'), +// MII_SLOT_REBOOT = FCC('r','b','o','o'), + MII_SLOT_NSC = FCC('n','s','c','l'), + MII_SLOT_TITAN = FCC('t','i','t','a'), + MII_SLOT_DRIVER_POP = FCC('d','r','v','p'), + MII_SLOT_DRIVER_CONF = FCC('d','r','v','c'), +}; + +static const struct { + char * label; + int has_config; +} _mii_slot_drivers[] = { + [MII_SLOT_DRIVER_NONE] = { "None", 0 }, + [MII_SLOT_DRIVER_SMARTPORT] = { "SmartPort", 1 }, + [MII_SLOT_DRIVER_DISK2] = { "Disk II", 1 }, + [MII_SLOT_DRIVER_MOUSE] = { "Mouse", 0 }, + [MII_SLOT_DRIVER_SUPERSERIAL] = { "Super Serial", 1 }, + [MII_SLOT_DRIVER_ROM1MB] = { "ROM 1MB", 1 }, + { NULL, 0 }, +}; + +static void +_mii_config_win_to_conf( + mii_ui_machine_config_t * m) +{ + mui_window_t * w = &m->win; + mii_machine_config_t * cf = &m->config; + +// cf->reboot_on_load = !!mui_control_get_value( +// mui_control_get_by_id(w, MII_SLOT_REBOOT)); + cf->titan_accelerator = !!mui_control_get_value( + mui_control_get_by_id(w, MII_SLOT_TITAN)); + cf->no_slot_clock = !!mui_control_get_value( + mui_control_get_by_id(w, MII_SLOT_NSC)); + + for (int slot_id = 1; slot_id < 8; slot_id++) { + mui_control_t * c = mui_control_get_by_id( + w, MII_SLOT_DRIVER_POP + slot_id); + mui_menu_items_t * items = mui_popupmenu_get_items(c); + int pop_val = mui_control_get_value(c); + mui_menu_item_t * item = &items->e[pop_val]; + // printf("%s popup slot %d %4.4s '%s'\n", __func__, + // slot_id, + // (char*)&item->uid, item->title); + mui_control_t * c2 = mui_control_get_by_id(w, + MII_SLOT_DRIVER_CONF + slot_id); + if (c2) { + mui_control_set_state(c2, + _mii_slot_drivers[pop_val].has_config ? + MUI_CONTROL_STATE_NORMAL : + MUI_CONTROL_STATE_DISABLED); + } + if (cf->slot[slot_id-1].driver != item->uid) { + cf->slot[slot_id-1].driver = item->uid; + memset(&cf->slot[slot_id-1].conf, 0, + sizeof(cf->slot[slot_id-1].conf)); + } + } +} + +static void +_mii_config_conf_to_win( + mii_ui_machine_config_t * m) +{ + mui_window_t * w = &m->win; + mii_machine_config_t * cf = &m->config; + +// mui_control_set_value(mui_control_get_by_id(w, MII_SLOT_REBOOT), +// cf->reboot_on_load); + mui_control_set_value(mui_control_get_by_id(w, MII_SLOT_TITAN), + cf->titan_accelerator); + mui_control_set_value(mui_control_get_by_id(w, MII_SLOT_NSC), + cf->no_slot_clock); + + for (int slot_id = 1; slot_id < 8; slot_id++) { + mui_control_t * c = mui_control_get_by_id( + w, MII_SLOT_DRIVER_POP + slot_id); + mui_control_set_value(c, cf->slot[slot_id-1].driver); + + mui_control_t * c2 = mui_control_get_by_id(w, + MII_SLOT_DRIVER_CONF + slot_id); + if (c2) { + mui_control_set_state(c2, + _mii_slot_drivers[cf->slot[slot_id-1].driver].has_config ? + MUI_CONTROL_STATE_NORMAL : + MUI_CONTROL_STATE_DISABLED); + } + } +} + +static int +_mii_config_sub_save_cb( + mui_window_t *win, // the sub window/dialog + void * cb_param, + uint32_t what, + void * param) +{ +// mii_ui_machine_config_t * m = cb_param; + printf("%s %4.4s\n", __func__, (char*)&what); + + return 0; +} + +mui_window_t * +mii_mui_configure_slot( + struct mui_t *mui, + mii_machine_config_t *config, + int slot_id) +{ + printf("%s config slot %d\n", __func__, slot_id); + mui_window_t * res = NULL; + switch (config->slot[slot_id].driver) { + case MII_SLOT_DRIVER_SMARTPORT: { + config->slot[slot_id].conf.smartport.slot_id = slot_id; + res = mii_mui_load_2dsk(mui, + &config->slot[slot_id].conf.smartport, + MII_2DSK_SMARTPORT); + } break; + case MII_SLOT_DRIVER_DISK2: { + config->slot[slot_id].conf.disk2.slot_id = slot_id; + res = mii_mui_load_2dsk(mui, + &config->slot[slot_id].conf.disk2, + MII_2DSK_DISKII); + } break; + case MII_SLOT_DRIVER_ROM1MB: { + config->slot[slot_id].conf.rom1mb.slot_id = slot_id; + res = mii_mui_load_1mbrom(mui, + &config->slot[slot_id].conf.rom1mb); + } break; + case MII_SLOT_DRIVER_SUPERSERIAL: { + mui_alert(mui, C2_PT(0,0), + "Not implemented yet", + "SSC is not implemented yet!\n" + "Driver is present just a placeholder, sorry.", + MUI_ALERT_FLAG_OK); + } break; + } + return res; +} + +static int +_mii_config_slot_action_cb( + mui_control_t * c, // control here + void * cb_param, // mii_ui_machine_config_t + uint32_t what, + void * param) // not used +{ + printf("%s %4.4s\n", __func__, (char*)&what); + mii_ui_machine_config_t * m = cb_param; + uint32_t uid = mui_control_get_uid(c); + + switch (what) { + case MUI_CONTROL_ACTION_SELECT: + printf("%s control %4.4s\n", __func__, (char*)&uid); + switch (uid) { + case MII_SLOT_SAVE: { + // save the config + printf("%s save\n", __func__); + _mii_config_win_to_conf(m); + if (m->dst) + *m->dst = m->config; + mui_window_action(&m->win, MII_MUI_SLOTS_SAVE, m->dst); + mui_window_dispose(&m->win); + } break; + case MII_SLOT_CANCEL: { + // cancel the config + printf("%s cancel\n", __func__); + mui_window_dispose(&m->win); + } break; + case MII_SLOT_DEFAULT: { + // set the default config + printf("%s default\n", __func__); + m->config = _default_config; + _mii_config_conf_to_win(m); + } break; +#if 0 + case MII_SLOT_REBOOT: { + // reboot the machine + printf("%s reboot\n", __func__); + m->config.reboot_on_load = !m->config.reboot_on_load; + } break; +#endif + case MII_SLOT_NSC: { + // no slot clock + printf("%s nsc\n", __func__); + m->config.no_slot_clock = !m->config.no_slot_clock; + } break; + case MII_SLOT_TITAN: { + // titan accelerator + printf("%s titan\n", __func__); + m->config.titan_accelerator = !m->config.titan_accelerator; + } break; + case MII_SLOT_DRIVER_CONF + 1 ... MII_SLOT_DRIVER_CONF + 7: { + // configure a slot + int slot_id = uid - MII_SLOT_DRIVER_CONF - 1; + printf("%s config slot %d\n", __func__, slot_id); + switch (m->config.slot[slot_id].driver) { + case MII_SLOT_DRIVER_SMARTPORT: { + m->config.slot[slot_id].conf.smartport.slot_id = slot_id; + mui_window_set_action( + mii_mui_load_2dsk(m->win.ui, + &m->config.slot[slot_id].conf.smartport, + MII_2DSK_SMARTPORT), + _mii_config_sub_save_cb, m); + } break; + case MII_SLOT_DRIVER_DISK2: { + m->config.slot[slot_id].conf.disk2.slot_id = slot_id; + mui_window_set_action( + mii_mui_load_2dsk(m->win.ui, + &m->config.slot[slot_id].conf.disk2, + MII_2DSK_DISKII), + _mii_config_sub_save_cb, m); + } break; + case MII_SLOT_DRIVER_ROM1MB: { + m->config.slot[slot_id].conf.rom1mb.slot_id = slot_id; + mui_window_set_action( + mii_mui_load_1mbrom(m->win.ui, + &m->config.slot[slot_id].conf.rom1mb), + _mii_config_sub_save_cb, m); + } break; + case MII_SLOT_DRIVER_SUPERSERIAL: { + mui_alert(m->win.ui, C2_PT(0,0), + "Not implemented yet", + "SSC is not implemented yet!\n" + "Driver is present just a placeholder, sorry.", + MUI_ALERT_FLAG_OK); + } break; + } + } break; + } + break; + case MUI_CONTROL_ACTION_VALUE_CHANGED: + // a popup menu changed + printf("%s popup %4.4s\n", __func__, (char*)&uid); + switch (uid) { + case MII_SLOT_DRIVER_POP + 1 ... MII_SLOT_DRIVER_POP + 7: { + _mii_config_win_to_conf(m); + } break; + } + break; + } + return 0; +} + +struct mui_window_t * +mii_mui_configure_slots( + struct mui_t *mui, + mii_machine_config_t *config) +{ + mui_t *ui = mui; + + mui_window_t *w = mui_window_get_by_id(mui, MII_SLOT_WINDOW_ID); + if (w) { + mui_window_select(w); + return w; + } + c2_pt_t where = {}; + c2_rect_t wpos = C2_RECT_WH(where.x, where.y, 520, 480); + if (where.x == 0 && where.y == 0) + c2_rect_offset(&wpos, + (ui->screen_size.x / 2) - (c2_rect_width(&wpos) / 2), + (ui->screen_size.y * 0.4) - (c2_rect_height(&wpos) / 2)); + w = mui_window_create(mui, + wpos, + NULL, 0, + "Configure " MUI_GLYPH_IIE " Slots", + sizeof(mii_ui_machine_config_t)); + mui_window_set_id(w, MII_SLOT_WINDOW_ID); + mii_ui_machine_config_t *m = (mii_ui_machine_config_t*)w; + + mui_control_t * c = NULL; + c2_rect_t cf; + cf = C2_RECT_WH(0, 0, 120, 40); + c2_rect_left_of(&cf, c2_rect_width(&w->content), 20); + c2_rect_top_of(&cf, c2_rect_height(&w->content), 20); + c = mui_button_new(w, + cf, MUI_BUTTON_STYLE_DEFAULT, + "Reboot", MII_SLOT_SAVE); + c->key_equ = MUI_KEY_EQU(0, 13); + c2_rect_left_of(&cf, cf.l, 20); + c = mui_button_new(w, + cf, MUI_BUTTON_STYLE_NORMAL, + "Cancel", MII_SLOT_CANCEL); + c->key_equ = MUI_KEY_EQU(0, 27); + + c2_rect_right_of(&cf, 0, 20); + cf.r = cf.l + 200; + #if 0 + c = mui_button_new(w, + cf, MUI_BUTTON_STYLE_CHECKBOX, + "Reboot Now", MII_SLOT_REBOOT); + #endif + c2_rect_top_of(&cf, cf.t, 36); + cf.r = cf.l + 120; + c = mui_button_new(w, + cf, MUI_BUTTON_STYLE_NORMAL, + "Default", MII_SLOT_DEFAULT); + c2_rect_right_of(&cf, cf.r, 30); + c = mui_button_new(w, + cf, MUI_BUTTON_STYLE_CHECKBOX, + "No Slot Clock", MII_SLOT_NSC); + c2_rect_right_of(&cf, cf.r, 30); + cf.r = cf.l + 200; + c = mui_button_new(w, + cf, MUI_BUTTON_STYLE_CHECKBOX, + "Titan Accelerator", MII_SLOT_TITAN); + + c2_rect_bottom_of(&cf, cf.b, 16); + cf.l = 60; + cf.r = c2_rect_width(&w->content) - 60; + cf.b = cf.t + 1; + c = mui_separator_new(w, cf); + + c2_rect_t slot_line_rect = C2_RECT_WH(0, 0, 500, 34); + cf = slot_line_rect; + cf.r = cf.l + 50; + c = mui_textbox_new(w, cf, "Slot", NULL, + MUI_TEXT_ALIGN_RIGHT | MUI_TEXT_ALIGN_MIDDLE); + c2_rect_right_of(&cf, cf.r, 6); + cf.r = cf.l + 240; + c = mui_textbox_new(w, cf, "Driver", NULL, + MUI_TEXT_ALIGN_CENTER | MUI_TEXT_ALIGN_MIDDLE); + c2_rect_right_of(&cf, cf.r, 30); + cf.r = cf.l + 150; + c = mui_textbox_new(w, cf, "Config", NULL, + MUI_TEXT_ALIGN_CENTER | MUI_TEXT_ALIGN_MIDDLE); + + c2_rect_offset(&slot_line_rect, 0, 36); + for (int i = 1; i < 8; i++) { + cf = slot_line_rect; + cf.r = cf.l + 50; + char idx[16]; + sprintf(idx, "%d:", i); + c = mui_textbox_new(w, cf, idx, NULL, + MUI_TEXT_ALIGN_RIGHT | MUI_TEXT_ALIGN_MIDDLE); + c2_rect_right_of(&cf, cf.r, 6); + cf.r = cf.l + 240; + c = mui_popupmenu_new(w, cf, + "Popup", MII_SLOT_DRIVER_POP + i); + mui_menu_items_t * items = mui_popupmenu_get_items(c); + mui_menu_items_clear(items); + for (int j = 0; _mii_slot_drivers[j].label; j++) { + mui_menu_item_t i = { + .title = strdup(_mii_slot_drivers[j].label), + .uid = j, + }; + mui_menu_items_push(items, i); + } + mui_menu_item_t z = {}; + mui_menu_items_push(items, z); + mui_popupmenu_prepare(c); + + c2_rect_right_of(&cf, cf.r, 30); + cf.r = cf.l + 150; + c = mui_button_new(w, + cf, MUI_BUTTON_STYLE_NORMAL, + "Config…", MII_SLOT_DRIVER_CONF + i); + c2_rect_offset(&slot_line_rect, 0, 38); + } + c = NULL; + TAILQ_FOREACH(c, &w->controls, self) { + if (mui_control_get_uid(c) == 0) + continue; + mui_control_set_action(c, _mii_config_slot_action_cb, m); + } + m->dst = config; + if (config) { + if (config->load_defaults) + m->config = _default_config; + else + m->config = *config; + } + _mii_config_conf_to_win(m); + return w; +} diff --git a/nuklear/mii_thread.c b/ui_gl/mii_thread.c similarity index 67% rename from nuklear/mii_thread.c rename to ui_gl/mii_thread.c index 5a57bfd..768928b 100644 --- a/nuklear/mii_thread.c +++ b/ui_gl/mii_thread.c @@ -5,7 +5,6 @@ * * SPDX-License-Identifier: MIT */ -#include #include #include #include @@ -15,7 +14,7 @@ #include #include #include - +#include // probably should wrap these into a HAVE_JOYSTICK define for non-linux #ifndef HAVE_JOYSTICK #define HAVE_JOYSTICK 1 @@ -24,6 +23,7 @@ #include "mii.h" #include "mii_thread.h" +#include typedef uint64_t mii_time_t; enum { MII_TIME_RES = 1, @@ -40,25 +40,44 @@ mii_get_time() return time; } -static bool mii_thread_running = false; static float default_fps = 60; mii_th_fifo_t signal_fifo; +int +mii_thread_set_fps( + int timerfd, + float fps) +{ + default_fps = fps; + long target_fps_us = 1000000 / default_fps; + struct itimerspec its = { + .it_interval = { .tv_sec = 0, .tv_nsec = target_fps_us * 1000 }, + .it_value = { .tv_sec = 0, .tv_nsec = target_fps_us * 1000 }, + }; + if (timerfd_settime(timerfd, 0, &its, NULL) < 0) { + perror(__func__); + return -1; + } + return 0; +} + static void * mii_thread_cpu_regulator( void *arg) { mii_t *mii = (mii_t *) arg; - mii_thread_running = true; - mii_cycles_t last_cycles = mii->cycles; uint32_t running = 1; - unsigned long target_fps_us = 1000000 / default_fps; - long sleep_time = target_fps_us; - //mii_time_t base = mii_get_time(NULL); + // use a timerfd as regulation + int timerfd = timerfd_create(CLOCK_MONOTONIC, 0); + if (timerfd < 0) { + perror(__func__); + return NULL; + } + mii_thread_set_fps(timerfd, default_fps); + mii->state = MII_RUNNING; uint32_t last_frame = mii->video.frame_count; - mii_time_t last_frame_stamp = mii_get_time(); - while (mii_thread_running) { + while (running) { mii_th_signal_t sig; while (!mii_th_fifo_isempty(&signal_fifo)) { sig = mii_th_fifo_read(&signal_fifo); @@ -77,23 +96,22 @@ mii_thread_cpu_regulator( break; case SIGNAL_RUN: mii->state = MII_RUNNING; - last_frame_stamp = mii_get_time(); running = 1; break; } } if (mii->state != MII_STOPPED) mii_run(mii); - + bool sleep = false; switch (mii->state) { case MII_STOPPED: - usleep(1000); + sleep = true; break; case MII_STEP: + sleep = true; if (running) { running--; mii_dump_trace_state(mii); - usleep(1000); running = 1; if (mii->trace.step_inst) mii->trace.step_inst--; @@ -102,37 +120,20 @@ mii_thread_cpu_regulator( } break; case MII_RUNNING: + sleep = mii->video.frame_count != last_frame; + if (sleep) + last_frame = mii->video.frame_count; break; case MII_TERMINATE: - mii_thread_running = false; + running = 0; break; } - - if (mii->video.frame_count != last_frame) { - sleep_time = target_fps_us; - mii_time_t now = mii_get_time(); - if (mii->state == MII_RUNNING) { - mii_time_t delta = now - last_frame_stamp; - // printf("frame time %d/%d sleep time %d\n", - // (int)delta, (int)target_fps_us, - // (int)target_fps_us - delta); - sleep_time = target_fps_us - delta; - if (sleep_time < 0) - sleep_time = 0; - last_frame = mii->video.frame_count; - while (last_frame_stamp <= now) - last_frame_stamp += target_fps_us; - - // calculate the MHz - mii_cycles_t cycles = mii->cycles; - mii_cycles_t delta_cycles = cycles - last_cycles; - last_cycles = cycles; - mii->speed_current = delta_cycles / (float)target_fps_us; - } - usleep(sleep_time); + if (sleep) { + uint64_t timer_v; + read(timerfd, &timer_v, sizeof(timer_v)); } } - mii_dispose(mii); + mii_dispose(mii); // this sets mii->state to MII_INIT return NULL; } @@ -154,7 +155,29 @@ mii_thread_joystick( perror(__func__); return NULL; } - + char name[128]; + if (ioctl(fd, JSIOCGNAME(sizeof(name)), name) == -1) { + perror(__func__); + return NULL; + } + printf("Joystick found: '%s'\n", name); +#if 0 + printf(" %d axes, %d buttons\n", axes, buttons); + // get and print mappings + struct js_corr corr[axes] = {}; + if (ioctl(fd, JSIOCGCORR, corr) == -1) { + perror(__func__); + // return NULL; + } + printf(" %d axes, %d buttons\n", axes, buttons); + for (int i = 0; i < axes; i++) { + printf("coor %d: type %d, prec %d: %d %d %d %d\n", i, + corr[i].type, corr[i].prec, + corr[i].coef[0], corr[i].coef[1], + corr[i].coef[2], corr[i].coef[3]); + } + printf("Joystick thread started: %d axis, %d buttons\n", axes, buttons); +#endif struct js_event event; mii_t *mii = (mii_t *)arg; mii->analog.v[0].value = 127; @@ -205,18 +228,19 @@ mii_thread_joystick( } #endif -void +pthread_t mii_thread_start( mii_t *mii) { const mii_th_fifo_t zero = {}; signal_fifo = zero; - pthread_t thread; - pthread_create(&thread, NULL, mii_thread_cpu_regulator, mii); + pthread_t cpu, joystick; + pthread_create(&cpu, NULL, mii_thread_cpu_regulator, mii); #if HAVE_JOYSTICK - pthread_create(&thread, NULL, mii_thread_joystick, mii); + pthread_create(&joystick, NULL, mii_thread_joystick, mii); #endif + return cpu; } struct mii_th_fifo_t* diff --git a/nuklear/mii_thread.h b/ui_gl/mii_thread.h similarity index 89% rename from nuklear/mii_thread.h rename to ui_gl/mii_thread.h index 7b9be87..1e34de3 100644 --- a/nuklear/mii_thread.h +++ b/ui_gl/mii_thread.h @@ -28,9 +28,13 @@ DEFINE_FIFO(mii_th_signal_t, mii_th_fifo); struct mii_t; -void +pthread_t mii_thread_start( struct mii_t *mii); struct mii_th_fifo_t* mii_thread_get_fifo( struct mii_t *mii); +int +mii_thread_set_fps( + int timerfd, + float fps); diff --git a/utils/icon-convert-tcc.c b/utils/icon-convert-tcc.c new file mode 100644 index 0000000..d03cbcd --- /dev/null +++ b/utils/icon-convert-tcc.c @@ -0,0 +1,80 @@ +/* + first convert the .png to binary files in contrib using imagemagik: + $ convert your_image.png -depth 8 rgba:your_image.bin + run this with + $ tcc -run icon-convert-tcc.c + */ +#include +#include +#include +#include +#include +#include +#include + +/* this loads each contrib/mii-icon-*.bin file which is raw RGBA and + add the header width/height, and convert the pixels to ARGB */ + +int main() { + glob_t g = {}; + glob("contrib/mii-icon-*.bin", 0, NULL, &g); + for (int i = 0; i < g.gl_pathc; i++) { + char * path = strdup(g.gl_pathv[i]); + char * name = basename(path); + char * dot = strrchr(name, '.'); + if (dot) + *dot = 0; + char * dup = strdup(name); + char * size = strrchr(dup, '-'); + if (size) + size++; + int w = 0, h = 0; + if (size) + w = h = atoi(size); + char *orig = g.gl_pathv[i]; + if (!w || !h) { + printf("bad size %s\n", orig); + continue; + } + FILE * f = fopen(orig, "rb"); + if (!f) { + perror(orig); + printf("bad file %s\n", orig); + continue; + } + fseek(f, 0, SEEK_END); + int size_ = ftell(f); + fseek(f, 0, SEEK_SET); + unsigned char * pixels = malloc(size_); + fread(pixels, 1, size_, f); + fclose(f); + char out[1024]; + snprintf(out, sizeof(out), "contrib/%s.h", name); + f = fopen(out, "wb"); + if (!f) { + printf("bad file %s\n", out); + continue; + } + fprintf(f, "/* this file is auto-generated by icon-convert-tcc.c */\n"); + // fprintf(f, "#include \n"); + fprintf(f, "static const unsigned long mii_icon%d[] = {\n", w); + fprintf(f, "\t%d,%d,\n", w, h); + for (int y = 0; y < h; y++) { + fprintf(f, "\t"); + for (int x = 0; x < w; x++) { + int i = (y * w + x) * 4; + int r = pixels[i + 0]; + int g = pixels[i + 1]; + int b = pixels[i + 2]; + int a = pixels[i + 3]; + fprintf(f, "0x%02x%02x%02x%02x, ", a, r, g, b); + if (x % 8 == 7) + fprintf(f, "\n\t"); + } + fprintf(f, "\n"); + } + fprintf(f, "};\n"); + fclose(f); + free(pixels); + } +} \ No newline at end of file