/*
   Modern
   microformat-shiv - v1.4.0
   Built: 2016-03-02 10:03 - http://microformat-shiv.com
   Copyright (c) 2016 Glenn Jones
   Licensed MIT
*/


var Microformats; // jshint ignore:line

(function (root, factory) {
    if (typeof define === 'function' && define.amd) {
        define([], factory);
    } else if (typeof exports === 'object') {
        module.exports = factory();
    } else {
        root.Microformats = factory();
  }
}(this, function () {

    var modules = {};


    modules.version = '1.4.0';
    modules.livingStandard = '2015-09-25T12:26:04Z';

    /**
     * constructor
     *
     */
    modules.Parser = function () {
        this.rootPrefix = 'h-';
        this.propertyPrefixes = ['p-', 'dt-', 'u-', 'e-'];
        this.excludeTags = ['br', 'hr'];
    };


    // create objects incase the v1 map modules don't load
    modules.maps = (modules.maps)? modules.maps : {};
    modules.rels = (modules.rels)? modules.rels : {};


    modules.Parser.prototype = {

        init: function() {
            this.rootNode = null;
            this.document = null;
            this.options = {
                'baseUrl': '',
                'filters': [],
                'textFormat': 'whitespacetrimmed',
                'dateFormat': 'auto', // html5 for testing
                'overlappingVersions': false,
                'impliedPropertiesByVersion': true,
                'parseLatLonGeo': false
            };
            this.rootID = 0;
            this.errors = [];
            this.noContentErr = 'No options.node or options.html was provided and no document object could be found.';
        },


        /**
         * internal parse function
         *
         * @param  {Object} options
         * @return {Object}
         */
        get: function(options) {
            var out = this.formatEmpty(),
                data = [],
                rels;

            this.init();
            options = (options)? options : {};
            this.mergeOptions(options);
            this.getDOMContext( options );

            // if we do not have any context create error
            if (!this.rootNode || !this.document) {
                this.errors.push(this.noContentErr);
            } else {

                // only parse h-* microformats if we need to
                // this is added to speed up parsing
                if (this.hasMicroformats(this.rootNode, options)) {
                    this.prepareDOM( options );

                    if (this.options.filters.length > 0) {
                        // parse flat list of items
                        var newRootNode = this.findFilterNodes(this.rootNode, this.options.filters);
                        data = this.walkRoot(newRootNode);
                    } else {
                        // parse whole document from root
                        data = this.walkRoot(this.rootNode);
                    }

                    out.items = data;
                    // don't clear-up DOM if it was cloned
                    if (modules.domUtils.canCloneDocument(this.document) === false) {
                        this.clearUpDom(this.rootNode);
                    }
                }

                // find any rels
                if (this.findRels) {
                    rels = this.findRels(this.rootNode);
                    out.rels = rels.rels;
                    out['rel-urls'] = rels['rel-urls'];
                }

            }

            if (this.errors.length > 0) {
                return this.formatError();
            }
            return out;
        },


        /**
         * parse to get parent microformat of passed node
         *
         * @param  {DOM Node} node
         * @param  {Object} options
         * @return {Object}
         */
        getParent: function(node, options) {
            this.init();
            options = (options)? options : {};

            if (node) {
                return this.getParentTreeWalk(node, options);
            }
            this.errors.push(this.noContentErr);
            return this.formatError();
        },


        /**
         * get the count of microformats
         *
         * @param  {DOM Node} rootNode
         * @return {Int}
         */
        count: function( options ) {
            var out = {},
                items,
                classItems,
                x,
                i;

            this.init();
            options = (options)? options : {};
            this.getDOMContext( options );

            // if we do not have any context create error
            if (!this.rootNode || !this.document) {
                return {'errors': [this.noContentErr]};
            }
            items = this.findRootNodes( this.rootNode, true );
            i = items.length;
            while (i--) {
                classItems = modules.domUtils.getAttributeList(items[i], 'class');
                x = classItems.length;
                while (x--) {
                    // find v2 names
                    if (modules.utils.startWith( classItems[x], 'h-' )) {
                        this.appendCount(classItems[x], 1, out);
                    }
                    // find v1 names
                    for (var key in modules.maps) {
                        // dont double count if v1 and v2 roots are present
                        if (modules.maps[key].root === classItems[x] && classItems.indexOf(key) === -1) {
                            this.appendCount(key, 1, out);
                        }
                    }
                }
            }
            var relCount = this.countRels( this.rootNode );
            if (relCount > 0) {
                out.rels = relCount;
            }

            return out;
        },


        /**
         * does a node have a class that marks it as a microformats root
         *
         * @param  {DOM Node} node
         * @param  {Objecte} options
         * @return {Boolean}
         */
        isMicroformat: function( node, options ) {
            var classes,
                i;

            if (!node) {
                return false;
            }

            // if documemt gets topmost node
            node = modules.domUtils.getTopMostNode( node );

            // look for h-* microformats
            classes = this.getUfClassNames(node);
            if (options && options.filters && modules.utils.isArray(options.filters)) {
                i = options.filters.length;
                while (i--) {
                    if (classes.root.indexOf(options.filters[i]) > -1) {
                        return true;
                    }
                }
                return false;
            }
            return (classes.root.length > 0);
        },


        /**
         * does a node or its children have microformats
         *
         * @param  {DOM Node} node
         * @param  {Objecte} options
         * @return {Boolean}
         */
        hasMicroformats: function( node, options ) {
            var items,
                i;

            if (!node) {
                return false;
            }

            // if browser based documemt get topmost node
            node = modules.domUtils.getTopMostNode( node );

            // returns all microformat roots
            items = this.findRootNodes( node, true );
            if (options && options.filters && modules.utils.isArray(options.filters)) {
                i = items.length;
                while (i--) {
                    if ( this.isMicroformat( items[i], options ) ) {
                        return true;
                    }
                }
                return false;
            }
            return (items.length > 0);
        },


        /**
         * add a new v1 mapping object to parser
         *
         * @param  {Array} maps
         */
        add: function( maps ) {
            maps.forEach(function(map) {
                if (map && map.root && map.name && map.properties) {
                modules.maps[map.name] = JSON.parse(JSON.stringify(map));
                }
            });
        },


        /**
         * internal parse to get parent microformats by walking up the tree
         *
         * @param  {DOM Node} node
         * @param  {Object} options
         * @param  {Int} recursive
         * @return {Object}
         */
        getParentTreeWalk: function (node, options, recursive) {
            options = (options)? options : {};

            // recursive calls
            if (recursive === undefined) {
                if (node.parentNode && node.nodeName !== 'HTML') {
                    return this.getParentTreeWalk(node.parentNode, options, true);
                }
                return this.formatEmpty();
            }
            if (node !== null && node !== undefined && node.parentNode) {
                if (this.isMicroformat( node, options )) {
                    // if we have a match return microformat
                    options.node = node;
                    return this.get( options );
                }
                return this.getParentTreeWalk(node.parentNode, options, true);
            }
            return this.formatEmpty();
        },



        /**
         * configures what are the base DOM objects for parsing
         *
         * @param  {Object} options
         */
        getDOMContext: function( options ) {
            var nodes = modules.domUtils.getDOMContext( options );
            this.rootNode = nodes.rootNode;
            this.document = nodes.document;
        },


        /**
         * prepares DOM before the parse begins
         *
         * @param  {Object} options
         * @return {Boolean}
         */
        prepareDOM: function( options ) {
            var baseTag,
                href;

            // use current document to define baseUrl, try/catch needed for IE10+ error
            try {
                if (!options.baseUrl && this.document && this.document.location) {
                    this.options.baseUrl = this.document.location.href;
                }
            } catch (e) {
                // there is no alt action
            }


            // find base tag to set baseUrl
            baseTag = modules.domUtils.querySelector(this.document, 'base');
            if (baseTag) {
                href = modules.domUtils.getAttribute(baseTag, 'href');
                if (href) {
                    this.options.baseUrl = href;
                }
            }

            // get path to rootNode
            // then clone document
            // then reset the rootNode to its cloned version in a new document
            var path,
                newDocument,
                newRootNode;

            path = modules.domUtils.getNodePath(this.rootNode);
            newDocument = modules.domUtils.cloneDocument(this.document);
            newRootNode = modules.domUtils.getNodeByPath(newDocument, path);

            // check results as early IE fails
            if (newDocument && newRootNode) {
                this.document = newDocument;
                this.rootNode = newRootNode;
            }

            // add includes
            if (this.addIncludes) {
                this.addIncludes( this.document );
            }

            return (this.rootNode && this.document);
        },


        /**
         * returns an empty structure with errors
         *
         *   @return {Object}
         */
        formatError: function() {
            var out = this.formatEmpty();
            out.errors = this.errors;
            return out;
        },


        /**
         * returns an empty structure
         *
         *   @return {Object}
         */
        formatEmpty: function() {
            return {
                'items': [],
                'rels': {},
                'rel-urls': {}
            };
        },


        // find microformats of a given type and return node structures
        findFilterNodes: function(rootNode, filters) {
            if (modules.utils.isString(filters)) {
                filters = [filters];
            }
            var newRootNode = modules.domUtils.createNode('div'),
                items = this.findRootNodes(rootNode, true),
                i = 0,
                x = 0,
                y = 0;

            // add v1 names
            y = filters.length;
            while (y--) {
                if (this.getMapping(filters[y])) {
                    var v1Name = this.getMapping(filters[y]).root;
                    filters.push(v1Name);
                }
            }

            if (items) {
                i = items.length;
                while (x < i) {
                    // append matching nodes into newRootNode
                    y = filters.length;
                    while (y--) {
                        if (modules.domUtils.hasAttributeValue(items[x], 'class', filters[y])) {
                            var clone = modules.domUtils.clone(items[x]);
                            modules.domUtils.appendChild(newRootNode, clone);
                            break;
                        }
                    }
                    x++;
                }
            }

            return newRootNode;
        },


        /**
         * appends data to output object for count
         *
         * @param  {string} name
         * @param  {Int} count
         * @param  {Object}
         */
        appendCount: function(name, count, out) {
            if (out[name]) {
                out[name] = out[name] + count;
            } else {
                out[name] = count;
            }
        },


        /**
         * is the microformats type in the filter list
         *
         * @param  {Object} uf
         * @param  {Array} filters
         * @return {Boolean}
         */
        shouldInclude: function(uf, filters) {
            var i;

            if (modules.utils.isArray(filters) && filters.length > 0) {
                i = filters.length;
                while (i--) {
                    if (uf.type[0] === filters[i]) {
                        return true;
                    }
                }
                return false;
            }
            return true;
        },


        /**
         * finds all microformat roots in a rootNode
         *
         * @param  {DOM Node} rootNode
         * @param  {Boolean} includeRoot
         * @return {Array}
         */
        findRootNodes: function(rootNode, includeRoot) {
            var arr = null,
                out = [],
                classList = [],
                items,
                x,
                i,
                y,
                key;


            // build an array of v1 root names
            for (key in modules.maps) {
                if (modules.maps.hasOwnProperty(key)) {
                    classList.push(modules.maps[key].root);
                }
            }

            // get all elements that have a class attribute
            includeRoot = (includeRoot) ? includeRoot : false;
            if (includeRoot && rootNode.parentNode) {
                arr = modules.domUtils.getNodesByAttribute(rootNode.parentNode, 'class');
            } else {
                arr = modules.domUtils.getNodesByAttribute(rootNode, 'class');
            }

            // loop elements that have a class attribute
            x = 0;
            i = arr.length;
            while (x < i) {

                items = modules.domUtils.getAttributeList(arr[x], 'class');

                // loop classes on an element
                y = items.length;
                while (y--) {
                    // match v1 root names
                    if (classList.indexOf(items[y]) > -1) {
                        out.push(arr[x]);
                        break;
                    }

                    // match v2 root name prefix
                    if (modules.utils.startWith(items[y], 'h-')) {
                        out.push(arr[x]);
                        break;
                    }
                }

                x++;
            }
            return out;
        },


        /**
         * starts the tree walk to find microformats
         *
         * @param  {DOM Node} node
         * @return {Array}
         */
        walkRoot: function(node) {
            var context = this,
                children = [],
                child,
                classes,
                items = [],
                out = [];

            classes = this.getUfClassNames(node);
            // if it is a root microformat node
            if (classes && classes.root.length > 0) {
                items = this.walkTree(node);

                if (items.length > 0) {
                    out = out.concat(items);
                }
            } else {
                // check if there are children and one of the children has a root microformat
                children = modules.domUtils.getChildren( node );
                if (children && children.length > 0 && this.findRootNodes(node, true).length > -1) {
                    for (var i = 0; i < children.length; i++) {
                        child = children[i];
                        items = context.walkRoot(child);
                        if (items.length > 0) {
                            out = out.concat(items);
                        }
                    }
                }
            }
            return out;
        },


        /**
         * starts the tree walking for a single microformat
         *
         * @param  {DOM Node} node
         * @return {Array}
         */
        walkTree: function(node) {
            var classes,
                out = [],
                obj,
                itemRootID;

            // loop roots found on one element
            classes = this.getUfClassNames(node);
            if (classes && classes.root.length && classes.root.length > 0) {

                this.rootID++;
                itemRootID = this.rootID;
                obj = this.createUfObject(classes.root, classes.typeVersion);

                this.walkChildren(node, obj, classes.root, itemRootID, classes);
                if (this.impliedRules) {
                    this.impliedRules(node, obj, classes);
                }
                out.push( this.cleanUfObject(obj) );


            }
            return out;
        },


        /**
         * finds child properties of microformat
         *
         * @param  {DOM Node} node
         * @param  {Object} out
         * @param  {String} ufName
         * @param  {Int} rootID
         * @param  {Object} parentClasses
         */
        walkChildren: function(node, out, ufName, rootID, parentClasses) {
            var context = this,
                children = [],
                rootItem,
                itemRootID,
                value,
                propertyName,
                propertyVersion,
                i,
                x,
                y,
                z,
                child;

            children = modules.domUtils.getChildren( node );

            y = 0;
            z = children.length;
            while (y < z) {
                child = children[y];

                // get microformat classes for this single element
                var classes = context.getUfClassNames(child, ufName);

                // a property which is a microformat
                if (classes.root.length > 0 && classes.properties.length > 0 && !child.addedAsRoot) {
                    // create object with type, property and value
                    rootItem = context.createUfObject(
                        classes.root,
                        classes.typeVersion,
                        modules.text.parse(this.document, child, context.options.textFormat)
                    );

                    // add the microformat as an array of properties
                    propertyName = context.removePropPrefix(classes.properties[0][0]);

                    // modifies value with "implied value rule"
                    if (parentClasses && parentClasses.root.length === 1 && parentClasses.properties.length === 1) {
                        if (context.impliedValueRule) {
                            out = context.impliedValueRule(out, parentClasses.properties[0][0], classes.properties[0][0], value);
                        }
                    }

                    if (out.properties[propertyName]) {
                        out.properties[propertyName].push(rootItem);
                    } else {
                        out.properties[propertyName] = [rootItem];
                    }

                    context.rootID++;
                    // used to stop duplication in heavily nested structures
                    child.addedAsRoot = true;


                    x = 0;
                    i = rootItem.type.length;
                    itemRootID = context.rootID;
                    while (x < i) {
                        context.walkChildren(child, rootItem, rootItem.type, itemRootID, classes);
                        x++;
                    }
                    if (this.impliedRules) {
                        context.impliedRules(child, rootItem, classes);
                    }
                    this.cleanUfObject(rootItem);

                }

                // a property which is NOT a microformat and has not been used for a given root element
                if (classes.root.length === 0 && classes.properties.length > 0) {

                    x = 0;
                    i = classes.properties.length;
                    while (x < i) {

                        value = context.getValue(child, classes.properties[x][0], out);
                        propertyName = context.removePropPrefix(classes.properties[x][0]);
                        propertyVersion = classes.properties[x][1];

                        // modifies value with "implied value rule"
                        if (parentClasses && parentClasses.root.length === 1 && parentClasses.properties.length === 1) {
                            if (context.impliedValueRule) {
                                out = context.impliedValueRule(out, parentClasses.properties[0][0], classes.properties[x][0], value);
                            }
                        }

                        // if we have not added this value into a property with the same name already
                        if (!context.hasRootID(child, rootID, propertyName)) {
                            // check the root and property is the same version or if overlapping versions are allowed
                            if ( context.isAllowedPropertyVersion( out.typeVersion, propertyVersion ) ) {
                                // add the property as an array of properties
                                if (out.properties[propertyName]) {
                                    out.properties[propertyName].push(value);
                                } else {
                                    out.properties[propertyName] = [value];
                                }
                                // add rootid to node so we can track its use
                                context.appendRootID(child, rootID, propertyName);
                            }
                        }

                        x++;
                    }

                    context.walkChildren(child, out, ufName, rootID, classes);
                }

                // if the node has no microformat classes, see if its children have
                if (classes.root.length === 0 && classes.properties.length === 0) {
                    context.walkChildren(child, out, ufName, rootID, classes);
                }

                // if the node is a child root add it to the children tree
                if (classes.root.length > 0 && classes.properties.length === 0) {

                    // create object with type, property and value
                    rootItem = context.createUfObject(
                        classes.root,
                        classes.typeVersion,
                        modules.text.parse(this.document, child, context.options.textFormat)
                    );

                    // add the microformat as an array of properties
                    if (!out.children) {
                        out.children =  [];
                    }

                    if (!context.hasRootID(child, rootID, 'child-root')) {
                        out.children.push( rootItem );
                        context.appendRootID(child, rootID, 'child-root');
                        context.rootID++;
                    }

                    x = 0;
                    i = rootItem.type.length;
                    itemRootID = context.rootID;
                    while (x < i) {
                        context.walkChildren(child, rootItem, rootItem.type, itemRootID, classes);
                        x++;
                    }
                    if (this.impliedRules) {
                        context.impliedRules(child, rootItem, classes);
                    }
                    context.cleanUfObject( rootItem );

                }



                y++;
            }

        },




        /**
         * gets the value of a property from a node
         *
         * @param  {DOM Node} node
         * @param  {String} className
         * @param  {Object} uf
         * @return {String || Object}
         */
        getValue: function(node, className, uf) {
            var value = '';

            if (modules.utils.startWith(className, 'p-')) {
                value = this.getPValue(node, true);
            }

            if (modules.utils.startWith(className, 'e-')) {
                value = this.getEValue(node);
            }

            if (modules.utils.startWith(className, 'u-')) {
                value = this.getUValue(node, true);
            }

            if (modules.utils.startWith(className, 'dt-')) {
                value = this.getDTValue(node, className, uf, true);
            }
            return value;
        },


        /**
         * gets the value of a node which contains a 'p-' property
         *
         * @param  {DOM Node} node
         * @param  {Boolean} valueParse
         * @return {String}
         */
        getPValue: function(node, valueParse) {
            var out = '';
            if (valueParse) {
                out = this.getValueClass(node, 'p');
            }

            if (!out && valueParse) {
                out = this.getValueTitle(node);
            }

            if (!out) {
                out = modules.domUtils.getAttrValFromTagList(node, ['abbr'], 'title');
            }

            if (!out) {
                out = modules.domUtils.getAttrValFromTagList(node, ['data', 'input'], 'value');
            }

            if (node.name === 'br' || node.name === 'hr') {
                out = '';
            }

            if (!out) {
                out = modules.domUtils.getAttrValFromTagList(node, ['img', 'area'], 'alt');
            }

            if (!out) {
                out = modules.text.parse(this.document, node, this.options.textFormat);
            }

            return (out) ? out : '';
        },


        /**
         * gets the value of a node which contains the 'e-' property
         *
         * @param  {DOM Node} node
         * @return {Object}
         */
        getEValue: function(node) {

            var out = {value: '', html: ''};

            this.expandURLs(node, 'src', this.options.baseUrl);
            this.expandURLs(node, 'href', this.options.baseUrl);

            out.value = modules.text.parse(this.document, node, this.options.textFormat);
            out.html = modules.html.parse(node);

            return out;
        },


        /**
         * gets the value of a node which contains the 'u-' property
         *
         * @param  {DOM Node} node
         * @param  {Boolean} valueParse
         * @return {String}
         */
        getUValue: function(node, valueParse) {
            var out = '';
            if (valueParse) {
                out = this.getValueClass(node, 'u');
            }

            if (!out && valueParse) {
                out = this.getValueTitle(node);
            }

            if (!out) {
                out = modules.domUtils.getAttrValFromTagList(node, ['a', 'area'], 'href');
            }

            if (!out) {
                out = modules.domUtils.getAttrValFromTagList(node, ['img', 'audio', 'video', 'source'], 'src');
            }

            if (!out) {
                out = modules.domUtils.getAttrValFromTagList(node, ['object'], 'data');
            }

            // if we have no protocol separator, turn relative url to absolute url
            if (out && out !== '' && out.indexOf('://') === -1) {
                out = modules.url.resolve(out, this.options.baseUrl);
            }

            if (!out) {
                out = modules.domUtils.getAttrValFromTagList(node, ['abbr'], 'title');
            }

            if (!out) {
                out = modules.domUtils.getAttrValFromTagList(node, ['data', 'input'], 'value');
            }

            if (!out) {
                out = modules.text.parse(this.document, node, this.options.textFormat);
            }

            return (out) ? out : '';
        },


        /**
         * gets the value of a node which contains the 'dt-' property
         *
         * @param  {DOM Node} node
         * @param  {String} className
         * @param  {Object} uf
         * @param  {Boolean} valueParse
         * @return {String}
         */
        getDTValue: function(node, className, uf, valueParse) {
            var out = '';

            if (valueParse) {
                out = this.getValueClass(node, 'dt');
            }

            if (!out && valueParse) {
                out = this.getValueTitle(node);
            }

            if (!out) {
                out = modules.domUtils.getAttrValFromTagList(node, ['time', 'ins', 'del'], 'datetime');
            }

            if (!out) {
                out = modules.domUtils.getAttrValFromTagList(node, ['abbr'], 'title');
            }

            if (!out) {
                out = modules.domUtils.getAttrValFromTagList(node, ['data', 'input'], 'value');
            }

            if (!out) {
                out = modules.text.parse(this.document, node, this.options.textFormat);
            }

            if (out) {
                if (modules.dates.isDuration(out)) {
                    // just duration
                    return out;
                } else if (modules.dates.isTime(out)) {
                    // just time or time+timezone
                    if (uf) {
                        uf.times.push([className, modules.dates.parseAmPmTime(out, this.options.dateFormat)]);
                    }
                    return modules.dates.parseAmPmTime(out, this.options.dateFormat);
                }
                // returns a date - microformat profile
                if (uf) {
                    uf.dates.push([className, new modules.ISODate(out).toString( this.options.dateFormat )]);
                }
                return new modules.ISODate(out).toString( this.options.dateFormat );
            }
            return '';
        },


        /**
         * appends a new rootid to a given node
         *
         * @param  {DOM Node} node
         * @param  {String} id
         * @param  {String} propertyName
         */
        appendRootID: function(node, id, propertyName) {
            if (this.hasRootID(node, id, propertyName) === false) {
                var rootids = [];
                if (modules.domUtils.hasAttribute(node, 'rootids')) {
                    rootids = modules.domUtils.getAttributeList(node, 'rootids');
                }
                rootids.push('id' + id + '-' + propertyName);
                modules.domUtils.setAttribute(node, 'rootids', rootids.join(' '));
            }
        },


        /**
         * does a given node already have a rootid
         *
         * @param  {DOM Node} node
         * @param  {String} id
         * @param  {String} propertyName
         * @return {Boolean}
         */
        hasRootID: function(node, id, propertyName) {
            var rootids = [];
            if (!modules.domUtils.hasAttribute(node, 'rootids')) {
                return false;
            }
            rootids = modules.domUtils.getAttributeList(node, 'rootids');
            return (rootids.indexOf('id' + id + '-' + propertyName) > -1);
        },



        /**
         * gets the text of any child nodes with a class value
         *
         * @param  {DOM Node} node
         * @param  {String} propertyName
         * @return {String || null}
         */
        getValueClass: function(node, propertyType) {
            var context = this,
                children = [],
                out = [],
                child,
                x,
                i;

            children = modules.domUtils.getChildren( node );

            x = 0;
            i = children.length;
            while (x < i) {
                child = children[x];
                var value = null;
                if (modules.domUtils.hasAttributeValue(child, 'class', 'value')) {
                    switch (propertyType) {
                    case 'p':
                        value = context.getPValue(child, false);
                        break;
                    case 'u':
                        value = context.getUValue(child, false);
                        break;
                    case 'dt':
                        value = context.getDTValue(child, '', null, false);
                        break;
                    }
                    if (value) {
                        out.push(modules.utils.trim(value));
                    }
                }
                x++;
            }
            if (out.length > 0) {
                if (propertyType === 'p') {
                    return modules.text.parseText( this.document, out.join(' '), this.options.textFormat);
                }
                if (propertyType === 'u') {
                    return out.join('');
                }
                if (propertyType === 'dt') {
                    return modules.dates.concatFragments(out, this.options.dateFormat).toString(this.options.dateFormat);
                }
                return undefined;
            }
            return null;
        },


        /**
         * returns a single string of the 'title' attr from all
         * the child nodes with the class 'value-title'
         *
         * @param  {DOM Node} node
         * @return {String}
         */
        getValueTitle: function(node) {
            var out = [],
                items,
                i,
                x;

            items = modules.domUtils.getNodesByAttributeValue(node, 'class', 'value-title');
            x = 0;
            i = items.length;
            while (x < i) {
                if (modules.domUtils.hasAttribute(items[x], 'title')) {
                    out.push(modules.domUtils.getAttribute(items[x], 'title'));
                }
                x++;
            }
            return out.join('');
        },


       /**
         * finds out whether a node has h-* class v1 and v2
         *
         * @param  {DOM Node} node
         * @return {Boolean}
         */
        hasHClass: function(node) {
            var classes = this.getUfClassNames(node);
            if (classes.root && classes.root.length > 0) {
                return true;
            }
            return false;
        },


        /**
         * get both the root and property class names from a node
         *
         * @param  {DOM Node} node
         * @param  {Array} ufNameArr
         * @return {Object}
         */
        getUfClassNames: function(node, ufNameArr) {
            var context = this,
                out = {
                    'root': [],
                    'properties': []
                },
                classNames,
                key,
                items,
                item,
                i,
                x,
                z,
                y,
                map,
                prop,
                propName,
                v2Name,
                impiedRel,
                ufName;

            // don't get classes from excluded list of tags
            if (modules.domUtils.hasTagName(node, this.excludeTags) === false) {

                // find classes for node
                classNames = modules.domUtils.getAttribute(node, 'class');
                if (classNames) {
                    items = classNames.split(' ');
                    x = 0;
                    i = items.length;
                    while (x < i) {

                        item = modules.utils.trim(items[x]);

                        // test for root prefix - v2
                        if (modules.utils.startWith(item, context.rootPrefix)) {
                            if (out.root.indexOf(item) === -1) {
                                out.root.push(item);
                            }
                            out.typeVersion = 'v2';
                        }

                        // test for property prefix - v2
                        z = context.propertyPrefixes.length;
                        while (z--) {
                            if (modules.utils.startWith(item, context.propertyPrefixes[z])) {
                                out.properties.push([item, 'v2']);
                            }
                        }

                        // test for mapped root classnames v1
                        for (key in modules.maps) {
                            if (modules.maps.hasOwnProperty(key)) {
                                // only add a root once
                                if (modules.maps[key].root === item && out.root.indexOf(key) === -1) {
                                    // if root map has subTree set to true
                                    // test to see if we should create a property or root
                                    if (modules.maps[key].subTree) {
                                        out.properties.push(['p-' + modules.maps[key].root, 'v1']);
                                    } else {
                                        out.root.push(key);
                                        if (!out.typeVersion) {
                                            out.typeVersion = 'v1';
                                        }
                                    }
                                }
                            }
                        }


                        // test for mapped property classnames v1
                        if (ufNameArr) {
                            for (var a = 0; a < ufNameArr.length; a++) {
                                ufName = ufNameArr[a];
                                // get mapped property v1 microformat
                                map = context.getMapping(ufName);
                                if (map) {
                                    for (key in map.properties) {
                                        if (map.properties.hasOwnProperty(key)) {

                                            prop = map.properties[key];
                                            propName = (prop.map) ? prop.map : 'p-' + key;

                                            if (key === item) {
                                                if (prop.uf) {
                                                    // loop all the classList make sure
                                                    //   1. this property is a root
                                                    //   2. that there is not already an equivalent v2 property i.e. url and u-url on the same element
                                                    y = 0;
                                                    while (y < i) {
                                                        v2Name = context.getV2RootName(items[y]);
                                                        // add new root
                                                        if (prop.uf.indexOf(v2Name) > -1 && out.root.indexOf(v2Name) === -1) {
                                                            out.root.push(v2Name);
                                                            out.typeVersion = 'v1';
                                                        }
                                                        y++;
                                                    }
                                                    // only add property once
                                                    if (out.properties.indexOf(propName) === -1) {
                                                        out.properties.push([propName, 'v1']);
                                                    }
                                                } else if (out.properties.indexOf(propName) === -1) {
                                                    out.properties.push([propName, 'v1']);
                                                }
                                            }
                                        }

                                    }
                                }
                            }

                        }

                        x++;

                    }
                }
            }


            // finds any alt rel=* mappings for a given node/microformat
            if (ufNameArr && this.findRelImpied) {
                for (var b = 0; b < ufNameArr.length; b++) {
                    ufName = ufNameArr[b];
                    impiedRel = this.findRelImpied(node, ufName);
                    if (impiedRel && out.properties.indexOf(impiedRel) === -1) {
                        out.properties.push([impiedRel, 'v1']);
                    }
                }
            }


            // if(out.root.length === 1 && out.properties.length === 1) {
            //  if(out.root[0].replace('h-','') === this.removePropPrefix(out.properties[0][0])) {
            //      out.typeVersion = 'v2';
            //  }
            // }

            return out;
        },


        /**
         * given a v1 or v2 root name, return mapping object
         *
         * @param  {String} name
         * @return {Object || null}
         */
        getMapping: function(name) {
            var key;
            for (key in modules.maps) {
                if (modules.maps[key].root === name || key === name) {
                    return modules.maps[key];
                }
            }
            return null;
        },


        /**
         * given a v1 root name returns a v2 root name i.e. vcard >>> h-card
         *
         * @param  {String} name
         * @return {String || null}
         */
        getV2RootName: function(name) {
            var key;
            for (key in modules.maps) {
                if (modules.maps[key].root === name) {
                    return key;
                }
            }
            return null;
        },


        /**
         * whether a property is the right microformats version for its root type
         *
         * @param  {String} typeVersion
         * @param  {String} propertyVersion
         * @return {Boolean}
         */
        isAllowedPropertyVersion: function(typeVersion, propertyVersion) {
            if (this.options.overlappingVersions === true) {
                return true;
            }
            return (typeVersion === propertyVersion);
        },


        /**
         * creates a blank microformats object
         *
         * @param  {String} name
         * @param  {String} value
         * @return {Object}
         */
        createUfObject: function(names, typeVersion, value) {
            var out = {};

            // is more than just whitespace
            if (value && modules.utils.isOnlyWhiteSpace(value) === false) {
                out.value = value;
            }
            // add type i.e. ["h-card", "h-org"]
            if (modules.utils.isArray(names)) {
                out.type = names;
            } else {
                out.type = [names];
            }
            out.properties = {};
            // metadata properties for parsing
            out.typeVersion = typeVersion;
            out.times = [];
            out.dates = [];
            out.altValue = null;

            return out;
        },


        /**
         * removes unwanted microformats property before output
         *
         * @param  {Object} microformat
         */
        cleanUfObject: function( microformat ) {
            delete microformat.times;
            delete microformat.dates;
            delete microformat.typeVersion;
            delete microformat.altValue;
            return microformat;
        },



        /**
         * removes microformat property prefixes from text
         *
         * @param  {String} text
         * @return {String}
         */
        removePropPrefix: function(text) {
            var i;

            i = this.propertyPrefixes.length;
            while (i--) {
                var prefix = this.propertyPrefixes[i];
                if (modules.utils.startWith(text, prefix)) {
                    text = text.substr(prefix.length);
                }
            }
            return text;
        },


        /**
         * expands all relative URLs to absolute ones where it can
         *
         * @param  {DOM Node} node
         * @param  {String} attrName
         * @param  {String} baseUrl
         */
        expandURLs: function(node, attrName, baseUrl) {
            var i,
                nodes,
                attr;

            nodes = modules.domUtils.getNodesByAttribute(node, attrName);
            i = nodes.length;
            while (i--) {
                try {
                    // the url parser can blow up if the format is not right
                    attr = modules.domUtils.getAttribute(nodes[i], attrName);
                    if (attr && attr !== '' && baseUrl !== '' && attr.indexOf('://') === -1) {
                        // attr = urlParser.resolve(baseUrl, attr);
                        attr = modules.url.resolve(attr, baseUrl);
                        modules.domUtils.setAttribute(nodes[i], attrName, attr);
                    }
                } catch (err) {
                    // do nothing - convert only the urls we can, leave the rest as they are
                }
            }
        },



        /**
         * merges passed and default options -single level clone of properties
         *
         * @param  {Object} options
         */
        mergeOptions: function(options) {
            var key;
            for (key in options) {
                if (options.hasOwnProperty(key)) {
                    this.options[key] = options[key];
                }
            }
        },


        /**
         * removes all rootid attributes
         *
         * @param  {DOM Node} rootNode
         */
        removeRootIds: function(rootNode) {
            var arr,
                i;

            arr = modules.domUtils.getNodesByAttribute(rootNode, 'rootids');
            i = arr.length;
            while (i--) {
                modules.domUtils.removeAttribute(arr[i], 'rootids');
            }
        },


        /**
         * removes all changes made to the DOM
         *
         * @param  {DOM Node} rootNode
         */
        clearUpDom: function(rootNode) {
            if (this.removeIncludes) {
                this.removeIncludes(rootNode);
            }
            this.removeRootIds(rootNode);
        }


    };


    modules.Parser.prototype.constructor = modules.Parser;


    // check parser module is loaded
    if (modules.Parser) {

        /**
         * applies "implied rules" microformat output structure i.e. feed-title, name, photo, url and date
         *
         * @param  {DOM Node} node
         * @param  {Object} uf (microformat output structure)
         * @param  {Object} parentClasses (classes structure)
         * @param  {Boolean} impliedPropertiesByVersion
         * @return {Object}
         */
         modules.Parser.prototype.impliedRules = function(node, uf, parentClasses) {
            var typeVersion = (uf.typeVersion)? uf.typeVersion: 'v2';

            // TEMP: override to allow v1 implied properties while spec changes
            if (this.options.impliedPropertiesByVersion === false) {
                typeVersion = 'v2';
            }

            if (node && uf && uf.properties) {
                uf = this.impliedBackwardComp( node, uf, parentClasses );
                if (typeVersion === 'v2') {
                    uf = this.impliedhFeedTitle( uf );
                    uf = this.impliedName( node, uf );
                    uf = this.impliedPhoto( node, uf );
                    uf = this.impliedUrl( node, uf );
                }
                uf = this.impliedValue( node, uf, parentClasses );
                uf = this.impliedDate( uf );

                // TEMP: flagged while spec changes are put forward
                if (this.options.parseLatLonGeo === true) {
                    uf = this.impliedGeo( uf );
                }
            }

            return uf;
        };


        /**
         * apply implied name rule
         *
         * @param  {DOM Node} node
         * @param  {Object} uf
         * @return {Object}
         */
        modules.Parser.prototype.impliedName = function(node, uf) {
            // implied name rule
            /*
                img.h-x[alt]                                        <img class="h-card" src="glenn.htm" alt="Glenn Jones"></a>
                area.h-x[alt]                                       <area class="h-card" href="glenn.htm" alt="Glenn Jones"></area>
                abbr.h-x[title]                                     <abbr class="h-card" title="Glenn Jones"GJ</abbr>

                .h-x>img:only-child[alt]:not[.h-*]                  <div class="h-card"><a src="glenn.htm" alt="Glenn Jones"></a></div>
                .h-x>area:only-child[alt]:not[.h-*]                 <div class="h-card"><area href="glenn.htm" alt="Glenn Jones"></area></div>
                .h-x>abbr:only-child[title]                         <div class="h-card"><abbr title="Glenn Jones">GJ</abbr></div>

                .h-x>:only-child>img:only-child[alt]:not[.h-*]      <div class="h-card"><span><img src="jane.html" alt="Jane Doe"/></span></div>
                .h-x>:only-child>area:only-child[alt]:not[.h-*]     <div class="h-card"><span><area href="jane.html" alt="Jane Doe"></area></span></div>
                .h-x>:only-child>abbr:only-child[title]             <div class="h-card"><span><abbr title="Jane Doe">JD</abbr></span></div>
            */
            var name,
                value;

            if (!uf.properties.name) {
                value = this.getImpliedProperty(node, ['img', 'area', 'abbr'], this.getNameAttr);
                var textFormat = this.options.textFormat;
                // if no value for tags/properties use text
                if (!value) {
                    name = [modules.text.parse(this.document, node, textFormat)];
                } else {
                    name = [modules.text.parseText(this.document, value, textFormat)];
                }
                if (name && name[0] !== '') {
                    uf.properties.name = name;
                }
            }

            return uf;
        };


        /**
         * apply implied photo rule
         *
         * @param  {DOM Node} node
         * @param  {Object} uf
         * @return {Object}
         */
        modules.Parser.prototype.impliedPhoto = function(node, uf) {
            // implied photo rule
            /*
                img.h-x[src]                                                <img class="h-card" alt="Jane Doe" src="jane.jpeg"/>
                object.h-x[data]                                            <object class="h-card" data="jane.jpeg"/>Jane Doe</object>
                .h-x>img[src]:only-of-type:not[.h-*]                        <div class="h-card"><img alt="Jane Doe" src="jane.jpeg"/></div>
                .h-x>object[data]:only-of-type:not[.h-*]                    <div class="h-card"><object data="jane.jpeg"/>Jane Doe</object></div>
                .h-x>:only-child>img[src]:only-of-type:not[.h-*]            <div class="h-card"><span><img alt="Jane Doe" src="jane.jpeg"/></span></div>
                .h-x>:only-child>object[data]:only-of-type:not[.h-*]        <div class="h-card"><span><object data="jane.jpeg"/>Jane Doe</object></span></div>
            */
            var value;
            if (!uf.properties.photo) {
                value = this.getImpliedProperty(node, ['img', 'object'], this.getPhotoAttr);
                if (value) {
                    // relative to absolute URL
                    if (value && value !== '' && this.options.baseUrl !== '' && value.indexOf('://') === -1) {
                        value = modules.url.resolve(value, this.options.baseUrl);
                    }
                    uf.properties.photo = [modules.utils.trim(value)];
                }
            }
            return uf;
        };


        /**
         * apply implied URL rule
         *
         * @param  {DOM Node} node
         * @param  {Object} uf
         * @return {Object}
         */
        modules.Parser.prototype.impliedUrl = function(node, uf) {
            // implied URL rule
            /*
                a.h-x[href]                             <a class="h-card" href="glenn.html">Glenn</a>
                area.h-x[href]                          <area class="h-card" href="glenn.html">Glenn</area>
                .h-x>a[href]:only-of-type:not[.h-*]     <div class="h-card" ><a href="glenn.html">Glenn</a><p>...</p></div>
                .h-x>area[href]:only-of-type:not[.h-*]  <div class="h-card" ><area href="glenn.html">Glenn</area><p>...</p></div>
            */
            var value;
            if (!uf.properties.url) {
                value = this.getImpliedProperty(node, ['a', 'area'], this.getURLAttr);
                if (value) {
                    // relative to absolute URL
                    if (value && value !== '' && this.options.baseUrl !== '' && value.indexOf('://') === -1) {
                        value = modules.url.resolve(value, this.options.baseUrl);
                    }
                    uf.properties.url = [modules.utils.trim(value)];
                }
            }
            return uf;
        };


        /**
         * apply implied date rule - if there is a time only property try to concat it with any date property
         *
         * @param  {DOM Node} node
         * @param  {Object} uf
         * @return {Object}
         */
        modules.Parser.prototype.impliedDate = function(uf) {
            // implied date rule
            // http://microformats.org/wiki/value-class-pattern#microformats2_parsers
            // http://microformats.org/wiki/microformats2-parsing-issues#implied_date_for_dt_properties_both_mf2_and_backcompat
            var newDate;
            if (uf.times.length > 0 && uf.dates.length > 0) {
                newDate = modules.dates.dateTimeUnion(uf.dates[0][1], uf.times[0][1], this.options.dateFormat);
                uf.properties[this.removePropPrefix(uf.times[0][0])][0] = newDate.toString(this.options.dateFormat);
            }
            // clean-up object
            delete uf.times;
            delete uf.dates;
            return uf;
        };


        /**
         * get an implied property value from pre-defined tag/attriubte combinations
         *
         * @param  {DOM Node} node
         * @param  {String} tagList (Array of tags from which an implied value can be pulled)
         * @param  {String} getAttrFunction (Function which can extract implied value)
         * @return {String || null}
         */
        modules.Parser.prototype.getImpliedProperty = function(node, tagList, getAttrFunction) {
            // i.e. img.h-card
            var value = getAttrFunction(node),
                descendant,
                child;

            if (!value) {
                // i.e. .h-card>img:only-of-type:not(.h-card)
                descendant = modules.domUtils.getSingleDescendantOfType( node, tagList);
                if (descendant && this.hasHClass(descendant) === false) {
                    value = getAttrFunction(descendant);
                }
                if (node.children.length > 0 ) {
                    // i.e.  .h-card>:only-child>img:only-of-type:not(.h-card)
                    child = modules.domUtils.getSingleDescendant(node);
                    if (child && this.hasHClass(child) === false) {
                        descendant = modules.domUtils.getSingleDescendantOfType(child, tagList);
                        if (descendant && this.hasHClass(descendant) === false) {
                            value = getAttrFunction(descendant);
                        }
                    }
                }
            }

            return value;
        };


        /**
         * get an implied name value from a node
         *
         * @param  {DOM Node} node
         * @return {String || null}
         */
        modules.Parser.prototype.getNameAttr = function(node) {
            var value = modules.domUtils.getAttrValFromTagList(node, ['img', 'area'], 'alt');
            if (!value) {
                value = modules.domUtils.getAttrValFromTagList(node, ['abbr'], 'title');
            }
            return value;
        };


        /**
         * get an implied photo value from a node
         *
         * @param  {DOM Node} node
         * @return {String || null}
         */
        modules.Parser.prototype.getPhotoAttr = function(node) {
            var value = modules.domUtils.getAttrValFromTagList(node, ['img'], 'src');
            if (!value && modules.domUtils.hasAttributeValue(node, 'class', 'include') === false) {
                value = modules.domUtils.getAttrValFromTagList(node, ['object'], 'data');
            }
            return value;
        };


        /**
         * get an implied photo value from a node
         *
         * @param  {DOM Node} node
         * @return {String || null}
         */
        modules.Parser.prototype.getURLAttr = function(node) {
            var value = null;
            if (modules.domUtils.hasAttributeValue(node, 'class', 'include') === false) {

                value = modules.domUtils.getAttrValFromTagList(node, ['a'], 'href');
                if (!value) {
                    value = modules.domUtils.getAttrValFromTagList(node, ['area'], 'href');
                }

            }
            return value;
        };


        /**
         *
         *
         * @param  {DOM Node} node
         * @param  {Object} uf
         * @return {Object}
         */
        modules.Parser.prototype.impliedValue = function(node, uf, parentClasses) {

            // intersection of implied name and implied value rules
            if (uf.properties.name) {
                if (uf.value && parentClasses.root.length > 0 && parentClasses.properties.length === 1) {
                    uf = this.getAltValue(uf, parentClasses.properties[0][0], 'p-name', uf.properties.name[0]);
                }
            }

            // intersection of implied URL and implied value rules
            if (uf.properties.url) {
                if (parentClasses && parentClasses.root.length === 1 && parentClasses.properties.length === 1) {
                    uf = this.getAltValue(uf, parentClasses.properties[0][0], 'u-url', uf.properties.url[0]);
                }
            }

            // apply alt value
            if (uf.altValue !== null) {
                uf.value = uf.altValue.value;
            }
            delete uf.altValue;


            return uf;
        };


        /**
         * get alt value based on rules about parent property prefix
         *
         * @param  {Object} uf
         * @param  {String} parentPropertyName
         * @param  {String} propertyName
         * @param  {String} value
         * @return {Object}
         */
        modules.Parser.prototype.getAltValue = function(uf, parentPropertyName, propertyName, value) {
            if (uf.value && !uf.altValue) {
                // first p-name of the h-* child
                if (modules.utils.startWith(parentPropertyName, 'p-') && propertyName === 'p-name') {
                    uf.altValue = {name: propertyName, value: value};
                }
                // if it's an e-* property element
                if (modules.utils.startWith(parentPropertyName, 'e-') && modules.utils.startWith(propertyName, 'e-')) {
                    uf.altValue = {name: propertyName, value: value};
                }
                // if it's an u-* property element
                if (modules.utils.startWith(parentPropertyName, 'u-') && propertyName === 'u-url') {
                    uf.altValue = {name: propertyName, value: value};
                }
            }
            return uf;
        };


        /**
         * if a h-feed does not have a title use the title tag of a page
         *
         * @param  {Object} uf
         * @return {Object}
         */
        modules.Parser.prototype.impliedhFeedTitle = function( uf ) {
            if (uf.type && uf.type.indexOf('h-feed') > -1) {
                // has no name property
                if (uf.properties.name === undefined || uf.properties.name[0] === '' ) {
                    // use the text from the title tag
                    var title = modules.domUtils.querySelector(this.document, 'title');
                    if (title) {
                        uf.properties.name = [modules.domUtils.textContent(title)];
                    }
                }
            }
            return uf;
        };



        /**
         * implied Geo from pattern <abbr class="p-geo" title="37.386013;-122.082932">
         *
         * @param  {Object} uf
         * @return {Object}
         */
        modules.Parser.prototype.impliedGeo = function( uf ) {
            var geoPair,
                parts,
                longitude,
                latitude,
                valid = true;

            if (uf.type && uf.type.indexOf('h-geo') > -1) {

                // has no latitude or longitude property
                if (uf.properties.latitude === undefined || uf.properties.longitude === undefined ) {

                    geoPair = (uf.properties.name)? uf.properties.name[0] : null;
                    geoPair = (!geoPair && uf.properties.value)? uf.properties.value : geoPair;

                    if (geoPair) {
                        // allow for the use of a ';' as in microformats and also ',' as in Geo URL
                        geoPair = geoPair.replace(';', ',');

                        // has sep char
                        if (geoPair.indexOf(',') > -1 ) {
                            parts = geoPair.split(',');

                            // only correct if we have two or more parts
                            if (parts.length > 1) {

                                // latitude no value outside the range -90 or 90
                                latitude = parseFloat( parts[0] );
                                if (modules.utils.isNumber(latitude) && latitude > 90 || latitude < -90) {
                                    valid = false;
                                }

                                // longitude no value outside the range -180 to 180
                                longitude = parseFloat( parts[1] );
                                if (modules.utils.isNumber(longitude) && longitude > 180 || longitude < -180) {
                                    valid = false;
                                }

                                if (valid) {
                                    uf.properties.latitude = [latitude];
                                    uf.properties.longitude  = [longitude];
                                }
                            }

                        }
                    }
                }
            }
            return uf;
        };


        /**
         * if a backwards compat built structure has no properties add name through this.impliedName
         *
         * @param  {Object} uf
         * @return {Object}
         */
        modules.Parser.prototype.impliedBackwardComp = function(node, uf, parentClasses) {

            // look for pattern in parent classes like "p-geo h-geo"
            // these are structures built from backwards compat parsing of geo
            if (parentClasses.root.length === 1 && parentClasses.properties.length === 1) {
                if (parentClasses.root[0].replace('h-', '') === this.removePropPrefix(parentClasses.properties[0][0])) {

                    // if microformat has no properties apply the impliedName rule to get value from containing node
                    // this will get value from html such as <abbr class="geo" title="30.267991;-97.739568">Brighton</abbr>
                    if ( modules.utils.hasProperties(uf.properties) === false ) {
                        uf = this.impliedName( node, uf );
                    }
                }
            }

            return uf;
        };



    }


    // check parser module is loaded
    if (modules.Parser) {


        /**
         * appends clones of include Nodes into the DOM structure
         *
         * @param  {DOM node} rootNode
         */
        modules.Parser.prototype.addIncludes = function(rootNode) {
            this.addAttributeIncludes(rootNode, 'itemref');
            this.addAttributeIncludes(rootNode, 'headers');
            this.addClassIncludes(rootNode);
        };


        /**
         * appends clones of include Nodes into the DOM structure for attribute based includes
         *
         * @param  {DOM node} rootNode
         * @param  {String} attributeName
         */
        modules.Parser.prototype.addAttributeIncludes = function(rootNode, attributeName) {
            var arr,
                idList,
                i,
                x,
                z,
                y;

            arr = modules.domUtils.getNodesByAttribute(rootNode, attributeName);
            x = 0;
            i = arr.length;
            while (x < i) {
                idList = modules.domUtils.getAttributeList(arr[x], attributeName);
                if (idList) {
                    z = 0;
                    y = idList.length;
                    while (z < y) {
                        this.apppendInclude(arr[x], idList[z]);
                        z++;
                    }
                }
                x++;
            }
        };


        /**
         * appends clones of include Nodes into the DOM structure for class based includes
         *
         * @param  {DOM node} rootNode
         */
        modules.Parser.prototype.addClassIncludes = function(rootNode) {
            var id,
                arr,
                x = 0,
                i;

            arr = modules.domUtils.getNodesByAttributeValue(rootNode, 'class', 'include');
            i = arr.length;
            while (x < i) {
                id = modules.domUtils.getAttrValFromTagList(arr[x], ['a'], 'href');
                if (!id) {
                    id = modules.domUtils.getAttrValFromTagList(arr[x], ['object'], 'data');
                }
                this.apppendInclude(arr[x], id);
                x++;
            }
        };


        /**
         * appends a clone of an include into another Node using Id
         *
         * @param  {DOM node} rootNode
         * @param  {Stringe} id
         */
        modules.Parser.prototype.apppendInclude = function(node, id) {
            var include,
                clone;

            id = modules.utils.trim(id.replace('#', ''));
            include = modules.domUtils.getElementById(this.document, id);
            if (include) {
                clone = modules.domUtils.clone(include);
                this.markIncludeChildren(clone);
                modules.domUtils.appendChild(node, clone);
            }
        };


        /**
         * adds an attribute marker to all the child microformat roots
         *
         * @param  {DOM node} rootNode
         */
        modules.Parser.prototype.markIncludeChildren = function(rootNode) {
            var arr,
                x,
                i;

            // loop the array and add the attribute
            arr = this.findRootNodes(rootNode);
            x = 0;
            i = arr.length;
            modules.domUtils.setAttribute(rootNode, 'data-include', 'true');
            modules.domUtils.setAttribute(rootNode, 'style', 'display:none');
            while (x < i) {
                modules.domUtils.setAttribute(arr[x], 'data-include', 'true');
                x++;
            }
        };


        /**
         * removes all appended include clones from DOM
         *
         * @param  {DOM node} rootNode
         */
        modules.Parser.prototype.removeIncludes = function(rootNode) {
            var arr,
                i;

            // remove all the items that were added as includes
            arr = modules.domUtils.getNodesByAttribute(rootNode, 'data-include');
            i = arr.length;
            while (i--) {
                modules.domUtils.removeChild(rootNode, arr[i]);
            }
        };


    }


    // check parser module is loaded
    if (modules.Parser) {

        /**
         * finds rel=* structures
         *
         * @param  {DOM node} rootNode
         * @return {Object}
         */
        modules.Parser.prototype.findRels = function(rootNode) {
            var out = {
                    'items': [],
                    'rels': {},
                    'rel-urls': {}
                },
                x,
                i,
                y,
                z,
                relList,
                items,
                item,
                value,
                arr;

            arr = modules.domUtils.getNodesByAttribute(rootNode, 'rel');
            x = 0;
            i = arr.length;
            while (x < i) {
                relList = modules.domUtils.getAttribute(arr[x], 'rel');

                if (relList) {
                    items = relList.split(' ');


                    // add rels
                    z = 0;
                    y = items.length;
                    while (z < y) {
                        item = modules.utils.trim(items[z]);

                        // get rel value
                        value = modules.domUtils.getAttrValFromTagList(arr[x], ['a', 'area'], 'href');
                        if (!value) {
                            value = modules.domUtils.getAttrValFromTagList(arr[x], ['link'], 'href');
                        }

                        // create the key
                        if (!out.rels[item]) {
                            out.rels[item] = [];
                        }

                        if (typeof this.options.baseUrl === 'string' && typeof value === 'string') {

                            var resolved = modules.url.resolve(value, this.options.baseUrl);
                            // do not add duplicate rels - based on resolved URLs
                            if (out.rels[item].indexOf(resolved) === -1) {
                                out.rels[item].push( resolved );
                            }
                        }
                        z++;
                    }


                    var url = null;
                    if (modules.domUtils.hasAttribute(arr[x], 'href')) {
                        url = modules.domUtils.getAttribute(arr[x], 'href');
                        if (url) {
                            url = modules.url.resolve(url, this.options.baseUrl );
                        }
                    }


                    // add to rel-urls
                    var relUrl = this.getRelProperties(arr[x]);
                    relUrl.rels = items;
                    // do not add duplicate rel-urls - based on resolved URLs
                    if (url && out['rel-urls'][url] === undefined) {
                        out['rel-urls'][url] = relUrl;
                    }


                }
                x++;
            }
            return out;
        };


        /**
         * gets the properties of a rel=*
         *
         * @param  {DOM node} node
         * @return {Object}
         */
        modules.Parser.prototype.getRelProperties = function(node) {
            var obj = {};

            if (modules.domUtils.hasAttribute(node, 'media')) {
                obj.media = modules.domUtils.getAttribute(node, 'media');
            }
            if (modules.domUtils.hasAttribute(node, 'type')) {
                obj.type = modules.domUtils.getAttribute(node, 'type');
            }
            if (modules.domUtils.hasAttribute(node, 'hreflang')) {
                obj.hreflang = modules.domUtils.getAttribute(node, 'hreflang');
            }
            if (modules.domUtils.hasAttribute(node, 'title')) {
                obj.title = modules.domUtils.getAttribute(node, 'title');
            }
            if (modules.utils.trim(this.getPValue(node, false)) !== '') {
                obj.text = this.getPValue(node, false);
            }

            return obj;
        };


        /**
         * finds any alt rel=* mappings for a given node/microformat
         *
         * @param  {DOM node} node
         * @param  {String} ufName
         * @return {String || undefined}
         */
        modules.Parser.prototype.findRelImpied = function(node, ufName) {
            var out,
                map,
                i;

            map = this.getMapping(ufName);
            if (map) {
                for (var key in map.properties) {
                    if (map.properties.hasOwnProperty(key)) {
                        var prop = map.properties[key],
                            propName = (prop.map) ? prop.map : 'p-' + key,
                            relCount = 0;

                        // is property an alt rel=* mapping
                        if (prop.relAlt && modules.domUtils.hasAttribute(node, 'rel')) {
                            i = prop.relAlt.length;
                            while (i--) {
                                if (modules.domUtils.hasAttributeValue(node, 'rel', prop.relAlt[i])) {
                                    relCount++;
                                }
                            }
                            if (relCount === prop.relAlt.length) {
                                out = propName;
                            }
                        }
                    }
                }
            }
            return out;
        };


        /**
         * returns whether a node or its children has rel=* microformat
         *
         * @param  {DOM node} node
         * @return {Boolean}
         */
        modules.Parser.prototype.hasRel = function(node) {
            return (this.countRels(node) > 0);
        };


        /**
         * returns the number of rel=* microformats
         *
         * @param  {DOM node} node
         * @return {Int}
         */
        modules.Parser.prototype.countRels = function(node) {
            if (node) {
                return modules.domUtils.getNodesByAttribute(node, 'rel').length;
            }
            return 0;
        };



    }


    modules.utils = {

        /**
         * is the object a string
         *
         * @param  {Object} obj
         * @return {Boolean}
         */
        isString: function( obj ) {
            return typeof( obj ) === 'string';
        },

        /**
         * is the object a number
         *
         * @param  {Object} obj
         * @return {Boolean}
         */
        isNumber: function( obj ) {
            return !isNaN(parseFloat( obj )) && isFinite( obj );
        },


        /**
         * is the object an array
         *
         * @param  {Object} obj
         * @return {Boolean}
         */
        isArray: function( obj ) {
            return obj && !( obj.propertyIsEnumerable( 'length' ) ) && typeof obj === 'object' && typeof obj.length === 'number';
        },


        /**
         * is the object a function
         *
         * @param  {Object} obj
         * @return {Boolean}
         */
        isFunction: function(obj) {
            return !!(obj && obj.constructor && obj.call && obj.apply);
        },


        /**
         * does the text start with a test string
         *
         * @param  {String} text
         * @param  {String} test
         * @return {Boolean}
         */
        startWith: function( text, test ) {
            return (text.indexOf(test) === 0);
        },


        /**
         * removes spaces at front and back of text
         *
         * @param  {String} text
         * @return {String}
         */
        trim: function( text ) {
            if (text && this.isString(text)) {
                return (text.trim())? text.trim() : text.replace(/^\s+|\s+$/g, '');
            }
            return '';
        },


        /**
         * replaces a character in text
         *
         * @param  {String} text
         * @param  {Int} index
         * @param  {String} character
         * @return {String}
         */
        replaceCharAt: function( text, index, character ) {
            if (text && text.length > index) {
               return text.substr(0, index) + character + text.substr(index+character.length);
            }
            return text;
        },


        /**
         * removes whitespace, tabs and returns from start and end of text
         *
         * @param  {String} text
         * @return {String}
         */
        trimWhitespace: function( text ) {
            if (text && text.length) {
                var i = text.length,
                    x = 0;

                // turn all whitespace chars at end into spaces
                while (i--) {
                    if (this.isOnlyWhiteSpace(text[i])) {
                        text = this.replaceCharAt( text, i, ' ' );
                    } else {
                        break;
                    }
                }

                // turn all whitespace chars at start into spaces
                i = text.length;
                while (x < i) {
                    if (this.isOnlyWhiteSpace(text[x])) {
                        text = this.replaceCharAt( text, i, ' ' );
                    } else {
                        break;
                    }
                    x++;
                }
            }
            return this.trim(text);
        },


        /**
         * does text only contain whitespace characters
         *
         * @param  {String} text
         * @return {Boolean}
         */
        isOnlyWhiteSpace: function( text ) {
            return !(/[^\t\n\r ]/.test( text ));
        },


        /**
         * removes whitespace from text (leaves a single space)
         *
         * @param  {String} text
         * @return {Sring}
         */
        collapseWhiteSpace: function( text ) {
            return text.replace(/[\t\n\r ]+/g, ' ');
        },


        /**
         * does an object have any of its own properties
         *
         * @param  {Object} obj
         * @return {Boolean}
         */
        hasProperties: function( obj ) {
            var key;
            for (key in obj) {
                if ( obj.hasOwnProperty( key ) ) {
                    return true;
                }
            }
            return false;
        },


        /**
         * a sort function - to sort objects in an array by a given property
         *
         * @param  {String} property
         * @param  {Boolean} reverse
         * @return {Int}
         */
        sortObjects: function(property, reverse) {
            reverse = (reverse) ? -1 : 1;
            return function (a, b) {
                a = a[property];
                b = b[property];
                if (a < b) {
                    return reverse * -1;
                }
                if (a > b) {
                    return reverse * 1;
                }
                return 0;
            };
        }

    };


    modules.domUtils = {

        // blank objects for DOM
        document: null,
        rootNode: null,


         /**
         * gets DOMParser object
         *
         * @return {Object || undefined}
         */
        getDOMParser: function () {
            if (typeof DOMParser === "undefined") {
                try {
                    return Components.classes["@mozilla.org/xmlextras/domparser;1"]
                        .createInstance(Components.interfaces.nsIDOMParser);
                } catch (e) {
                    return undefined;
                }
            } else {
                return new DOMParser();
            }
        },


         /**
         * configures what are the base DOM objects for parsing
         *
         * @param  {Object} options
         * @return {DOM Node} node
         */
        getDOMContext: function( options ) {

            // if a node is passed
            if (options.node) {
                this.rootNode = options.node;
            }


            // if a html string is passed
            if (options.html) {
                // var domParser = new DOMParser();
                var domParser = this.getDOMParser();
                this.rootNode = domParser.parseFromString( options.html, 'text/html' );
            }


            // find top level document from rootnode
            if (this.rootNode !== null) {
                if (this.rootNode.nodeType === 9) {
                    this.document = this.rootNode;
                    this.rootNode = modules.domUtils.querySelector(this.rootNode, 'html');
                } else {
                    // if it's DOM node get parent DOM Document
                    this.document = modules.domUtils.ownerDocument(this.rootNode);
                }
            }


            // use global document object
            if (!this.rootNode && document) {
                this.rootNode = modules.domUtils.querySelector(document, 'html');
                this.document = document;
            }


            if (this.rootNode && this.document) {
                return {document: this.document, rootNode: this.rootNode};
            }

            return {document: null, rootNode: null};
        },



        /**
        * gets the first DOM node
        *
        * @param  {Dom Document}
        * @return {DOM Node} node
        */
        getTopMostNode: function( node ) {
            // var doc = this.ownerDocument(node);
            // if(doc && doc.nodeType && doc.nodeType === 9 && doc.documentElement){
            //  return doc.documentElement;
            // }
            return node;
        },



         /**
         * abstracts DOM ownerDocument
         *
         * @param  {DOM Node} node
         * @return {Dom Document}
         */
        ownerDocument: function(node) {
            return node.ownerDocument;
        },


        /**
         * abstracts DOM textContent
         *
         * @param  {DOM Node} node
         * @return {String}
         */
        textContent: function(node) {
            if (node.textContent) {
                return node.textContent;
            } else if (node.innerText) {
                return node.innerText;
            }
            return '';
        },


        /**
         * abstracts DOM innerHTML
         *
         * @param  {DOM Node} node
         * @return {String}
         */
        innerHTML: function(node) {
            return node.innerHTML;
        },


        /**
         * abstracts DOM hasAttribute
         *
         * @param  {DOM Node} node
         * @param  {String} attributeName
         * @return {Boolean}
         */
        hasAttribute: function(node, attributeName) {
            return node.hasAttribute(attributeName);
        },


        /**
         * does an attribute contain a value
         *
         * @param  {DOM Node} node
         * @param  {String} attributeName
         * @param  {String} value
         * @return {Boolean}
         */
        hasAttributeValue: function(node, attributeName, value) {
            return (this.getAttributeList(node, attributeName).indexOf(value) > -1);
        },


        /**
         * abstracts DOM getAttribute
         *
         * @param  {DOM Node} node
         * @param  {String} attributeName
         * @return {String || null}
         */
        getAttribute: function(node, attributeName) {
            return node.getAttribute(attributeName);
        },


        /**
         * abstracts DOM setAttribute
         *
         * @param  {DOM Node} node
         * @param  {String} attributeName
         * @param  {String} attributeValue
         */
        setAttribute: function(node, attributeName, attributeValue) {
            node.setAttribute(attributeName, attributeValue);
        },


        /**
         * abstracts DOM removeAttribute
         *
         * @param  {DOM Node} node
         * @param  {String} attributeName
         */
        removeAttribute: function(node, attributeName) {
            node.removeAttribute(attributeName);
        },


        /**
         * abstracts DOM getElementById
         *
         * @param  {DOM Node || DOM Document} node
         * @param  {String} id
         * @return {DOM Node}
         */
        getElementById: function(docNode, id) {
            return docNode.querySelector( '#' + id );
        },


        /**
         * abstracts DOM querySelector
         *
         * @param  {DOM Node || DOM Document} node
         * @param  {String} selector
         * @return {DOM Node}
         */
        querySelector: function(docNode, selector) {
            return docNode.querySelector( selector );
        },


        /**
         * get value of a Node attribute as an array
         *
         * @param  {DOM Node} node
         * @param  {String} attributeName
         * @return {Array}
         */
        getAttributeList: function(node, attributeName) {
            var out = [],
                attList;

            attList = node.getAttribute(attributeName);
            if (attList && attList !== '') {
                if (attList.indexOf(' ') > -1) {
                    out = attList.split(' ');
                } else {
                    out.push(attList);
                }
            }
            return out;
        },


        /**
         * gets all child nodes with a given attribute
         *
         * @param  {DOM Node} node
         * @param  {String} attributeName
         * @return {NodeList}
         */
        getNodesByAttribute: function(node, attributeName) {
            var selector = '[' + attributeName + ']';
            return node.querySelectorAll(selector);
        },


        /**
         * gets all child nodes with a given attribute containing a given value
         *
         * @param  {DOM Node} node
         * @param  {String} attributeName
         * @return {DOM NodeList}
         */
        getNodesByAttributeValue: function(rootNode, name, value) {
            var arr = [],
                x = 0,
                i,
                out = [];

            arr = this.getNodesByAttribute(rootNode, name);
            if (arr) {
                i = arr.length;
                while (x < i) {
                    if (this.hasAttributeValue(arr[x], name, value)) {
                        out.push(arr[x]);
                    }
                    x++;
                }
            }
            return out;
        },


        /**
         * gets attribute value from controlled list of tags
         *
         * @param  {Array} tagNames
         * @param  {String} attributeName
         * @return {String || null}
         */
        getAttrValFromTagList: function(node, tagNames, attributeName) {
            var i = tagNames.length;

            while (i--) {
                if (node.tagName.toLowerCase() === tagNames[i]) {
                    var attrValue = this.getAttribute(node, attributeName);
                    if (attrValue && attrValue !== '') {
                        return attrValue;
                    }
                }
            }
            return null;
        },


       /**
         * get node if it has no siblings. CSS equivalent is :only-child
         *
         * @param  {DOM Node} rootNode
         * @param  {Array} tagNames
         * @return {DOM Node || null}
         */
        getSingleDescendant: function(node) {
            return this.getDescendant( node, null, false );
        },


        /**
         * get node if it has no siblings of the same type. CSS equivalent is :only-of-type
         *
         * @param  {DOM Node} rootNode
         * @param  {Array} tagNames
         * @return {DOM Node || null}
         */
        getSingleDescendantOfType: function(node, tagNames) {
            return this.getDescendant( node, tagNames, true );
        },


        /**
         * get child node limited by presence of siblings - either CSS :only-of-type or :only-child
         *
         * @param  {DOM Node} rootNode
         * @param  {Array} tagNames
         * @return {DOM Node || null}
         */
        getDescendant: function( node, tagNames, onlyOfType ) {
            var i = node.children.length,
                countAll = 0,
                countOfType = 0,
                child,
                out = null;

            while (i--) {
                child = node.children[i];
                if (child.nodeType === 1) {
                    if (tagNames) {
                        // count just only-of-type
                        if (this.hasTagName(child, tagNames)) {
                            out = child;
                            countOfType++;
                        }
                    } else {
                        // count all elements
                        out = child;
                        countAll++;
                    }
                }
            }
            if (onlyOfType === true) {
                return (countOfType === 1)? out : null;
            }
            return (countAll === 1)? out : null;
        },


       /**
         * is a node one of a list of tags
         *
         * @param  {DOM Node} rootNode
         * @param  {Array} tagNames
         * @return {Boolean}
         */
        hasTagName: function(node, tagNames) {
            var i = tagNames.length;
            while (i--) {
                if (node.tagName.toLowerCase() === tagNames[i]) {
                    return true;
                }
            }
            return false;
        },


       /**
         * abstracts DOM appendChild
         *
         * @param  {DOM Node} node
         * @param  {DOM Node} childNode
         * @return {DOM Node}
         */
        appendChild: function(node, childNode) {
            return node.appendChild(childNode);
        },


       /**
         * abstracts DOM removeChild
         *
         * @param  {DOM Node} childNode
         * @return {DOM Node || null}
         */
        removeChild: function(childNode) {
            if (childNode.parentNode) {
                return childNode.parentNode.removeChild(childNode);
            }
            return null;
        },


        /**
         * abstracts DOM cloneNode
         *
         * @param  {DOM Node} node
         * @return {DOM Node}
         */
        clone: function(node) {
            var newNode = node.cloneNode(true);
            newNode.removeAttribute('id');
            return newNode;
        },


        /**
         * gets the text of a node
         *
         * @param  {DOM Node} node
         * @return {String}
         */
        getElementText: function( node ) {
            if (node && node.data) {
                return node.data;
            }
            return '';
        },


        /**
         * gets the attributes of a node - ordered by sequence in html
         *
         * @param  {DOM Node} node
         * @return {Array}
         */
        getOrderedAttributes: function( node ) {
            var nodeStr = node.outerHTML,
                attrs = [];

            for (var i = 0; i < node.attributes.length; i++) {
                var attr = node.attributes[i];
                    attr.indexNum = nodeStr.indexOf(attr.name);

                attrs.push( attr );
            }
            return attrs.sort( modules.utils.sortObjects( 'indexNum' ) );
        },


        /**
         * decodes html entities in given text
         *
         * @param  {DOM Document} doc
         * @param  String} text
         * @return {String}
         */
        decodeEntities: function( doc, text ) {
            // return text;
            return doc.createTextNode( text ).nodeValue;
        },


        /**
         * clones a DOM document
         *
         * @param  {DOM Document} document
         * @return {DOM Document}
         */
        cloneDocument: function( document ) {
            var newNode,
                newDocument = null;

            if ( this.canCloneDocument( document )) {
                newDocument = document.implementation.createHTMLDocument('');
                newNode = newDocument.importNode( document.documentElement, true );
                newDocument.replaceChild(newNode, newDocument.querySelector('html'));
            }
            return (newNode && newNode.nodeType && newNode.nodeType === 1)? newDocument : document;
        },


        /**
         * can environment clone a DOM document
         *
         * @param  {DOM Document} document
         * @return {Boolean}
         */
        canCloneDocument: function( document ) {
            return (document && document.importNode && document.implementation && document.implementation.createHTMLDocument);
        },


        /**
         * get the child index of a node. Used to create a node path
         *
         *   @param  {DOM Node} node
         *   @return {Int}
         */
        getChildIndex: function (node) {
            var parent = node.parentNode,
                i = -1,
                child;
            while (parent && (child = parent.childNodes[++i])) {
                 if (child === node) {
                     return i;
                 }
            }
            return -1;
        },


        /**
         * get a node's path
         *
         *   @param  {DOM Node} node
         *   @return {Array}
         */
        getNodePath: function  (node) {
            var parent = node.parentNode,
                path = [],
                index = this.getChildIndex(node);

          if (parent && (path = this.getNodePath(parent))) {
               if (index > -1) {
                   path.push(index);
               }
          }
          return path;
        },


        /**
         * get a node from a path.
         *
         *   @param  {DOM document} document
         *   @param  {Array} path
         *   @return {DOM Node}
         */
        getNodeByPath: function (document, path) {
            var node = document.documentElement,
                i = 0,
                index;
          while ((index = path[++i]) > -1) {
              node = node.childNodes[index];
          }
          return node;
        },


        /**
        * get an array/nodeList of child nodes
        *
        *   @param  {DOM node} node
        *   @return {Array}
        */
        getChildren: function( node ) {
            return node.children;
        },


        /**
        * create a node
        *
        *   @param  {String} tagName
        *   @return {DOM node}
        */
        createNode: function( tagName ) {
            return this.document.createElement(tagName);
        },


        /**
        * create a node with text content
        *
        *   @param  {String} tagName
        *   @param  {String} text
        *   @return {DOM node}
        */
        createNodeWithText: function( tagName, text ) {
            var node = this.document.createElement(tagName);
            node.innerHTML = text;
            return node;
        }



    };


    modules.url = {


        /**
         * creates DOM objects needed to resolve URLs
         */
        init: function() {
            // this._domParser = new DOMParser();
            this._domParser = modules.domUtils.getDOMParser();
            // do not use a head tag it does not work with IE9
            this._html = '<base id="base" href=""></base><a id="link" href=""></a>';
            this._nodes = this._domParser.parseFromString( this._html, 'text/html' );
            this._baseNode =  modules.domUtils.getElementById(this._nodes, 'base');
            this._linkNode =  modules.domUtils.getElementById(this._nodes, 'link');
        },


        /**
         * resolves url to absolute version using baseUrl
         *
         * @param  {String} url
         * @param  {String} baseUrl
         * @return {String}
         */
        resolve: function(url, baseUrl) {
            // use modern URL web API where we can
            if (modules.utils.isString(url) && modules.utils.isString(baseUrl) && url.indexOf('://') === -1) {
                // this try catch is required as IE has an URL object but no constuctor support
                // http://glennjones.net/articles/the-problem-with-window-url
                try {
                    var resolved = new URL(url, baseUrl).toString();
                    // deal with early Webkit not throwing an error - for Safari
                    if (resolved === '[object URL]') {
                        resolved = URI.resolve(baseUrl, url);
                    }
                    return resolved;
                } catch (e) {
                    // otherwise fallback to DOM
                    if (this._domParser === undefined) {
                        this.init();
                    }

                    // do not use setAttribute it does not work with IE9
                    this._baseNode.href = baseUrl;
                    this._linkNode.href = url;

                    // dont use getAttribute as it returns orginal value not resolved
                    return this._linkNode.href;
                }
            } else {
                if (modules.utils.isString(url)) {
                    return url;
                }
                return '';
            }
        },

    };


    /**
     * constructor
     * parses text to find just the date element of an ISO date/time string i.e. 2008-05-01
     *
     * @param  {String} dateString
     * @param  {String} format
     * @return {String}
     */
    modules.ISODate = function ( dateString, format ) {
        this.clear();

        this.format = (format)? format : 'auto'; // auto or W3C or RFC3339 or HTML5
        this.setFormatSep();

        // optional should be full iso date/time string
        if (arguments[0]) {
            this.parse(dateString, format);
        }
    };


    modules.ISODate.prototype = {


        /**
         * clear all states
         *
         */
        clear: function() {
            this.clearDate();
            this.clearTime();
            this.clearTimeZone();
            this.setAutoProfileState();
        },


        /**
         * clear date states
         *
         */
        clearDate: function() {
            this.dY = -1;
            this.dM = -1;
            this.dD = -1;
            this.dDDD = -1;
        },


        /**
         * clear time states
         *
         */
        clearTime: function() {
            this.tH = -1;
            this.tM = -1;
            this.tS = -1;
            this.tD = -1;
        },


        /**
         * clear timezone states
         *
         */
        clearTimeZone: function() {
            this.tzH = -1;
            this.tzM = -1;
            this.tzPN = '+';
            this.z = false;
        },


        /**
         * resets the auto profile state
         *
         */
        setAutoProfileState: function() {
            this.autoProfile = {
               sep: 'T',
               dsep: '-',
               tsep: ':',
               tzsep: ':',
               tzZulu: 'Z'
            };
        },


        /**
         * parses text to find ISO date/time string i.e. 2008-05-01T15:45:19Z
         *
         * @param  {String} dateString
         * @param  {String} format
         * @return {String}
         */
        parse: function( dateString, format ) {
            this.clear();

            var parts = [],
                tzArray = [],
                position = 0,
                datePart = '',
                timePart = '',
                timeZonePart = '';

            if (format) {
                this.format = format;
            }



            // discover date time separtor for auto profile
            // Set to 'T' by default
            if (dateString.indexOf('t') > -1) {
                this.autoProfile.sep = 't';
            }
            if (dateString.indexOf('z') > -1) {
                this.autoProfile.tzZulu = 'z';
            }
            if (dateString.indexOf('Z') > -1) {
                this.autoProfile.tzZulu = 'Z';
            }
            if (dateString.toUpperCase().indexOf('T') === -1) {
                this.autoProfile.sep = ' ';
            }


            dateString = dateString.toUpperCase().replace(' ', 'T');

            // break on 'T' divider or space
            if (dateString.indexOf('T') > -1) {
                parts = dateString.split('T');
                datePart = parts[0];
                timePart = parts[1];

                // zulu UTC
                if (timePart.indexOf( 'Z' ) > -1) {
                    this.z = true;
                }

                // timezone
                if (timePart.indexOf( '+' ) > -1 || timePart.indexOf( '-' ) > -1) {
                    tzArray = timePart.split( 'Z' ); // incase of incorrect use of Z
                    timePart = tzArray[0];
                    timeZonePart = tzArray[1];

                    // timezone
                    if (timePart.indexOf( '+' ) > -1 || timePart.indexOf( '-' ) > -1) {
                        position = 0;

                        if (timePart.indexOf( '+' ) > -1) {
                            position = timePart.indexOf( '+' );
                        } else {
                            position = timePart.indexOf( '-' );
                        }

                        timeZonePart = timePart.substring( position, timePart.length );
                        timePart = timePart.substring( 0, position );
                    }
                }

            } else {
                datePart = dateString;
            }

            if (datePart !== '') {
                this.parseDate( datePart );
                if (timePart !== '') {
                    this.parseTime( timePart );
                    if (timeZonePart !== '') {
                        this.parseTimeZone( timeZonePart );
                    }
                }
            }
            return this.toString( format );
        },


        /**
         * parses text to find just the date element of an ISO date/time string i.e. 2008-05-01
         *
         * @param  {String} dateString
         * @param  {String} format
         * @return {String}
         */
        parseDate: function( dateString, format ) {
            this.clearDate();

            var parts = [];

            // discover timezone separtor for auto profile // default is ':'
            if (dateString.indexOf('-') === -1) {
                this.autoProfile.tsep = '';
            }

            // YYYY-DDD
            parts = dateString.match( /(\d\d\d\d)-(\d\d\d)/ );
            if (parts) {
                if (parts[1]) {
                    this.dY = parts[1];
                }
                if (parts[2]) {
                    this.dDDD = parts[2];
                }
            }

            if (this.dDDD === -1) {
                // YYYY-MM-DD ie 2008-05-01 and YYYYMMDD ie 20080501
                parts = dateString.match( /(\d\d\d\d)?-?(\d\d)?-?(\d\d)?/ );
                if (parts[1]) {
                    this.dY = parts[1];
                }
                if (parts[2]) {
                    this.dM = parts[2];
                }
                if (parts[3]) {
                    this.dD = parts[3];
                }
            }
            return this.toString(format);
        },


        /**
         * parses text to find just the time element of an ISO date/time string i.e. 13:30:45
         *
         * @param  {String} timeString
         * @param  {String} format
         * @return {String}
         */
        parseTime: function( timeString, format ) {
            this.clearTime();
            var parts = [];

            // discover date separtor for auto profile // default is ':'
            if (timeString.indexOf(':') === -1) {
                this.autoProfile.tsep = '';
            }

            // finds timezone HH:MM:SS and HHMMSS  ie 13:30:45, 133045 and 13:30:45.0135
            parts = timeString.match( /(\d\d)?:?(\d\d)?:?(\d\d)?.?([0-9]+)?/ );
            if (parts[1]) {
                this.tH = parts[1];
            }
            if (parts[2]) {
                this.tM = parts[2];
            }
            if (parts[3]) {
                this.tS = parts[3];
            }
            if (parts[4]) {
                this.tD = parts[4];
            }
            return this.toTimeString(format);
        },


        /**
         * parses text to find just the time element of an ISO date/time string i.e. +08:00
         *
         * @param  {String} timeString
         * @param  {String} format
         * @return {String}
         */
        parseTimeZone: function( timeString, format ) {
            this.clearTimeZone();
            var parts = [];

            if (timeString.toLowerCase() === 'z') {
                this.z = true;
                // set case for z
                this.autoProfile.tzZulu = (timeString === 'z')? 'z' : 'Z';
            } else {

                // discover timezone separtor for auto profile // default is ':'
                if (timeString.indexOf(':') === -1) {
                    this.autoProfile.tzsep = '';
                }

                // finds timezone +HH:MM and +HHMM  ie +13:30 and +1330
                parts = timeString.match( /([\-\+]{1})?(\d\d)?:?(\d\d)?/ );
                if (parts[1]) {
                    this.tzPN = parts[1];
                }
                if (parts[2]) {
                    this.tzH = parts[2];
                }
                if (parts[3]) {
                    this.tzM = parts[3];
                }


            }
            this.tzZulu = 'z';
            return this.toTimeString( format );
        },


        /**
         * returns ISO date/time string in W3C Note, RFC 3339, HTML5, or auto profile
         *
         * @param  {String} format
         * @return {String}
         */
        toString: function( format ) {
            var output = '';

            if (format) {
                this.format = format;
            }
            this.setFormatSep();

            if (this.dY  > -1) {
                output = this.dY;
                if (this.dM > 0 && this.dM < 13) {
                    output += this.dsep + this.dM;
                    if (this.dD > 0 && this.dD < 32) {
                        output += this.dsep + this.dD;
                        if (this.tH > -1 && this.tH < 25) {
                            output += this.sep + this.toTimeString( format );
                        }
                    }
                }
                if (this.dDDD > -1) {
                    output += this.dsep + this.dDDD;
                }
            } else if (this.tH > -1) {
                output += this.toTimeString( format );
            }

            return output;
        },


        /**
         * returns just the time string element of an ISO date/time
         * in W3C Note, RFC 3339, HTML5, or auto profile
         *
         * @param  {String} format
         * @return {String}
         */
        toTimeString: function( format ) {
            var out = '';

            if (format) {
                this.format = format;
            }
            this.setFormatSep();

            // time can only be created with a full date
            if (this.tH) {
                if (this.tH > -1 && this.tH < 25) {
                    out += this.tH;
                    if (this.tM > -1 && this.tM < 61) {
                        out += this.tsep + this.tM;
                        if (this.tS > -1 && this.tS < 61) {
                            out += this.tsep + this.tS;
                            if (this.tD > -1) {
                                out += '.' + this.tD;
                            }
                        }
                    }



                    // time zone offset
                    if (this.z) {
                        out += this.tzZulu;
                    } else if (this.tzH && this.tzH > -1 && this.tzH < 25) {
                        out += this.tzPN + this.tzH;
                        if (this.tzM > -1 && this.tzM < 61) {
                            out += this.tzsep + this.tzM;
                        }
                    }
                }
            }
            return out;
        },


        /**
         * set the current profile to W3C Note, RFC 3339, HTML5, or auto profile
         *
         */
        setFormatSep: function() {
            switch ( this.format.toLowerCase() ) {
                case 'rfc3339':
                    this.sep = 'T';
                    this.dsep = '';
                    this.tsep = '';
                    this.tzsep = '';
                    this.tzZulu = 'Z';
                    break;
                case 'w3c':
                    this.sep = 'T';
                    this.dsep = '-';
                    this.tsep = ':';
                    this.tzsep = ':';
                    this.tzZulu = 'Z';
                    break;
                case 'html5':
                    this.sep = ' ';
                    this.dsep = '-';
                    this.tsep = ':';
                    this.tzsep = ':';
                    this.tzZulu = 'Z';
                    break;
                default:
                    // auto - defined by format of input string
                    this.sep = this.autoProfile.sep;
                    this.dsep = this.autoProfile.dsep;
                    this.tsep = this.autoProfile.tsep;
                    this.tzsep = this.autoProfile.tzsep;
                    this.tzZulu = this.autoProfile.tzZulu;
            }
        },


        /**
         * does current data contain a full date i.e. 2015-03-23
         *
         * @return {Boolean}
         */
        hasFullDate: function() {
            return (this.dY !== -1 && this.dM !== -1 && this.dD !== -1);
        },


        /**
         * does current data contain a minimum date which is just a year number i.e. 2015
         *
         * @return {Boolean}
         */
        hasDate: function() {
            return (this.dY !== -1);
        },


        /**
         * does current data contain a minimum time which is just a hour number i.e. 13
         *
         * @return {Boolean}
         */
        hasTime: function() {
            return (this.tH !== -1);
        },

        /**
         * does current data contain a minimum timezone i.e. -1 || +1 || z
         *
         * @return {Boolean}
         */
        hasTimeZone: function() {
            return (this.tzH !== -1);
        }

    };

    modules.ISODate.prototype.constructor = modules.ISODate;


    modules.dates = {


        /**
         * does text contain am
         *
         * @param  {String} text
         * @return {Boolean}
         */
        hasAM: function( text ) {
            text = text.toLowerCase();
            return (text.indexOf('am') > -1 || text.indexOf('a.m.') > -1);
        },


        /**
         * does text contain pm
         *
         * @param  {String} text
         * @return {Boolean}
         */
        hasPM: function( text ) {
            text = text.toLowerCase();
            return (text.indexOf('pm') > -1 || text.indexOf('p.m.') > -1);
        },


        /**
         * remove am and pm from text and return it
         *
         * @param  {String} text
         * @return {String}
         */
        removeAMPM: function( text ) {
            return text.replace('pm', '').replace('p.m.', '').replace('am', '').replace('a.m.', '');
        },


       /**
         * simple test of whether ISO date string is a duration  i.e.  PY17M or PW12
         *
         * @param  {String} text
         * @return {Boolean}
         */
        isDuration: function( text ) {
            if (modules.utils.isString( text )) {
                text = text.toLowerCase();
                if (modules.utils.startWith(text, 'p') ) {
                    return true;
                }
            }
            return false;
        },


       /**
         * is text a time or timezone
         * i.e. HH-MM-SS or z+-HH-MM-SS 08:43 | 15:23:00:0567 | 10:34pm | 10:34 p.m. | +01:00:00 | -02:00 | z15:00 | 0843
         *
         * @param  {String} text
         * @return {Boolean}
         */
        isTime: function( text ) {
            if (modules.utils.isString(text)) {
                text = text.toLowerCase();
                text = modules.utils.trim( text );
                // start with timezone char
                if ( text.match(':') && ( modules.utils.startWith(text, 'z') || modules.utils.startWith(text, '-')  || modules.utils.startWith(text, '+') )) {
                    return true;
                }
                // has ante meridiem or post meridiem
                if ( text.match(/^[0-9]/) &&
                    ( this.hasAM(text) || this.hasPM(text) )) {
                    return true;
                }
                // contains time delimiter but not datetime delimiter
                if ( text.match(':') && !text.match(/t|\s/) ) {
                    return true;
                }

                // if it's a number of 2, 4 or 6 chars
                if (modules.utils.isNumber(text)) {
                    if (text.length === 2 || text.length === 4 || text.length === 6) {
                        return true;
                    }
                }
            }
            return false;
        },


        /**
         * parses a time from text and returns 24hr time string
         * i.e. 5:34am = 05:34:00 and 1:52:04p.m. = 13:52:04
         *
         * @param  {String} text
         * @return {String}
         */
        parseAmPmTime: function( text ) {
            var out = text,
                times = [];

            // if the string has a text : or am or pm
            if (modules.utils.isString(out)) {
                // text = text.toLowerCase();
                text = text.replace(/[ ]+/g, '');

                if (text.match(':') || this.hasAM(text) || this.hasPM(text)) {

                    if (text.match(':')) {
                        times = text.split(':');
                    } else {
                        // single number text i.e. 5pm
                        times[0] = text;
                        times[0] = this.removeAMPM(times[0]);
                    }

                    // change pm hours to 24hr number
                    if (this.hasPM(text)) {
                        if (times[0] < 12) {
                            times[0] = parseInt(times[0], 10) + 12;
                        }
                    }

                    // add leading zero's where needed
                    if (times[0] && times[0].length === 1) {
                        times[0] = '0' + times[0];
                    }

                    // rejoin text elements together
                    if (times[0]) {
                        text = times.join(':');
                    }
                }
            }

            // remove am/pm strings
            return this.removeAMPM(text);
        },


       /**
         * overlays a time on a date to return the union of the two
         *
         * @param  {String} date
         * @param  {String} time
         * @param  {String} format ( Modules.ISODate profile format )
         * @return {Object} Modules.ISODate
         */
        dateTimeUnion: function(date, time, format) {
            var isodate = new modules.ISODate(date, format),
                isotime = new modules.ISODate();

            isotime.parseTime(this.parseAmPmTime(time), format);
            if (isodate.hasFullDate() && isotime.hasTime()) {
                isodate.tH = isotime.tH;
                isodate.tM = isotime.tM;
                isodate.tS = isotime.tS;
                isodate.tD = isotime.tD;
                return isodate;
            }
            if (isodate.hasFullDate()) {
                return isodate;
            }
            return new modules.ISODate();
        },


       /**
         * concatenate an array of date and time text fragments to create an ISODate object
         * used for microformat value and value-title rules
         *
         * @param  {Array} arr ( Array of Strings )
         * @param  {String} format ( Modules.ISODate profile format )
         * @return {Object} Modules.ISODate
         */
        concatFragments: function (arr, format) {
            var out = new modules.ISODate(),
                i = 0,
                value = '';

            // if the fragment already contains a full date just return it once
            if (arr[0].toUpperCase().match('T')) {
                return new modules.ISODate(arr[0], format);
            }
            for (i = 0; i < arr.length; i++) {
                value = arr[i];

                // date pattern
                if ( value.charAt(4) === '-' && out.hasFullDate() === false ) {
                    out.parseDate(value);
                }

                // time pattern
                if ( (value.indexOf(':') > -1 || modules.utils.isNumber( this.parseAmPmTime(value) )) && out.hasTime() === false ) {
                    // split time and timezone
                    var items = this.splitTimeAndZone(value);
                    value = items[0];

                    // parse any use of am/pm
                    value = this.parseAmPmTime(value);
                    out.parseTime(value);

                    // parse any timezone
                    if (items.length > 1) {
                         out.parseTimeZone(items[1], format);
                    }
                }

                // timezone pattern
                if (value.charAt(0) === '-' || value.charAt(0) === '+' || value.toUpperCase() === 'Z') {
                    if ( out.hasTimeZone() === false ) {
                        out.parseTimeZone(value);
                    }
                }

            }
            return out;
        },


       /**
         * parses text by splitting it into an array of time and timezone strings
         *
         * @param  {String} text
         * @return {Array} Modules.ISODate
         */
        splitTimeAndZone: function ( text ) {
           var out = [text],
               chars = ['-', '+', 'z', 'Z'],
               i = chars.length;

            while (i--) {
              if (text.indexOf(chars[i]) > -1) {
                  out[0] = text.slice( 0, text.indexOf(chars[i]) );
                  out.push( text.slice( text.indexOf(chars[i]) ) );
                  break;
               }
            }
           return out;
        }

    };


    modules.text = {

        // normalised or whitespace or whitespacetrimmed
        textFormat: 'whitespacetrimmed',

        // block level tags, used to add line returns
        blockLevelTags: ['h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'p', 'hr', 'pre', 'table',
            'address', 'article', 'aside', 'blockquote', 'caption', 'col', 'colgroup', 'dd', 'div',
            'dt', 'dir', 'fieldset', 'figcaption', 'figure', 'footer', 'form',  'header', 'hgroup', 'hr',
            'li', 'map', 'menu', 'nav', 'optgroup', 'option', 'section', 'tbody', 'testarea',
            'tfoot', 'th', 'thead', 'tr', 'td', 'ul', 'ol', 'dl', 'details'],

        // tags to exclude
        excludeTags: ['noframe', 'noscript', 'template', 'script', 'style', 'frames', 'frameset'],


        /**
         * parses the text from the DOM Node
         *
         * @param  {DOM Node} node
         * @param  {String} textFormat
         * @return {String}
         */
        parse: function(doc, node, textFormat) {
            var out;
            this.textFormat = (textFormat)? textFormat : this.textFormat;
            if (this.textFormat === 'normalised') {
                out = this.walkTreeForText( node );
                if (out !== undefined) {
                    return this.normalise( doc, out );
                }
                return '';
            }
            return this.formatText( doc, modules.domUtils.textContent(node), this.textFormat );
        },


        /**
         * parses the text from a html string
         *
         * @param  {DOM Document} doc
         * @param  {String} text
         * @param  {String} textFormat
         * @return {String}
         */
        parseText: function( doc, text, textFormat ) {
           var node = modules.domUtils.createNodeWithText( 'div', text );
           return this.parse( doc, node, textFormat );
        },


        /**
         * parses the text from a html string - only for whitespace or whitespacetrimmed formats
         *
         * @param  {String} text
         * @param  {String} textFormat
         * @return {String}
         */
        formatText: function( doc, text, textFormat ) {
           this.textFormat = (textFormat)? textFormat : this.textFormat;
           if (text) {
              var out = '',
                  regex = /(<([^>]+)>)/ig;

              out = text.replace(regex, '');
              if (this.textFormat === 'whitespacetrimmed') {
                 out = modules.utils.trimWhitespace( out );
              }

              // return entities.decode( out, 2 );
              return modules.domUtils.decodeEntities( doc, out );
           }
           return '';
        },


        /**
         * normalises whitespace in given text
         *
         * @param  {String} text
         * @return {String}
         */
        normalise: function( doc, text ) {
            text = text.replace( /&nbsp;/g, ' ') ;    // exchanges html entity for space into space char
            text = modules.utils.collapseWhiteSpace( text );     // removes linefeeds, tabs and addtional spaces
            text = modules.domUtils.decodeEntities( doc, text );  // decode HTML entities
            text = text.replace( '–', '-' );          // correct dash decoding
            return modules.utils.trim( text );
        },


        /**
         * walks DOM tree parsing the text from DOM Nodes
         *
         * @param  {DOM Node} node
         * @return {String}
         */
        walkTreeForText: function( node ) {
            var out = '',
                j = 0;

            if (node.tagName && this.excludeTags.indexOf( node.tagName.toLowerCase() ) > -1) {
                return out;
            }

            // if node is a text node get its text
            if (node.nodeType && node.nodeType === 3) {
                out += modules.domUtils.getElementText( node );
            }

            // get the text of the child nodes
            if (node.childNodes && node.childNodes.length > 0) {
                for (j = 0; j < node.childNodes.length; j++) {
                    var text = this.walkTreeForText( node.childNodes[j] );
                    if (text !== undefined) {
                        out += text;
                    }
                }
            }

            // if it's a block level tag add an additional space at the end
            if (node.tagName && this.blockLevelTags.indexOf( node.tagName.toLowerCase() ) !== -1) {
                out += ' ';
            }

            return (out === '')? undefined : out ;
        }

    };


    modules.html = {

        // elements which are self-closing
        selfClosingElt: ['area', 'base', 'br', 'col', 'hr', 'img', 'input', 'link', 'meta', 'param', 'command', 'keygen', 'source'],


        /**
         * parse the html string from DOM Node
         *
         * @param  {DOM Node} node
         * @return {String}
         */
        parse: function( node ) {
            var out = '',
                j = 0;

            // we do not want the outer container
            if (node.childNodes && node.childNodes.length > 0) {
                for (j = 0; j < node.childNodes.length; j++) {
                    var text = this.walkTreeForHtml( node.childNodes[j] );
                    if (text !== undefined) {
                        out += text;
                    }
                }
            }

            return out;
        },


        /**
         * walks the DOM tree parsing the html string from the nodes
         *
         * @param  {DOM Document} doc
         * @param  {DOM Node} node
         * @return {String}
         */
        walkTreeForHtml: function( node ) {
            var out = '',
                j = 0;

            // if node is a text node get its text
            if (node.nodeType && node.nodeType === 3) {
                out += modules.domUtils.getElementText( node );
            }


            // exclude text which has been added with include pattern  -
            if (node.nodeType && node.nodeType === 1 && modules.domUtils.hasAttribute(node, 'data-include') === false) {

                // begin tag
                out += '<' + node.tagName.toLowerCase();

                // add attributes
                var attrs = modules.domUtils.getOrderedAttributes(node);
                for (j = 0; j < attrs.length; j++) {
                    out += ' ' + attrs[j].name +  '=' + '"' + attrs[j].value + '"';
                }

                if (this.selfClosingElt.indexOf(node.tagName.toLowerCase()) === -1) {
                    out += '>';
                }

                // get the text of the child nodes
                if (node.childNodes && node.childNodes.length > 0) {

                    for (j = 0; j < node.childNodes.length; j++) {
                        var text = this.walkTreeForHtml( node.childNodes[j] );
                        if (text !== undefined) {
                            out += text;
                        }
                    }
                }

                // end tag
                if (this.selfClosingElt.indexOf(node.tagName.toLowerCase()) > -1) {
                    out += ' />';
                } else {
                    out += '</' + node.tagName.toLowerCase() + '>';
                }
            }

            return (out === '')? undefined : out;
        }


    };


    modules.maps['h-adr'] = {
        root: 'adr',
        name: 'h-adr',
        properties: {
            'post-office-box': {},
            'street-address': {},
            'extended-address': {},
            'locality': {},
            'region': {},
            'postal-code': {},
            'country-name': {}
        }
    };


    modules.maps['h-card'] =  {
        root: 'vcard',
        name: 'h-card',
        properties: {
            'fn': {
                'map': 'p-name'
            },
            'adr': {
                'map': 'p-adr',
                'uf': ['h-adr']
            },
            'agent': {
                'uf': ['h-card']
            },
            'bday': {
                'map': 'dt-bday'
            },
            'class': {},
            'category': {
                'map': 'p-category',
                'relAlt': ['tag']
            },
            'email': {
                'map': 'u-email'
            },
            'geo': {
                'map': 'p-geo',
                'uf': ['h-geo']
            },
            'key': {
                'map': 'u-key'
            },
            'label': {},
            'logo': {
                'map': 'u-logo'
            },
            'mailer': {},
            'honorific-prefix': {},
            'given-name': {},
            'additional-name': {},
            'family-name': {},
            'honorific-suffix': {},
            'nickname': {},
            'note': {}, // could be html i.e. e-note
            'org': {},
            'p-organization-name': {},
            'p-organization-unit': {},
            'photo': {
                'map': 'u-photo'
            },
            'rev': {
                'map': 'dt-rev'
            },
            'role': {},
            'sequence': {},
            'sort-string': {},
            'sound': {
                'map': 'u-sound'
            },
            'title': {
                'map': 'p-job-title'
            },
            'tel': {},
            'tz': {},
            'uid': {
                'map': 'u-uid'
            },
            'url': {
                'map': 'u-url'
            }
        }
    };


    modules.maps['h-entry'] = {
        root: 'hentry',
        name: 'h-entry',
        properties: {
            'entry-title': {
                'map': 'p-name'
            },
            'entry-summary': {
                'map': 'p-summary'
            },
            'entry-content': {
                'map': 'e-content'
            },
            'published': {
                'map': 'dt-published'
            },
            'updated': {
                'map': 'dt-updated'
            },
            'author': {
                'uf': ['h-card']
            },
            'category': {
                'map': 'p-category',
                'relAlt': ['tag']
            },
            'geo': {
                'map': 'p-geo',
                'uf': ['h-geo']
            },
            'latitude': {},
            'longitude': {},
            'url': {
                'map': 'u-url',
                'relAlt': ['bookmark']
            }
        }
    };


    modules.maps['h-event'] = {
        root: 'vevent',
        name: 'h-event',
        properties: {
            'summary': {
                'map': 'p-name'
            },
            'dtstart': {
                'map': 'dt-start'
            },
            'dtend': {
                'map': 'dt-end'
            },
            'description': {},
            'url': {
                'map': 'u-url'
            },
            'category': {
                'map': 'p-category',
                'relAlt': ['tag']
            },
            'location': {
                'uf': ['h-card']
            },
            'geo': {
                'uf': ['h-geo']
            },
            'latitude': {},
            'longitude': {},
            'duration': {
                'map': 'dt-duration'
            },
            'contact': {
                'uf': ['h-card']
            },
            'organizer': {
                'uf': ['h-card']},
            'attendee': {
                'uf': ['h-card']},
            'uid': {
                'map': 'u-uid'
            },
            'attach': {
                'map': 'u-attach'
            },
            'status': {},
            'rdate': {},
            'rrule': {}
        }
    };


    modules.maps['h-feed'] = {
        root: 'hfeed',
        name: 'h-feed',
        properties: {
            'category': {
                'map': 'p-category',
                'relAlt': ['tag']
            },
            'summary': {
                'map': 'p-summary'
            },
            'author': {
                'uf': ['h-card']
            },
            'url': {
                'map': 'u-url'
            },
            'photo': {
                'map': 'u-photo'
            },
        }
    };


    modules.maps['h-geo'] = {
        root: 'geo',
        name: 'h-geo',
        properties: {
            'latitude': {},
            'longitude': {}
        }
    };


    modules.maps['h-item'] = {
        root: 'item',
        name: 'h-item',
        subTree: false,
        properties: {
            'fn': {
                'map': 'p-name'
            },
            'url': {
                'map': 'u-url'
            },
            'photo': {
                'map': 'u-photo'
            }
        }
    };


    modules.maps['h-listing'] = {
            root: 'hlisting',
            name: 'h-listing',
            properties: {
                'version': {},
                'lister': {
                    'uf': ['h-card']
                },
                'dtlisted': {
                    'map': 'dt-listed'
                },
                'dtexpired': {
                    'map': 'dt-expired'
                },
                'location': {},
                'price': {},
                'item': {
                    'uf': ['h-card', 'a-adr', 'h-geo']
                },
                'summary': {
                    'map': 'p-name'
                },
                'description': {
                    'map': 'e-description'
                },
                'listing': {}
            }
        };


    modules.maps['h-news'] = {
            root: 'hnews',
            name: 'h-news',
            properties: {
                'entry': {
                    'uf': ['h-entry']
                },
                'geo': {
                    'uf': ['h-geo']
                },
                'latitude': {},
                'longitude': {},
                'source-org': {
                    'uf': ['h-card']
                },
                'dateline': {
                    'uf': ['h-card']
                },
                'item-license': {
                    'map': 'u-item-license'
                },
                'principles': {
                    'map': 'u-principles',
                    'relAlt': ['principles']
                }
            }
        };


    modules.maps['h-org'] = {
        root: 'h-x-org',  // drop this from v1 as it causes issue with fn org hcard pattern
        name: 'h-org',
        childStructure: true,
        properties: {
            'organization-name': {},
            'organization-unit': {}
        }
    };


    modules.maps['h-product'] = {
            root: 'hproduct',
            name: 'h-product',
            properties: {
                'brand': {
                    'uf': ['h-card']
                },
                'category': {
                    'map': 'p-category',
                    'relAlt': ['tag']
                },
                'price': {},
                'description': {
                    'map': 'e-description'
                },
                'fn': {
                    'map': 'p-name'
                },
                'photo': {
                    'map': 'u-photo'
                },
                'url': {
                    'map': 'u-url'
                },
                'review': {
                    'uf': ['h-review', 'h-review-aggregate']
                },
                'listing': {
                    'uf': ['h-listing']
                },
                'identifier': {
                    'map': 'u-identifier'
                }
            }
        };


    modules.maps['h-recipe'] = {
            root: 'hrecipe',
            name: 'h-recipe',
            properties: {
                'fn': {
                    'map': 'p-name'
                },
                'ingredient': {
                    'map': 'e-ingredient'
                },
                'yield': {},
                'instructions': {
                    'map': 'e-instructions'
                },
                'duration': {
                    'map': 'dt-duration'
                },
                'photo': {
                    'map': 'u-photo'
                },
                'summary': {},
                'author': {
                    'uf': ['h-card']
                },
                'published': {
                    'map': 'dt-published'
                },
                'nutrition': {},
                'category': {
                    'map': 'p-category',
                    'relAlt': ['tag']
                },
            }
        };


    modules.maps['h-resume'] = {
        root: 'hresume',
        name: 'h-resume',
        properties: {
            'summary': {},
            'contact': {
                'uf': ['h-card']
            },
            'education': {
                'uf': ['h-card', 'h-event']
            },
            'experience': {
                'uf': ['h-card', 'h-event']
            },
            'skill': {},
            'affiliation': {
                'uf': ['h-card']
            }
        }
    };


    modules.maps['h-review-aggregate'] = {
        root: 'hreview-aggregate',
        name: 'h-review-aggregate',
        properties: {
            'summary': {
                'map': 'p-name'
            },
            'item': {
                'map': 'p-item',
                'uf': ['h-item', 'h-geo', 'h-adr', 'h-card', 'h-event', 'h-product']
            },
            'rating': {},
            'average': {},
            'best': {},
            'worst': {},
            'count': {},
            'votes': {},
            'category': {
                'map': 'p-category',
                'relAlt': ['tag']
            },
            'url': {
                'map': 'u-url',
                'relAlt': ['self', 'bookmark']
            }
        }
    };


    modules.maps['h-review'] = {
        root: 'hreview',
        name: 'h-review',
        properties: {
            'summary': {
                'map': 'p-name'
            },
            'description': {
                'map': 'e-description'
            },
            'item': {
                'map': 'p-item',
                'uf': ['h-item', 'h-geo', 'h-adr', 'h-card', 'h-event', 'h-product']
            },
            'reviewer': {
                'uf': ['h-card']
            },
            'dtreviewer': {
                'map': 'dt-reviewer'
            },
            'rating': {},
            'best': {},
            'worst': {},
            'category': {
                'map': 'p-category',
                'relAlt': ['tag']
            },
            'url': {
                'map': 'u-url',
                'relAlt': ['self', 'bookmark']
            }
        }
    };


    modules.rels = {
        // xfn
        'friend': [ 'yes', 'external'],
        'acquaintance': [ 'yes', 'external'],
        'contact': [ 'yes', 'external'],
        'met': [ 'yes', 'external'],
        'co-worker': [ 'yes', 'external'],
        'colleague': [ 'yes', 'external'],
        'co-resident': [ 'yes', 'external'],
        'neighbor': [ 'yes', 'external'],
        'child': [ 'yes', 'external'],
        'parent': [ 'yes', 'external'],
        'sibling': [ 'yes', 'external'],
        'spouse': [ 'yes', 'external'],
        'kin': [ 'yes', 'external'],
        'muse': [ 'yes', 'external'],
        'crush': [ 'yes', 'external'],
        'date': [ 'yes', 'external'],
        'sweetheart': [ 'yes', 'external'],
        'me': [ 'yes', 'external'],

        // other rel=*
        'license': [ 'yes', 'yes'],
        'nofollow': [ 'no', 'external'],
        'tag': [ 'no', 'yes'],
        'self': [ 'no', 'external'],
        'bookmark': [ 'no', 'external'],
        'author': [ 'no', 'external'],
        'home': [ 'no', 'external'],
        'directory': [ 'no', 'external'],
        'enclosure': [ 'no', 'external'],
        'pronunciation': [ 'no', 'external'],
        'payment': [ 'no', 'external'],
        'principles': [ 'no', 'external']

    };



    var External = {
        version: modules.version,
        livingStandard: modules.livingStandard
    };


    External.get = function(options) {
        var parser = new modules.Parser();
        addV1(parser, options);
        return parser.get( options );
    };


    External.getParent = function(node, options) {
        var parser = new modules.Parser();
        addV1(parser, options);
        return parser.getParent( node, options );
    };


    External.count = function(options) {
        var parser = new modules.Parser();
        addV1(parser, options);
        return parser.count( options );
    };


    External.isMicroformat = function( node, options ) {
        var parser = new modules.Parser();
        addV1(parser, options);
        return parser.isMicroformat( node, options );
    };


    External.hasMicroformats = function( node, options ) {
        var parser = new modules.Parser();
        addV1(parser, options);
        return parser.hasMicroformats( node, options );
    };


    function addV1(parser, options) {
        if (options && options.maps) {
            if (Array.isArray(options.maps)) {
                parser.add(options.maps);
            } else {
                parser.add([options.maps]);
            }
        }
    }


    return External;


}));
try {
    // mozilla jsm support
    Components.utils.importGlobalProperties(["URL"]);
} catch (e) {}
this.EXPORTED_SYMBOLS = ['Microformats'];