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; 015import java.util.Collections; 016 017import gololang.ir.CollectionLiteral.Type; 018 019/** 020 * Represents a collection literal. 021 * 022 * <p>For instance code such as: 023 * <pre class="listing"><code class="lang-golo" data-lang="golo"> 024 * list[1, 2, 3] 025 * </code></pre> 026 * @see Type 027 */ 028public final class CollectionLiteral extends ExpressionStatement<CollectionLiteral> { 029 030 public enum Type { 031 array, list, set, map, tuple, vector, range 032 } 033 034 private final Type type; 035 private final List<ExpressionStatement<?>> expressions = new LinkedList<>(); 036 037 private CollectionLiteral(Type type) { 038 super(); 039 this.type = type; 040 } 041 042 /** 043 * Creates a collection literal of the given type. 044 * 045 * <p>For instance: 046 * <pre class="listing"><code class="lang-java" data-lang="java"> 047 * collection(CollectionLiteral.Type.array, constant(1), constant(2), constant(3)) 048 * </code></pre> 049 * creates 050 * <pre class="listing"><code class="lang-golo" data-lang="golo"> 051 * array[1, 2, 3] 052 * </code></pre> 053 * 054 * @param type the type 055 * @param values expressions to add to the collection 056 */ 057 public static CollectionLiteral create(Object type, Object... values) { 058 CollectionLiteral col = new CollectionLiteral( 059 type instanceof CollectionLiteral.Type 060 ? (CollectionLiteral.Type) type 061 : CollectionLiteral.Type.valueOf(type.toString())); 062 for (Object v : values) { 063 col.add(v); 064 } 065 return col; 066 } 067 068 069 protected CollectionLiteral self() { return this; } 070 071 /** 072 * Adds an expression to the collection. 073 * 074 * @param expression the expression to add. 075 * @return the collection literal. 076 */ 077 public CollectionLiteral add(Object expression) { 078 this.expressions.add(makeParentOf(ExpressionStatement.of(expression))); 079 return this; 080 } 081 082 public Type getType() { 083 return type; 084 } 085 086 public List<ExpressionStatement<?>> getExpressions() { 087 return Collections.unmodifiableList(expressions); 088 } 089 090 @Override 091 public String toString() { 092 return this.type.toString() + this.expressions.toString(); 093 } 094 095 /** 096 * {@inheritDoc} 097 */ 098 @Override 099 public void accept(GoloIrVisitor visitor) { 100 visitor.visitCollectionLiteral(this); 101 } 102 103 /** 104 * {@inheritDoc} 105 */ 106 @Override 107 public List<GoloElement<?>> children() { 108 return Collections.unmodifiableList(expressions); 109 } 110 111 /** 112 * {@inheritDoc} 113 */ 114 @Override 115 protected void replaceElement(GoloElement<?> original, GoloElement<?> newElement) { 116 if (expressions.contains(original) && newElement instanceof ExpressionStatement) { 117 expressions.set(expressions.indexOf(original), ExpressionStatement.of(newElement)); 118 } else { 119 throw cantReplace(original, newElement); 120 } 121 } 122}