mirror of
https://github.com/0intro/wmii
synced 2025-01-21 17:42:07 +03:00
236 lines
4.2 KiB
C
236 lines
4.2 KiB
C
/*
|
|
* (C)opyright MMIV-MMV Anselm R. Garbe <garbeam at gmail dot com>
|
|
* See LICENSE file for license details.
|
|
*/
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include "cext.h"
|
|
|
|
static CItem zero_item = { 0 };
|
|
|
|
static int comp_ptr(void *p1, void *p2)
|
|
{
|
|
return p1 == p2;
|
|
}
|
|
|
|
static void detach_from_stack(CItem *i)
|
|
{
|
|
/* remove from stack */
|
|
if (i->up)
|
|
i->up->down = i->down;
|
|
if (i->down)
|
|
i->down->up = i->up;
|
|
}
|
|
|
|
static void attach_to_stack(Container *c, CItem *i)
|
|
{
|
|
if (!c->stack)
|
|
c->stack = i;
|
|
else {
|
|
i->down = c->stack;
|
|
c->stack->up = i;
|
|
c->stack = i;
|
|
}
|
|
}
|
|
|
|
void cext_attach_item(Container *c, void *item)
|
|
{
|
|
CItem *i, *new = cext_emalloc(sizeof(CItem));
|
|
*new = zero_item;
|
|
new->item = item;
|
|
for (i = c->list; i && i->next; i = i->next);
|
|
if (!c->list)
|
|
c->list = new;
|
|
else
|
|
i->next = new;
|
|
|
|
attach_to_stack(c, new);
|
|
}
|
|
|
|
void cext_detach_item(Container *c, void *item)
|
|
{
|
|
CItem *i = c->list;
|
|
|
|
if (!i)
|
|
return;
|
|
|
|
/* remove from list */
|
|
if (i->item == item)
|
|
c->list = c->list->next;
|
|
else {
|
|
for (; i->next && i->next->item != item; i = i->next);
|
|
if (!i->next)
|
|
return;
|
|
i->next = i->next->next;
|
|
}
|
|
|
|
detach_from_stack(i);
|
|
free(i);
|
|
}
|
|
|
|
void *cext_find_item(Container *c, void *pattern, int (*comp)(void *pattern, void *item))
|
|
{
|
|
CItem *i;
|
|
for (i = c->list; i && comp(pattern, i->item); i = i->next);
|
|
return i ? i->item : nil;
|
|
}
|
|
|
|
void cext_iterate(Container *c, void (*doit)(void *))
|
|
{
|
|
CItem *i;
|
|
for (i = c->list; i; i = i->next)
|
|
doit(i->item);
|
|
}
|
|
|
|
void cext_top_item(Container *c, void *item)
|
|
{
|
|
CItem *i = cext_find_item(c, item, comp_ptr);
|
|
if (!i)
|
|
return;
|
|
|
|
detach_from_stack(i);
|
|
attach_to_stack(c, i);
|
|
}
|
|
|
|
void *cext_get_top_item(Container *c)
|
|
{
|
|
return c->stack ? c->stack->item : nil;
|
|
}
|
|
|
|
void *cext_get_down_item(Container *c, void *item)
|
|
{
|
|
CItem *i = cext_find_item(c, item, comp_ptr);
|
|
if (!i)
|
|
return nil;
|
|
return i->down ? i->down->item : c->stack->item;
|
|
}
|
|
|
|
void *cext_get_up_item(Container *c, void *item)
|
|
{
|
|
CItem *i = cext_find_item(c, item, comp_ptr);
|
|
CItem *bottom;
|
|
if (!i)
|
|
return nil;
|
|
for (bottom = c->stack; bottom && bottom->down; bottom = bottom->down);
|
|
return i->up ? i->up->item : bottom->item;
|
|
}
|
|
|
|
void *cext_get_item(Container *c, size_t index)
|
|
{
|
|
size_t idx = 0;
|
|
CItem *i;
|
|
|
|
for (i = c->list; i && index != idx; i = i->next)
|
|
idx++;
|
|
|
|
return i ? i->item : nil;
|
|
}
|
|
|
|
int cext_get_item_index(Container *c, void *item)
|
|
{
|
|
int idx = 0;
|
|
CItem *i;
|
|
|
|
for (i = c->list; i && i->item != item; i = i->next)
|
|
idx++;
|
|
|
|
return i ? idx : -1;
|
|
}
|
|
|
|
size_t cext_sizeof(Container *c)
|
|
{
|
|
size_t idx = 0;
|
|
CItem *i;
|
|
|
|
for (i = c->list; i; i = i->next)
|
|
idx++;
|
|
|
|
return idx;
|
|
}
|
|
|
|
/* old obsolete stuff follows */
|
|
|
|
void **attach_item_begin(void **old, void *item, size_t size_item)
|
|
{
|
|
int i, size_old;
|
|
void **result = 0;
|
|
for (size_old = 0; old && old[size_old]; size_old++);
|
|
result = cext_emalloc(size_item * (size_old + 2));
|
|
result[0] = item;
|
|
for (i = 0; old && old[i]; i++)
|
|
result[i + 1] = old[i];
|
|
result[i + 1] = 0;
|
|
if (old)
|
|
free(old);
|
|
return result;
|
|
}
|
|
|
|
void **attach_item_end(void **old, void *item, size_t size_item)
|
|
{
|
|
int i, size_old;
|
|
void **result = 0;
|
|
for (size_old = 0; old && old[size_old]; size_old++);
|
|
result = cext_emalloc(size_item * (size_old + 2));
|
|
for (i = 0; old && old[i]; i++)
|
|
result[i] = old[i];
|
|
result[i++] = item;
|
|
result[i] = 0;
|
|
if (old)
|
|
free(old);
|
|
return result;
|
|
}
|
|
|
|
void **detach_item(void **old, void *item, size_t size_item)
|
|
{
|
|
int size_old, i, j = 0;
|
|
void **result = 0;
|
|
for (size_old = 0; old && old[size_old]; size_old++);
|
|
if (size_old != 1) {
|
|
result = cext_emalloc(size_item * size_old);
|
|
for (i = 0; old[i]; i++)
|
|
if (old[i] != item)
|
|
result[j++] = old[i];
|
|
result[j] = 0;
|
|
}
|
|
if (old)
|
|
free(old);
|
|
return result;
|
|
}
|
|
|
|
int index_item(void **items, void *item)
|
|
{
|
|
int i = 0;
|
|
for (i = 0; items && items[i] && (items[i] != item); i++);
|
|
return items[i] ? i : -1;
|
|
}
|
|
|
|
int count_items(void **items)
|
|
{
|
|
int i;
|
|
for (i = 0; items && items[i]; i++);
|
|
return i;
|
|
}
|
|
|
|
int index_next_item(void **items, void *item)
|
|
{
|
|
int idx = index_item(items, item);
|
|
if (idx == -1)
|
|
return idx;
|
|
if (idx == count_items(items) - 1)
|
|
return 0;
|
|
else
|
|
return idx + 1;
|
|
}
|
|
|
|
int index_prev_item(void **items, void *item)
|
|
{
|
|
int idx = index_item(items, item);
|
|
if (idx == -1)
|
|
return idx;
|
|
if (idx == 0)
|
|
return count_items(items) - 1;
|
|
else
|
|
return idx - 1;
|
|
}
|