This commit is contained in:
Ken Miller 2014-10-04 22:27:06 -05:00
commit c8705bef4e
2 changed files with 429 additions and 171 deletions

View File

@ -14,7 +14,7 @@ library | lastest version | category | description
**stb_textedit.h** | 1.5 | UI | guts of a text editor for games etc implementing them from scratch **stb_textedit.h** | 1.5 | UI | guts of a text editor for games etc implementing them from scratch
**stb_dxt.h** | 1.04 | 3D graphics | Fabian "ryg" Giesen's real-time DXT compressor **stb_dxt.h** | 1.04 | 3D graphics | Fabian "ryg" Giesen's real-time DXT compressor
**stb_perlin.h** | 0.2 | 3D graphics | revised Perlin noise (3D input, 1D output) **stb_perlin.h** | 0.2 | 3D graphics | revised Perlin noise (3D input, 1D output)
**stb_tilemap_editor.h** | 0.10 | games | embeddable tilemap editor **stb_tilemap_editor.h** | 0.20 | games | embeddable tilemap editor
**stb_herringbone_wang_tile.h** | 0.6 | games | herringbone Wang tile map generator **stb_herringbone_wang_tile.h** | 0.6 | games | herringbone Wang tile map generator
**stb_c_lexer.h** | 0.06 | parsing | simplify writing parsers for C-like languages **stb_c_lexer.h** | 0.06 | parsing | simplify writing parsers for C-like languages
**stb_divide.h** | 0.91 | math | more useful 32-bit modulus e.g. "euclidean divide" **stb_divide.h** | 0.91 | math | more useful 32-bit modulus e.g. "euclidean divide"

View File

@ -1,4 +1,4 @@
// stb_tilemap_editor.h - v0.10 - Sean Barrett - http://nothings.org/stb // stb_tilemap_editor.h - v0.20 - Sean Barrett - http://nothings.org/stb
// placed in the public domain - not copyrighted - first released 2014-09 // placed in the public domain - not copyrighted - first released 2014-09
// //
// Embeddable tilemap editor for C/C++ // Embeddable tilemap editor for C/C++
@ -70,9 +70,13 @@
// stbte_tilemap keeps its own undo state. (The clipboard is global, so // stbte_tilemap keeps its own undo state. (The clipboard is global, so
// either approach allows cut&pasting between levels.) // either approach allows cut&pasting between levels.)
// //
// REVISION HISTORY
//
// 0.20 - 2014-09-27 - eraser tool, bugfixes, new colorscheme
// 0.10 - 2014-09-23 - initial release
//
// TODO // TODO
// //
// Eraser!!!
// Separate scroll state for each category // Separate scroll state for each category
// Implement paint bucket // Implement paint bucket
// Support STBTE_HITTEST_TILE above // Support STBTE_HITTEST_TILE above
@ -263,45 +267,138 @@ extern void stbte_set_layername(stbte_tilemap *tm, int layer, const char *layern
#error "Undo buffer size must be a power of 2" #error "Undo buffer size must be a power of 2"
#endif #endif
#define STBTE_COLOR_TOOLBAR_BACKGROUND 0x606060 static int *stbte__colors;
enum
{
STBTE__base,
STBTE__outline,
STBTE__text,
STBTE__num_color_aspects,
};
enum
{
STBTE__idle,
STBTE__over,
STBTE__down,
STBTE__over_down,
STBTE__selected,
STBTE__selected_over,
STBTE__disabled,
STBTE__num_color_states,
};
enum
{
STBTE__cexpander,
STBTE__ctoolbar,
STBTE__ctoolbar_button,
STBTE__cpanel,
STBTE__cpanel_sider,
STBTE__cpanel_sizer,
STBTE__cscrollbar,
STBTE__cmapsize,
STBTE__clayer_button,
STBTE__clayer_hide,
STBTE__clayer_lock,
STBTE__clayer_solo,
STBTE__ccategory_button,
STBTE__num_color_modes,
};
#ifdef STBTE__COLORPICKER
static char *stbte__color_names[] =
{
"expander", "toolbar", "tool button", "panel",
"panel c1", "panel c2", "scollbar", "map button",
"layer", "hide", "lock", "solo",
"category",
};
#endif // STBTE__COLORPICKER
// idle, over, down, over&down, selected, sel&over, disabled
static int stbte__color_table[STBTE__num_color_modes][STBTE__num_color_aspects][STBTE__num_color_states] =
{
{
{ 0x000000, 0x84987c, 0xdcdca8, 0xdcdca8, 0x40c040, 0x60d060, 0x505050, },
{ 0xa4b090, 0xe0ec80, 0xffffc0, 0xffffc0, 0x80ff80, 0x80ff80, 0x606060, },
{ 0xffffff, 0xffffff, 0xffffff, 0xffffff, 0xffffff, 0xffffff, 0x909090, },
}, {
{ 0x808890, 0x606060, 0x606060, 0x606060, 0x606060, 0x606060, 0x606060, },
{ 0x605860, 0x606060, 0x606060, 0x606060, 0x606060, 0x606060, 0x606060, },
{ 0x000000, 0x000000, 0x000000, 0x000000, 0x000000, 0x000000, 0x000000, },
}, {
{ 0x3c5068, 0x7088a8, 0x647488, 0x94b4dc, 0x8890c4, 0x9caccc, 0x404040, },
{ 0x889cb8, 0x889cb8, 0x889cb8, 0x889cb8, 0x84c4e8, 0xacc8ff, 0x0c0c08, },
{ 0xbcc4cc, 0xffffff, 0xffffff, 0xffffff, 0xffffff, 0xffffff, 0x707074, },
}, {
{ 0x403848, 0x403010, 0x403010, 0x403010, 0x403010, 0x403010, 0x303024, },
{ 0x68546c, 0xc08040, 0xc08040, 0xc08040, 0xc08040, 0xc08040, 0x605030, },
{ 0xf4e4ff, 0xffffff, 0xffffff, 0xffffff, 0xffffff, 0xffffff, 0x909090, },
}, {
{ 0xb4b04c, 0xacac60, 0xc0ffc0, 0xc0ffc0, 0x40c040, 0x60d060, 0x505050, },
{ 0xa0a04c, 0xd0d04c, 0xffff80, 0xffff80, 0x80ff80, 0x80ff80, 0x606060, },
{ 0xffffff, 0xffffff, 0xffffff, 0xffffff, 0xffffff, 0xffffff, 0x909090, },
}, {
{ 0x40c440, 0x60d060, 0xc0ffc0, 0xc0ffc0, 0x40c040, 0x60d060, 0x505050, },
{ 0x40c040, 0x80ff80, 0x80ff80, 0x80ff80, 0x80ff80, 0x80ff80, 0x606060, },
{ 0xffffff, 0xffffff, 0xffffff, 0xffffff, 0xffffff, 0xffffff, 0x909090, },
}, {
{ 0x9090ac, 0xa0a0b8, 0xbcb8cc, 0xbcb8cc, 0x909040, 0x909040, 0x909040, },
{ 0xa0a0b8, 0xb0b4d0, 0xa0a0b8, 0xa0a0b8, 0xa0a050, 0xa0a050, 0xa0a050, },
{ 0x808088, 0x808030, 0x808030, 0x808030, 0x808030, 0x808030, 0x808030, },
}, {
{ 0x704c70, 0x885c8c, 0x9c68a4, 0xb870bc, 0xb490bc, 0xb490bc, 0x302828, },
{ 0x646064, 0xcca8d4, 0xc060c0, 0xa07898, 0xe0b8e0, 0xe0b8e0, 0x403838, },
{ 0xdccce4, 0xffffff, 0xffffff, 0xffffff, 0xffffff, 0xffffff, 0x909090, },
}, {
{ 0x704c70, 0x885c8c, 0x9c68a4, 0xb870bc, 0xb490bc, 0xb490bc, 0x302828, },
{ 0xb09cb4, 0xcca8d4, 0xc060c0, 0xa07898, 0xe0b8e0, 0xe0b8e0, 0x403838, },
{ 0xdccce4, 0xffffff, 0xffffff, 0xffffff, 0xffffff, 0xffffff, 0x909090, },
}, {
{ 0x646494, 0x888cb8, 0xb0b0b0, 0xb0b0cc, 0x9c9cf4, 0x8888b0, 0x50506c, },
{ 0x9090a4, 0xb0b4d4, 0xb0b0dc, 0xb0b0cc, 0xd0d0fc, 0xd0d4f0, 0x606060, },
{ 0xb4b4d4, 0xe4e4ff, 0xffffff, 0xffffff, 0xe0e4ff, 0xececff, 0x909090, },
}, {
{ 0x646444, 0x888c64, 0xb0b0b0, 0xb0b088, 0xaca858, 0x88886c, 0x505050, },
{ 0x88886c, 0xb0b490, 0xb0b0b0, 0xb0b088, 0xd8d898, 0xd0d4b0, 0x606060, },
{ 0xb4b49c, 0xffffd8, 0xffffff, 0xffffd4, 0xffffdc, 0xffffcc, 0x909090, },
}, {
{ 0x906464, 0xb48c8c, 0xd4b0b0, 0xdcb0b0, 0xff9c9c, 0xc88888, 0x505050, },
{ 0xb47c80, 0xd4b4b8, 0xc4a8a8, 0xdcb0b0, 0xffc0c0, 0xfce8ec, 0x606060, },
{ 0xe0b4b4, 0xffdcd8, 0xffd8d4, 0xffe0e4, 0xffece8, 0xffffff, 0x909090, },
}, {
{ 0x403848, 0x403848, 0x403848, 0x886894, 0x7c80c8, 0x7c80c8, 0x302828, },
{ 0x403848, 0x403848, 0x403848, 0x403848, 0x7c80c8, 0x7c80c8, 0x403838, },
{ 0xc8c4c8, 0xffffff, 0xffffff, 0xffffff, 0xe8e8ec, 0xffffff, 0x909090, },
},
};
#define STBTE_COLOR_TILEMAP_BACKGROUND 0x000000 #define STBTE_COLOR_TILEMAP_BACKGROUND 0x000000
#define STBTE_COLOR_TILEMAP_BORDER 0x203060 #define STBTE_COLOR_TILEMAP_BORDER 0x203060
#define STBTE_COLOR_TILEMAP_HIGHLIGHT 0xffffff #define STBTE_COLOR_TILEMAP_HIGHLIGHT 0xffffff
#define STBTE_COLOR_PANEL_BACKGROUND 0x403010 #define STBTE_COLOR_GRID 0x404040
#define STBTE_COLOR_PANEL_OUTLINE 0xc08040
#define STBTE_COLOR_PANEL_TEXT 0xffffff
#define STBTE_COLOR_BUTTON_BACKGROUND 0x703870
#define STBTE_COLOR_BUTTON_OUTLINE 0xc060c0
#define STBTE_COLOR_BUTTON_TEXT 0xffffff
#define STBTE_COLOR_BUTTON_DOWN 0xe080e0
#define STBTE_COLOR_BUTTON_OVER 0xffc0ff
#define STBTE_COLOR_BUTTON_TEXT_SELECTED 0x000000
#define STBTE_COLOR_MICROBUTTON 0x40c040
#define STBTE_COLOR_MICROBUTTON_DOWN 0xc0ffc0
#define STBTE_COLOR_MICROBUTTON_FRAME 0x00ff00
#define STBTE_COLOR_MICROBUTTON_OVER 0x80ff80
#define STBTE_COLOR_TILEPALETTE_OUTLINE 0xffffff
#define STBTE_COLOR_TILEPALETTE_BACKGROUND 0x000000
#define STBTE_COLOR_MINIBUTTON_ICON 0xffffff
#define STBTE_COLOR_SELECTION_OUTLINE1 0xdfdfdf #define STBTE_COLOR_SELECTION_OUTLINE1 0xdfdfdf
#define STBTE_COLOR_SELECTION_OUTLINE2 0x303030 #define STBTE_COLOR_SELECTION_OUTLINE2 0x303030
#define STBTE_COLOR_GRID 0x404040 #define STBTE_COLOR_TILEPALETTE_OUTLINE 0xffffff
#define STBTE_COLOR_TILEPALETTE_BACKGROUND 0x000000
#define STBTE_COLOR_LAYERCONTROL 0x6f6f6f // disabled, selected, down, over
#define STBTE_COLOR_LAYERCONTROL_OVER 0xcfcfcf static unsigned char stbte__state_to_index[2][2][2][2] =
#define STBTE_COLOR_LAYERCONTROL_DOWN 0xffffff {
#define STBTE_COLOR_LAYERCONTROL_TOGGLED 0xbfbfbf {
#define STBTE_COLOR_LAYERCONTROL_DISABLED 0x404040 { { STBTE__idle , STBTE__over }, { STBTE__down , STBTE__over_down }, },
#define STBTE_COLOR_LAYERCONTROL_OUTLINE 0xffffff { { STBTE__selected, STBTE__selected_over }, { STBTE__down , STBTE__over_down }, },
#define STBTE_COLOR_LAYERCONTROL_OUTLINE_DISABLED 0x202020 },{
#define STBTE_COLOR_LAYERCONTROL_TEXT 0xffffff { { STBTE__disabled, STBTE__disabled }, { STBTE__disabled, STBTE__disabled }, },
#define STBTE_COLOR_LAYERCONTROL_TEXT_DOWN 0x5f5f5f { { STBTE__selected, STBTE__selected_over }, { STBTE__disabled, STBTE__disabled }, },
#define STBTE_COLOR_LAYERCONTROL_TEXT_TOGGLED 0x000000 }
#define STBTE_COLOR_LAYERCONTROL_TEXT_DISABLED 0x606060 };
#define STBTE__INDEX_FOR_STATE(disable,select,down,over) stbte__state_to_index[disable][select][down][over]
#define STBTE_COLOR_LAYERMASK_HIDE 0xffff55 #define STBTE__INDEX_FOR_ID(id,disable,select) STBTE__INDEX_FOR_STATE(disable,select,STBTE__IS_ACTIVE(id),STBTE__IS_HOT(id))
#define STBTE_COLOR_LAYERMASK_LOCK 0x5f55ff
#define STBTE_COLOR_LAYERMASK_SOLO 0xff5f55
#define STBTE__FONT_HEIGHT 9 #define STBTE__FONT_HEIGHT 9
static short stbte__font_offset[95+16]; static short stbte__font_offset[95+16];
@ -358,6 +455,7 @@ typedef short stbte__tiledata;
enum enum
{ {
STBTE__panel_toolbar, STBTE__panel_toolbar,
STBTE__panel_colorpick,
STBTE__panel_info, STBTE__panel_info,
STBTE__panel_layers, STBTE__panel_layers,
STBTE__panel_categories, STBTE__panel_categories,
@ -378,6 +476,7 @@ enum
{ {
STBTE__tool_select, STBTE__tool_select,
STBTE__tool_brush, STBTE__tool_brush,
STBTE__tool_erase,
STBTE__tool_rect, STBTE__tool_rect,
STBTE__tool_eyedrop, STBTE__tool_eyedrop,
STBTE__tool_fill, STBTE__tool_fill,
@ -391,7 +490,7 @@ enum
}; };
// icons are stored in the 0-31 range of ASCII in the font // icons are stored in the 0-31 range of ASCII in the font
static int toolchar[] = { 26,24,20,23,22, 19,29,28, }; static int toolchar[] = { 26,24,25,20,23,22, 19,29,28, };
enum enum
{ {
@ -514,7 +613,8 @@ static void stbte__init_gui(void)
stbte__ui.panel[i].delta_height = 0; stbte__ui.panel[i].delta_height = 0;
stbte__ui.panel[i].side = STBTE__side_left; stbte__ui.panel[i].side = STBTE__side_left;
} }
stbte__ui.panel[STBTE__panel_toolbar].side = STBTE__side_top; stbte__ui.panel[STBTE__panel_toolbar ].side = STBTE__side_top;
stbte__ui.panel[STBTE__panel_colorpick].side = STBTE__side_right;
if (stbte__ui.left_width == 0) if (stbte__ui.left_width == 0)
stbte__ui.left_width = 80; stbte__ui.left_width = 80;
@ -1069,7 +1169,33 @@ static int stbte__button_core(int id)
return 0; return 0;
} }
static int stbte__button(char *label, int x, int y, int textoff, int width, int id, int toggled) static void stbte__draw_box(int x0, int y0, int x1, int y1, int colormode, int colorindex)
{
stbte__draw_rect (x0,y0,x1,y1, stbte__color_table[colormode][STBTE__base ][colorindex]);
stbte__draw_frame(x0,y0,x1,y1, stbte__color_table[colormode][STBTE__outline][colorindex]);
}
static void stbte__draw_textbox(int x0, int y0, int x1, int y1, char *text, int xoff, int yoff, int colormode, int colorindex)
{
stbte__draw_box(x0,y0,x1,y1,colormode,colorindex);
stbte__draw_text(x0+xoff,y0+yoff, text, x1-x0-xoff-1, stbte__color_table[colormode][STBTE__text][colorindex]);
}
static int stbte__button(int colormode, char *label, int x, int y, int textoff, int width, int id, int toggled, int disabled)
{
int x0=x,y0=y, x1=x+width,y1=y+STBTE__BUTTON_HEIGHT;
int s = STBTE__BUTTON_INTERNAL_SPACING;
int over = !disabled && stbte__hittest(x0,y0,x1,y1,id);
if (stbte__ui.event == STBTE__paint)
stbte__draw_textbox(x0,y0,x1,y1, label,s+textoff,s, colormode, STBTE__INDEX_FOR_ID(id,disabled,toggled));
if (disabled)
return 0;
return (stbte__button_core(id) == 1);
}
static int stbte__button_icon(int colormode, char ch, int x, int y, int width, int id, int toggled)
{ {
int x0=x,y0=y, x1=x+width,y1=y+STBTE__BUTTON_HEIGHT; int x0=x,y0=y, x1=x+width,y1=y+STBTE__BUTTON_HEIGHT;
int s = STBTE__BUTTON_INTERNAL_SPACING; int s = STBTE__BUTTON_INTERNAL_SPACING;
@ -1077,104 +1203,55 @@ static int stbte__button(char *label, int x, int y, int textoff, int width, int
int over = stbte__hittest(x0,y0,x1,y1,id); int over = stbte__hittest(x0,y0,x1,y1,id);
if (stbte__ui.event == STBTE__paint) { if (stbte__ui.event == STBTE__paint) {
stbte__draw_rect (x0, y0, x1, y1, STBTE__IS_ACTIVE(id) || toggled ? STBTE_COLOR_BUTTON_DOWN : STBTE_COLOR_BUTTON_BACKGROUND);
stbte__draw_frame(x0, y0, x1, y1, STBTE__IS_HOT(id) || toggled ? STBTE_COLOR_BUTTON_OVER : STBTE_COLOR_BUTTON_OUTLINE);
stbte__draw_text (x0+s+textoff, y0+s, label ,width-s*2, toggled ? STBTE_COLOR_BUTTON_TEXT : STBTE_COLOR_BUTTON_TEXT_SELECTED);
}
return (stbte__button_core(id) == 1);
}
static int stbte__button_icon(char ch, int x, int y, int width, int id, int toggled)
{
int x0=x,y0=y, x1=x+width,y1=y+STBTE__BUTTON_HEIGHT;
int s = STBTE__BUTTON_INTERNAL_SPACING, pad;
char label[2] = { ch, 0 }; char label[2] = { ch, 0 };
int pad = (9 - stbte__get_char_width(ch))/2;
int over = stbte__hittest(x0,y0,x1,y1,id); stbte__draw_textbox(x0,y0,x1,y1, label,s+pad,s, colormode, STBTE__INDEX_FOR_ID(id,0,toggled));
if (stbte__ui.event == STBTE__paint) {
stbte__draw_rect (x0, y0, x1, y1, STBTE__IS_ACTIVE(id) || toggled ? STBTE_COLOR_BUTTON_DOWN : STBTE_COLOR_BUTTON_BACKGROUND);
stbte__draw_frame(x0, y0, x1, y1, STBTE__IS_HOT(id) || toggled ? STBTE_COLOR_BUTTON_OVER : STBTE_COLOR_BUTTON_OUTLINE);
pad = (9 - stbte__get_char_width(ch))/2;
stbte__draw_text (x0+s+pad, y0+s, label ,9, toggled ? STBTE_COLOR_BUTTON_TEXT : STBTE_COLOR_BUTTON_TEXT_SELECTED);
} }
return (stbte__button_core(id) == 1); return (stbte__button_core(id) == 1);
} }
static int stbte__minibutton(int x, int y, int ch, int id) static int stbte__minibutton(int colormode, int x, int y, int ch, int id)
{ {
int x0 = x, y0 = y, x1 = x+8, y1 = y+7; int x0 = x, y0 = y, x1 = x+8, y1 = y+7;
int over = stbte__hittest(x0,y0,x1,y1,id); int over = stbte__hittest(x0,y0,x1,y1,id);
if (stbte__ui.event == STBTE__paint) { if (stbte__ui.event == STBTE__paint) {
char str[2] = { ch,0 }; char str[2] = { ch,0 };
stbte__draw_rect (x0,y0,x1,y1, STBTE__IS_ACTIVE(id) ? STBTE_COLOR_MICROBUTTON_DOWN : STBTE_COLOR_MICROBUTTON); stbte__draw_textbox(x0,y0,x1,y1, str,1,0,colormode, STBTE__INDEX_FOR_ID(id,0,0));
stbte__draw_frame(x0,y0,x1,y1, STBTE__IS_HOT(id) ? STBTE_COLOR_MICROBUTTON_OVER : STBTE_COLOR_MICROBUTTON_FRAME);
stbte__draw_text (x0+1,y0,str,99, STBTE_COLOR_MINIBUTTON_ICON);
} }
return stbte__button_core(id); return stbte__button_core(id);
} }
static int stbte__layerbutton(int x, int y, int ch, int id, int toggled, int disabled, int color) static int stbte__layerbutton(int x, int y, int ch, int id, int toggled, int disabled, int colormode)
{ {
int x0 = x, y0 = y, x1 = x+10, y1 = y+11; int x0 = x, y0 = y, x1 = x+10, y1 = y+11;
int over = stbte__hittest(x0,y0,x1,y1,id); int over = !disabled && stbte__hittest(x0,y0,x1,y1,id);
if (stbte__ui.event == STBTE__paint) { if (stbte__ui.event == STBTE__paint) {
int rc = STBTE_COLOR_LAYERCONTROL;
int rf = STBTE_COLOR_LAYERCONTROL_OUTLINE;
int rt = STBTE_COLOR_LAYERCONTROL_TEXT;
if (toggled) {
rc = STBTE_COLOR_LAYERCONTROL_TOGGLED;
rt = STBTE_COLOR_LAYERCONTROL_TEXT_TOGGLED;
}
if (STBTE__IS_HOT(id)) {
rc = STBTE_COLOR_LAYERCONTROL_OVER;
}
if (STBTE__IS_ACTIVE(id)) {
rc = STBTE_COLOR_LAYERCONTROL_DOWN;
rt = STBTE_COLOR_LAYERCONTROL_TEXT_DOWN;
}
rc &= color;
rf &= color;
rt &= color;
if (disabled) {
rc = STBTE_COLOR_LAYERCONTROL_DISABLED;
rf = STBTE_COLOR_LAYERCONTROL_OUTLINE_DISABLED;
rt = STBTE_COLOR_LAYERCONTROL_TEXT_DISABLED;
}
stbte__draw_rect (x0,y0,x1,y1, rc);
stbte__draw_frame(x0,y0,x1,y1, rf);
{
char str[2] = { ch,0 }; char str[2] = { ch,0 };
int off = (9-stbte__get_char_width(ch))/2; int off = (9-stbte__get_char_width(ch))/2;
stbte__draw_text (x0+1+off,y0+2,str,99, rt); stbte__draw_textbox(x0,y0,x1,y1, str, off+1,2, colormode, STBTE__INDEX_FOR_ID(id,disabled,toggled));
}
} }
if (disabled) if (disabled)
return 0; return 0;
return stbte__button_core(id); return stbte__button_core(id);
} }
static int stbte__microbutton(int x, int y, int size, int id, int colormode)
static int stbte__microbutton(int x, int y, int size, int id, int c1, int c2, int toggled)
{ {
int x0 = x, y0 = y, x1 = x+size, y1 = y+size; int x0 = x, y0 = y, x1 = x+size, y1 = y+size;
int over = stbte__hittest(x0,y0,x1,y1,id); int over = stbte__hittest(x0,y0,x1,y1,id);
if (stbte__ui.event == STBTE__paint) { if (stbte__ui.event == STBTE__paint) {
stbte__draw_rect (x0,y0,x1,y1, STBTE__IS_ACTIVE(id) || toggled ? c2 : c1 ); stbte__draw_box(x0,y0,x1,y1, colormode, STBTE__INDEX_FOR_ID(id,0,0));
stbte__draw_frame(x0,y0,x1,y1, STBTE__IS_HOT(id) ? STBTE_COLOR_MICROBUTTON_OVER : STBTE_COLOR_MICROBUTTON_FRAME);
} }
return stbte__button_core(id); return stbte__button_core(id);
} }
static int stbte__microbutton_dragger(int x, int y, int size, int id, int c1, int c2, int toggled, int *pos) static int stbte__microbutton_dragger(int x, int y, int size, int id, int *pos)
{ {
int x0 = x, y0 = y, x1 = x+size, y1 = y+size; int x0 = x, y0 = y, x1 = x+size, y1 = y+size;
int over = stbte__hittest(x0,y0,x1,y1,id); int over = stbte__hittest(x0,y0,x1,y1,id);
switch (stbte__ui.event) { switch (stbte__ui.event) {
case STBTE__paint: case STBTE__paint:
stbte__draw_rect (x0,y0,x1,y1, STBTE__IS_ACTIVE(id) || toggled ? c2 : c1 ); stbte__draw_box(x0,y0,x1,y1, STBTE__cexpander, STBTE__INDEX_FOR_ID(id,0,0));
stbte__draw_frame(x0,y0,x1,y1, STBTE__IS_HOT(id) ? STBTE_COLOR_MICROBUTTON_OVER : STBTE_COLOR_MICROBUTTON_FRAME);
break; break;
case STBTE__leftdown: case STBTE__leftdown:
if (STBTE__IS_HOT(id) && STBTE__INACTIVE()) { if (STBTE__IS_HOT(id) && STBTE__INACTIVE()) {
@ -1204,17 +1281,12 @@ static int stbte__category_button(char *label, int x, int y, int width, int id,
int over = stbte__hittest(x0,y0,x1,y1,id); int over = stbte__hittest(x0,y0,x1,y1,id);
if (stbte__ui.event == STBTE__paint) { if (stbte__ui.event == STBTE__paint)
stbte__draw_rect (x0, y0, x1, y1, toggled ? STBTE_COLOR_BUTTON_DOWN : STBTE_COLOR_BUTTON_BACKGROUND); stbte__draw_textbox(x0,y0,x1,y1, label, s,s, STBTE__ccategory_button, STBTE__INDEX_FOR_ID(id,0,toggled));
stbte__draw_text (x0+s, y0+s, label ,width-s*2, STBTE__IS_HOT(id) ? STBTE_COLOR_BUTTON_TEXT : STBTE_COLOR_BUTTON_TEXT_SELECTED);
}
return (stbte__button_core(id) == 1); return (stbte__button_core(id) == 1);
} }
#define STBTE_COLOR_SCROLLBAR_TRACK 0x808030
#define STBTE_COLOR_SCROLLBAR_THUMB 0x909040
static void stbte__scrollbar(int x, int y0, int y1, int *val, int v0, int v1, int num_vis, int id) static void stbte__scrollbar(int x, int y0, int y1, int *val, int v0, int v1, int num_vis, int id)
{ {
int over; int over;
@ -1229,8 +1301,8 @@ static void stbte__scrollbar(int x, int y0, int y1, int *val, int v0, int v1, in
over = stbte__hittest(x-1,y0,x+2,y1,id); over = stbte__hittest(x-1,y0,x+2,y1,id);
switch (stbte__ui.event) { switch (stbte__ui.event) {
case STBTE__paint: case STBTE__paint:
stbte__draw_rect(x,y0,x+1,y1, STBTE_COLOR_SCROLLBAR_TRACK); stbte__draw_rect(x,y0,x+1,y1, stbte__color_table[STBTE__cscrollbar][STBTE__text][STBTE__idle]);
stbte__draw_rect(x-1,thumbpos-3,x+2,thumbpos+4, STBTE_COLOR_SCROLLBAR_THUMB); stbte__draw_box(x-1,thumbpos-3,x+2,thumbpos+4, STBTE__cscrollbar, STBTE__INDEX_FOR_ID(id,0,0));
break; break;
case STBTE__leftdown: case STBTE__leftdown:
if (STBTE__IS_HOT(id) && STBTE__INACTIVE()) { if (STBTE__IS_HOT(id) && STBTE__INACTIVE()) {
@ -1288,9 +1360,9 @@ static void stbte__compute_panel_locations(stbte_tilemap *tm)
int i, limit, w, k; int i, limit, w, k;
int window_width = stbte__ui.x1 - stbte__ui.x0; int window_width = stbte__ui.x1 - stbte__ui.x0;
int window_height = stbte__ui.y1 - stbte__ui.y0; int window_height = stbte__ui.y1 - stbte__ui.y0;
int min_width[STBTE__num_panel]={0,0,0,0,0}; int min_width[STBTE__num_panel]={0,0,0,0,0,0};
int height[STBTE__num_panel]={0,0,0,0,0}; int height[STBTE__num_panel]={0,0,0,0,0,0};
int panel_active[STBTE__num_panel]={1,1,1,1,1}; int panel_active[STBTE__num_panel]={1,0,1,1,1,1};
int vpos[4] = { 0,0,0,0 }; int vpos[4] = { 0,0,0,0 };
stbte__panel *p = stbte__ui.panel; stbte__panel *p = stbte__ui.panel;
stbte__panel *pt = &p[STBTE__panel_toolbar]; stbte__panel *pt = &p[STBTE__panel_toolbar];
@ -1307,9 +1379,13 @@ static void stbte__compute_panel_locations(stbte_tilemap *tm)
// determine which panels are active // determine which panels are active
panel_active[STBTE__panel_categories] = tm->num_categories != 0; panel_active[STBTE__panel_categories] = tm->num_categories != 0;
panel_active[STBTE__panel_layers ] = tm->num_layers > 1; panel_active[STBTE__panel_layers ] = tm->num_layers > 1;
#ifdef STBTE__COLORPICKER
panel_active[STBTE__panel_colorpick ] = 1;
#endif
// compute minimum widths for each panel (assuming they're on sides not top) // compute minimum widths for each panel (assuming they're on sides not top)
min_width[STBTE__panel_info ] = 8 + 11 + 7*tm->digits+17+7; // estimate min width of "w:0000" min_width[STBTE__panel_info ] = 8 + 11 + 7*tm->digits+17+7; // estimate min width of "w:0000"
min_width[STBTE__panel_colorpick ] = 120;
min_width[STBTE__panel_tiles ] = 4 + tm->palette_spacing_x + 5; // 5 for scrollbar min_width[STBTE__panel_tiles ] = 4 + tm->palette_spacing_x + 5; // 5 for scrollbar
min_width[STBTE__panel_categories] = 4 + 42 + 5; // 42 is enough to show ~7 chars; 5 for scrollbar min_width[STBTE__panel_categories] = 4 + 42 + 5; // 42 is enough to show ~7 chars; 5 for scrollbar
min_width[STBTE__panel_layers ] = 4 + 54 + 30*tm->has_layer_names; // 2 digits plus 3 buttons plus scrollbar min_width[STBTE__panel_layers ] = 4 + 54 + 30*tm->has_layer_names; // 2 digits plus 3 buttons plus scrollbar
@ -1351,6 +1427,9 @@ static void stbte__compute_panel_locations(stbte_tilemap *tm)
stbte__region[i].x = (i == STBTE__side_left) ? stbte__ui.x0 - anim : stbte__ui.x1 - stbte__region[i].width + anim; stbte__region[i].x = (i == STBTE__side_left) ? stbte__ui.x0 - anim : stbte__ui.x1 - stbte__region[i].width + anim;
} }
// color picker
height[STBTE__panel_colorpick] = 300;
// info panel // info panel
w = stbte__region[p[STBTE__panel_info].side].width; w = stbte__region[p[STBTE__panel_info].side].width;
p[STBTE__panel_info].mode = (w >= 8 + (11+7*tm->digits+17)*2 + 4); p[STBTE__panel_info].mode = (w >= 8 + (11+7*tm->digits+17)*2 + 4);
@ -1417,6 +1496,7 @@ enum
STBTE__panel_mover, // p1 is panel ID, p2 is destination side STBTE__panel_mover, // p1 is panel ID, p2 is destination side
STBTE__panel_sizer, // param panel ID STBTE__panel_sizer, // param panel ID
STBTE__scrollbar_id, STBTE__scrollbar_id,
STBTE__colorpick_id,
}; };
// id is: [ 24-bit data : 7-bit identifer ] // id is: [ 24-bit data : 7-bit identifer ]
@ -1440,6 +1520,10 @@ static void stbte__alert(const char *msg)
stbte__ui.alert_timer = 3; stbte__ui.alert_timer = 3;
} }
#define STBTE__BG(tm,layer) ((layer) == 0 ? (tm)->background_tile : STBTE__NO_TILE)
static void stbte__brush_predict(stbte_tilemap *tm, short result[]) static void stbte__brush_predict(stbte_tilemap *tm, short result[])
{ {
int layer_to_paint = tm->cur_layer; int layer_to_paint = tm->cur_layer;
@ -1457,8 +1541,6 @@ static void stbte__brush_predict(stbte_tilemap *tm, short result[])
continue; continue;
if (i != tm->solo_layer) { if (i != tm->solo_layer) {
short bg;
// if there's a selected layer, can only paint on that // if there's a selected layer, can only paint on that
if (tm->cur_layer >= 0 && i != tm->cur_layer) if (tm->cur_layer >= 0 && i != tm->cur_layer)
continue; continue;
@ -1471,9 +1553,8 @@ static void stbte__brush_predict(stbte_tilemap *tm, short result[])
if (tm->layerinfo[i].locked == STBTE__locked) if (tm->layerinfo[i].locked == STBTE__locked)
continue; continue;
bg = i == 0 ? tm->background_tile : STBTE__NO_TILE;
// if the layer is non-empty and protected, can't write to it // if the layer is non-empty and protected, can't write to it
if (tm->layerinfo[i].locked == STBTE__protected && result[i] != bg) if (tm->layerinfo[i].locked == STBTE__protected && result[i] != STBTE__BG(tm,i))
continue; continue;
} }
@ -1500,8 +1581,6 @@ static void stbte__brush(stbte_tilemap *tm, int x, int y)
continue; continue;
if (i != tm->solo_layer) { if (i != tm->solo_layer) {
short bg;
// if there's a selected layer, can only paint on that // if there's a selected layer, can only paint on that
if (tm->cur_layer >= 0 && i != tm->cur_layer) if (tm->cur_layer >= 0 && i != tm->cur_layer)
continue; continue;
@ -1514,9 +1593,8 @@ static void stbte__brush(stbte_tilemap *tm, int x, int y)
if (tm->layerinfo[i].locked == STBTE__locked) if (tm->layerinfo[i].locked == STBTE__locked)
continue; continue;
bg = i == 0 ? tm->background_tile : STBTE__NO_TILE;
// if the layer is non-empty and protected, can't write to it // if the layer is non-empty and protected, can't write to it
if (tm->layerinfo[i].locked == STBTE__protected && tm->data[y][x][i] != bg) if (tm->layerinfo[i].locked == STBTE__protected && tm->data[y][x][i] != STBTE__BG(tm,i))
continue; continue;
} }
@ -1533,6 +1611,7 @@ enum
STBTE__erase_none = -1, STBTE__erase_none = -1,
STBTE__erase_brushonly = 0, STBTE__erase_brushonly = 0,
STBTE__erase_any = 1, STBTE__erase_any = 1,
STBTE__erase_all = 2,
}; };
static int stbte__erase_predict(stbte_tilemap *tm, short result[], int allow_any) static int stbte__erase_predict(stbte_tilemap *tm, short result[], int allow_any)
@ -1571,7 +1650,7 @@ static int stbte__erase_predict(stbte_tilemap *tm, short result[], int allow_any
// if multiple layers are legit, first scan all for brush data // if multiple layers are legit, first scan all for brush data
if (ti) { if (ti && allow_any != STBTE__erase_all) {
for (i=tm->num_layers-1; i >= 0; --i) { for (i=tm->num_layers-1; i >= 0; --i) {
if (result[i] != ti->id) if (result[i] != ti->id)
continue; continue;
@ -1579,7 +1658,7 @@ static int stbte__erase_predict(stbte_tilemap *tm, short result[], int allow_any
continue; continue;
if (i == 0 && result[i] == tm->background_tile) if (i == 0 && result[i] == tm->background_tile)
return STBTE__erase_none; return STBTE__erase_none;
result[i] = (i == 0 ? tm->background_tile : STBTE__NO_TILE); result[i] = STBTE__BG(tm,i);
return STBTE__erase_brushonly; return STBTE__erase_brushonly;
} }
} }
@ -1595,10 +1674,13 @@ static int stbte__erase_predict(stbte_tilemap *tm, short result[], int allow_any
continue; continue;
if (i == 0 && result[i] == tm->background_tile) if (i == 0 && result[i] == tm->background_tile)
return STBTE__erase_none; return STBTE__erase_none;
result[i] = (i == 0 ? tm->background_tile : STBTE__NO_TILE); result[i] = STBTE__BG(tm,i);
if (allow_any != STBTE__erase_all)
return STBTE__erase_any; return STBTE__erase_any;
} }
if (allow_any == STBTE__erase_all)
return allow_any;
return STBTE__erase_none; return STBTE__erase_none;
} }
@ -1641,7 +1723,7 @@ static int stbte__erase(stbte_tilemap *tm, int x, int y, int allow_any)
// if multiple layers are legit, first scan all for brush data // if multiple layers are legit, first scan all for brush data
if (ti) { if (ti && allow_any != STBTE__erase_all) {
for (i=tm->num_layers-1; i >= 0; --i) { for (i=tm->num_layers-1; i >= 0; --i) {
if (tm->data[y][x][i] != ti->id) if (tm->data[y][x][i] != ti->id)
continue; continue;
@ -1650,7 +1732,7 @@ static int stbte__erase(stbte_tilemap *tm, int x, int y, int allow_any)
if (i == 0 && tm->data[y][x][i] == tm->background_tile) if (i == 0 && tm->data[y][x][i] == tm->background_tile)
return STBTE__erase_none; return STBTE__erase_none;
stbte__undo_record(tm,x,y,i,tm->data[y][x][i]); stbte__undo_record(tm,x,y,i,tm->data[y][x][i]);
tm->data[y][x][i] = (i == 0 ? tm->background_tile : STBTE__NO_TILE); tm->data[y][x][i] = STBTE__BG(tm,i);
return STBTE__erase_brushonly; return STBTE__erase_brushonly;
} }
} }
@ -1667,10 +1749,12 @@ static int stbte__erase(stbte_tilemap *tm, int x, int y, int allow_any)
if (i == 0 && tm->data[y][x][i] == tm->background_tile) if (i == 0 && tm->data[y][x][i] == tm->background_tile)
return STBTE__erase_none; return STBTE__erase_none;
stbte__undo_record(tm,x,y,i,tm->data[y][x][i]); stbte__undo_record(tm,x,y,i,tm->data[y][x][i]);
tm->data[y][x][i] = (i == 0 ? tm->background_tile : STBTE__NO_TILE); tm->data[y][x][i] = STBTE__BG(tm,i);
if (allow_any != STBTE__erase_all)
return STBTE__erase_any; return STBTE__erase_any;
} }
if (allow_any == STBTE__erase_all)
return allow_any;
return STBTE__erase_none; return STBTE__erase_none;
} }
@ -1735,24 +1819,25 @@ static void stbte__paste_stack(stbte_tilemap *tm, short result[], short dest[],
// check that we're allowed to write to it // check that we're allowed to write to it
if (tm->layerinfo[i].hidden) return; if (tm->layerinfo[i].hidden) return;
if (tm->layerinfo[i].locked == STBTE__locked) return; if (tm->layerinfo[i].locked == STBTE__locked) return;
// if dragging w/o copy, we have to be allowed to erase // if protected, dest has to be empty
if (tm->layerinfo[i].locked == STBTE__protected && dest[i] != STBTE__BG(tm,i)) return;
// if dragging w/o copy, we will try to erase stuff, which protection disallows
if (dragging && tm->layerinfo[i].locked == STBTE__protected) if (dragging && tm->layerinfo[i].locked == STBTE__protected)
return; return;
} }
result[i] = dest[i]; result[i] = dest[i];
if (src[i] != STBTE__NO_TILE) if (src[i] != STBTE__BG(tm,i))
result[i] = src[i]; result[i] = src[i];
return; return;
} }
for (i=0; i < tm->num_layers; ++i) { for (i=0; i < tm->num_layers; ++i) {
result[i] = dest[i]; result[i] = dest[i];
if (src[i] != STBTE__NO_TILE) { if (src[i] != STBTE__NO_TILE)
if (!tm->layerinfo[i].hidden && tm->layerinfo[i].locked != STBTE__locked) if (!tm->layerinfo[i].hidden && tm->layerinfo[i].locked != STBTE__locked)
if (!dragging || tm->layerinfo[i].locked == STBTE__unlocked) if (tm->layerinfo[i].locked == STBTE__unlocked || (!dragging && dest[i] == STBTE__BG(tm,i)))
result[i] = src[i]; result[i] = src[i];
} }
}
} }
// compute the result of dragging away from a tile // compute the result of dragging away from a tile
@ -1763,14 +1848,12 @@ static void stbte__clear_stack(stbte_tilemap *tm, short result[])
i = tm->cur_layer; i = tm->cur_layer;
if (tm->solo_layer >= 0) if (tm->solo_layer >= 0)
i = tm->solo_layer; i = tm->solo_layer;
if (i >= 0) { if (i >= 0)
result[i] = (i == 0 ? tm->background_tile : STBTE__NO_TILE); result[i] = STBTE__BG(tm,i);
} else { else
for (i=0; i < tm->num_layers; ++i) { for (i=0; i < tm->num_layers; ++i)
if (!tm->layerinfo[i].hidden && tm->layerinfo[i].locked == STBTE__unlocked) if (!tm->layerinfo[i].hidden && tm->layerinfo[i].locked == STBTE__unlocked)
result[i] = (i == 0 ? tm->background_tile : STBTE__NO_TILE); result[i] = STBTE__BG(tm,i);
}
}
} }
// check if some map square is active // check if some map square is active
@ -2133,7 +2216,10 @@ static void stbte__tile(stbte_tilemap *tm, int sx, int sy, int mapx, int mapy)
if (STBTE__IS_HOT(id) && STBTE__INACTIVE()) { if (STBTE__IS_HOT(id) && STBTE__INACTIVE()) {
stbte__activate(id); stbte__activate(id);
stbte__begin_undo(tm); stbte__begin_undo(tm);
stbte__ui.brush_state = stbte__erase(tm, mapx, mapy, 1); if (stbte__erase(tm, mapx, mapy, STBTE__erase_any) == STBTE__erase_brushonly)
stbte__ui.brush_state = STBTE__erase_brushonly;
else
stbte__ui.brush_state = STBTE__erase_any;
} }
break; break;
case STBTE__leftup: case STBTE__leftup:
@ -2146,6 +2232,28 @@ static void stbte__tile(stbte_tilemap *tm, int sx, int sy, int mapx, int mapy)
} }
break; break;
case STBTE__tool_erase:
switch (stbte__ui.event) {
case STBTE__mousemove:
if (STBTE__IS_MAP_ACTIVE() && over)
stbte__erase(tm, mapx, mapy, STBTE__erase_all);
break;
case STBTE__leftdown:
if (STBTE__IS_HOT(id) && STBTE__INACTIVE()) {
stbte__activate(id);
stbte__begin_undo(tm);
stbte__erase(tm, mapx, mapy, STBTE__erase_all);
}
break;
case STBTE__leftup:
if (STBTE__IS_MAP_ACTIVE()) {
stbte__end_undo(tm);
stbte__activate(0);
}
break;
}
break;
case STBTE__tool_select: case STBTE__tool_select:
if (STBTE__IS_HOT(id)) { if (STBTE__IS_HOT(id)) {
switch (stbte__ui.event) { switch (stbte__ui.event) {
@ -2237,11 +2345,13 @@ static void stbte__toolbar(stbte_tilemap *tm, int x0, int y0, int w, int h)
for (i=0; i < STBTE__num_tool; ++i) { for (i=0; i < STBTE__num_tool; ++i) {
int highlight=0; int highlight=0;
highlight = (stbte__ui.tool == i); highlight = (stbte__ui.tool == i);
if (i == STBTE__tool_undo || i == STBTE__tool_grid)
x += 8;
if (i == STBTE__tool_grid && stbte__ui.show_grid) if (i == STBTE__tool_grid && stbte__ui.show_grid)
highlight=1; highlight=1;
if (i == STBTE__tool_fill) if (i == STBTE__tool_fill)
continue; continue;
if (stbte__button_icon(toolchar[i], x, y, 13, STBTE__ID(STBTE__toolbarA, i), highlight)) { if (stbte__button_icon(STBTE__ctoolbar_button, toolchar[i], x, y, 13, STBTE__ID(STBTE__toolbarA, i), highlight)) {
switch (i) { switch (i) {
case STBTE__tool_eyedrop: case STBTE__tool_eyedrop:
stbte__ui.eyedrop_last_layer = tm->num_layers; // flush eyedropper state stbte__ui.eyedrop_last_layer = tm->num_layers; // flush eyedropper state
@ -2262,22 +2372,20 @@ static void stbte__toolbar(stbte_tilemap *tm, int x0, int y0, int w, int h)
} }
} }
x += 13; x += 13;
if (i+1 == STBTE__tool_undo || i+1 == STBTE__tool_grid)
x += 8;
} }
x += 8; x += 8;
if (stbte__button("cut" , x, y,10, 40, STBTE__ID(STBTE__toolbarB,0), 0)) { if (stbte__button(STBTE__ctoolbar_button, "cut" , x, y,10, 40, STBTE__ID(STBTE__toolbarB,0), 0, !stbte__ui.has_selection)) {
if (stbte__ui.has_selection) if (stbte__ui.has_selection)
stbte__copy_cut(tm, 1); stbte__copy_cut(tm, 1);
} }
x += 42; x += 42;
if (stbte__button("copy" , x, y, 5, 40, STBTE__ID(STBTE__toolbarB,1), 0)) { if (stbte__button(STBTE__ctoolbar_button, "copy" , x, y, 5, 40, STBTE__ID(STBTE__toolbarB,1), 0, !stbte__ui.has_selection)) {
if (stbte__ui.has_selection) if (stbte__ui.has_selection)
stbte__copy_cut(tm, 0); stbte__copy_cut(tm, 0);
} }
x += 42; x += 42;
if (stbte__button("paste", x, y, 0, 40, STBTE__ID(STBTE__toolbarB,2), stbte__ui.pasting)) { if (stbte__button(STBTE__ctoolbar_button, "paste", x, y, 0, 40, STBTE__ID(STBTE__toolbarB,2), stbte__ui.pasting, !stbte__ui.has_copy)) {
if (stbte__ui.has_copy) { if (stbte__ui.has_copy) {
stbte__ui.pasting = 1; stbte__ui.pasting = 1;
stbte__activate(STBTE__ID(STBTE__toolbarB,3)); stbte__activate(STBTE__ID(STBTE__toolbarB,3));
@ -2285,20 +2393,22 @@ static void stbte__toolbar(stbte_tilemap *tm, int x0, int y0, int w, int h)
} }
} }
#define STBTE__TEXTCOLOR(n) stbte__color_table[n][STBTE__text][STBTE__idle]
static int stbte__info_value(char *label, int x, int y, int val, int digits, int id) static int stbte__info_value(char *label, int x, int y, int val, int digits, int id)
{ {
if (stbte__ui.event == STBTE__paint) { if (stbte__ui.event == STBTE__paint) {
int off = 9-stbte__get_char_width(label[0]); int off = 9-stbte__get_char_width(label[0]);
char text[16]; char text[16];
sprintf(text, label, digits, val); sprintf(text, label, digits, val);
stbte__draw_text_core(x+off,y, text, 999, STBTE_COLOR_PANEL_TEXT,1); stbte__draw_text_core(x+off,y, text, 999, STBTE__TEXTCOLOR(STBTE__cpanel),1);
} }
if (id) { if (id) {
x += 9+7*digits+4; x += 9+7*digits+4;
if (stbte__minibutton(x,y, '+', id + (0<<19))) if (stbte__minibutton(STBTE__cmapsize, x,y, '+', STBTE__ID2(id,1,0)))
val += (stbte__ui.shift ? 10 : 1); val += (stbte__ui.shift ? 10 : 1);
x += 9; x += 9;
if (stbte__minibutton(x,y, '-', id + (1<<19))) if (stbte__minibutton(STBTE__cmapsize, x,y, '-', STBTE__ID2(id,2,0)))
val -= (stbte__ui.shift ? 10 : 1); val -= (stbte__ui.shift ? 10 : 1);
if (val < 1) val = 1; else if (val > 4096) val = 4096; if (val < 1) val = 1; else if (val > 4096) val = 4096;
} }
@ -2331,7 +2441,7 @@ static void stbte__info(stbte_tilemap *tm, int x0, int y0, int w, int h)
stbte__info_value(in_region ? "y:%*d" : "y:",x,y, (stbte__ui.hot_id>> 7)&4095, tm->digits, 0); stbte__info_value(in_region ? "y:%*d" : "y:",x,y, (stbte__ui.hot_id>> 7)&4095, tm->digits, 0);
y += 15; y += 15;
x = x0+2; x = x0+2;
stbte__draw_text(x,y,"brush:",40,STBTE_COLOR_PANEL_TEXT); stbte__draw_text(x,y,"brush:",40,STBTE__TEXTCOLOR(STBTE__cpanel));
if (tm->cur_tile >= 0) if (tm->cur_tile >= 0)
STBTE_DRAW_TILE(x+43,y-3,tm->tiles[tm->cur_tile].id,1); STBTE_DRAW_TILE(x+43,y-3,tm->tiles[tm->cur_tile].id,1);
} }
@ -2347,7 +2457,7 @@ static void stbte__layers(stbte_tilemap *tm, int x0, int y0, int w, int h)
y0 += 5; y0 += 5;
if (!tm->has_layer_names) { if (!tm->has_layer_names) {
if (stbte__ui.event == STBTE__paint) { if (stbte__ui.event == STBTE__paint) {
stbte__draw_text(x0,y0, "Layers", w-4, STBTE_COLOR_PANEL_TEXT); stbte__draw_text(x0,y0, "Layers", w-4, STBTE__TEXTCOLOR(STBTE__cpanel));
} }
y0 += 11; y0 += 11;
} }
@ -2361,13 +2471,13 @@ static void stbte__layers(stbte_tilemap *tm, int x0, int y0, int w, int h)
if (i-tm->layer_scroll >= 0 && i-tm->layer_scroll < num_rows) { if (i-tm->layer_scroll >= 0 && i-tm->layer_scroll < num_rows) {
if (str == NULL) if (str == NULL)
sprintf(str=text, "%2d", i+1); sprintf(str=text, "%2d", i+1);
if (stbte__button(str, x0,y,(i+1<10)*2,xoff-2, STBTE__ID(STBTE__layer,i), tm->cur_layer==i)) if (stbte__button(STBTE__clayer_button, str, x0,y,(i+1<10)*2,xoff-2, STBTE__ID(STBTE__layer,i), tm->cur_layer==i,0))
tm->cur_layer = (tm->cur_layer == i ? -1 : i); tm->cur_layer = (tm->cur_layer == i ? -1 : i);
if (stbte__layerbutton(x0+xoff + 0,y+1,'H',STBTE__ID(STBTE__hide,i), tm->layerinfo[i].hidden,disabled,STBTE_COLOR_LAYERMASK_HIDE)) if (stbte__layerbutton(x0+xoff + 0,y+1,'H',STBTE__ID(STBTE__hide,i), tm->layerinfo[i].hidden,disabled,STBTE__clayer_hide))
tm->layerinfo[i].hidden = !tm->layerinfo[i].hidden; tm->layerinfo[i].hidden = !tm->layerinfo[i].hidden;
if (stbte__layerbutton(x0+xoff + 12,y+1,lockedchar[locked],STBTE__ID(STBTE__lock,i), locked!=0,disabled,STBTE_COLOR_LAYERMASK_LOCK)) if (stbte__layerbutton(x0+xoff + 12,y+1,lockedchar[locked],STBTE__ID(STBTE__lock,i), locked!=0,disabled,STBTE__clayer_lock))
tm->layerinfo[i].locked = (locked+1)%3; tm->layerinfo[i].locked = (locked+1)%3;
if (stbte__layerbutton(x0+xoff + 24,y+1,'S',STBTE__ID(STBTE__solo,i), tm->solo_layer==i,0,STBTE_COLOR_LAYERMASK_SOLO)) if (stbte__layerbutton(x0+xoff + 24,y+1,'S',STBTE__ID(STBTE__solo,i), tm->solo_layer==i,0,STBTE__clayer_solo))
tm->solo_layer = (tm->solo_layer == i ? -1 : i); tm->solo_layer = (tm->solo_layer == i ? -1 : i);
y += 15; y += 15;
} }
@ -2427,12 +2537,17 @@ static void stbte__palette_of_tiles(stbte_tilemap *tm, int x0, int y0, int w, in
int i,x,y; int i,x,y;
int num_vis_rows = (h-6) / tm->palette_spacing_y; int num_vis_rows = (h-6) / tm->palette_spacing_y;
int num_columns = (w-2-6) / tm->palette_spacing_x; int num_columns = (w-2-6) / tm->palette_spacing_x;
int num_total_rows = (tm->cur_palette_count + num_columns-1) / num_columns; // ceil() int num_total_rows;
int column,row; int column,row;
int x1 = x0+w, y1=y0+h; int x1 = x0+w, y1=y0+h;
x = x0+2; x = x0+2;
y = y0+6; y = y0+6;
if (num_columns == 0)
return;
num_total_rows = (tm->cur_palette_count + num_columns-1) / num_columns; // ceil()
column = 0; column = 0;
row = -tm->palette_scroll; row = -tm->palette_scroll;
for (i=0; i < tm->num_tiles; ++i) { for (i=0; i < tm->num_tiles; ++i) {
@ -2459,6 +2574,140 @@ static void stbte__palette_of_tiles(stbte_tilemap *tm, int x0, int y0, int w, in
stbte__scrollbar(x1-4, y0+6, y1-2, &tm->palette_scroll, 0, num_total_rows, num_vis_rows, STBTE__ID(STBTE__scrollbar_id, STBTE__palette)); stbte__scrollbar(x1-4, y0+6, y1-2, &tm->palette_scroll, 0, num_total_rows, num_vis_rows, STBTE__ID(STBTE__scrollbar_id, STBTE__palette));
} }
static int stbte__cp_mode, stbte__cp_aspect, stbte__cp_state, stbte__cp_index, stbte__save, stbte__cp_altered, stbte__color_copy;
#ifdef STBTE__COLORPICKER
static void stbte__dump_colorstate(void)
{
int i,j,k;
printf("static int stbte__color_table[STBTE__num_color_modes][STBTE__num_color_aspects][STBTE__num_color_states] =\n");
printf("{\n");
printf(" {\n");
for (k=0; k < STBTE__num_color_modes; ++k) {
for (j=0; j < STBTE__num_color_aspects; ++j) {
printf(" { ");
for (i=0; i < STBTE__num_color_states; ++i) {
printf("0x%06x, ", stbte__color_table[k][j][i]);
}
printf("},\n");
}
if (k+1 < STBTE__num_color_modes)
printf(" }, {\n");
else
printf(" },\n");
}
printf("};\n");
}
static void stbte__slider(int x0, int w, int y, int *value, int id)
{
int x1 = x0+w;
int pos = *value * w / 256;
int over = stbte__hittest(x0,y-2,x1,y+3,id);
switch (stbte__ui.event) {
case STBTE__paint:
stbte__draw_rect(x0,y,x1,y+1, 0x808080);
stbte__draw_rect(x0+pos-1,y-1,x0+pos+2,y+2, 0xffffff);
break;
case STBTE__leftdown:
if (STBTE__IS_HOT(id) && STBTE__INACTIVE())
stbte__activate(id);
// fall through
case STBTE__mousemove:
if (STBTE__IS_ACTIVE(id)) {
int v = (stbte__ui.mx-x0)*256/w;
if (v < 0) v = 0; else if (v > 255) v = 255;
*value = v;
}
break;
case STBTE__leftup:
if (STBTE__IS_ACTIVE(id)) {
stbte__activate(0);
stbte__dump_colorstate();
}
break;
}
}
static void stbte__colorpicker(int x0, int y0, int w, int h)
{
int x1 = x0+w, y1 = y0+h, x,y, i;
x = x0+2; y = y0+6;
y += 5;
x += 8;
{
int color = stbte__color_table[stbte__cp_mode][stbte__cp_aspect][stbte__cp_index];
int rgb[3];
if (stbte__cp_altered && stbte__cp_index == STBTE__idle)
color = stbte__save;
if (stbte__minibutton(STBTE__cmapsize, x1-20,y+ 5, 'C', STBTE__ID2(STBTE__colorpick_id,4,0)))
stbte__color_copy = color;
if (stbte__minibutton(STBTE__cmapsize, x1-20,y+15, 'P', STBTE__ID2(STBTE__colorpick_id,4,1)))
color = stbte__color_copy;
rgb[0] = color >> 16; rgb[1] = (color>>8)&255; rgb[2] = color & 255;
for (i=0; i < 3; ++i) {
stbte__slider(x+8,64, y, rgb+i, STBTE__ID2(STBTE__colorpick_id,3,i));
y += 15;
}
if (stbte__ui.event != STBTE__paint && stbte__ui.event != STBTE__tick)
stbte__color_table[stbte__cp_mode][stbte__cp_aspect][stbte__cp_index] = (rgb[0]<<16)|(rgb[1]<<8)|(rgb[2]);
}
y += 5;
// states
x = x0+2+35;
if (stbte__ui.event == STBTE__paint) {
static char *states[] = { "idle", "over", "down", "down&over", "selected", "selected&over", "disabled" };
stbte__draw_text(x, y+1, states[stbte__cp_index], x1-x-1, 0xffffff);
}
x = x0+24; y += 12;
for (i=3; i >= 0; --i) {
int state = 0 != (stbte__cp_state & (1 << i));
if (stbte__layerbutton(x,y, "OASD"[i], STBTE__ID2(STBTE__colorpick_id, 0,i), state,0, STBTE__clayer_button)) {
stbte__cp_state ^= (1 << i);
stbte__cp_index = stbte__state_to_index[0][0][0][stbte__cp_state];
}
x += 16;
}
x = x0+2; y += 18;
for (i=0; i < 3; ++i) {
static char *labels[] = { "Base", "Edge", "Text" };
if (stbte__button(STBTE__ctoolbar_button, labels[i], x,y,0,36, STBTE__ID2(STBTE__colorpick_id,1,i), stbte__cp_aspect==i,0))
stbte__cp_aspect = i;
x += 40;
}
y += 18;
x = x0+2;
for (i=0; i < STBTE__num_color_modes; ++i) {
if (stbte__button(STBTE__ctoolbar_button, stbte__color_names[i], x, y, 0,80, STBTE__ID2(STBTE__colorpick_id,2,i), stbte__cp_mode == i,0))
stbte__cp_mode = i;
y += 12;
}
// make the currently selected aspect flash, unless we're actively dragging color slider etc
if (stbte__ui.event == STBTE__tick) {
stbte__save = stbte__color_table[stbte__cp_mode][stbte__cp_aspect][STBTE__idle];
if ((stbte__ui.active_id & 127) != STBTE__colorpick_id) {
if ((stbte__ui.ms_time & 2047) < 200) {
stbte__color_table[stbte__cp_mode][stbte__cp_aspect][STBTE__idle] ^= 0x1f1f1f;
stbte__cp_altered = 1;
}
}
}
}
#endif
static void stbte__editor_traverse(stbte_tilemap *tm) static void stbte__editor_traverse(stbte_tilemap *tm)
{ {
@ -2523,15 +2772,14 @@ static void stbte__editor_traverse(stbte_tilemap *tm)
for (i=0; i < STBTE__num_panel; ++i) { for (i=0; i < STBTE__num_panel; ++i) {
stbte__panel *p = &stbte__ui.panel[i]; stbte__panel *p = &stbte__ui.panel[i];
if (stbte__ui.event == STBTE__paint) { if (stbte__ui.event == STBTE__paint) {
stbte__draw_rect (p->x0,p->y0,p->x0+p->width,p->y0+p->height, STBTE_COLOR_PANEL_BACKGROUND); stbte__draw_box(p->x0,p->y0,p->x0+p->width,p->y0+p->height, STBTE__cpanel, STBTE__idle);
stbte__draw_frame(p->x0,p->y0,p->x0+p->width,p->y0+p->height, STBTE_COLOR_PANEL_OUTLINE);
} }
// obscure tilemap data underneath panel // obscure tilemap data underneath panel
stbte__hittest(p->x0,p->y0,p->x0+p->width,p->y0+p->height, STBTE__ID2(STBTE__panel, i, 0)); stbte__hittest(p->x0,p->y0,p->x0+p->width,p->y0+p->height, STBTE__ID2(STBTE__panel, i, 0));
switch (i) { switch (i) {
case STBTE__panel_toolbar: case STBTE__panel_toolbar:
if (stbte__ui.event == STBTE__paint) if (stbte__ui.event == STBTE__paint)
stbte__draw_rect(p->x0,p->y0,p->x0+p->width,p->y0+p->height, STBTE_COLOR_TOOLBAR_BACKGROUND); stbte__draw_rect(p->x0,p->y0,p->x0+p->width,p->y0+p->height, stbte__color_table[STBTE__ctoolbar][STBTE__base][STBTE__idle]);
stbte__toolbar(tm,p->x0,p->y0,p->width,p->height); stbte__toolbar(tm,p->x0,p->y0,p->width,p->height);
break; break;
case STBTE__panel_info: case STBTE__panel_info:
@ -2543,10 +2791,15 @@ static void stbte__editor_traverse(stbte_tilemap *tm)
case STBTE__panel_categories: case STBTE__panel_categories:
stbte__categories(tm,p->x0,p->y0,p->width,p->height); stbte__categories(tm,p->x0,p->y0,p->width,p->height);
break; break;
case STBTE__panel_colorpick:
#ifdef STBTE__COLORPICKER
stbte__colorpicker(p->x0,p->y0,p->width,p->height);
#endif
break;
case STBTE__panel_tiles: case STBTE__panel_tiles:
// erase boundary between categories and tiles if they're on same side // erase boundary between categories and tiles if they're on same side
if (stbte__ui.event == STBTE__paint && p->side == stbte__ui.panel[STBTE__panel_categories].side) if (stbte__ui.event == STBTE__paint && p->side == stbte__ui.panel[STBTE__panel_categories].side)
stbte__draw_rect(p->x0+1,p->y0-1,p->x0+p->width-1,p->y0+1, STBTE_COLOR_PANEL_BACKGROUND); stbte__draw_rect(p->x0+1,p->y0-1,p->x0+p->width-1,p->y0+1, stbte__color_table[STBTE__cpanel][STBTE__base][STBTE__idle]);
stbte__palette_of_tiles(tm,p->x0,p->y0,p->width,p->height); stbte__palette_of_tiles(tm,p->x0,p->y0,p->width,p->height);
break; break;
} }
@ -2554,7 +2807,7 @@ static void stbte__editor_traverse(stbte_tilemap *tm)
for (j=0; j < 2; ++j) { for (j=0; j < 2; ++j) {
int result; int result;
if (i == STBTE__panel_toolbar) continue; if (i == STBTE__panel_toolbar) continue;
result = stbte__microbutton(p->x0+p->width - 1 - 2*4 + 4*j,p->y0+2,3, STBTE__ID2(STBTE__panel, i, j+1), 0x808080,0xc0c0c0, 0); result = stbte__microbutton(p->x0+p->width - 1 - 2*4 + 4*j,p->y0+2,3, STBTE__ID2(STBTE__panel, i, j+1), STBTE__cpanel_sider+j);
if (result) { if (result) {
switch (j) { switch (j) {
case 0: p->side = result > 0 ? STBTE__side_left : STBTE__side_right; break; case 0: p->side = result > 0 ? STBTE__side_left : STBTE__side_right; break;
@ -2577,7 +2830,7 @@ static void stbte__editor_traverse(stbte_tilemap *tm)
width = stbte__ui.left_width , x += stbte__region[i].width + 1; width = stbte__ui.left_width , x += stbte__region[i].width + 1;
else else
width = -stbte__ui.right_width, x -= 6; width = -stbte__ui.right_width, x -= 6;
if (stbte__microbutton_dragger(x, stbte__region[i].y+2, 5, STBTE__ID(STBTE__region,i), 0x206020,0xffffff, 0, &width)) { if (stbte__microbutton_dragger(x, stbte__region[i].y+2, 5, STBTE__ID(STBTE__region,i), &width)) {
// if non-0, it is expanding, so retract it // if non-0, it is expanding, so retract it
if (stbte__region[i].retracted == 0.0) if (stbte__region[i].retracted == 0.0)
stbte__region[i].retracted = 0.01f; stbte__region[i].retracted = 0.01f;
@ -2601,7 +2854,7 @@ static void stbte__editor_traverse(stbte_tilemap *tm)
if (stbte__ui.event == STBTE__paint && stbte__ui.alert_msg) { if (stbte__ui.event == STBTE__paint && stbte__ui.alert_msg) {
int w = stbte__text_width(stbte__ui.alert_msg); int w = stbte__text_width(stbte__ui.alert_msg);
int x = (stbte__ui.x0+stbte__ui.x1)/2; int x = (stbte__ui.x0+stbte__ui.x1)/2;
int y = (stbte__ui.y0+stbte__ui.y1)/2; int y = (stbte__ui.y0+stbte__ui.y1)*5/6;
stbte__draw_rect (x-w/2-4,y-8, x+w/2+4,y+8, 0x604020); stbte__draw_rect (x-w/2-4,y-8, x+w/2+4,y+8, 0x604020);
stbte__draw_frame(x-w/2-4,y-8, x+w/2+4,y+8, 0x906030); stbte__draw_frame(x-w/2-4,y-8, x+w/2+4,y+8, 0x906030);
stbte__draw_text (x-w/2,y-4, stbte__ui.alert_msg, w+1, 0xff8040); stbte__draw_text (x-w/2,y-4, stbte__ui.alert_msg, w+1, 0xff8040);
@ -2614,6 +2867,11 @@ static void stbte__editor_traverse(stbte_tilemap *tm)
stbte__ui.alert_msg = 0; stbte__ui.alert_msg = 0;
} }
} }
if (stbte__ui.event == STBTE__paint) {
stbte__color_table[stbte__cp_mode][stbte__cp_aspect][STBTE__idle] = stbte__save;
stbte__cp_altered = 0;
}
} }
static void stbte__do_event(stbte_tilemap *tm) static void stbte__do_event(stbte_tilemap *tm)