Dette scriptet burde ikke installeres direkte. Det er et bibliotek for andre script å inkludere med det nye metadirektivet // @require https://update.greasyfork.org/scripts/17641/111510/rsel-exprparser-basic.js
      
  // ==UserScript==
// @name            rsel-exprparser-basic
// @namespace       https://greasyfork.org/users/11629-TheLastTaterTot
// @version         0.2.6
// @description     Parses RSel-specific expression text and rebuilds it in the UI.
// @author          TheLastTaterTot
// @include         https://editor-beta.waze.com/*editor/*
// @include         https://www.waze.com/*editor/*
// @exclude         https://www.waze.com/*user/editor/*
// @grant           none
// @run-at          document-end
// ==/UserScript==
// Main usage: RSelExprParser.updateExpression(<rsel expression text>)
var RSelExprParser = {
    new__EXPR_DEBUGINFO: function(m, exprWord, exprPhrase) {
        return {
            m: m,
            exprMatches: exprWord,
            exprMatchPhrases: exprPhrase,
            exprBuild: {},
            err: null ,
            errorMsg: null
        };
    },
    //reset for error debugging
    _rsel: {
        getSelectionIndex: function(selector, selText) {
            return selector.map(function(i) {
                if (new RegExp(selText,'i').test(this.innerText))
                    return this.value
            }).get(0);
        },
        getSelectOptions: function(selector) {
            var opts = [];
            selector.map(function(i, a) {
                opts.push(a.innerText.toLowerCase());
            });
            return opts;
        },
        getNewExprBuild: function() {
            return {
                cond: null ,
                op: null ,
                op2: null ,
                val: null ,
                val2: null ,
                condmod: null ,
                errorCode: 0
            }
        }
    },
    /*Using RSel DOM elements rather than requesting dev to provide direct modifiction of RSel's expr object.
        This is so the RSel dev can feel free to significantly change his object storage structure if needed. */
    rselBtns: {
        lfParens: function() {
            try {
                $('#btnRSLBkt').click();
            } catch (err) {}
        },
        rtParens: function() {
            try {
                $('#btnRSRBkt').click();
            } catch (err) {}
        },
        and: function() {
            try {
                $('#btnRSAnd').click()
            } catch (err) {}
        },
        or: function() {
            try {
                $('#btnRSOr').click()
            } catch (err) {}
        },
        not: function() {
            try {
                $('#btnRSNot').click()
            } catch (err) {}
        },
        clear: function() {
            try {
                $('#btnRSClear').click()
            } catch (err) {}
        }
    },
    rselCond: {
        country: {
            op: function(selText) {
                $('#opRSCountry').val(RSelExprParser._rsel.getSelectionIndex($('#opRSCountry option'), selText));
            },
            val: function(selText) {
                $('#selRSCountry').val(RSelExprParser._rsel.getSelectionIndex($('#selRSCountry option'), selText));
            },
            add: function() {
                $('#btnRSAddCountry').click();
            }
        },
        state: {
            op: function(selText) {
                $('#opRSState').val(RSelExprParser._rsel.getSelectionIndex($('#opRSState option'), selText));
            },
            val: function(val) {
                $('#inRSState').val(val);
            },
            add: function() {
                $('#btnRSAddState').click();
            }
        },
        city: {
            op: function(selText) {
                $('#opRSCity').val(RSelExprParser._rsel.getSelectionIndex($('#opRSCity option'), selText));
            },
            val: function(val) {
                $('#inRSCity').val(val);
            },
            condmod: function(val) {
                $('#selRSAltCity').val(val);
            },
            add: function() {
                $('#btnRSAddCity').click();
            }
        },
        street: {
            op: function(selText) {
                $('#opRSStreet').val(RSelExprParser._rsel.getSelectionIndex($('#opRSStreet option'), selText));
            },
            val: function(val) {
                $('#inRSStreet').val(val);
            },
            condmod: function(val) {
                $('#selRSAlttStreet').val(val);
            },
            add: function() {
                $('#btnRSAddStreet').click();
            }
        },
        unnamed: {
            op: function(chkVal) {
                $('#cbRSNoName').prop('checked', chkVal);
            },
            //checked - has no name
            op2: function(chkVal) {
                $('#cbRSAltNoName').prop('checked', chkVal);
            },
            //checked - alt name
            add: function() {
                $('#btnRSAddNoName').click();
            }
        },
        road: {
            op: function(selText) {
                $('#opRSRoadType').val(RSelExprParser._rsel.getSelectionIndex($('#opRSRoadType option'), selText));
            },
            val: function(selText) {
                $('#selRSRoadType').val(RSelExprParser._rsel.getSelectionIndex($('#selRSRoadType option'), selText));
            },
            add: function() {
                $('#btnRSAddRoadType').click();
            }
        },
        direction: {
            op: function(selText) {
                $('#opRSDirection').val(RSelExprParser._rsel.getSelectionIndex($('#opRSDirection option'), selText));
            },
            val: function(selText) {
                $('#selRSDirection').val(RSelExprParser._rsel.getSelectionIndex($('#selRSDirection option'), selText));
            },
            add: function() {
                $('#btnRSAddDirection').click();
            }
        },
        elevation: {
            op: function(selText) {
                $('#opRSElevation').val(RSelExprParser._rsel.getSelectionIndex($('#opRSElevation option'), selText));
            },
            val: function(selText) {
                $('#selRSElevation').val(RSelExprParser._rsel.getSelectionIndex($('#selRSElevation option'), selText));
            },
            add: function() {
                $('#btnRSAddElevation').click();
            }
        },
        manlock: {
            op: function(selText) {
                $('#opRSManLock').val(RSelExprParser._rsel.getSelectionIndex($('#opRSManLock option'), selText));
            },
            val: function(val) {
                $('#selRSManLock').val(val);
            },
            add: function() {
                $('#btnRSAddManLock').click();
            }
        },
        traflock: {
            op: function(selText) {
                $('#opRSTrLock').val(RSelExprParser._rsel.getSelectionIndex($('#opRSTrLock option'), selText));
            },
            val: function(val) {
                $('#selRSTrLock').val(val);
            },
            add: function() {
                $('#btnRSAddTrLock').click();
            }
        },
        speed: {
            opOptNodes: $('#opRSSpeed option'),
            op: function(selText) {
                $('#opRSSpeed').val(RSelExprParser._rsel.getSelectionIndex($('#opRSSpeed option'), selText));
            },
            val: function(val) {
                $('#inRSSpeed').val(val);
            },
            add: function() {
                $('#btnRSAddSpeed').click();
            }
        },
        closure: {
            op: function(checked) {
                $('#cbRSClsr').prop('checked', checked);
            },
            op2: function(selText) {
                $('#opRSClsrStrtEnd').val(RSelExprParser._rsel.getSelectionIndex($('#opRSClsrStrtEnd option'), selText));
            },
            val: function(val) {
                $('#inRSClsrDays').val(val);
            },
            condmod: function(selText) {
                $('#opRSClsrBeforeAter').val(RSelExprParser._rsel.getSelectionIndex($('#opRSClsrBeforeAter option'), selText));
            },
            add: function() {
                $('#btnRSAddClsr').click();
            }
        },
        updatedby: {
            op: function(selText) {
                $('#opRSUpdtd').val(RSelExprParser._rsel.getSelectionIndex($('#opRSUpdtd option'), selText));
            },
            val: function(val) {
                $('#inRSUpdtd').val(val);
            },
            add: function() {
                $('#btnRSAddUpdtd').click();
            }
        },
        createdby: {
            op: function(selText) {
                $('#opRSCrtd').val(RSelExprParser._rsel.getSelectionIndex($('#opRSCrtd option'), selText));
            },
            val: function(val) {
                $('#inRSCrtd').val(val);
            },
            add: function() {
                $('#btnRSAddCrtd').click();
            }
        },
        last: {
            op: function(selText) {
                $('#opRSLastU').val(RSelExprParser._rsel.getSelectionIndex($('#opRSLastU option'), selText));
            },
            val: function(val) {
                $('#inRSLastU').val(val);
            },
            add: function() {
                $('#btnRSAddLastU').click();
            }
        },
        length: {
            op: function(selText) {
                $('#opRSLength').val(RSelExprParser._rsel.getSelectionIndex($('#opRSLength option'), selText));
            },
            val: function(val) {
                $('#inRSLength').val(val);
            },
            condmod: function(selText) {
                $('#unitRSLength').val(RSelExprParser._rsel.getSelectionIndex($('#unitRSLength option'), selText));
            },
            add: function() {
                $('#btnRSAddLength').click();
            }
        },
        id: {
            op: function(selText) {
                $('#opRSSegId').val(RSelExprParser._rsel.getSelectionIndex($('#opRSSegId option'), selText));
            },
            val: function(val) {
                $('#inRSSegId').val(val);
            },
            add: function() {
                $('#btnRSAddSegId').click();
            }
        },
        roundabout: {
            op: function(checked) {
                $('#cbRSIsRound').prop('checked', checked);
            },
            add: function() {
                $('#btnRSAddIsRound').click();
            }
        },
        toll: {
            op: function(checked) {
                $('#cbRSIsToll').prop('checked', checked);
            },
            add: function() {
                $('#btnRSAddIsToll').click();
            }
        },
        tunnel: {
            op: function(checked) {
                $('#cbRSTunnel').prop('checked', checked);
            },
            add: function() {
                $('#btnRSAddTunnel').click();
            }
        },
        new: {
            op: function(checked) {
                $('#cbRSIsNew').prop('checked', checked);
            },
            add: function() {
                $('#btnRSAddIsNew').click();
            }
        },
        changed: {
            op: function(checked) {
                $('#cbRSIsChngd').prop('checked', checked);
            },
            add: function() {
                $('#btnRSAddIsChngd').click();
            }
        },
        screen: {
            op: function(checked) {
                $('#cbRSOnScr').prop('checked', checked);
            },
            add: function() {
                $('#btnRSAddOnScr').click();
            }
        },
        restriction: {
            op: function(checked) {
                $('#cbRSRestr').prop('checked', checked);
            },
            add: function() {
                $('#btnRSAddRestr').click();
            }
        },
        editable: {
            op: function(checked) {
                $('#cbRSEdtbl').prop('checked', checked);
            },
            add: function() {
                $('#btnRSAddEdtbl').click();
            }
        }
    },
    addExpr: function(eb) {
        var checkKeys = false;
        Object.keys(this.rselCond).map(function(a, i) {
            if (a === eb.cond)
                checkKeys = true;
        });
        if (checkKeys) {
            try {
                this.rselCond[eb.cond].op(eb.op);
                if (eb.op2 !== null )
                    this.rselCond[eb.cond].op2(eb.op2);
                if (eb.condmod !== null )
                    this.rselCond[eb.cond].condmod(eb.condmod);
                if (eb.val2 === null ) {
                    if (eb.val !== null )
                        this.rselCond[eb.cond].val(eb.val);
                    this.rselCond[eb.cond].add();
                } else {
                    this.rselBtns.lfParens();
                    this.rselCond[eb.cond].val(eb.val);
                    this.rselCond[eb.cond].add();
                    this.rselBtns.or();
                    this.rselCond[eb.cond].val(eb.val2);
                    this.rselCond[eb.cond].add();
                    this.rselBtns.rtParens();
                }
            } catch (err) {
                return {
                    errorCode: 101,
                    errorMsg: 'Error: Unable to parse expression text.',
                    err: err
                };
            }
        } else {
            return {
                errorCode: 3,
                errorMsg: 'Selection condition was not recognized'
            };
            //
        }
        return {
            errorCode: 0
        };
    },
    //=============================================================================
    parseExpr: function(parseThis) {
        //---------------------------------------------------------------
        parseThis = parseThis.replace(/\bpri?m?(?:ary|\.)?\s?(?:or)\s?alt(?:ern|s)?(?:\.)?/ig, 'any');
        parseThis = parseThis.replace(/\b((?:un)?name[ds]?)\b|\b(road) type\b|\b(last) update\b|\b(speed) limits?\b/ig, '$1$2$3$4')
        parseThis = parseThis.replace(/\b(man)ual (lock)s?\b|\b(traf)[fic]* (lock)s?\b/ig, '$1$2$3$4');
        parseThis = parseThis.replace(/\b(created|updated)\s(by)\b/ig, '$1$2');
        parseThis = parseThis.replace(/\bon screen/ig, 'onscreen');
        //\b(?:in|on|off|out|outside)(?: of)?[- ]?screen\b
        parseThis = parseThis.replace(/\b(?:off|out)(?: of)?[- ]?screen/ig, 'offscreen');
        var parseExprArray = parseThis.match(
        /(\(['"].*?['"]\)|".*?"|'.*?')|\bno[\s-]alt|\b(?:street[\s-]?)?name\(s\)|\bstreet(?:\snames?)\b|\btoll(?:[-\s]?ro?a?d)?\b|\bdoes(?:\s?n[o']t)\b|(?:!\s?)?contains?\b|!=|>=|<=|[ab][<->]{2}[ab]|\w+(\(s\))?|&&|\|\||!=|[|&<>=()!~]/gi
        ),
            parseExprHistory = [],
            condMatches = [],
            condMatchPhrases = [],
            exprMatches = [],
            exprMatchPhrases = [],
            exprFragment, unwantedWordsSearch,
            e, f, b, fLength;
        // The following parses the expression text into unique chunks within separate array elements
        e = parseExprArray.length;
        while (e-- > 0) {
            try {
                exprFragment = parseExprArray.shift();
                //console.info(exprFragment);
                // Find operators that join individual expressions (AND|OR|!|parenthesis)
                if (/^(?:and|or|&&|\|\||!=|[=&|()!])$/i.test(exprFragment)) {
                    exprMatches.push(exprFragment.toLowerCase());
                    exprMatchPhrases.push(exprFragment.toLowerCase());
                }
                // Identify elements that contain selection condition names
                if (
                /^country|^state|^city|^street|^(?:un|street[\s-]?)?name|^road|^round|^toll|^speed|^dir|^elevation|^tun|^manlock|^traflock|^speed|^new|^changed|screen$|^restrict|^clos|^createdby|^last|^updatedby|^length|^id|^editable/i
                .test(exprFragment)) {
                    condMatches.push(exprFragment.toLowerCase());
                    // lists specific selection conditions
                    exprMatches.push(exprFragment.toLowerCase());
                    //same as condMatches, but includes operations as separate array elements
                    try {
                        //search phrase fowards
                        fLength = parseExprArray.length;
                        f = 0;
                        while (!(/^(and|or|&&|\|\||[&|)])$/i.test(parseExprArray[f])) && (++f < fLength)) {}
                        //search phrase backwards
                        b = parseExprHistory.length;
                        while (!(/^(and|or|&&|\|\||[&|(])$/i.test(parseExprHistory[b - 1])) && (--b > 0)) {}
                        condMatchPhrases.push(parseExprHistory.slice(b).concat(exprFragment, parseExprArray.slice(0, f)));
                        //list specific selection conditions and its criteria
                        unwantedWordsSearch = parseExprHistory.slice(b);
                        if (unwantedWordsSearch && unwantedWordsSearch.length) {
                            unwantedWordsSearch = unwantedWordsSearch.filter(function(a) {
                                return !/\b(has|have|is|=|are|does|was|were)\b/i.test(a)
                            });
                        }
                        if (/!|!=/.test(unwantedWordsSearch[0]))
                            unwantedWordsSearch.splice(0, 1);
                        exprMatchPhrases.push(unwantedWordsSearch.concat(parseExprArray.slice(0, f)));
                        //excludes the match cond
                        parseExprHistory = parseExprHistory.concat(exprFragment, parseExprArray.slice(0, f));
                        parseExprArray = parseExprArray.slice(f);
                        e -= f;
                    } catch (err) {
                        return {
                            errorCode: 101,
                            errorMsg: 'Error parsing expression at ' + exprFragment,
                            err: err
                        };
                    }
                } else {
                    parseExprHistory.push(exprFragment);
                }
            } catch (err) {
                return {
                    errorCode: 101,
                    errdebug: 'Error parsing expression at ' + exprFragment,
                    err: err
                };
            }
        }
        //while
        //---------------------------------------------------------------
        // Quick crude check for unmatched parentheses
        var nOpenParens = exprMatches.toString().match(/\(/g),
        nCloseParens = exprMatches.toString().match(/\)/g);
        if (!nOpenParens) nOpenParens = [];
        if (!nCloseParens) nCloseParens = [];
        if (nOpenParens.length !== nCloseParens.length)
            return {
                errorCode: 1,
                errorMsg: 'Warning: Open and close paretheses may be unmatched.'
            };
        //---------------------------------------------------------------
        return {
            errorCode: 0,
            exprMatches: exprMatches,
            exprMatchPhrases: exprMatchPhrases,
            condMatches: condMatches,
            condMatchPhrases: condMatchPhrases
        };
    },
    buildExpr: function(exprWord, exprPhrase) {
        var exprBuild = RSelExprParser._rsel.getNewExprBuild();
        exprBuild.cond = exprWord;
        //if (m===10) debugger;
        //============================================================
        // Where the magic happens... sort of.
        //============================================================
        switch (true) {
        case exprWord === '(':
            this.rselBtns.lfParens();
            return false;
        case exprWord === ')':
            this.rselBtns.rtParens();
            return false;
        case 'and' === exprWord:
            this.rselBtns.and();
            return false;
        case 'or' === exprWord:
            this.rselBtns.or();
            return false;
        case /no alt/i.test(exprPhrase):
            exprBuild.cond = 'unnamed';
            exprBuild.op = true;
            exprBuild.op2 = true;
            return exprBuild;
        case '!' === exprWord:
            this.rselBtns.not();
            return false;
        case /^unnamed/.test(exprBuild.cond):
            exprBuild.cond = 'unnamed';
            exprBuild.op = true;
            exprBuild.op2 = false;
            return exprBuild;
            // SPEED LIMITS
        case 'speed' === exprBuild.cond:
            try {
                if (exprPhrase.length < 2 && /\bnot?\b|!|!=/i.test(exprPhrase[0])) {
                    exprBuild.op = 'none';
                } else {
                    exprPhrase = exprPhrase.join(' ');
                    if (/\bnot?\b|!|!=/i.test(exprPhrase))
                        RSelExprParser.rselBtns.not();
                    var optionText = RSelExprParser._rsel.getSelectOptions(RSelExprParser.rselCond.speed.opOptNodes);
                    optionText = RegExp(optionText.join('|'), 'i').exec(exprPhrase);
                    if (optionText)
                        exprBuild.op = optionText[0];
                    else
                        exprBuild.op = 'any';
                }
                var speedVal = exprPhrase.match(/(\d+)\s?mph|(\d+)\s?km/i);
                if (speedVal && speedVal.length === 2)
                    exprBuild.val = speedVal[1];
            } catch (err) {
                exprBuild.errorCode = 101;
                exprBuild.err = err;
                return exprBuild;
            }
            return exprBuild;
            // BINARY CONDITIONS:
        case exprPhrase.length === 0 || //suggests binary
        /^(screen|roundabout|toll|tun|new|changed|restrict|editable)/.test(exprBuild.cond) || //binary selection conditions
        (/^name.*|^closure/i.test(exprBuild.cond) && exprPhrase.length <= 1):
            //selection conditions that have both binary and multiple options
            exprPhrase = exprPhrase.join(' ');
            exprBuild.cond = exprBuild.cond.replace(/^name.*/, 'name');
            exprBuild.cond = exprBuild.cond.replace(/^toll\s.*/, 'toll');
            if (/\bnot?\b|!|!=/i.test(exprPhrase)) {
                exprBuild.op = false;
            } else {
                exprBuild.op = true;
            }
            switch (exprBuild.cond) {
            case 'name':
                try {
                    if (/alt/i.test(exprPhrase)) {
                        exprBuild.cond = 'unnamed';
                        exprBuild.op = false;
                        exprBuild.op2 = true;
                    } else {
                        exprBuild.cond = 'unnamed';
                        exprBuild.op = false;
                        exprBuild.op2 = false;
                    }
                    return exprBuild;
                } catch (err) {
                    exprBuild.errorCode = 101;
                    exprBuild.err = err;
                    return exprBuild;
                }
            case 'closure':
                exprBuild.op2 = '---';
                return exprBuild;
            case 'onscreen':
                exprBuild.cond = 'screen';
                exprBuild.op = true;
                return exprBuild;
            case 'offscreen':
                exprBuild.cond = 'screen';
                exprBuild.op = false;
                return exprBuild;
            case 'roundabout':
            case 'toll':
            case 'tunnel':
            case 'new':
            case 'changed':
            case 'restriction':
            case 'editable':
                return exprBuild;
            default:
                exprBuild.errorCode = 101;
                exprBuild.errorMsg = 'Error: Presumed binary selector had no match.';
                return exprBuild;
            }
            //switch
            //--------------------------------------------------------------------
        case /^closure/.test(exprBuild.cond):
            try {
                exprPhrase = exprPhrase.join().toLowerCase();
                exprBuild.op = !(/does\s?n['o]t|!|!=/.test(exprPhrase));
                //checkbox
                exprBuild.op2 = /start|end/.exec(exprPhrase) + 's';
                //starts/ends
                exprBuild.condmod = /before|after|\bin\b/.exec(exprPhrase) + '';
                //in/before/after
                if (!exprBuild.condmod)
                    exprBuild.condmod = 'in';
                exprBuild.val = /\d+/.exec(exprPhrase) + '';
                //days ago
            } catch (err) {
                exprBuild.errorCode = 101;
                exprBuild.err = err;
                return exprBuild;
            }
            return exprBuild;
        default:
            // CONDITION NAME MATCHING (TYPE OF SELECTION)
            try {
                if (/^(str.*|cit.*)/.test(exprBuild.cond)) {
                    exprBuild.cond = exprBuild.cond.replace(/^str.*/, 'street');
                    exprBuild.cond = exprBuild.cond.replace(/^cit.*/, 'city');
                    var exprStart = exprPhrase.slice(0, -1), //don't include last element bc it should be the name itself
                    prim, alt;
                    if (exprStart) {
                        //exprStart = exprStart.toString().toLowerCase();
                        prim = /\bprim?(?:ary|\.)?\b/i.test(exprStart);
                        alt = /\balt(?:ern\w*|\.)?\b/i.test(exprStart);
                        exprPhrase = exprStart.filter(function(a) {
                            return !/^pri|^alt/i.test(a)
                        }).concat(exprPhrase.slice(-1));
                    } else {
                        prim = false;
                        alt = false;
                    }
                    if (prim && alt)
                        exprBuild.condmod = 2;
                    else if (prim)
                        exprBuild.condmod = 0;
                    else if (alt)
                        exprBuild.condmod = 1;
                    else
                        exprBuild.condmod = 0;
                }
            } catch (err) {
                exprBuild.errorCode = 101;
                exprBuild.err = err;
                return exprBuild;
            }
            // COMPARATOR OPERATION MATCHING
            try {
                // Convert natural lang representation to standard comparator operations
                var exprPhraseStr = exprPhrase.join(' ').replace(/\bcontains?/i, 'contains').replace(/(?:\bdo(?:es)?\s?n[o']t\s|!\s?)(contains)/i, '! $1');
                //.replace(/\b(?:do(?:es)?\s?n[o']t\s|!\s?)contains?/i, '!^').replace(/\bcontains?/i,'\u220b');
                // Comparator operations with standard representation
                exprBuild.op = /(?:! )?contains|[!<>=~]{1,2}/i.exec(exprPhraseStr) + '';
            } catch (err) {
                exprBuild.errorCode = 101;
                exprBuild.err = err;
                return exprBuild;
            }
            // SELECTION VALUE MATCHING
            try {
                if (/^length|^last/.test(exprBuild.cond)) {
                    exprBuild.val = exprPhraseStr.match(/\b\d+/) + ''
                } else {
                    try {
                        // The following line is kind of elaborate bc it needed to grab text between parens/quotes while keeping the inner quotes
                        exprBuild.val = exprPhraseStr.replace(new RegExp('^(?:\\s?' + exprBuild.op + '\\s)(.*)','i'), '$1').replace(/^\(["'](.*?)['"]\)$|^\s?["'](.*)["']$|^["'](.*)['"]$|\b(\w*?)\b/, '$1$2$3$4').replace(/(") (\w) (")/, '$1$2$3');
                    } catch (err) {
                        exprBuild.errorCode = 2;
                        exprBuild.err = err;
                        return exprBuild;
                    }
                    if (/^direction/.test(exprBuild.cond)) {
                        exprBuild.val = exprBuild.val.match(/A[<>-\s]*B|B[<>-\s]*A|unknown/i) + '';
                        //reduce to unique key words...
                    }
                }
                return exprBuild;
            } catch (err) {
                exprBuild.errorCode = 101;
                exprBuild.err = err;
                return exprBuild;
            }
        }
        //switch
    },
    //parseExpr()
    updateExpression: function(parseThis) {
        console.info('*** Begin parsing expression... ***');
        this.rselBtns.clear();
        var parsed = this.parseExpr(parseThis);
        if (parsed && !parsed.errorCode) {
            var exprMatches = parsed.exprMatches,
            exprMatchPhrases = parsed.exprMatchPhrases,
            exprFragment, exprFragPhrase, mLength, m, __EXPR_DEBUGINFO;
            mLength = exprMatchPhrases.length;
            for (m = 0; m < mLength; m++) {
                __EXPR_DEBUGINFO = this.new__EXPR_DEBUGINFO(m, exprMatches[m], exprMatchPhrases[m]);
                //if (m > 3) debugger;
                exprFragment = exprMatches[m];
                exprFragPhrase = exprMatchPhrases[m];
                if (exprFragPhrase.constructor !== Array) exprFragPhrase = [exprFragPhrase];
                var exprBuild = this.buildExpr(exprFragment, exprFragPhrase);
                if (exprBuild && !exprBuild.errorCode) {
                    __EXPR_DEBUGINFO.errorStatus = this.addExpr(exprBuild);
                    if (__EXPR_DEBUGINFO.errorStatus && __EXPR_DEBUGINFO.errorStatus.errorCode) {
                        console.warn('updateExpression() may have partly failed. Check results.');
                        __EXPR_DEBUGINFO.exprBuild = exprBuild;
                        console.debug(__EXPR_DEBUGINFO);
                    }
                } else if (exprBuild && exprBuild.errorCode) {
                    console.warn('updateExpression() may have partly failed. Check results.');
                    __EXPR_DEBUGINFO.exprBuild = exprBuild;
                    console.debug(__EXPR_DEBUGINFO);
                }
            }
            //for each condition matched
        } else {
            console.debug(parsed);
        }
    }
};