All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.squarespace.less.model.Quoted 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.LessUtils.safeEquals;
import static com.squarespace.less.model.NodeType.QUOTED;

import java.util.ArrayList;
import java.util.List;

import com.squarespace.less.LessException;
import com.squarespace.less.core.Buffer;
import com.squarespace.less.core.LessUtils;
import com.squarespace.less.exec.ExecEnv;


/**
 * A quoted string which may contain embedded variables.
 */
public class Quoted extends BaseNode {

  /**
   * Default capacity for the components in the string.
   */
  private static final int DEFAULT_CAPACITY = 3;

  /**
   * Delimiter for the quoted string, either \' or \"
   */
  protected final char delim;

  /**
   * Components that make up the string. Some of these may be variable
   * references.
   */
  protected List parts;

  /**
   * Indicates whether the string is escaped. If true, it will be emitted
   * without the start/end delimiters.
   */
  protected boolean escaped;

  /**
   * Indicates whether any of the parts require evaluation.
   */
  protected boolean evaluate;

  /**
   * Hashcode of the quoted string.
   */
  private int hashCode;

  /**
   * Constructs an empty string with the given delimiter character.
   */
  public Quoted(char delim) {
    this(delim, false);
  }

  /**
   * Constructs an empty string with the given delimiter character, and setting
   * the value for the escape flag.
   */
  public Quoted(char delim, boolean escape) {
    this(delim, escape, null);
  }

  /**
   * Constructs an empty string with the given delimiter character, setting
   * the value for the escape flag, and setting the initial parts.
   */
  public Quoted(char delim, boolean escape, List parts) {
    this.delim = delim;
    this.escaped = escape;
    this.parts = parts;
    if (parts != null) {
      for (Node part : parts) {
        evaluate |= part.needsEval();
      }
    }
  }

  /**
   * Returns the delimiter character.
   */
  public char delimiter() {
    return delim;
  }

  /**
   * Indicates whether the string is escaped. See {@link #escaped}.
   */
  public boolean escaped() {
    return escaped;
  }

  /**
   * Returns the list of component parts for the string.
   */
  public List parts() {
    return parts;
  }

  /**
   * Sets the escape flag.
   */
  public void setEscape(boolean flag) {
    this.escaped = flag;
  }

  /**
   * Appends a part to the string.
   */
  public void append(Node node) {
    this.parts = LessUtils.initList(parts, DEFAULT_CAPACITY);
    this.parts.add(node);
    this.evaluate |= node.needsEval();
  }

  /**
   * Copies the string.
   */
  public Quoted copy() {
    Quoted res = new Quoted(delim);
    for (Node part : parts) {
      res.append(part);
    }
    return res;
  }

  /**
   * See {@link Node#type()}
   */
  @Override
  public NodeType type() {
    return QUOTED;
  }

  /**
   * See {@link Node#needsEval()}
   */
  @Override
  public boolean needsEval() {
    return evaluate;
  }

  /**
   * See {@link Node#eval(ExecEnv)}
   */
  @Override
  public Node eval(ExecEnv env) throws LessException {
    if (!evaluate) {
      return this;
    }
    int size = parts.size();
    List result = new ArrayList<>(size);
    for (int i = 0; i < size; i++) {
      result.add(parts.get(i).eval(env));
    }
    return new Quoted(delim, escaped, result);
  }

  /**
   * See {@link Node#repr(Buffer)}
   */
  @Override
  public void repr(Buffer buf) {
    if (escaped) {
      buf.append('~');
    }
    buf.append(delim);
    int size = parts.size();
    for (int i = 0; i < size; i++) {
      parts.get(i).repr(buf);
    }
    buf.append(delim);
  }

  /**
   * See {@link Node#modelRepr(Buffer)}
   */
  @Override
  public void modelRepr(Buffer buf) {
    typeRepr(buf);
    posRepr(buf);
    buf.append(" delim=").append(delim == '"' ? "DOUBLE" : "SINGLE");
    if (escaped) {
      buf.append(" [escaped]");
    }
    buf.append('\n').incrIndent();
    ReprUtils.modelRepr(buf, "\n", true, parts);
    buf.decrIndent();
  }

  @Override
  public boolean equals(Object obj) {
    if (obj instanceof Quoted) {
      Quoted other = (Quoted)obj;
      // NOTE: delimiter characters are not compared for equals().
      return escaped == other.escaped && safeEquals(parts, other.parts);
    }
    return false;
  }

  @Override
  public int hashCode() {
    // NOTE: delimiter character is not part of the hash.
    return hashCode == 0 ? buildHashCode(escaped, parts) : hashCode;
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy