diff options
author | John MacFarlane <jgm@berkeley.edu> | 2015-01-10 23:11:07 -0800 |
---|---|---|
committer | John MacFarlane <jgm@berkeley.edu> | 2015-01-10 23:11:07 -0800 |
commit | bfe4d148098717f1603cbe12ba1cf306db09ce3e (patch) | |
tree | 09890f62d62f7aaa093b84f254f4b172ae4da545 /src/iterator.c | |
parent | 4221ca8f33e2f3fc96e34de26418c04747888e76 (diff) | |
parent | fdfbe19d21822d30778a54a808b414dd280a8de6 (diff) |
Merge pull request #277 from nwellnhof/iterator
Rework iterators
Diffstat (limited to 'src/iterator.c')
-rw-r--r-- | src/iterator.c | 123 |
1 files changed, 69 insertions, 54 deletions
diff --git a/src/iterator.c b/src/iterator.c index a3ae415..7f90cc7 100644 --- a/src/iterator.c +++ b/src/iterator.c @@ -1,3 +1,4 @@ +#include <assert.h> #include <stdlib.h> #include "config.h" @@ -5,18 +6,32 @@ #include "cmark.h" #include "iterator.h" +static const int S_leaf_mask = + (1 << CMARK_NODE_HTML) | + (1 << CMARK_NODE_HRULE) | + (1 << CMARK_NODE_CODE_BLOCK) | + (1 << CMARK_NODE_TEXT) | + (1 << CMARK_NODE_SOFTBREAK) | + (1 << CMARK_NODE_LINEBREAK) | + (1 << CMARK_NODE_CODE) | + (1 << CMARK_NODE_INLINE_HTML); + cmark_iter* cmark_iter_new(cmark_node *root) { + if (root == NULL) { + return NULL; + } cmark_iter *iter = (cmark_iter*)malloc(sizeof(cmark_iter)); if (iter == NULL) { return NULL; - } else { - iter->root = root; - iter->current = root; - iter->event_type = CMARK_EVENT_ENTER; - return iter; } + iter->root = root; + iter->cur.ev_type = CMARK_EVENT_NONE; + iter->cur.node = NULL; + iter->next.ev_type = CMARK_EVENT_ENTER; + iter->next.node = root; + return iter; } void @@ -25,72 +40,72 @@ cmark_iter_free(cmark_iter *iter) free(iter); } -cmark_event_type -cmark_iter_next(cmark_iter *iter) +static bool +S_is_leaf(cmark_node *node) { - return iter->event_type; + return (1 << node->type) & S_leaf_mask; } -int S_is_leaf(cmark_node *node) +cmark_event_type +cmark_iter_next(cmark_iter *iter) { - switch (cmark_node_get_type(node)) { - case CMARK_NODE_HTML: - case CMARK_NODE_HRULE: - case CMARK_NODE_CODE_BLOCK: - case CMARK_NODE_TEXT: - case CMARK_NODE_SOFTBREAK: - case CMARK_NODE_LINEBREAK: - case CMARK_NODE_CODE: - case CMARK_NODE_INLINE_HTML: - return 1; - default: - return 0; + cmark_event_type ev_type = iter->next.ev_type; + cmark_node *node = iter->next.node; + + iter->cur.ev_type = ev_type; + iter->cur.node = node; + + if (ev_type == CMARK_EVENT_DONE) { + return ev_type; } + + /* roll forward to next item, setting both fields */ + if (ev_type == CMARK_EVENT_ENTER && !S_is_leaf(node)) { + if (node->first_child == NULL) { + /* stay on this node but exit */ + iter->next.ev_type = CMARK_EVENT_EXIT; + } else { + iter->next.ev_type = CMARK_EVENT_ENTER; + iter->next.node = node->first_child; + } + } else if (node == iter->root) { + /* don't move past root */ + iter->next.ev_type = CMARK_EVENT_DONE; + iter->next.node = NULL; + } else if (node->next) { + iter->next.ev_type = CMARK_EVENT_ENTER; + iter->next.node = node->next; + } else if (node->parent) { + iter->next.ev_type = CMARK_EVENT_EXIT; + iter->next.node = node->parent; + } else { + assert(false); + iter->next.ev_type = CMARK_EVENT_DONE; + iter->next.node = NULL; + } + + return ev_type; } void cmark_iter_reset(cmark_iter *iter, cmark_node *current, cmark_event_type event_type) { - iter->event_type = event_type; - iter->current = current; + iter->next.ev_type = event_type; + iter->next.node = current; + cmark_iter_next(iter); } cmark_node* cmark_iter_get_node(cmark_iter *iter) { - /* we'll return current */ - cmark_node *cur = iter->current; - - if (cur == NULL || iter->event_type == CMARK_EVENT_DONE) { - return NULL; - } - - /* roll forward to next item, setting both fields */ - if (iter->event_type == CMARK_EVENT_ENTER && !S_is_leaf(cur)) { - if (cur->first_child == NULL) { - /* stay on this node but exit */ - iter->event_type = CMARK_EVENT_EXIT; - } else { - iter->current = cur->first_child; - iter->event_type = CMARK_EVENT_ENTER; - } - } else if (cur == iter->root) { - /* don't move past root */ - iter->event_type = CMARK_EVENT_DONE; - iter->current = NULL; - } else if (cur->next) { - iter->event_type = CMARK_EVENT_ENTER; - iter->current = cur->next; - } else if (cur->parent) { - iter->event_type = CMARK_EVENT_EXIT; - iter->current = cur->parent; - } else { - iter->event_type = CMARK_EVENT_DONE; - iter->current = NULL; - } + return iter->cur.node; +} - return cur; +cmark_event_type +cmark_iter_get_event_type(cmark_iter *iter) +{ + return iter->cur.ev_type; } |