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
016import static java.util.Collections.unmodifiableList;
017
018/**
019 * A {@code match} node.
020 * <p> This node describe tree such as:
021 * <pre class="listing"><code class="lang-golo" data-lang="golo">
022 *  match {
023 *    when condition_1 then expression_1
024 *    when condition_2 then expression_2
025 *    otherwise default_expression
026 *  }
027 * </code></pre>
028 */
029public final class MatchExpression extends ExpressionStatement<MatchExpression> implements Alternatives<ExpressionStatement<?>> {
030
031  private ExpressionStatement<?> otherwise;
032  private final LinkedList<WhenClause<ExpressionStatement<?>>> clauses = new LinkedList<>();
033
034  MatchExpression() {
035    super();
036  }
037
038  /**
039   * Creates an empty match expression.
040   */
041  public static MatchExpression match() {
042    return new MatchExpression();
043  }
044
045  protected MatchExpression self() { return this; }
046
047  /**
048   * {@inheritDoc}
049   */
050  @Override
051  public MatchExpression when(Object cond) {
052    if (cond instanceof WhenClause) {
053      @SuppressWarnings("unchecked")
054      WhenClause<ExpressionStatement<?>> clause = (WhenClause<ExpressionStatement<?>>) cond;
055      this.clauses.add(makeParentOf(clause));
056    } else {
057      this.clauses.add(makeParentOf(new WhenClause<ExpressionStatement<?>>(ExpressionStatement.of(cond), null)));
058    }
059    return this;
060  }
061
062  /**
063   * {@inheritDoc}
064   *
065   * @param action a {@link gololang.ir.ExpressionStatement} containing the expression to return.
066   */
067  @Override
068  public MatchExpression then(Object action) {
069    this.clauses.getLast().then(ExpressionStatement.of(action));
070    return this;
071  }
072
073  /**
074   * {@inheritDoc}
075   *
076   * @param action a {@link gololang.ir.ExpressionStatement} containing the expression to return.
077   */
078  @Override
079  public MatchExpression otherwise(Object action) {
080    this.otherwise = makeParentOf(ExpressionStatement.of(action));
081    return this;
082  }
083
084  public List<WhenClause<ExpressionStatement<?>>> getClauses() {
085    return unmodifiableList(this.clauses);
086  }
087
088  public ExpressionStatement<?> getOtherwise() {
089    return this.otherwise;
090  }
091
092  /**
093   * {@inheritDoc}
094   */
095  @Override
096  public void accept(GoloIrVisitor visitor) {
097    visitor.visitMatchExpression(this);
098  }
099
100  /**
101   * {@inheritDoc}
102   */
103  @Override
104  public List<GoloElement<?>> children() {
105    LinkedList<GoloElement<?>> children = new LinkedList<>(clauses);
106    children.add(otherwise);
107    return children;
108  }
109
110  /**
111   * {@inheritDoc}
112   */
113  @Override
114  protected void replaceElement(GoloElement<?> original, GoloElement<?> newElement) {
115    if (!(newElement instanceof ExpressionStatement || newElement instanceof WhenClause)) {
116      throw cantConvert("ExpressionStatement or WhenClause", newElement);
117    }
118    if (otherwise.equals(original)) {
119      otherwise(newElement);
120      return;
121    }
122    if (clauses.contains(original)) {
123      @SuppressWarnings("unchecked")
124      WhenClause<ExpressionStatement<?>> when = (WhenClause<ExpressionStatement<?>>) newElement;
125      clauses.set(clauses.indexOf(original), makeParentOf(when));
126      return;
127    }
128    throw doesNotContain(original);
129  }
130}