Javascript объект, который принимает свойство другого объекта как параметр, так или иначе изменяет его - PullRequest
0 голосов
/ 27 апреля 2018

Я работал над проектом Math Parser. Я почти закончил, но столкнулся с проблемой, которую не смог найти причину.

У меня есть 3 файла javascript:

main.js

var button1 = document.getElementById("button");
var text1 = document.getElementById("text");
var output = document.getElementById("result");

button1.addEventListener("click", function(){

    var textInput = text1.value;
    var cleared = new clearingUnwantedCharacters(textInput);
    var marked = new subgroupMarking(cleared.output);
    var prioritizedList = new subgroupPrioritization(marked.parenGroups);
    for(var i = 0; i <= 360; i++){
        var splittedPieces = new splittingOperators(prioritizedList.prioritizedGroups);
        var calculatedPieces = new calculatePieces(splittedPieces.seperatedPieces, i, true);
        console.log(calculatedPieces.result);
    }

});

objects.js

function clearingUnwantedCharacters(inputText){
    this.output = inputText;
    this.output = this.output.replace(/\s+/g, '');
    this.output = this.output.toLowerCase();
}

function subgroupMarking(inputText){    
    this.parenGroups = [inputText];

    this.locLeftParen = markChar(this.parenGroups[0], "(");
    this.locRigthParen = markChar(this.parenGroups[0], ")");        

    for(var i = this.locLeftParen.length-1; i >= 0; i--){ 
        var leftParen = this.locLeftParen[i];
        var rightParen = this.locRigthParen.find(function(res){return res > leftParen});
        this.parenGroups[i+1] = this.parenGroups[0].substring(leftParen+1, rightParen);
        this.parenGroups[0] = changeAt(this.parenGroups[0], leftParen, rightParen+1, "{_"+(i+1)+"}");
        this.locLeftParen = markChar(this.parenGroups[0], "(");
        this.locRigthParen = markChar(this.parenGroups[0], ")");
    }
}

function subgroupPrioritization(subgroupList){  
    var withParenGroup = [];
    var withoutParenGroup = [];

    for(var i = 0; i < subgroupList.length; i++){ 
        var content = subgroupList[i].toString();
        var hasParenGroup = content.search(/{_\d+}/g);  
        if(hasParenGroup == -1){
            withoutParenGroup.push(content);
        }else{
            withParenGroup.push(content);
        }

    }

    this.prioritizedGroups = withParenGroup.concat(withoutParenGroup);
}

function splittingOperators(prioritizedGroupList){
    this.seperatedPieces = [];

    for(var i = prioritizedGroupList.length-1; i >= 0; i--){
        var pieces = prioritizedGroupList[i].split(/(\+|\*|\/|\^|\b(?=-)|(?<=})(?=-))/g);

        for(var k = 0; k < pieces.length; k++){
                if(pieces[k] == ""){
                    pieces[k] = "+";
                }
        }

        this.seperatedPieces.unshift(pieces);

    }
}

function calculatePieces(pieceList, x, degreeOption){

    this.result = pieceList;

    for(var i = this.result.length-1; i >= 0; i--){ // ["34", "+", "2{_2}"] 
            for(var k = 0; k < this.result[i].length; k++){ // 34, +, 2{_2} 
                var specialFlag = false;
                var totalMultiplication = (countChar(this.result[i][k], "-")%2 == 0) ? 1 : -1;  
                var has = {
                    x : this.result[i][k].includes("x"),
                    subgroup : (this.result[i][k].search(/{_\d+}/g) != -1),
                    logarithm : this.result[i][k].includes("log"),
                    trigonometry : (this.result[i][k].includes("sin") || this.result[i][k].includes("cos") || this.result[i][k].includes("tan") || this.result[i][k].includes("cot"))
                };
                if(has.trigonometry){
                    specialFlag = true;

                    var trigSubgroupIndexes = [];
                    var trigTypes = [];

                    var trigsMultiplied = 1;

                    this.result[i][k] = this.result[i][k].replace(/(?:arc)?(sin|cos|tan|cot){_(\d+)}/g, function(match, type, index){
                        trigSubgroupIndexes.push(parseInt(index));
                        if(match.includes("arc")){                      
                            trigTypes.push([true, type]);
                        }else{
                            trigTypes.push([false, type]);
                        }   
                        return " ";
                    }); 
                    for(var l = 0; l < trigTypes.length; l++){                  
                        var inverseness = trigTypes[l][0];
                        var type = trigTypes[l][1];
                        var parameter = (degreeOption) ? convertToRadian(parseFloat(this.result[trigSubgroupIndexes[l]])) : parseFloat(this.result[trigSubgroupIndexes[l]]);
                        var result;
                        if(inverseness){
                            switch(type){
                                case "sin":
                                    result = Math.asin(parameter);
                                break;case "cos":
                                    result = Math.acos(parameter);
                                break;case "tan":
                                    result = Math.atan(parameter);
                                break;case "cot":
                                    result = 1 / Math.atan(parameter);
                                break;
                            }
                        }else{
                            switch(type){
                                case "sin":
                                    result = Math.sin(parameter);
                                break;case "cos":
                                    result = Math.cos(parameter);
                                break;case "tan":
                                    result = Math.tan(parameter);
                                break;case "cot":
                                    result = 1 / Math.tan(parameter);
                                break;
                            }
                        }
                        trigsMultiplied *= result;
                    }                   
                    totalMultiplication *= trigsMultiplied;
                }
                if(has.logarithm){
                    specialFlag = true;
                    var logSubgroupIndexes = [];
                    var logsMultiplied = 1;
                    this.result[i][k] = this.result[i][k].replace(/log{_(\d+)}{_(\d+)}/g, function(match, firstIndex, secondIndex){
                        logSubgroupIndexes.push([parseInt(firstIndex), parseInt(secondIndex)]);
                        return " ";
                    });
                    for(var l = 0; l < logSubgroupIndexes.length; l++){                 
                        var base = this.result[logSubgroupIndexes[l][0]];
                        var power = this.result[logSubgroupIndexes[l][1]];
                        logsMultiplied *= (Math.log(power) / Math.log(base));
                    }                   
                    totalMultiplication *= logsMultiplied;
                }
                if(has.subgroup){
                    specialFlag = true;
                    var subgroupIndexes = [];
                    var groupsMultiplied = 1;                   
                    this.result[i][k] = this.result[i][k].replace(/{_(\d+)}/g, function(match, index){
                        subgroupIndexes.push(parseInt(index));
                        return " ";
                    });
                    for(var l = 0; l < subgroupIndexes.length; l++){                    
                        groupsMultiplied *= this.result[subgroupIndexes[l]];
                    }
                    totalMultiplication *= groupsMultiplied;
                }
                if(has.x){
                    specialFlag = true;
                    var powerX = countChar(this.result[i][k], "x");
                    this.result[i][k] = this.result[i][k].replace("x", " ");
                    totalMultiplication *= Math.pow(x, powerX);
                }
                if(specialFlag == true){
                    var factors = this.result[i][k].match(/\d+/g);
                    if(factors !== null){
                        for(var l = 0; l < factors.length; l++){
                            totalMultiplication *= parseFloat(factors[l]);
                        }
                    }
                    if(!isFinite(totalMultiplication) || checkNum(this.result[i][k]) || this.result[i][k].includes("(") || this.result[i][k].includes(")")){
                        this.result[i][k] = NaN;                        
                    }else{
                        this.result[i][k] = totalMultiplication;
                    }
                }


            }

            var operators = ["^","*","/","+","-"];      
            for(var k = 0; k < operators.length; k++){
                var search = this.result[i].indexOf(operators[k]);

                while(search != -1){
                    switch(operators[k]){
                        case "+":   
                            this.result[i].splice(search-1, 3, checkParse(this.result[i][search-1]) + checkParse(this.result[i][search+1]));
                        break;case "-":
                            this.result[i].splice(search-1, 3, checkParse(this.result[i][search-1]) - checkParse(this.result[i][search+1]));
                        break;case "*":
                            this.result[i].splice(search-1, 3, checkParse(this.result[i][search-1]) * checkParse(this.result[i][search+1]));
                        break;case "/":
                            this.result[i].splice(search-1, 3, checkParse(this.result[i][search-1]) / checkParse(this.result[i][search+1]));
                        break;case "^":
                            this.result[i].splice(search-1, 3, Math.pow(checkParse(this.result[i][search-1]) , checkParse(this.result[i][search+1])));
                        break;
                    }   
                    if(this.result[i].indexOf(operators[k], search) == -1){                     
                        break;
                    }else{
                        search = this.result[i].indexOf(operators[k], search);      
                    }
                }
            }   
    }
    if(isNaN(this.result[0][0])){
        this.result = "An error has occured, please make sure the parameters are correct.";
    }else{
        this.result = this.result[0][0];
    }
}

functions.js

function countChar(str, character){
    return str.toString().split(character).length-1;
}
function markChar(str, character){

    var charCount = str.split(character).length;
    var result = [];

    for(var i = 0; i < charCount - 1; i++){
        if(result.length == 0){
            result.push(str.indexOf(character));
        }else{
            result.push(str.indexOf(character, result[i-1] + 1));
        }   
    }
    return result;
}
function changeAt(str, startingIndex, endingIndex, replacement){ // Returns the given string with the part between startingIndex and endingIndex replaced with replacement.
    return str.substring(0, startingIndex) + replacement + str.substring(endingIndex, str.length);  
}
function checkParse(str){
    if(/[a-z]/.test(str) || countChar(str, ".") > 1){
        return NaN;
    }else{
        return parseFloat(str);
    }
}
function checkNum(str){
    return (/[a-z]/.test(str) || countChar(str, ".") > 1);
}
function convertToRadian(degrees) {

    return degrees * Math.PI / 180;
}

Он работает, как задумано, когда сделано, как указано выше. Он помещает значения в x в соответствии с диапазоном, указанным в цикле for в main.js и console.logs результатов. Но я не хочу создавать объект splittingOperators каждый раз, когда я пересчитываю с другим значением x. В этом нет необходимости, поскольку фактическая формула остается неизменной.

var button1 = document.getElementById("button");
var text1 = document.getElementById("text");
var output = document.getElementById("result");

button1.addEventListener("click", function(){

    var textInput = text1.value;
    var cleared = new clearingUnwantedCharacters(textInput);
    var marked = new subgroupMarking(cleared.output);
    var prioritizedList = new subgroupPrioritization(marked.parenGroups);
    var splittedPieces = new splittingOperators(prioritizedList.prioritizedGroups);
    for(var i = 0; i <= 360; i++){
        var calculatedPieces = new calculatePieces(splittedPieces.seperatedPieces, i, true);
        console.log(calculatedPieces.result);
    }

});

Когда я помещаю создание объекта вне цикла for, он вычисляет и записывает в журнал первое значение x, но дает

Uncaught TypeError: this.result [i] [k] .includes не является функцией

как ошибка после этого. Я подозреваю, что это потому, что он меняет splittedPieces.seperatedPieces, когда это не должно. Я не мог найти причины для этого, или, может быть, я что-то упустил.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...