com.squarespace.less.model.Operation Maven / Gradle / Ivy
/**
* 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.model;
import static com.squarespace.less.core.ExecuteErrorMaker.badColorMath;
import static com.squarespace.less.core.LessUtils.safeEquals;
import com.squarespace.less.LessErrorInfo;
import com.squarespace.less.LessException;
import com.squarespace.less.LessOptions;
import com.squarespace.less.core.Buffer;
import com.squarespace.less.exec.ExecEnv;
/**
* Represents an operator being applied to a pair of operands.
*/
public class Operation extends BaseNode {
/**
* Operator to be applied to the operands.
*/
protected final Operator operator;
/**
* Left-hand operand.
*/
protected final Node left;
/**
* Right-hand operand.
*/
protected final Node right;
/**
* Indicates operation is a sub-expression wrapped in parenthesis.
*/
protected boolean isSubExpression;
/**
* Indicates this operation is inside an expression that requires
* strict math mode to evaluate.
*/
protected boolean requireStrictMath;
/**
* Constructs an operation for the given operator and operands.
*/
public Operation(Operator operator, Node left, Node right) {
this(operator, left, right, false);
}
/**
* Constructs an operation for the given operator, operands and sub-expression flag.
*/
public Operation(Operator operator, Node left, Node right, boolean isSubExpression) {
this.operator = operator;
this.left = left;
this.right = right;
this.isSubExpression = isSubExpression;
}
/**
* Returns the operator.
*/
public Operator operator() {
return operator;
}
/**
* Returns the left-hand operand.
*/
public Node left() {
return left;
}
/**
* Returns the right-hand operand.
*/
public Node right() {
return right;
}
/**
* Indicates whether this operation requires strict math mode to evaluate.
*/
public boolean requiresStrictMath() {
return requireStrictMath;
}
/**
* Mark this operation as being inside a sub-expression.
*/
public void setSubExpression(boolean flag) {
this.isSubExpression = flag;
}
/**
* Mark this operation as being inside an expression which requires
* strict math to evaluate.
*/
public void setRequireStrictMath(boolean flag) {
this.requireStrictMath = flag;
}
/**
* See {@link Node#type()}
*/
@Override
public NodeType type() {
return NodeType.OPERATION;
}
/**
* See {@link Node#needsEval()}
*/
@Override
public boolean needsEval() {
return true;
}
/**
* See {@link Node#eval(ExecEnv)}
*/
@Override
public Node eval(ExecEnv env) throws LessException {
Node op0 = left.needsEval() ? left.eval(env) : left;
Node op1 = right.needsEval() ? right.eval(env) : right;
if (requireStrictMath && env.isStrictMath() && !isSubExpression) {
return new Operation(operator, op0, op1);
}
// Check if we can cast the node to a friendlier type.
op0 = cast(op0);
op1 = cast(op1);
LessOptions opts = env.context().options();
if ((op0 instanceof Dimension) && (op1 instanceof BaseColor)) {
if (operator == Operator.MULTIPLY || operator == Operator.ADD) {
Node temp = op0;
op0 = op1;
op1 = temp;
} else {
LessErrorInfo info = badColorMath(operator, op0);
if (opts.strict()) {
throw new LessException(info);
} else if (!opts.hideWarnings()) {
env.addWarning(info.getMessage() + ".. ignoring right-hand operand.");
}
return op0;
}
}
return op0.operate(env, operator, op1);
}
/**
* See {@link Node#repr(Buffer)}
*/
@Override
public void repr(Buffer buf) {
boolean noStrict = !requireStrictMath;
if (noStrict) {
buf.append('(');
}
left.repr(buf);
if (noStrict) {
buf.append(' ');
}
buf.append(operator.toString());
if (noStrict) {
buf.append(' ');
}
right.repr(buf);
if (noStrict) {
buf.append(')');
}
}
/**
* See {@link Node#modelRepr(Buffer)}
*/
@Override
public void modelRepr(Buffer buf) {
typeRepr(buf);
posRepr(buf);
buf.append(' ').append(operator.toString()).append('\n');
buf.incrIndent();
buf.indent();
left.modelRepr(buf);
buf.append('\n').indent();
right.modelRepr(buf);
buf.decrIndent();
}
/**
* Casts the node to a more specific type.
*/
private Node cast(Node node) {
if (node instanceof Keyword) {
Keyword kwd = (Keyword)node;
RGBColor color = RGBColor.fromName(kwd.value());
return color == null ? kwd : color;
}
return node;
}
@Override
public boolean equals(Object obj) {
if (obj instanceof Operation) {
Operation other = (Operation)obj;
return operator == other.operator
&& safeEquals(left, other.left)
&& safeEquals(right, other.right);
}
return false;
}
@Override
public int hashCode() {
return hashCode == 0 ? buildHashCode(operator, left, right) : hashCode;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy