TIC-80-guile/src/history.c

194 lines
3.8 KiB
C

// 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 "history.h"
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
typedef struct
{
u8* buffer;
u32 start;
u32 end;
} Data;
typedef struct Item Item;
struct Item
{
Item* next;
Item* prev;
Data data;
};
static void list_delete(Item* list, Item* from)
{
Item* it = from;
while(it)
{
Item* next = it->next;
if(it->data.buffer) free(it->data.buffer);
if(it) free(it);
it = next;
}
}
static Item* list_insert(Item* list, Data* data)
{
Item* item = (Item*)malloc(sizeof(Item));
item->next = NULL;
item->prev = NULL;
item->data = *data;
if(list)
{
list_delete(list, list->next);
list->next = item;
item->prev = list;
}
return item;
}
static Item* list_first(Item* list)
{
Item* it = list;
while(it->prev) it = it->prev;
return it;
}
struct History
{
Item* list;
u32 size;
u8* state;
void* data;
};
History* history_create(void* data, u32 size)
{
History* history = (History*)malloc(sizeof(History));
history->data = data;
history->list = NULL;
history->size = size;
history->state = malloc(size);
memcpy(history->state, data, history->size);
// empty diff
history->list = list_insert(history->list, &(Data){NULL, 0, 0});
return history;
}
void history_delete(History* history)
{
if(history)
{
free(history->state);
list_delete(history->list, list_first(history->list));
free(history);
}
}
static void history_diff(History* history, Data* data)
{
for (u32 i = data->start, k = 0; i < data->end; ++i, ++k)
history->state[i] ^= data->buffer[k];
}
static u32 trim_left(u8* data, u32 size)
{
for(u32 i = 0; i < size; i++)
if(data[i]) return i;
return size;
}
static u32 trim_right(u8* data, u32 size)
{
for(u32 i = 0; i < size; i++)
if(data[size - i - 1]) return size - i;
return 0;
}
bool history_add(History* history)
{
if (memcmp(history->state, history->data, history->size) == 0) return false;
history_diff(history, &(Data){history->data, 0, history->size});
{
Data data;
data.start = trim_left(history->state, history->size);
data.end = trim_right(history->state, history->size);
u32 size = data.end - data.start;
data.buffer = malloc(size);
memcpy(data.buffer, (u8*)history->state + data.start, size);
history->list = list_insert(history->list, &data);
}
memcpy(history->state, history->data, history->size);
return true;
}
void history_undo(History* history)
{
if(history->list->prev)
{
history_diff(history, &history->list->data);
history->list = history->list->prev;
}
memcpy(history->data, history->state, history->size);
}
void history_redo(History* history)
{
if(history->list->next)
{
history->list = history->list->next;
history_diff(history, &history->list->data);
}
memcpy(history->data, history->state, history->size);
}