com.squarespace.less.exec.ExecEnv Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of less-core Show documentation
Show all versions of less-core Show documentation
Less compiler in Java, based on less.js
/**
* Copyright (c) 2014 SQUARESPACE, Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.squarespace.less.exec;
import com.squarespace.less.LessContext;
import com.squarespace.less.LessException;
import com.squarespace.less.core.Buffer;
import com.squarespace.less.core.FlexList;
import com.squarespace.less.model.Block;
import com.squarespace.less.model.BlockNode;
import com.squarespace.less.model.Definition;
/**
* Represents the current execution environment.
*/
public class ExecEnv {
/**
* Context for the current compile.
*/
protected final LessContext ctx;
/**
* Stack frames for execution.
*/
protected final FlexList frames;
/**
* List of warnings emitted during execution.
*/
protected FlexList warnings;
/**
* Exception that terminated execution, if any.
*/
private LessException error;
/**
* Indicates whether strict math mode is currently enabled.
*/
private boolean strictMath;
/**
* When evaluating a rule we clear this flag and evaluate the rule's
* right-hand side expression. If any definition is encountered which
* is marked important, it will set this flag.
*/
private boolean importantFlagged;
/**
* Constructs an instance associated with the given compile context.
*/
public ExecEnv(LessContext ctx) {
this(ctx, new FlexList(64), null);
}
/**
* Constructs an instance associated with the given compile context and
* initial stack contents.
*/
public ExecEnv(LessContext ctx, FlexList initialStack) {
this(ctx, initialStack, null);
}
/**
* Constructs an instance associated with the given compile context,
* initial stack contents, and warning list.
*/
public ExecEnv(LessContext ctx, FlexList initialStack, FlexList warnings) {
this.ctx = ctx;
this.frames = initialStack;
this.warnings = warnings;
}
/**
* Returns the context associated with this compile.
*/
public LessContext context() {
return ctx;
}
/**
* Returns a new {@link ExecEnv} instance with a copy of the stack frames and warnings.
*/
public ExecEnv copy() {
return new ExecEnv(ctx, frames.copy(), warnings);
}
/**
* Current stack depth.
*/
public int depth() {
return frames.size();
}
/**
* Indicates an error has been produced.
*/
public boolean hasError() {
return error != null;
}
/**
* Indicates whether strict math mode is currently enabled.
*/
public boolean isStrictMath() {
return strictMath;
}
/**
* Enables / disables strict math mode.
*/
public void setStrictMath(boolean flag) {
this.strictMath = flag;
}
/**
* Sets the important flag.
*/
public void setImportantFlag(boolean flag) {
this.importantFlagged = flag;
}
/**
* Indicates whether the important flag was set during evaluation.
*/
public boolean importantFlagged() {
return importantFlagged;
}
/**
* Returns the execution error, if any.
*/
public LessException error() {
return error;
}
/**
* Sets the execution error.
*/
public void error(LessException exc) {
error = exc;
}
/**
* Pushes a list of frames onto the stack.
*/
public void append(FlexList other) {
frames.append(other);
}
/**
* Adds a warning to the list.
*/
public void addWarning(String warning) {
if (warnings == null) {
warnings = new FlexList<>();
}
warnings.append(warning);
}
/**
* Return the formatted list of warnings.
*/
public String warnings() {
if (warnings == null || warnings.isEmpty()) {
return null;
}
StringBuilder buf = new StringBuilder();
int size = warnings.size();
for (int i = 0; i < size; i++) {
if (i > 0) {
buf.append(", ");
}
buf.append(warnings.get(i));
}
warnings.clear();
return buf.toString();
}
/**
* Returns the stack frames.
*/
public FlexList frames() {
return frames;
}
/**
* Iterate up the stack, trying to find the given variable definition in each block.
*/
public Definition resolveDefinition(String name) throws LessException {
int size = frames.size();
for (int i = size - 1; i >= 0; i--) {
Definition def = frames.get(i).resolveDefinition(name);
// Future: pragma to skip over circular references, looking in a higher stack frame.
// To do that we check if the definition is evaluating (circular) and skip it:
//
// if (def != null && pragmaSkipCircular() && !def.evaluating()) {
// ...
// If definition exists, return it
if (def != null) {
return def;
}
}
return null;
}
/**
* Iterate up the stack, trying to resolve the mixin against each block.
*/
public boolean resolveMixins(MixinResolver resolver) throws LessException {
int size = frames.size();
for (int i = size - 1; i >= 0; i--) {
if (resolver.match(frames.get(i))) {
return true;
}
}
return false;
}
/**
* Push a block node onto the stack.
*/
public void push(BlockNode blockNode) throws LessException {
frames.push(blockNode.block());
}
/**
* Push a block onto the stack.
*/
public void push(Block block) throws LessException {
frames.push(block);
}
/**
* Pop the current block off the stack.
*/
public void pop() {
frames.pop();
}
/**
* Dump all variable definitions in each block of the stack. Used for
* debugging in a pinch.
*/
public String dumpDefinitions() {
Buffer buf = new Buffer(4);
int size = frames.size();
for (int i = size - 1; i >= 0; i--) {
frames.get(i).dumpDefs(buf);
buf.incrIndent();
}
return buf.toString();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy