diff options
Diffstat (limited to 'src/LinkedList.cc')
| -rw-r--r-- | src/LinkedList.cc | 356 |
1 files changed, 356 insertions, 0 deletions
diff --git a/src/LinkedList.cc b/src/LinkedList.cc new file mode 100644 index 00000000..8e066965 --- /dev/null +++ b/src/LinkedList.cc @@ -0,0 +1,356 @@ +// LinkedList.cc for Openbox +// 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.h" + +#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; i < elements; i++) + remove(0); + + if (iterators) { + __llist_node *n = iterators->_first; + + while (n) { + __llist_iterator *p = (__llist_iterator *) n->getData(); + p->list = (__llist *) 0; + p->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... + + __llist_node *nnode = new __llist_node; + if (! nnode) return -1; + + if ((! _first) || (! _last)) { + // list is empty... insert the item as the first item, regardless of the + // index given + _first = nnode; + _first->setData(d); + _first->setNext((__llist_node *) 0); + _last = _first; + } else { + if (index == 0) { + // if index is 0... prepend the data on the list + + nnode->setData(d); + nnode->setNext(_first); + + _first = nnode; + } else if ((index == -1) || (index == elements)) { + // if index is -1... append the data on the list + + 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 *inode = _first; + + for (register int i = 1; i < index; i++) { + if (inode) { + inode = inode->getNext(); + } else { + delete nnode; + return -1; + } + } + + nnode->setData(d); + + 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; + + 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(); + if (index == (elements - 1)) // return the last item + return last(); + + __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(); +} + + +void *__llist::first(void) { + if (_first) + return _first->getData(); + + return (void *) 0; +} + + +void *__llist::last(void) { + if (_last) + return _last->getData(); + + return (void *) 0; +} |
