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