UNPKG

@designliquido/delegua

Version:

Linguagem de programação simples e moderna usando português estruturado.

444 lines 16.7 kB
"use strict"; var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); exports.Lexador = void 0; const browser_process_hrtime_1 = __importDefault(require("browser-process-hrtime")); const simbolo_1 = require("./simbolo"); const palavras_reservadas_1 = require("./palavras-reservadas"); const delegua_1 = __importDefault(require("../tipos-de-simbolos/delegua")); /** * O Lexador é responsável por transformar o código em uma coleção de tokens de linguagem. * Cada token de linguagem é representado por um tipo, um lexema e informações da linha de código em que foi expresso. * Também é responsável por mapear as palavras reservadas da linguagem, que não podem ser usadas por outras * estruturas, tais como nomes de variáveis, funções, literais, classes e assim por diante. */ class Lexador { constructor(performance = false) { this.performance = performance; this.simbolos = []; this.erros = []; this.hashArquivo = -1; this.inicioSimbolo = 0; this.atual = 0; this.linha = 0; } eDigito(caractere) { return caractere >= '0' && caractere <= '9'; } eAlfabeto(caractere) { const acentuacoes = [ 'á', 'Á', 'ã', 'Ã', 'â', 'Â', 'à', 'À', 'é', 'É', 'ê', 'Ê', 'í', 'Í', 'ó', 'Ó', 'õ', 'Õ', 'ô', 'Ô', 'ú', 'Ú', 'ç', 'Ç', '_', ]; return ((caractere >= 'a' && caractere <= 'z') || (caractere >= 'A' && caractere <= 'Z') || acentuacoes.includes(caractere)); } eAlfabetoOuDigito(caractere) { return this.eDigito(caractere) || this.eAlfabeto(caractere); } eFinalDaLinha() { if (this.codigo.length === this.linha) { return true; } return this.atual >= this.codigo[this.linha].length; } /** * Indica se o código está na última linha. * @returns Verdadeiro se contador de linhas está na última linha. * Falso caso contrário. */ eUltimaLinha() { return this.linha >= this.codigo.length - 1; } eFinalDoCodigo() { return this.eUltimaLinha() && this.codigo[this.codigo.length - 1].length <= this.atual; } avancar() { this.atual += 1; if (this.eFinalDaLinha() && !this.eUltimaLinha()) { this.linha++; this.atual = 0; } } adicionarSimbolo(tipo, literal = null) { const texto = this.codigo[this.linha].substring(this.inicioSimbolo, this.atual); this.simbolos.push(new simbolo_1.Simbolo(tipo, literal || texto, literal, this.linha + 1, this.hashArquivo)); } simboloAtual() { if (this.eFinalDaLinha()) return '\0'; return this.codigo[this.linha].charAt(this.atual); } comentarioMultilinha() { let conteudo = ''; while (!this.eFinalDoCodigo()) { this.avancar(); conteudo += this.codigo[this.linha].charAt(this.atual); if (this.simboloAtual() === '*' && this.proximoSimbolo() === '/') { const linhas = conteudo.split('\0'); for (let linha of linhas) { this.adicionarSimbolo(delegua_1.default.LINHA_COMENTARIO, linha.trim()); } // Remove o asterisco da última linha let lexemaUltimaLinha = this.simbolos[this.simbolos.length - 1].lexema; lexemaUltimaLinha = lexemaUltimaLinha.substring(0, lexemaUltimaLinha.length - 1); this.simbolos[this.simbolos.length - 1].lexema = lexemaUltimaLinha; this.simbolos[this.simbolos.length - 1].literal = lexemaUltimaLinha; this.avancar(); this.avancar(); break; } } } comentarioUmaLinha() { this.avancar(); const linhaAtual = this.linha; let ultimoAtual = this.atual; while (linhaAtual === this.linha && !this.eFinalDoCodigo()) { ultimoAtual = this.atual; this.avancar(); } const conteudo = this.codigo[linhaAtual].substring(this.inicioSimbolo + 2, ultimoAtual); this.adicionarSimbolo(delegua_1.default.COMENTARIO, conteudo.trim()); } proximoSimbolo() { return this.codigo[this.linha].charAt(this.atual + 1); } simboloAnterior() { return this.codigo[this.linha].charAt(this.atual - 1); } analisarTexto(delimitador = '"') { while (this.simboloAtual() !== delimitador && !this.eFinalDoCodigo()) { this.avancar(); } if (this.eFinalDoCodigo()) { this.erros.push({ linha: this.linha + 1, caractere: this.simboloAnterior(), mensagem: 'Texto não finalizado.', }); return; } const valor = this.codigo[this.linha].substring(this.inicioSimbolo + 1, this.atual); this.adicionarSimbolo(delegua_1.default.TEXTO, valor); } analisarNumero() { while (this.eDigito(this.simboloAtual())) { this.avancar(); } if (this.simboloAtual() == '.' && this.eDigito(this.proximoSimbolo())) { this.avancar(); while (this.eDigito(this.simboloAtual())) { this.avancar(); } } const numeroCompleto = this.codigo[this.linha].substring(this.inicioSimbolo, this.atual); this.adicionarSimbolo(delegua_1.default.NUMERO, parseFloat(numeroCompleto)); } identificarPalavraChave() { while (this.eAlfabetoOuDigito(this.simboloAtual())) { this.avancar(); } const codigo = this.codigo[this.linha].substring(this.inicioSimbolo, this.atual); const tipo = codigo in palavras_reservadas_1.palavrasReservadas ? palavras_reservadas_1.palavrasReservadas[codigo] : delegua_1.default.IDENTIFICADOR; this.adicionarSimbolo(tipo); } analisarToken() { const caractere = this.simboloAtual(); switch (caractere) { case '@': this.adicionarSimbolo(delegua_1.default.ARROBA, '@'); this.avancar(); break; case '[': this.adicionarSimbolo(delegua_1.default.COLCHETE_ESQUERDO); this.avancar(); break; case ']': this.adicionarSimbolo(delegua_1.default.COLCHETE_DIREITO); this.avancar(); break; case '(': this.adicionarSimbolo(delegua_1.default.PARENTESE_ESQUERDO); this.avancar(); break; case ')': this.adicionarSimbolo(delegua_1.default.PARENTESE_DIREITO); this.avancar(); break; case '{': this.adicionarSimbolo(delegua_1.default.CHAVE_ESQUERDA); this.avancar(); break; case '}': this.adicionarSimbolo(delegua_1.default.CHAVE_DIREITA); this.avancar(); break; case ',': this.adicionarSimbolo(delegua_1.default.VIRGULA); this.avancar(); break; case '.': this.adicionarSimbolo(delegua_1.default.PONTO); this.avancar(); break; case '-': this.inicioSimbolo = this.atual; this.avancar(); if (this.simboloAtual() === '=') { this.adicionarSimbolo(delegua_1.default.MENOS_IGUAL, '-='); this.avancar(); } else if (this.simboloAtual() === '-') { this.adicionarSimbolo(delegua_1.default.DECREMENTAR, '--'); this.avancar(); } else { this.adicionarSimbolo(delegua_1.default.SUBTRACAO); } break; case '+': this.inicioSimbolo = this.atual; this.avancar(); if (this.simboloAtual() === '=') { this.adicionarSimbolo(delegua_1.default.MAIS_IGUAL, '+='); this.avancar(); } else if (this.simboloAtual() === '+') { this.adicionarSimbolo(delegua_1.default.INCREMENTAR, '++'); this.avancar(); } else { this.adicionarSimbolo(delegua_1.default.ADICAO); } break; case ':': this.adicionarSimbolo(delegua_1.default.DOIS_PONTOS); this.avancar(); break; case '%': this.inicioSimbolo = this.atual; this.avancar(); switch (this.simboloAtual()) { case '=': this.avancar(); this.adicionarSimbolo(delegua_1.default.MODULO_IGUAL, '%='); break; default: this.adicionarSimbolo(delegua_1.default.MODULO); break; } break; case '*': this.inicioSimbolo = this.atual; this.avancar(); switch (this.simboloAtual()) { case '*': this.avancar(); this.adicionarSimbolo(delegua_1.default.EXPONENCIACAO, '**'); break; case '=': this.avancar(); this.adicionarSimbolo(delegua_1.default.MULTIPLICACAO_IGUAL, '*='); break; default: this.adicionarSimbolo(delegua_1.default.MULTIPLICACAO); break; } break; case '!': this.avancar(); if (this.simboloAtual() === '=') { this.adicionarSimbolo(delegua_1.default.DIFERENTE, '!='); this.avancar(); } else { this.adicionarSimbolo(delegua_1.default.NEGACAO); } break; case '=': this.avancar(); if (this.simboloAtual() === '=') { this.adicionarSimbolo(delegua_1.default.IGUAL_IGUAL, '=='); this.avancar(); } else { this.adicionarSimbolo(delegua_1.default.IGUAL); } break; case '&': this.adicionarSimbolo(delegua_1.default.BIT_AND); this.avancar(); break; case '~': this.adicionarSimbolo(delegua_1.default.BIT_NOT); this.avancar(); break; case '|': this.avancar(); if (this.simboloAtual() === '|') { this.adicionarSimbolo(delegua_1.default.EXPRESSAO_REGULAR, '||'); this.avancar(); } else { this.adicionarSimbolo(delegua_1.default.BIT_OR); } break; case '^': this.adicionarSimbolo(delegua_1.default.BIT_XOR); this.avancar(); break; case '<': this.avancar(); if (this.simboloAtual() === '=') { this.adicionarSimbolo(delegua_1.default.MENOR_IGUAL, '<='); this.avancar(); } else if (this.simboloAtual() === '<') { this.adicionarSimbolo(delegua_1.default.MENOR_MENOR, '<<'); this.avancar(); } else { this.adicionarSimbolo(delegua_1.default.MENOR); } break; case '>': this.avancar(); if (this.simboloAtual() === '=') { this.adicionarSimbolo(delegua_1.default.MAIOR_IGUAL, '>='); this.avancar(); } else if (this.simboloAtual() === '>') { this.adicionarSimbolo(delegua_1.default.MAIOR_MAIOR, '>>'); this.avancar(); } else { this.adicionarSimbolo(delegua_1.default.MAIOR); } break; case '/': this.avancar(); switch (this.simboloAtual()) { case '/': this.comentarioUmaLinha(); break; case '*': this.comentarioMultilinha(); break; case '=': this.adicionarSimbolo(delegua_1.default.DIVISAO_IGUAL, '/='); this.avancar(); break; default: this.adicionarSimbolo(delegua_1.default.DIVISAO); break; } break; case '\\': this.inicioSimbolo = this.atual; this.avancar(); switch (this.simboloAtual()) { case '=': this.adicionarSimbolo(delegua_1.default.DIVISAO_INTEIRA_IGUAL, '\\='); this.avancar(); break; default: this.adicionarSimbolo(delegua_1.default.DIVISAO_INTEIRA); break; } break; // Esta sessão ignora espaços em branco (ou similares) na tokenização. case ' ': case '\0': case '\r': case '\t': this.avancar(); break; // Ponto-e-vírgula é opcional em Delégua, mas em alguns casos pode ser // necessário. Por exemplo, declaração de `para` sem inicializador. case ';': this.adicionarSimbolo(delegua_1.default.PONTO_E_VIRGULA); this.avancar(); break; case '"': this.avancar(); this.analisarTexto('"'); this.avancar(); break; case "'": this.avancar(); this.analisarTexto("'"); this.avancar(); break; default: if (this.eDigito(caractere)) this.analisarNumero(); else if (this.eAlfabeto(caractere)) this.identificarPalavraChave(); else { this.erros.push({ linha: this.linha + 1, caractere: caractere, mensagem: 'Caractere inesperado.', }); this.avancar(); } } } mapear(codigo, hashArquivo) { const inicioMapeamento = (0, browser_process_hrtime_1.default)(); this.erros = []; this.simbolos = []; this.inicioSimbolo = 0; this.atual = 0; this.linha = 0; this.codigo = codigo || ['']; if (codigo.length === 0) { this.codigo = ['']; } this.hashArquivo = hashArquivo; for (let iterador = 0; iterador < this.codigo.length; iterador++) { this.codigo[iterador] += '\0'; } while (!this.eFinalDoCodigo()) { this.inicioSimbolo = this.atual; this.analisarToken(); } if (this.performance) { const deltaMapeamento = (0, browser_process_hrtime_1.default)(inicioMapeamento); console.log(`[Lexador] Tempo para mapeamento: ${deltaMapeamento[0] * 1e9 + deltaMapeamento[1]}ns`); } return { simbolos: this.simbolos, erros: this.erros, }; } } exports.Lexador = Lexador; //# sourceMappingURL=lexador.js.map