| 1 | // MIT License |
| 2 | |
| 3 | // Copyright (c) 2017 Vadim Grigoruk @nesbox // grigoruk@gmail.com |
| 4 | |
| 5 | // Permission is hereby granted, free of charge, to any person obtaining a copy |
| 6 | // of this software and associated documentation files (the "Software"), to deal |
| 7 | // in the Software without restriction, including without limitation the rights |
| 8 | // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell |
| 9 | // copies of the Software, and to permit persons to whom the Software is |
| 10 | // furnished to do so, subject to the following conditions: |
| 11 | |
| 12 | // The above copyright notice and this permission notice shall be included in all |
| 13 | // copies or substantial portions of the Software. |
| 14 | |
| 15 | // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
| 16 | // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
| 17 | // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |
| 18 | // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
| 19 | // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |
| 20 | // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE |
| 21 | // SOFTWARE. |
| 22 | |
| 23 | #include "history.h" |
| 24 | |
| 25 | #include <stdlib.h> |
| 26 | #include <stdio.h> |
| 27 | #include <string.h> |
| 28 | |
| 29 | typedef struct |
| 30 | { |
| 31 | u8* buffer; |
| 32 | u32 start; |
| 33 | u32 end; |
| 34 | } Data; |
| 35 | |
| 36 | typedef struct Item Item; |
| 37 | |
| 38 | struct Item |
| 39 | { |
| 40 | Item* next; |
| 41 | Item* prev; |
| 42 | |
| 43 | Data data; |
| 44 | }; |
| 45 | |
| 46 | static void list_delete(Item* list, Item* from) |
| 47 | { |
| 48 | Item* it = from; |
| 49 | |
| 50 | while(it) |
| 51 | { |
| 52 | Item* next = it->next; |
| 53 | |
| 54 | if(it->data.buffer) free(it->data.buffer); |
| 55 | |
| 56 | free(it); |
| 57 | |
| 58 | it = next; |
| 59 | } |
| 60 | } |
| 61 | |
| 62 | static Item* list_insert(Item* list, Data* data) |
| 63 | { |
| 64 | Item* item = (Item*)malloc(sizeof(Item)); |
| 65 | item->next = NULL; |
| 66 | item->prev = NULL; |
| 67 | item->data = *data; |
| 68 | |
| 69 | if(list) |
| 70 | { |
| 71 | list_delete(list, list->next); |
| 72 | |
| 73 | list->next = item; |
| 74 | item->prev = list; |
| 75 | } |
| 76 | |
| 77 | return item; |
| 78 | } |
| 79 | |
| 80 | static Item* list_first(Item* list) |
| 81 | { |
| 82 | Item* it = list; |
| 83 | while(it->prev) it = it->prev; |
| 84 | |
| 85 | return it; |
| 86 | } |
| 87 | |
| 88 | struct History |
| 89 | { |
| 90 | Item* list; |
| 91 | |
| 92 | u32 size; |
| 93 | u8* state; |
| 94 | |
| 95 | void* data; |
| 96 | }; |
| 97 | |
| 98 | History* history_create(void* data, u32 size) |
| 99 | { |
| 100 | History* history = (History*)malloc(sizeof(History)); |
| 101 | history->data = data; |
| 102 | |
| 103 | history->list = NULL; |
| 104 | history->size = size; |
| 105 | |
| 106 | history->state = malloc(size); |
| 107 | memcpy(history->state, data, history->size); |
| 108 | |
| 109 | // empty diff |
| 110 | history->list = list_insert(history->list, &(Data){NULL, 0, 0}); |
| 111 | |
| 112 | return history; |
| 113 | } |
| 114 | |
| 115 | void history_delete(History* history) |
| 116 | { |
| 117 | if(history) |
| 118 | { |
| 119 | free(history->state); |
| 120 | |
| 121 | list_delete(history->list, list_first(history->list)); |
| 122 | |
| 123 | free(history); |
| 124 | } |
| 125 | } |
| 126 | |
| 127 | static void history_diff(History* history, Data* data) |
| 128 | { |
| 129 | for (u32 i = data->start, k = 0; i < data->end; ++i, ++k) |
| 130 | history->state[i] ^= data->buffer[k]; |
| 131 | } |
| 132 | |
| 133 | static u32 trim_left(u8* data, u32 size) |
| 134 | { |
| 135 | for(u32 i = 0; i < size; i++) |
| 136 | if(data[i]) return i; |
| 137 | |
| 138 | return size; |
| 139 | } |
| 140 | |
| 141 | static u32 trim_right(u8* data, u32 size) |
| 142 | { |
| 143 | for(u32 i = 0; i < size; i++) |
| 144 | if(data[size - i - 1]) return size - i; |
| 145 | |
| 146 | return 0; |
| 147 | } |
| 148 | |
| 149 | bool history_add(History* history) |
| 150 | { |
| 151 | if (memcmp(history->state, history->data, history->size) == 0) return false; |
| 152 | |
| 153 | history_diff(history, &(Data){history->data, 0, history->size}); |
| 154 | |
| 155 | { |
| 156 | Data data; |
| 157 | data.start = trim_left(history->state, history->size); |
| 158 | data.end = trim_right(history->state, history->size); |
| 159 | u32 size = data.end - data.start; |
| 160 | data.buffer = malloc(size); |
| 161 | |
| 162 | memcpy(data.buffer, (u8*)history->state + data.start, size); |
| 163 | |
| 164 | history->list = list_insert(history->list, &data); |
| 165 | } |
| 166 | |
| 167 | memcpy(history->state, history->data, history->size); |
| 168 | |
| 169 | return true; |
| 170 | } |
| 171 | |
| 172 | void history_undo(History* history) |
| 173 | { |
| 174 | if(history->list->prev) |
| 175 | { |
| 176 | history_diff(history, &history->list->data); |
| 177 | |
| 178 | history->list = history->list->prev; |
| 179 | } |
| 180 | |
| 181 | memcpy(history->data, history->state, history->size); |
| 182 | } |
| 183 | |
| 184 | void history_redo(History* history) |
| 185 | { |
| 186 | if(history->list->next) |
| 187 | { |
| 188 | history->list = history->list->next; |
| 189 | |
| 190 | history_diff(history, &history->list->data); |
| 191 | } |
| 192 | |
| 193 | memcpy(history->data, history->state, history->size); |
| 194 | } |
| 195 | |