// CodeMirror, copyright (c) by Marijn Haverbeke and others
// Distributed under an MIT license: http://codemirror.net/LICENSE
// Yacas mode copyright (c) 2015 by Grzegorz Mazur
// Loosely based on mathematica mode by Calin Barbat
if (typeof exports == "object" && typeof module == "object") // CommonJS
mod(require("../../lib/codemirror"));
else if (typeof define == "function" && define.amd) // AMD
define(["../../lib/codemirror"], mod);
else // Plain browser env
})(function(CodeMirror) {
CodeMirror.defineMode('yacas', function(_config, _parserConfig) {
var obj = {}, words = str.split(" ");
for (var i = 0; i < words.length; ++i) obj[words[i]] = true;
var bodiedOps = words("Assert BackQuote D Defun Deriv For ForEach FromFile " +
"FromString Function Integrate InverseTaylor Limit " +
"LocalSymbols Macro MacroRule MacroRulePattern " +
"NIntegrate Rule RulePattern Subst TD TExplicitSum " +
"TSum Taylor Taylor1 Taylor2 Taylor3 ToFile " +
"ToStdout ToString TraceRule Until While");
var pFloatForm = "(?:(?:\\.\\d+|\\d+\\.\\d*|\\d+)(?:[eE][+-]?\\d+)?)";
var pIdentifier = "(?:[a-zA-Z\\$'][a-zA-Z0-9\\$']*)";
var reFloatForm = new RegExp(pFloatForm);
var reIdentifier = new RegExp(pIdentifier);
var rePattern = new RegExp(pIdentifier + "?_" + pIdentifier);
var reFunctionLike = new RegExp(pIdentifier + "\\s*\\(");
function tokenBase(stream, state) {
state.tokenize = tokenString;
return state.tokenize(stream, state);
state.tokenize = tokenComment;
return state.tokenize(stream, state);
var m = stream.match(/^(\w+)\s*\(/, false);
if (m !== null && bodiedOps.hasOwnProperty(m[1]))
state.scopes.push('bodied');
var scope = currentScope(state);
if (scope === 'bodied' && ch === '[')
if (ch === '[' || ch === '{' || ch === '(')
scope = currentScope(state);
if (scope === '[' && ch === ']' ||
scope === '{' && ch === '}' ||
scope === '(' && ch === ')')
while (scope === 'bodied') {
scope = currentScope(state);
// look for ordered rules
if (stream.match(/\d+ *#/, true, false)) {
if (stream.match(reFloatForm, true, false)) {
if (stream.match(rePattern, true, false)) {
// match all braces separately
if (stream.match(/(?:\[|\]|{|}|\(|\))/, true, false)) {
// literals looking like function calls
if (stream.match(reFunctionLike, true, false)) {
if (stream.match(reIdentifier, true, false)) {
// operators; note that operators like @@ or /; are matched separately for each symbol.
if (stream.match(/(?:\\|\+|\-|\*|\/|,|;|\.|:|@|~|=|>|<|&|\||_|`|'|\^|\?|!|%)/, true, false)) {
// everything else is an error
function tokenString(stream, state) {
var next, end = false, escaped = false;
while ((next = stream.next()) != null) {
if (next === '"' && !escaped) {
escaped = !escaped && next === '\\';
state.tokenize = tokenBase;
function tokenComment(stream, state) {
while((next = stream.next()) != null) {
if (prev === '*' && next === '/') {
state.tokenize = tokenBase;
function currentScope(state) {
if (state.scopes.length > 0)
scope = state.scopes[state.scopes.length - 1];
token: function(stream, state) {
if (stream.eatSpace()) return null;
return state.tokenize(stream, state);
indent: function(state, textAfter) {
if (state.tokenize !== tokenBase && state.tokenize !== null)
if (textAfter === ']' || textAfter === '];' ||
textAfter === '}' || textAfter === '};' ||
return (state.scopes.length + delta) * _config.indentUnit;
electricChars: "{}[]();",
CodeMirror.defineMIME('text/x-yacas', {