fluxbox/src/LinkedList.cc

355 lines
8.2 KiB
C++
Raw Normal View History

2001-12-11 20:47:02 +00:00
// LinkedList.cc for Blackbox - an X11 Window manager
// Copyright (c) 1997 - 2000 Brad Hughes (bhughes@tcac.net)
//
// 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.
// stupid macros needed to access some functions in version 2 of the GNU C
// library
#ifndef _GNU_SOURCE
#define _GNU_SOURCE
#endif // _GNU_SOURCE
#include "LinkedList.hh"
#ifdef HAVE_CONFIG_H
# include "../config.h"
#endif // HAVE_CONFIG_H
#ifdef HAVE_STDIO_H
# include <stdio.h>
#endif // HAVE_STDIO_H
__llist_iterator::__llist_iterator(__llist *l) {
// initialize the iterator...
list = l;
if (list) {
if (! list->iterators)
list->iterators = new __llist;
list->iterators->insert(this);
}
reset();
}
__llist_iterator::~__llist_iterator(void) {
if (list && list->iterators)
list->iterators->remove(this);
}
void *__llist_iterator::current(void) {
// return the current node data... if any
return ((node) ? node->getData() : 0);
}
void __llist_iterator::reset(void) {
// update the iterator's current node to the first node in the linked list
if (list)
node = list->_first;
}
const int __llist_iterator::set(const int index) {
// set the current node to index
if (list) {
if (index < list->elements && index >= 0 && list->_first) {
node = list->_first;
for (register int i = 0; i < index; i++)
node = node->getNext();
return 1;
}
}
node = (__llist_node *) 0;
return 0;
}
void __llist_iterator::operator++(void) {
// iterate to the next node in the list...
node = ((node) ? node->getNext() : 0);
}
void __llist_iterator::operator++(int) {
// iterate to the next node in the list...
node = ((node) ? node->getNext() : 0);
}
__llist::__llist(void *d) {
// initialize the linked list...
_first = (__llist_node *) 0;
_last = (__llist_node *) 0;
iterators = (__llist *) 0;
elements = 0;
if (d) insert(d);
}
__llist::~__llist(void) {
// remove all the items in the list...
for (register int i = 0, r = elements; i < r; i++)
remove(0);
if (iterators) {
__llist_node *n = iterators->_first;
while (n) {
((__llist_iterator *) n->getData())->list = (__llist *) 0;
((__llist_iterator *) n->getData())->node = (__llist_node *) 0;
n = n->getNext();
}
delete iterators;
}
}
const int __llist::insert(void *d, int index) {
// insert item into linked list at specified index...
if ((! _first) || (! _last)) {
// list is empty... insert the item as the first item, regardless of the
// index given
_first = new __llist_node;
_first->setData(d);
_first->setNext((__llist_node *) 0);
_last = _first;
} else {
if (index == 0) {
// if index is 0... prepend the data on the list
__llist_node *nnode = new __llist_node;
nnode->setData(d);
nnode->setNext(_first);
_first = nnode;
} else if ((index == -1) || (index == elements)) {
// if index is -1... append the data on the list
__llist_node *nnode = new __llist_node;
nnode->setData(d);
nnode->setNext((__llist_node *) 0);
_last->setNext(nnode);
_last = nnode;
} else if (index < elements) {
// otherwise... insert the item at the position specified by index
__llist_node *nnode = new __llist_node, *inode = _first->getNext();
if (! nnode)
return -1;
nnode->setData(d);
for (register int i = 1; i < index; i++)
if (inode)
inode = inode->getNext();
else {
delete nnode;
return -1;
}
if ((! inode) || inode == _last) {
nnode->setNext((__llist_node *) 0);
_last->setNext(nnode);
_last = nnode;
} else {
nnode->setNext(inode->getNext());
inode->setNext(nnode);
}
}
}
return ++elements;
}
const int __llist::remove(void *d) {
// remove list item whose data pointer address matches the pointer address
// given
if ((! _first) || (! _last))
return -1;
else if (_first->getData() == d) {
// remove the first item in the list...
__llist_node *node = _first;
_first = _first->getNext();
if (iterators && iterators->_first) {
__llist_node *n = iterators->_first;
while (n) {
((__llist_iterator *) n->getData())->reset();
n = n->getNext();
}
}
--elements;
delete node;
return 0;
} else {
// iterate through the list and remove the first occurance of the item
// NOTE: we don't validate _first in this assignment, because it is checked
// for validity above...
__llist_node *rnode = _first->getNext(), *prev = _first;
for (register int i = 1; i < elements; i++)
if (rnode)
if (rnode->getData() == d) {
// we found the item... update the previous node and delete the
// now useless rnode...
prev->setNext(rnode->getNext());
if (rnode == _last)
_last = prev;
if (iterators && iterators->_first) {
__llist_node *n = iterators->_first;
while (n) {
((__llist_iterator *) n->getData())->reset();
n = n->getNext();
}
}
--elements;
delete rnode;
return i;
} else {
prev = rnode;
rnode = rnode->getNext();
}
return -1;
}
}
void *__llist::remove(const int index) {
if (index >= elements || index < 0 || (! _first) || (! _last))
return (void *) 0;
// remove list item at specified index within the list
if (index == 0) {
// remove the first item in the list...
__llist_node *node = _first;
void *data_return = _first->getData();
_first = _first->getNext();
if (iterators && iterators->_first) {
__llist_node *n = iterators->_first;
while (n) {
((__llist_iterator *) n->getData())->reset();
n = n->getNext();
}
}
--elements;
delete node;
return data_return;
} else {
__llist_node *rnode = _first->getNext(), *prev = _first;
void *data_return = (void *) 0;
for (register int i = 1; i < index; i++)
if (rnode) {
prev = rnode;
rnode = rnode->getNext();
} else
return (void *) 0;
if (! rnode) return (void *) 0;
prev->setNext(rnode->getNext());
data_return = rnode->getData();
if (rnode == _last)
_last = prev;
if (iterators && iterators->_first) {
__llist_node *n = iterators->_first;
while (n) {
((__llist_iterator *) n->getData())->reset();
n = n->getNext();
}
}
--elements;
data_return = rnode->getData();
delete rnode;
return data_return;
}
return (void *) 0;
}
void *__llist::find(const int index) {
if (index >= elements || index < 0 || (! _first) || (! _last))
return (void *) 0;
if (index == 0) {
// return the first item
return first();
} else if (index == (elements - 1)) {
// return the last item
return last();
} else {
__llist_node *fnode = _first->getNext();
for (register int i = 1; i < index; i++)
if (fnode)
fnode = fnode->getNext();
else
return (void *) 0;
return fnode->getData();
}
return (void *) 0;
}
void *__llist::first(void) {
if (_first)
return _first->getData();
return (void *) 0;
}
void *__llist::last(void) {
if (_last)
return _last->getData();
return (void *) 0;
}