0
1
mirror of https://github.com/profanity-im/profanity.git synced 2025-07-26 12:14:28 -04:00

More work on history

This commit is contained in:
James Booth 2012-04-30 19:52:56 +01:00
parent f21cb52ab0
commit 4531aebd29
2 changed files with 107 additions and 37 deletions

View File

@ -32,17 +32,26 @@ PHistory p_history_new(unsigned int size)
void p_history_append(PHistory history, char *item) void p_history_append(PHistory history, char *item)
{ {
char *copied = strdup(item); char *copied = "";
if (item != NULL) {
copied = strdup(item);
}
// if not editing history (no session) // if no history yet, just add the item
if (history->items == NULL) {
history->items = g_list_append(history->items, copied);
return;
}
// if not editing history (no session started)
if (history->session == NULL) { if (history->session == NULL) {
// if already at max size // lose first element if hit max size
if (g_list_length(history->items) == history->max_size) { if (g_list_length(history->items) == history->max_size) {
// remove first element // remove first element
GList *first = g_list_first(history->items); GList *first = g_list_first(history->items);
const char *first_item = (char *) first->data; char *first_item = first->data;
history->items = g_list_remove(history->items, first_item); history->items = g_list_remove(history->items, first_item);
} }
@ -52,18 +61,22 @@ void p_history_append(PHistory history, char *item)
// if editing history (session exists with possible changes) // if editing history (session exists with possible changes)
} else { } else {
// if adding a new element, copy the session over the history // if adding a new item, copy the session over the history
if (history->sess_curr == history->sess_new) { if ((history->sess_new != NULL) &&
(history->sess_curr == history->sess_new)) {
_replace_history_with_session(history); _replace_history_with_session(history);
// otherwise, adding edited history item // otherwise, adding edited history item
} else { } else {
// remove the new item, from the session
// discard the new item if there was one
if (history->sess_new != NULL) {
history->session = g_list_reverse(history->session); history->session = g_list_reverse(history->session);
GList *first = g_list_first(history->session); GList *first = g_list_first(history->session);
const char *first_item = (char *) first->data; char *first_item = first->data;
history->session = g_list_remove(history->session, first_item); history->session = g_list_remove(history->session, first_item);
history->session = g_list_reverse(history->session); history->session = g_list_reverse(history->session);
}
// copy sess_curr to the end of the session // copy sess_curr to the end of the session
char *new_data = strdup(history->sess_curr->data); char *new_data = strdup(history->sess_curr->data);
@ -99,45 +112,77 @@ static void _replace_history_with_session(PHistory history)
char * p_history_previous(PHistory history, char *item) char * p_history_previous(PHistory history, char *item)
{ {
// no history
if (history->items == NULL) { if (history->items == NULL) {
return item; return item;
} }
char *copied = "";
if (item != NULL) {
copied = strdup(item);
}
// no session, create one
if (history->session == NULL) { if (history->session == NULL) {
history->session = g_list_copy(history->items); history->session = g_list_copy(history->items);
history->sess_curr = g_list_last(history->session); history->sess_curr = g_list_last(history->session);
history->items_curr = g_list_last(history->items);
// add the new item including empty string
g_list_append(history->session, copied);
history->sess_new = g_list_last(history->session);
char *result = strdup(history->sess_curr->data);
return result;
// session exists
} else { } else {
// update the currently pointed to item with passed data
history->sess_curr->data = copied;
// move to previous
history->sess_curr = g_list_previous(history->sess_curr); history->sess_curr = g_list_previous(history->sess_curr);
} history->items_curr = g_list_previous(history->items_curr);
// set to first if rolled over beginning // set to first if rolled over beginning
if (history->sess_curr == NULL) { if (history->sess_curr == NULL) {
history->sess_curr = g_list_first(history->session); history->sess_curr = g_list_first(history->session);
history->items_curr = g_list_first(history->items);
}
} }
char *curr = history->sess_curr->data; char *result = strdup(history->sess_curr->data);
char *result = malloc((strlen(curr) + 1) * sizeof(char));
strcpy(result, curr);
return result; return result;
} }
char * p_history_next(PHistory history, char *item) char * p_history_next(PHistory history, char *item)
{ {
if (history->session == NULL) {
return NULL; // no history, or no session, return item
} else { if ((history->items == NULL) || (history->session == NULL)) {
history->sess_curr = g_list_next(history->sess_curr); return item;
} }
// set to last if rolled over end char *copied = "";
if (item != NULL) {
copied = strdup(item);
}
// session exists
// update the currently pointed to item with passed data
history->sess_curr->data = copied;
// move to next
history->sess_curr = g_list_next(history->sess_curr);
history->items_curr = g_list_next(history->items_curr);
// set to last if rolled over beginning
if (history->sess_curr == NULL) { if (history->sess_curr == NULL) {
history->sess_curr = g_list_last(history->session); history->sess_curr = g_list_last(history->session);
history->items_curr = NULL;
} }
char *curr = history->sess_curr->data; char *result = strdup(history->sess_curr->data);
char *result = malloc((strlen(curr) + 1) * sizeof(char));
strcpy(result, curr);
return result; return result;
} }

View File

@ -10,12 +10,12 @@ void previous_on_empty_returns_current(void)
assert_string_equals("inp", item); assert_string_equals("inp", item);
} }
void next_on_empty_returns_null(void) void next_on_empty_returns_current(void)
{ {
PHistory history = p_history_new(10); PHistory history = p_history_new(10);
char *item = p_history_next(history, "inp"); char *item = p_history_next(history, "inp");
assert_is_null(item); assert_string_equals("inp", item);
} }
void previous_once_returns_last(void) void previous_once_returns_last(void)
@ -71,7 +71,7 @@ void previous_goes_to_correct_element(void)
assert_string_equals("going", item3); assert_string_equals("going", item3);
} }
void prev_then_next_returns_null(void) void prev_then_next_returns_empty(void)
{ {
PHistory history = p_history_new(10); PHistory history = p_history_new(10);
p_history_append(history, "Hello"); p_history_append(history, "Hello");
@ -79,17 +79,42 @@ void prev_then_next_returns_null(void)
char *item1 = p_history_previous(history, NULL); char *item1 = p_history_previous(history, NULL);
char *item2 = p_history_next(history, item1); char *item2 = p_history_next(history, item1);
assert_is_null(item2); assert_string_equals("", item2);
}
void prev_with_val_then_next_returns_val(void)
{
PHistory history = p_history_new(10);
p_history_append(history, "Hello");
char *item1 = p_history_previous(history, "Oioi");
char *item2 = p_history_next(history, item1);
assert_string_equals("Oioi", item2);
}
void prev_with_val_then_next_twice_returns_val(void)
{
PHistory history = p_history_new(10);
p_history_append(history, "Hello");
char *item1 = p_history_previous(history, "Oioi");
char *item2 = p_history_next(history, item1);
char *item3 = p_history_next(history, item2);
assert_string_equals("Oioi", item3);
} }
void register_prof_history_tests(void) void register_prof_history_tests(void)
{ {
TEST_MODULE("prof_history tests"); TEST_MODULE("prof_history tests");
TEST(previous_on_empty_returns_current); TEST(previous_on_empty_returns_current);
TEST(next_on_empty_returns_null); TEST(next_on_empty_returns_current);
TEST(previous_once_returns_last); TEST(previous_once_returns_last);
TEST(previous_twice_when_one_returns_first); TEST(previous_twice_when_one_returns_first);
TEST(previous_always_stops_at_first); TEST(previous_always_stops_at_first);
TEST(previous_goes_to_correct_element); TEST(previous_goes_to_correct_element);
TEST(prev_then_next_returns_null); TEST(prev_then_next_returns_empty);
TEST(prev_with_val_then_next_returns_val);
TEST(prev_with_val_then_next_twice_returns_val);
} }