TIC-80-guile/src/map.c

1153 lines
25 KiB
C
Raw Normal View History

2017-09-26 08:59:34 +02:00
// MIT License
// Copyright (c) 2017 Vadim Grigoruk @nesbox // grigoruk@gmail.com
// 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 "map.h"
#include "history.h"
#define SHEET_COLS (TIC_SPRITESHEET_SIZE / TIC_SPRITESIZE)
#define MAP_WIDTH (TIC80_WIDTH)
2017-10-20 09:32:31 +02:00
#define MAP_HEIGHT (TIC80_HEIGHT - TOOLBAR_SIZE)
2017-09-26 08:59:34 +02:00
#define MAP_X (0)
2017-10-20 09:32:31 +02:00
#define MAP_Y (TOOLBAR_SIZE)
2017-09-26 08:59:34 +02:00
#define MAX_SCROLL_X (TIC_MAP_WIDTH * TIC_SPRITESIZE)
#define MAX_SCROLL_Y (TIC_MAP_HEIGHT * TIC_SPRITESIZE)
#define ICON_SIZE 7
#define MIN_SCALE 1
#define MAX_SCALE 4
#define FILL_STACK_SIZE (TIC_MAP_WIDTH*TIC_MAP_HEIGHT)
static void normalizeMap(s32* x, s32* y)
{
while(*x < 0) *x += MAX_SCROLL_X;
while(*y < 0) *y += MAX_SCROLL_Y;
while(*x >= MAX_SCROLL_X) *x -= MAX_SCROLL_X;
while(*y >= MAX_SCROLL_Y) *y -= MAX_SCROLL_Y;
}
2018-02-01 17:34:42 +01:00
static tic_point getTileOffset(Map* map)
2017-09-26 08:59:34 +02:00
{
2018-02-01 17:34:42 +01:00
return (tic_point){(map->sheet.rect.w - 1)*TIC_SPRITESIZE / 2, (map->sheet.rect.h - 1)*TIC_SPRITESIZE / 2};
2017-09-26 08:59:34 +02:00
}
static void getMouseMap(Map* map, s32* x, s32* y)
{
2018-02-01 17:34:42 +01:00
tic_point offset = getTileOffset(map);
2017-09-26 08:59:34 +02:00
s32 mx = getMouseX() + map->scroll.x - offset.x;
s32 my = getMouseY() + map->scroll.y - offset.y;
normalizeMap(&mx, &my);
*x = mx / TIC_SPRITESIZE;
*y = my / TIC_SPRITESIZE;
}
static s32 drawWorldButton(Map* map, s32 x, s32 y)
{
static const u8 WorldIcon[] =
{
0b00000000,
0b00011100,
0b00100010,
0b01001001,
0b00100010,
0b00011100,
0b00000000,
0b00000000,
};
enum{Size = 8};
x -= Size;
2018-02-01 17:34:42 +01:00
tic_rect rect = {x, y, Size, ICON_SIZE};
2017-09-26 08:59:34 +02:00
bool over = false;
if(checkMousePos(&rect))
{
2018-02-06 20:15:56 +01:00
setCursor(tic_cursor_hand);
2017-09-26 08:59:34 +02:00
over = true;
showTooltip("WORLD MAP [tab]");
2018-02-01 16:32:31 +01:00
if(checkMouseClick(&rect, tic_mouse_left))
2017-09-26 08:59:34 +02:00
setStudioMode(TIC_WORLD_MODE);
}
2017-10-20 09:37:30 +02:00
drawBitIcon(x, y, WorldIcon, over ? (tic_color_dark_gray) : (tic_color_light_blue));
2017-09-26 08:59:34 +02:00
return x;
}
static s32 drawGridButton(Map* map, s32 x, s32 y)
{
static const u8 GridIcon[] =
{
0b00000000,
0b00101000,
0b01111100,
0b00101000,
0b01111100,
0b00101000,
0b00000000,
0b00000000,
};
x -= ICON_SIZE;
2018-02-01 17:34:42 +01:00
tic_rect rect = {x, y, ICON_SIZE, ICON_SIZE};
2017-09-26 08:59:34 +02:00
bool over = false;
if(checkMousePos(&rect))
{
2018-02-06 20:15:56 +01:00
setCursor(tic_cursor_hand);
2017-09-26 08:59:34 +02:00
over = true;
showTooltip("SHOW/HIDE GRID [`]");
2018-02-01 16:32:31 +01:00
if(checkMouseClick(&rect, tic_mouse_left))
2017-09-26 08:59:34 +02:00
map->canvas.grid = !map->canvas.grid;
}
2017-10-20 09:37:30 +02:00
drawBitIcon(x, y, GridIcon, map->canvas.grid ? (tic_color_black) : over ? (tic_color_dark_gray) : (tic_color_light_blue));
2017-09-26 08:59:34 +02:00
return x;
}
static bool sheetVisible(Map* map)
{
tic_mem* tic = map->tic;
return tic->api.key(tic, tic_key_shift) || map->sheet.show;
}
2017-09-26 08:59:34 +02:00
static s32 drawSheetButton(Map* map, s32 x, s32 y)
{
static const u8 DownIcon[] =
{
0b00000000,
0b00000000,
0b01111100,
0b00111000,
0b00010000,
0b00000000,
0b00000000,
0b00000000,
};
static const u8 UpIcon[] =
{
0b00000000,
0b00000000,
0b00010000,
0b00111000,
0b01111100,
0b00000000,
0b00000000,
0b00000000,
};
x -= ICON_SIZE;
2018-02-01 17:34:42 +01:00
tic_rect rect = {x, y, ICON_SIZE, ICON_SIZE};
2017-09-26 08:59:34 +02:00
bool over = false;
if(checkMousePos(&rect))
{
2018-02-06 20:15:56 +01:00
setCursor(tic_cursor_hand);
2017-09-26 08:59:34 +02:00
over = true;
showTooltip("SHOW TILES [shift]");
2018-02-01 16:32:31 +01:00
if(checkMouseClick(&rect, tic_mouse_left))
2017-09-26 08:59:34 +02:00
{
map->sheet.show = !map->sheet.show;
}
}
drawBitIcon(rect.x, rect.y, sheetVisible(map) ? UpIcon : DownIcon, over ? (tic_color_dark_gray) : (tic_color_light_blue));
2017-09-26 08:59:34 +02:00
return x;
}
static s32 drawToolButton(Map* map, s32 x, s32 y, const u8* Icon, s32 width, const char* tip, s32 mode)
{
x -= width;
2018-02-01 17:34:42 +01:00
tic_rect rect = {x, y, width, ICON_SIZE};
2017-09-26 08:59:34 +02:00
bool over = false;
if(checkMousePos(&rect))
{
2018-02-06 20:15:56 +01:00
setCursor(tic_cursor_hand);
2017-09-26 08:59:34 +02:00
over = true;
showTooltip(tip);
2018-02-01 16:32:31 +01:00
if(checkMouseClick(&rect, tic_mouse_left))
2017-09-26 08:59:34 +02:00
{
map->mode = mode;
}
}
2017-10-20 09:37:30 +02:00
drawBitIcon(rect.x, rect.y, Icon, map->mode == mode ? (tic_color_black) : over ? (tic_color_dark_gray) : (tic_color_light_blue));
2017-09-26 08:59:34 +02:00
return x;
}
static s32 drawFillButton(Map* map, s32 x, s32 y)
{
static const u8 Icon[] =
{
0b00000000,
0b00001000,
0b00000100,
0b00111110,
0b01011100,
0b01001000,
0b00000000,
0b00000000,
};
enum{Size = 8};
return drawToolButton(map, x, y, Icon, Size, "FILL [4]", MAP_FILL_MODE);
}
static s32 drawSelectButton(Map* map, s32 x, s32 y)
{
static const u8 Icon[] =
{
0b00000000,
0b01010100,
0b00000000,
0b01000100,
0b00000000,
0b01010100,
0b00000000,
0b00000000,
};
return drawToolButton(map, x, y, Icon, ICON_SIZE, "SELECT [3]", MAP_SELECT_MODE);
}
static s32 drawHandButton(Map* map, s32 x, s32 y)
{
static const u8 Icon[] =
{
0b00000000,
0b00011000,
0b00011100,
0b01011100,
0b00111100,
0b00011000,
0b00000000,
0b00000000,
};
return drawToolButton(map, x, y, Icon, ICON_SIZE, "DRAG MAP [2]", MAP_DRAG_MODE);
}
static s32 drawPenButton(Map* map, s32 x, s32 y)
{
static const u8 Icon[] =
{
0b00000000,
0b00001000,
0b00010100,
0b00101000,
0b01010000,
0b01100000,
0b00000000,
0b00000000,
};
return drawToolButton(map, x, y, Icon, ICON_SIZE, "DRAW [1]", MAP_DRAW_MODE);
}
static void drawTileIndex(Map* map, s32 x, s32 y)
{
s32 index = -1;
if(sheetVisible(map))
2017-09-26 08:59:34 +02:00
{
2018-02-01 17:34:42 +01:00
tic_rect rect = {TIC80_WIDTH - TIC_SPRITESHEET_SIZE - 1, TOOLBAR_SIZE, TIC_SPRITESHEET_SIZE, TIC_SPRITESHEET_SIZE};
2017-09-26 08:59:34 +02:00
if(checkMousePos(&rect))
{
s32 mx = getMouseX() - rect.x;
s32 my = getMouseY() - rect.y;
mx /= TIC_SPRITESIZE;
my /= TIC_SPRITESIZE;
index = mx + my * SHEET_COLS;
}
}
else
{
2018-02-01 17:34:42 +01:00
tic_rect rect = {MAP_X, MAP_Y, MAP_WIDTH, MAP_HEIGHT};
2017-09-26 08:59:34 +02:00
if(checkMousePos(&rect))
{
s32 tx = 0, ty = 0;
getMouseMap(map, &tx, &ty);
2017-12-14 15:08:04 +01:00
index = map->tic->api.map_get(map->tic, map->src, tx, ty);
2017-09-26 08:59:34 +02:00
}
}
if(index >= 0)
{
char buf[] = "#999";
sprintf(buf, "#%03i", index);
2017-10-20 09:37:30 +02:00
map->tic->api.text(map->tic, buf, x, y, (tic_color_light_blue));
2017-09-26 08:59:34 +02:00
}
}
static void drawMapToolbar(Map* map, s32 x, s32 y)
{
tic_mem* tic = map->tic;
2017-10-20 09:37:30 +02:00
map->tic->api.rect(map->tic, 0, 0, TIC80_WIDTH, TOOLBAR_SIZE, (tic_color_white));
2017-09-26 08:59:34 +02:00
drawTileIndex(map, TIC80_WIDTH/2 - tic->font.width, y);
2017-09-26 08:59:34 +02:00
x = drawSheetButton(map, TIC80_WIDTH, 0);
x = drawFillButton(map, x, 0);
x = drawSelectButton(map, x, 0);
x = drawHandButton(map, x, 0);
x = drawPenButton(map, x, 0);
x = drawGridButton(map, x - 5, 0);
x = drawWorldButton(map, x, 0);
}
static void drawSheet(Map* map, s32 x, s32 y)
{
if(!sheetVisible(map))return;
2017-09-26 08:59:34 +02:00
2018-02-01 17:34:42 +01:00
tic_rect rect = {x, y, TIC_SPRITESHEET_SIZE, TIC_SPRITESHEET_SIZE};
2017-09-26 08:59:34 +02:00
2017-10-20 09:37:30 +02:00
map->tic->api.rect_border(map->tic, rect.x - 1, rect.y - 1, rect.w + 2, rect.h + 2, (tic_color_white));
2017-12-08 11:54:01 +01:00
}
static void drawSheetOvr(Map* map, s32 x, s32 y)
{
if(!sheetVisible(map))return;
2017-12-08 11:54:01 +01:00
2018-02-01 17:34:42 +01:00
tic_rect rect = {x, y, TIC_SPRITESHEET_SIZE, TIC_SPRITESHEET_SIZE};
2017-09-26 08:59:34 +02:00
if(checkMousePos(&rect))
{
2018-02-06 20:15:56 +01:00
setCursor(tic_cursor_hand);
2017-09-26 08:59:34 +02:00
2018-02-01 16:32:31 +01:00
if(checkMouseDown(&rect, tic_mouse_left))
2017-09-26 08:59:34 +02:00
{
s32 mx = getMouseX() - rect.x;
s32 my = getMouseY() - rect.y;
mx /= TIC_SPRITESIZE;
my /= TIC_SPRITESIZE;
if(map->sheet.drag)
{
2018-02-06 20:15:56 +01:00
s32 rl = MIN(mx, map->sheet.start.x);
s32 rt = MIN(my, map->sheet.start.y);
s32 rr = MAX(mx, map->sheet.start.x);
s32 rb = MAX(my, map->sheet.start.y);
2017-09-26 08:59:34 +02:00
2018-02-01 17:34:42 +01:00
map->sheet.rect = (tic_rect){rl, rt, rr-rl+1, rb-rt+1};
2017-09-26 08:59:34 +02:00
map->mode = MAP_DRAW_MODE;
}
else
{
map->sheet.drag = true;
2018-02-01 17:34:42 +01:00
map->sheet.start = (tic_point){mx, my};
2017-09-26 08:59:34 +02:00
}
}
else
{
if(map->sheet.drag)
map->sheet.show = false;
map->sheet.drag = false;
}
}
for(s32 j = 0, index = 0; j < rect.h; j += TIC_SPRITESIZE)
for(s32 i = 0; i < rect.w; i += TIC_SPRITESIZE, index++)
2017-12-13 21:24:31 +01:00
map->tic->api.sprite(map->tic, getBankTiles(), index, x + i, y + j, NULL, 0);
2017-09-26 08:59:34 +02:00
{
s32 bx = map->sheet.rect.x * TIC_SPRITESIZE - 1 + x;
s32 by = map->sheet.rect.y * TIC_SPRITESIZE - 1 + y;
s32 bw = map->sheet.rect.w * TIC_SPRITESIZE + 2;
s32 bh = map->sheet.rect.h * TIC_SPRITESIZE + 2;
2017-10-20 09:37:30 +02:00
map->tic->api.rect_border(map->tic, bx, by, bw, bh, (tic_color_white));
2017-09-26 08:59:34 +02:00
}
}
static void drawCursorPos(Map* map, s32 x, s32 y)
{
tic_mem* tic = map->tic;
2017-09-26 08:59:34 +02:00
char pos[] = "999:999";
s32 tx = 0, ty = 0;
getMouseMap(map, &tx, &ty);
sprintf(pos, "%03i:%03i", tx, ty);
2017-10-20 09:37:30 +02:00
s32 width = map->tic->api.text(map->tic, pos, TIC80_WIDTH, 0, (tic_color_gray));
2017-09-26 08:59:34 +02:00
s32 px = x + (TIC_SPRITESIZE + 3);
if(px + width >= TIC80_WIDTH) px = x - (width + 2);
s32 py = y - (tic->font.height + 2);
2017-10-20 09:32:31 +02:00
if(py <= TOOLBAR_SIZE) py = y + (TIC_SPRITESIZE + 3);
map->tic->api.rect(map->tic, px - 1, py - 1, width + 1, tic->font.height + 1, (tic_color_white));
2017-10-20 09:37:30 +02:00
map->tic->api.text(map->tic, pos, px, py, (tic_color_light_blue));
2017-09-26 08:59:34 +02:00
}
static void setMapSprite(Map* map, s32 x, s32 y)
{
s32 mx = map->sheet.rect.x;
s32 my = map->sheet.rect.y;
for(s32 j = 0; j < map->sheet.rect.h; j++)
for(s32 i = 0; i < map->sheet.rect.w; i++)
2017-12-14 15:08:04 +01:00
map->tic->api.map_set(map->tic, map->src, (x+i)%TIC_MAP_WIDTH, (y+j)%TIC_MAP_HEIGHT, (mx+i) + (my+j) * SHEET_COLS);
2017-09-26 08:59:34 +02:00
history_add(map->history);
}
static void drawTileCursor(Map* map)
{
if(map->scroll.active)
return;
2018-02-01 17:34:42 +01:00
tic_point offset = getTileOffset(map);
2017-09-26 08:59:34 +02:00
s32 mx = getMouseX() + map->scroll.x - offset.x;
s32 my = getMouseY() + map->scroll.y - offset.y;
mx -= mx % TIC_SPRITESIZE;
my -= my % TIC_SPRITESIZE;
mx += -map->scroll.x;
my += -map->scroll.y;
s32 width = map->sheet.rect.w * TIC_SPRITESIZE + 2;
s32 height = map->sheet.rect.h * TIC_SPRITESIZE + 2;
map->tic->api.rect_border(map->tic, mx - 1, my - 1,
2017-10-20 09:37:30 +02:00
width, height, (tic_color_white));
2017-09-26 08:59:34 +02:00
{
s32 sx = map->sheet.rect.x;
s32 sy = map->sheet.rect.y;
for(s32 j = 0, ty=my; j < map->sheet.rect.h; j++, ty+=TIC_SPRITESIZE)
for(s32 i = 0, tx=mx; i < map->sheet.rect.w; i++, tx+=TIC_SPRITESIZE)
2017-12-13 21:24:31 +01:00
map->tic->api.sprite(map->tic, getBankTiles(), (sx+i) + (sy+j) * SHEET_COLS, tx, ty, NULL, 0);
2017-09-26 08:59:34 +02:00
}
drawCursorPos(map, mx, my);
}
static void processMouseDrawMode(Map* map)
{
2018-02-01 17:34:42 +01:00
tic_rect rect = {MAP_X, MAP_Y, MAP_WIDTH, MAP_HEIGHT};
2017-09-26 08:59:34 +02:00
2018-02-06 20:15:56 +01:00
setCursor(tic_cursor_hand);
2017-09-26 08:59:34 +02:00
drawTileCursor(map);
2018-02-01 16:32:31 +01:00
if(checkMouseDown(&rect, tic_mouse_left))
2017-09-26 08:59:34 +02:00
{
s32 tx = 0, ty = 0;
getMouseMap(map, &tx, &ty);
if(map->canvas.draw)
{
s32 w = tx - map->canvas.start.x;
s32 h = ty - map->canvas.start.y;
if(w % map->sheet.rect.w == 0 && h % map->sheet.rect.h == 0)
setMapSprite(map, tx, ty);
}
else
{
map->canvas.draw = true;
2018-02-01 17:34:42 +01:00
map->canvas.start = (tic_point){tx, ty};
2017-09-26 08:59:34 +02:00
}
}
else
{
map->canvas.draw = false;
}
2018-02-06 20:15:56 +01:00
if(checkMouseDown(&rect, tic_mouse_middle))
2017-09-26 08:59:34 +02:00
{
s32 tx = 0, ty = 0;
getMouseMap(map, &tx, &ty);
2017-12-14 15:08:04 +01:00
s32 index = map->tic->api.map_get(map->tic, map->src, tx, ty);
2017-09-26 08:59:34 +02:00
2018-02-01 17:34:42 +01:00
map->sheet.rect = (tic_rect){index % SHEET_COLS, index / SHEET_COLS, 1, 1};
2017-09-26 08:59:34 +02:00
}
}
static void processScrolling(Map* map, bool pressed)
{
2018-02-01 17:34:42 +01:00
tic_rect rect = {MAP_X, MAP_Y, MAP_WIDTH, MAP_HEIGHT};
2017-09-26 08:59:34 +02:00
if(map->scroll.active)
{
if(pressed)
{
map->scroll.x = map->scroll.start.x - getMouseX();
map->scroll.y = map->scroll.start.y - getMouseY();
normalizeMap(&map->scroll.x, &map->scroll.y);
2018-02-06 20:15:56 +01:00
setCursor(tic_cursor_hand);
2017-09-26 08:59:34 +02:00
}
else map->scroll.active = false;
}
else if(checkMousePos(&rect))
{
if(pressed)
{
map->scroll.active = true;
map->scroll.start.x = getMouseX() + map->scroll.x;
map->scroll.start.y = getMouseY() + map->scroll.y;
}
}
}
static void processMouseDragMode(Map* map)
{
2018-02-01 17:34:42 +01:00
tic_rect rect = {MAP_X, MAP_Y, MAP_WIDTH, MAP_HEIGHT};
2017-09-26 08:59:34 +02:00
2018-02-01 16:32:31 +01:00
processScrolling(map, checkMouseDown(&rect, tic_mouse_left) ||
checkMouseDown(&rect, tic_mouse_right));
2017-09-26 08:59:34 +02:00
}
static void resetSelection(Map* map)
{
2018-02-01 17:34:42 +01:00
map->select.rect = (tic_rect){0,0,0,0};
2017-09-26 08:59:34 +02:00
}
static void drawSelectionRect(Map* map, s32 x, s32 y, s32 w, s32 h)
{
enum{Step = 3};
2017-10-20 09:37:30 +02:00
u8 color = (tic_color_white);
2017-09-26 08:59:34 +02:00
s32 index = map->tickCounter / 10;
2017-10-20 09:31:38 +02:00
for(s32 i = x; i < (x+w); i++) {map->tic->api.pixel(map->tic, i, y, index++ % Step ? color : 0);} index++;
for(s32 i = y; i < (y+h); i++) {map->tic->api.pixel(map->tic, x + w-1, i, index++ % Step ? color : 0);} index++;
for(s32 i = (x+w-1); i >= x; i--) {map->tic->api.pixel(map->tic, i, y + h-1, index++ % Step ? color : 0);} index++;
for(s32 i = (y+h-1); i >= y; i--) {map->tic->api.pixel(map->tic, x, i, index++ % Step ? color : 0);}
2017-09-26 08:59:34 +02:00
}
static void drawPasteData(Map* map)
{
s32 w = map->paste[0];
s32 h = map->paste[1];
u8* data = map->paste + 2;
s32 mx = getMouseX() + map->scroll.x - (w - 1)*TIC_SPRITESIZE / 2;
s32 my = getMouseY() + map->scroll.y - (h - 1)*TIC_SPRITESIZE / 2;
2018-02-01 17:34:42 +01:00
tic_rect rect = {MAP_X, MAP_Y, MAP_WIDTH, MAP_HEIGHT};
2017-09-26 08:59:34 +02:00
2018-02-01 16:32:31 +01:00
if(checkMouseClick(&rect, tic_mouse_left))
2017-09-26 08:59:34 +02:00
{
normalizeMap(&mx, &my);
mx /= TIC_SPRITESIZE;
my /= TIC_SPRITESIZE;
for(s32 j = 0; j < h; j++)
for(s32 i = 0; i < w; i++)
2017-12-14 15:08:04 +01:00
map->tic->api.map_set(map->tic, map->src, (mx+i)%TIC_MAP_WIDTH, (my+j)%TIC_MAP_HEIGHT, data[i + j * w]);
2017-09-26 08:59:34 +02:00
history_add(map->history);
2018-02-06 20:15:56 +01:00
free(map->paste);
2017-09-26 08:59:34 +02:00
map->paste = NULL;
}
else
{
mx -= mx % TIC_SPRITESIZE;
my -= my % TIC_SPRITESIZE;
mx += -map->scroll.x;
my += -map->scroll.y;
drawSelectionRect(map, mx-1, my-1, w*TIC_SPRITESIZE+2, h*TIC_SPRITESIZE+2);
for(s32 j = 0; j < h; j++)
for(s32 i = 0; i < w; i++)
2017-12-13 21:24:31 +01:00
map->tic->api.sprite(map->tic, getBankTiles(), data[i + j * w], mx + i*TIC_SPRITESIZE, my + j*TIC_SPRITESIZE, NULL, 0);
2017-09-26 08:59:34 +02:00
}
}
static void normalizeMapRect(s32* x, s32* y)
{
while(*x < 0) *x += TIC_MAP_WIDTH;
while(*y < 0) *y += TIC_MAP_HEIGHT;
while(*x >= TIC_MAP_WIDTH) *x -= TIC_MAP_WIDTH;
while(*y >= TIC_MAP_HEIGHT) *y -= TIC_MAP_HEIGHT;
}
static void processMouseSelectMode(Map* map)
{
2018-02-01 17:34:42 +01:00
tic_rect rect = {MAP_X, MAP_Y, MAP_WIDTH, MAP_HEIGHT};
2017-09-26 08:59:34 +02:00
if(checkMousePos(&rect))
{
if(map->paste)
drawPasteData(map);
else
{
2018-02-01 16:32:31 +01:00
if(checkMouseDown(&rect, tic_mouse_left))
2017-09-26 08:59:34 +02:00
{
s32 mx = getMouseX() + map->scroll.x;
s32 my = getMouseY() + map->scroll.y;
mx /= TIC_SPRITESIZE;
my /= TIC_SPRITESIZE;
normalizeMapRect(&mx, &my);
if(map->select.drag)
{
2018-02-06 20:15:56 +01:00
s32 rl = MIN(mx, map->select.start.x);
s32 rt = MIN(my, map->select.start.y);
s32 rr = MAX(mx, map->select.start.x);
s32 rb = MAX(my, map->select.start.y);
2017-09-26 08:59:34 +02:00
2018-02-01 17:34:42 +01:00
map->select.rect = (tic_rect){rl, rt, rr - rl + 1, rb - rt + 1};
2017-09-26 08:59:34 +02:00
}
else
{
map->select.drag = true;
2018-02-01 17:34:42 +01:00
map->select.start = (tic_point){mx, my};
map->select.rect = (tic_rect){map->select.start.x, map->select.start.y, 1, 1};
2017-09-26 08:59:34 +02:00
}
}
else if(map->select.drag)
{
map->select.drag = false;
if(map->select.rect.w <= 1 && map->select.rect.h <= 1)
resetSelection(map);
}
}
}
}
typedef struct
{
2018-02-01 17:34:42 +01:00
tic_point* data;
tic_point* head;
2017-09-26 08:59:34 +02:00
} FillStack;
static bool push(FillStack* stack, s32 x, s32 y)
{
if(stack->head == NULL)
{
stack->head = stack->data;
stack->head->x = x;
stack->head->y = y;
return true;
}
if(stack->head < (stack->data + FILL_STACK_SIZE-1))
{
stack->head++;
stack->head->x = x;
stack->head->y = y;
return true;
}
return false;
}
static bool pop(FillStack* stack, s32* x, s32* y)
{
if(stack->head > stack->data)
{
*x = stack->head->x;
*y = stack->head->y;
stack->head--;
return true;
}
if(stack->head == stack->data)
{
*x = stack->head->x;
*y = stack->head->y;
stack->head = NULL;
return true;
}
return false;
}
static void fillMap(Map* map, s32 x, s32 y, u8 tile)
{
if(tile == (map->sheet.rect.x + map->sheet.rect.y * SHEET_COLS)) return;
static FillStack stack = {NULL, NULL};
if(!stack.data)
2018-02-06 20:15:56 +01:00
stack.data = (tic_point*)malloc(FILL_STACK_SIZE * sizeof(tic_point));
2017-09-26 08:59:34 +02:00
stack.head = NULL;
static const s32 dx[4] = {0, 1, 0, -1};
static const s32 dy[4] = {-1, 0, 1, 0};
if(!push(&stack, x, y)) return;
s32 mx = map->sheet.rect.x;
s32 my = map->sheet.rect.y;
struct
{
s32 l;
s32 t;
s32 r;
s32 b;
}clip = { 0, 0, TIC_MAP_WIDTH, TIC_MAP_HEIGHT };
if (map->select.rect.w > 0 && map->select.rect.h > 0)
{
clip.l = map->select.rect.x;
clip.t = map->select.rect.y;
clip.r = map->select.rect.x + map->select.rect.w;
clip.b = map->select.rect.y + map->select.rect.h;
}
while(pop(&stack, &x, &y))
{
for(s32 j = 0; j < map->sheet.rect.h; j++)
for(s32 i = 0; i < map->sheet.rect.w; i++)
2017-12-14 15:08:04 +01:00
map->tic->api.map_set(map->tic, map->src, x+i, y+j, (mx+i) + (my+j) * SHEET_COLS);
2017-09-26 08:59:34 +02:00
for(s32 i = 0; i < COUNT_OF(dx); i++)
{
s32 nx = x + dx[i]*map->sheet.rect.w;
s32 ny = y + dy[i]*map->sheet.rect.h;
if(nx >= clip.l && nx < clip.r && ny >= clip.t && ny < clip.b)
{
bool match = true;
for(s32 j = 0; j < map->sheet.rect.h; j++)
for(s32 i = 0; i < map->sheet.rect.w; i++)
2017-12-14 15:08:04 +01:00
if(map->tic->api.map_get(map->tic, map->src, nx+i, ny+j) != tile)
2017-09-26 08:59:34 +02:00
match = false;
if(match)
{
if(!push(&stack, nx, ny)) return;
}
}
}
}
}
static void processMouseFillMode(Map* map)
{
2018-02-01 17:34:42 +01:00
tic_rect rect = {MAP_X, MAP_Y, MAP_WIDTH, MAP_HEIGHT};
2017-09-26 08:59:34 +02:00
2018-02-06 20:15:56 +01:00
setCursor(tic_cursor_hand);
2017-09-26 08:59:34 +02:00
drawTileCursor(map);
2018-02-01 16:32:31 +01:00
if(checkMouseClick(&rect, tic_mouse_left))
2017-09-26 08:59:34 +02:00
{
s32 tx = 0, ty = 0;
getMouseMap(map, &tx, &ty);
2017-12-14 15:08:04 +01:00
fillMap(map, tx, ty, map->tic->api.map_get(map->tic, map->src, tx, ty));
2017-09-26 08:59:34 +02:00
history_add(map->history);
}
}
static void drawSelection(Map* map)
{
2018-02-01 17:34:42 +01:00
tic_rect* sel = &map->select.rect;
2017-09-26 08:59:34 +02:00
if(sel->w > 0 && sel->h > 0)
{
s32 x = sel->x * TIC_SPRITESIZE - map->scroll.x;
s32 y = sel->y * TIC_SPRITESIZE - map->scroll.y;
s32 w = sel->w * TIC_SPRITESIZE;
s32 h = sel->h * TIC_SPRITESIZE;
while(x+w<0)x+=MAX_SCROLL_X;
while(y+h<0)y+=MAX_SCROLL_Y;
while(x+w>=MAX_SCROLL_X)x-=MAX_SCROLL_X;
while(y+h>=MAX_SCROLL_Y)y-=MAX_SCROLL_Y;
drawSelectionRect(map, x-1, y-1, w+2, h+2);
}
}
static void drawGrid(Map* map)
{
2017-12-08 11:54:01 +01:00
tic_mem* tic = map->tic;
2017-09-26 08:59:34 +02:00
s32 scrollX = map->scroll.x % TIC_SPRITESIZE;
s32 scrollY = map->scroll.y % TIC_SPRITESIZE;
for(s32 j = -scrollY; j <= TIC80_HEIGHT-scrollY; j += TIC_SPRITESIZE)
{
if(j >= 0 && j < TIC80_HEIGHT)
for(s32 i = 0; i < TIC80_WIDTH; i++)
2017-12-08 11:54:01 +01:00
{
u8 color = tic->api.get_pixel(tic, i, j);
tic->api.pixel(tic, i, j, (color+1)%TIC_PALETTE_SIZE);
2017-09-26 08:59:34 +02:00
}
}
for(s32 j = -scrollX; j <= TIC80_WIDTH-scrollX; j += TIC_SPRITESIZE)
{
if(j >= 0 && j < TIC80_WIDTH)
for(s32 i = 0; i < TIC80_HEIGHT; i++)
{
if((i+scrollY) % TIC_SPRITESIZE)
{
2017-12-08 11:54:01 +01:00
u8 color = tic->api.get_pixel(tic, j, i);
tic->api.pixel(tic, j, i, (color+1)%TIC_PALETTE_SIZE);
2017-09-26 08:59:34 +02:00
}
}
}
}
2017-12-08 11:54:01 +01:00
static void drawMapOvr(Map* map)
2017-09-26 08:59:34 +02:00
{
2018-02-01 17:34:42 +01:00
tic_rect rect = {MAP_X, MAP_Y, MAP_WIDTH, MAP_HEIGHT};
2017-09-26 08:59:34 +02:00
s32 scrollX = map->scroll.x % TIC_SPRITESIZE;
s32 scrollY = map->scroll.y % TIC_SPRITESIZE;
2018-01-24 17:09:22 +01:00
tic_mem* tic = map->tic;
tic->api.map(tic, map->src, getBankTiles(), map->scroll.x / TIC_SPRITESIZE, map->scroll.y / TIC_SPRITESIZE,
2017-12-08 11:54:01 +01:00
TIC_MAP_SCREEN_WIDTH + 1, TIC_MAP_SCREEN_HEIGHT + 1, -scrollX, -scrollY, -1, 1);
2017-09-26 08:59:34 +02:00
if(map->canvas.grid || map->scroll.active)
drawGrid(map);
{
s32 screenScrollX = map->scroll.x % TIC80_WIDTH;
s32 screenScrollY = map->scroll.y % TIC80_HEIGHT;
2018-01-24 17:09:22 +01:00
tic->api.line(tic, 0, TIC80_HEIGHT - screenScrollY, TIC80_WIDTH, TIC80_HEIGHT - screenScrollY, (tic_color_gray));
tic->api.line(tic, TIC80_WIDTH - screenScrollX, 0, TIC80_WIDTH - screenScrollX, TIC80_HEIGHT, (tic_color_gray));
2017-09-26 08:59:34 +02:00
}
if(!sheetVisible(map) && checkMousePos(&rect))
2017-09-26 08:59:34 +02:00
{
2018-01-24 17:09:22 +01:00
if(tic->api.key(tic, tic_key_space))
2017-09-26 08:59:34 +02:00
{
2018-02-01 16:32:31 +01:00
processScrolling(map, checkMouseDown(&rect, tic_mouse_left) || checkMouseDown(&rect, tic_mouse_right));
2017-09-26 08:59:34 +02:00
}
else
{
static void(*const Handlers[])(Map*) = {processMouseDrawMode, processMouseDragMode, processMouseSelectMode, processMouseFillMode};
Handlers[map->mode](map);
if(map->mode != MAP_DRAG_MODE)
2018-02-01 16:32:31 +01:00
processScrolling(map, checkMouseDown(&rect, tic_mouse_right));
2017-09-26 08:59:34 +02:00
}
}
drawSelection(map);
}
static void undo(Map* map)
{
history_undo(map->history);
}
static void redo(Map* map)
{
history_redo(map->history);
}
static void copySelectionToClipboard(Map* map)
{
2018-02-01 17:34:42 +01:00
tic_rect* sel = &map->select.rect;
2017-09-26 08:59:34 +02:00
if(sel->w > 0 && sel->h > 0)
{
s32 size = sel->w * sel->h + 2;
2018-02-06 20:15:56 +01:00
u8* buffer = malloc(size);
2017-09-26 08:59:34 +02:00
if(buffer)
{
buffer[0] = sel->w;
buffer[1] = sel->h;
u8* ptr = buffer + 2;
for(s32 j = sel->y; j < sel->y+sel->h; j++)
for(s32 i = sel->x; i < sel->x+sel->w; i++)
{
s32 x = i, y = j;
normalizeMapRect(&x, &y);
s32 index = x + y * TIC_MAP_WIDTH;
2017-12-14 15:08:04 +01:00
*ptr++ = map->src->data[index];
2017-09-26 08:59:34 +02:00
}
toClipboard(buffer, size, true);
2018-02-06 20:15:56 +01:00
free(buffer);
2017-09-26 08:59:34 +02:00
}
}
}
static void copyToClipboard(Map* map)
{
copySelectionToClipboard(map);
resetSelection(map);
}
static void deleteSelection(Map* map)
{
2018-02-01 17:34:42 +01:00
tic_rect* sel = &map->select.rect;
2017-09-26 08:59:34 +02:00
if(sel->w > 0 && sel->h > 0)
{
for(s32 j = sel->y; j < sel->y+sel->h; j++)
for(s32 i = sel->x; i < sel->x+sel->w; i++)
{
s32 x = i, y = j;
normalizeMapRect(&x, &y);
s32 index = x + y * TIC_MAP_WIDTH;
2017-12-14 15:08:04 +01:00
map->src->data[index] = 0;
2017-09-26 08:59:34 +02:00
}
history_add(map->history);
}
}
static void cutToClipboard(Map* map)
{
copySelectionToClipboard(map);
deleteSelection(map);
resetSelection(map);
}
static void copyFromClipboard(Map* map)
{
2018-02-14 12:52:10 +01:00
if(getSystem()->hasClipboardText())
2017-09-26 08:59:34 +02:00
{
2018-02-14 12:52:10 +01:00
char* clipboard = getSystem()->getClipboardText();
2017-09-26 08:59:34 +02:00
if(clipboard)
{
s32 size = strlen(clipboard)/2;
if(size > 2)
{
2018-02-06 20:15:56 +01:00
u8* data = malloc(size);
2017-09-26 08:59:34 +02:00
2017-11-11 12:44:41 +01:00
str2buf(clipboard, strlen(clipboard), data, true);
2017-09-26 08:59:34 +02:00
if(data[0] * data[1] == size - 2)
{
map->paste = data;
map->mode = MAP_SELECT_MODE;
}
2018-02-06 20:15:56 +01:00
else free(data);
2017-09-26 08:59:34 +02:00
}
getSystem()->freeClipboardText(clipboard);
2017-09-26 08:59:34 +02:00
}
}
}
2018-02-12 17:49:07 +01:00
static void processKeyboard(Map* map)
{
tic_mem* tic = map->tic;
2018-02-15 10:43:00 +01:00
if(tic->ram.input.keyboard.data == 0) return;
2018-02-12 17:49:07 +01:00
bool ctrl = tic->api.key(tic, tic_key_ctrl);
switch(getClipboardEvent())
{
case TIC_CLIPBOARD_CUT: cutToClipboard(map); break;
case TIC_CLIPBOARD_COPY: copyToClipboard(map); break;
case TIC_CLIPBOARD_PASTE: copyFromClipboard(map); break;
default: break;
}
2017-09-26 08:59:34 +02:00
2018-02-12 17:49:07 +01:00
if(ctrl)
{
2018-02-13 11:29:33 +01:00
if(keyWasPressed(tic_key_z)) undo(map);
else if(keyWasPressed(tic_key_y)) redo(map);
2018-02-12 17:49:07 +01:00
}
else
{
2018-02-13 11:29:33 +01:00
if(keyWasPressed(tic_key_tab)) setStudioMode(TIC_WORLD_MODE);
else if(keyWasPressed(tic_key_1)) map->mode = MAP_DRAW_MODE;
else if(keyWasPressed(tic_key_2)) map->mode = MAP_DRAG_MODE;
else if(keyWasPressed(tic_key_3)) map->mode = MAP_SELECT_MODE;
else if(keyWasPressed(tic_key_4)) map->mode = MAP_FILL_MODE;
else if(keyWasPressed(tic_key_delete)) deleteSelection(map);
else if(keyWasPressed(tic_key_grave)) map->canvas.grid = !map->canvas.grid;
2018-02-12 17:49:07 +01:00
}
enum{Step = 1};
if(tic->api.key(tic, tic_key_up)) map->scroll.y -= Step;
if(tic->api.key(tic, tic_key_down)) map->scroll.y += Step;
if(tic->api.key(tic, tic_key_left)) map->scroll.x -= Step;
if(tic->api.key(tic, tic_key_right)) map->scroll.x += Step;
static const tic_key Keycodes[] = {tic_key_up, tic_key_down, tic_key_left, tic_key_right};
for(s32 i = 0; i < COUNT_OF(Keycodes); i++)
if(tic->api.key(tic, Keycodes[i]))
{
normalizeMap(&map->scroll.x, &map->scroll.y);
break;
}
}
2017-09-26 08:59:34 +02:00
static void tick(Map* map)
{
tic_mem* tic = map->tic;
2017-09-26 08:59:34 +02:00
map->tickCounter++;
processKeyboard(map);
map->tic->api.clear(map->tic, TIC_COLOR_BG);
2017-10-20 09:32:31 +02:00
drawSheet(map, TIC80_WIDTH - TIC_SPRITESHEET_SIZE - 1, TOOLBAR_SIZE);
drawMapToolbar(map, TIC80_WIDTH - 9*tic->font.width, 1);
2017-09-26 08:59:34 +02:00
drawToolbar(map->tic, TIC_COLOR_BG, false);
}
static void onStudioEvent(Map* map, StudioEvent event)
{
switch(event)
{
case TIC_TOOLBAR_CUT: cutToClipboard(map); break;
case TIC_TOOLBAR_COPY: copyToClipboard(map); break;
case TIC_TOOLBAR_PASTE: copyFromClipboard(map); break;
case TIC_TOOLBAR_UNDO: undo(map); break;
case TIC_TOOLBAR_REDO: redo(map); break;
default: break;
}
}
static void scanline(tic_mem* tic, s32 row, void* data)
{
if(row == 0)
2018-03-12 07:40:48 +01:00
memcpy(tic->ram.vram.palette.data, getConfig()->cart->bank0.palette.data, sizeof(tic_palette));
}
static void overline(tic_mem* tic, void* data)
2017-10-20 09:16:38 +02:00
{
2017-12-08 11:54:01 +01:00
Map* map = (Map*)data;
tic->api.clip(tic, 0, TOOLBAR_SIZE, TIC80_WIDTH - (sheetVisible(map) ? TIC_SPRITESHEET_SIZE+2 : 0), TIC80_HEIGHT - TOOLBAR_SIZE);
2017-12-08 11:54:01 +01:00
drawMapOvr(map);
tic->api.clip(tic, 0, 0, TIC80_WIDTH, TIC80_HEIGHT);
drawSheetOvr(map, TIC80_WIDTH - TIC_SPRITESHEET_SIZE - 1, TOOLBAR_SIZE);
2017-10-20 09:16:38 +02:00
}
2017-12-14 15:08:04 +01:00
void initMap(Map* map, tic_mem* tic, tic_map* src)
2017-09-26 08:59:34 +02:00
{
if(map->history) history_delete(map->history);
*map = (Map)
{
.tic = tic,
.tick = tick,
2017-12-14 15:08:04 +01:00
.src = src,
2017-09-26 08:59:34 +02:00
.mode = MAP_DRAW_MODE,
.canvas =
{
.grid = true,
.draw = false,
.start = {0, 0},
},
.sheet =
{
.show = false,
.rect = {0, 0, 1, 1},
.start = {0, 0},
.drag = false,
},
.select =
{
.rect = {0, 0, 0, 0},
.start = {0, 0},
.drag = false,
},
.paste = NULL,
.tickCounter = 0,
.scroll =
{
.x = 0,
.y = 0,
.active = false,
.gesture = false,
.start = {0, 0},
},
2017-12-14 15:08:04 +01:00
.history = history_create(src, sizeof(tic_map)),
2017-09-26 08:59:34 +02:00
.event = onStudioEvent,
.overline = overline,
.scanline = scanline,
2017-09-26 08:59:34 +02:00
};
normalizeMap(&map->scroll.x, &map->scroll.y);
}