001/*
002 * Copyright (c) 2012-2021 Institut National des Sciences Appliquées de Lyon (INSA Lyon) and others
003 *
004 * This program and the accompanying materials are made available under the
005 * terms of the Eclipse Public License 2.0 which is available at
006 * http://www.eclipse.org/legal/epl-2.0.
007 *
008 * SPDX-License-Identifier: EPL-2.0
009 */
010
011package gololang.ir;
012
013import java.util.List;
014import java.util.LinkedList;
015
016/**
017 * A destructuring assignment.
018 *
019 * <p>For instance:
020 * <pre class="listing"><code class="lang-golo" data-lang="golo">
021 * let a, b = [1, 2]
022 * </code></pre>
023 */
024public final class DestructuringAssignment extends GoloAssignment<DestructuringAssignment> {
025
026  private final List<LocalReference> references = new LinkedList<>();
027  private boolean isVarargs = false;
028
029  private DestructuringAssignment() {
030    super();
031  }
032
033  /**
034   * Create a destructuring assignment.
035   *
036   * <p>Typical usage:
037   * <pre class="listing"><code class="lang-java" data-lang="java">
038   * destruct(tuple(1, 2)).to("a", "b").declaring(true)
039   * </code></pre>
040   * creates
041   * <pre class="listing"><code class="lang-golo" data-lang="golo">
042   * let a, b = [1, 2]
043   * </code></pre>
044   *
045   * @param expr the expression to destructure
046   * @see #as(Object)
047   */
048  public static DestructuringAssignment destruct(Object expr) {
049    return new DestructuringAssignment().as(expr);
050  }
051
052  /**
053   * Creates a uninitialized destructuring assignment.
054   */
055  public static DestructuringAssignment create() {
056    return new DestructuringAssignment();
057  }
058
059  protected DestructuringAssignment self() { return this; }
060
061  /**
062   * Checks if this destructuring is a varargs one.
063   *
064   * @see #varargs(boolean)
065   */
066  public boolean isVarargs() {
067    return this.isVarargs;
068  }
069
070  /**
071   * Defines if this destructuring is a varargs one.
072   *
073   * <p>For instance:
074   * <pre class="listing"><code class="lang-golo" data-lang="golo">
075   * let a, b... = list[1, 2, 3, 4]
076   * </code></pre>
077   * <p>This is a builder method.
078   */
079  public DestructuringAssignment varargs(boolean varargs) {
080    this.isVarargs = varargs;
081    return this;
082  }
083
084  /**
085   * Defines this destructuring as a varargs one.
086   *
087   * <p>Same as {@code varargs(true)}.
088   */
089  public DestructuringAssignment varargs() {
090    return varargs(true);
091  }
092
093  /**
094   * {@inheritDoc}
095   */
096  @Override
097  public DestructuringAssignment variable() {
098    for (LocalReference ref : references) {
099      ref.variable();
100    }
101    return this;
102  }
103
104  /**
105   * {@inheritDoc}
106   */
107  @Override
108  public boolean isConstant() {
109    if (!references.isEmpty()) {
110      return references.get(0).isConstant();
111    }
112    return false;
113  }
114
115  /**
116   * {@inheritDoc}
117   */
118  @Override
119  public LocalReference[] getReferences() {
120    return this.references.toArray(new LocalReference[references.size()]);
121  }
122
123  /**
124   * {@inheritDoc}
125   */
126  @Override
127  public int getReferencesCount() {
128    return this.references.size();
129  }
130
131  /**
132   * @inheritDoc
133   */
134  @Override
135  public DestructuringAssignment to(Object... refs) {
136    for (Object o : refs) {
137      references.add(LocalReference.of(o));
138    }
139    return this;
140  }
141
142  @Override
143  public String toString() {
144    List<String> names = new LinkedList<>();
145    for (LocalReference r : getReferences()) {
146      names.add(r.toString());
147    }
148    return String.join(", ", names) + " = " + expression().toString();
149  }
150
151  /**
152   * {@inheritDoc}
153   */
154  @Override
155  public void accept(GoloIrVisitor visitor) {
156    visitor.visitDestructuringAssignment(this);
157  }
158
159  /**
160   * {@inheritDoc}
161   */
162  @Override
163  public void walk(GoloIrVisitor visitor) {
164    for (LocalReference ref : references) {
165      ref.accept(visitor);
166    }
167    super.walk(visitor);
168  }
169}