/* * Copyright 2013 Google 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.google.template.soy.basicfunctions; import com.google.common.base.Preconditions; import com.google.common.collect.ImmutableSet; import com.google.template.soy.data.SanitizedContent; import com.google.template.soy.data.SoyValue; import com.google.template.soy.data.restricted.IntegerData; import com.google.template.soy.data.restricted.StringData; import com.google.template.soy.jssrc.restricted.JsExpr; import com.google.template.soy.jssrc.restricted.JsExprUtils; import com.google.template.soy.jssrc.restricted.SoyJsSrcFunction; import com.google.template.soy.pysrc.restricted.PyExpr; import com.google.template.soy.pysrc.restricted.PyStringExpr; import com.google.template.soy.pysrc.restricted.SoyPySrcFunction; import com.google.template.soy.shared.restricted.SoyJavaFunction; import com.google.template.soy.shared.restricted.SoyPureFunction; import java.util.List; import java.util.Set; import javax.inject.Inject; import javax.inject.Singleton; /** * A function that returns a substring of a given string. * * <p><code>strSub(expr1, expr2, expr3)</code> requires <code>expr1</code> to be of type string or * {@link com.google.template.soy.data.SanitizedContent} and <code>expr2</code> and <code>expr3 * </code> to be of type integer. <code>expr3</code> is optional. * * <p>This function returns a new string that is a substring of <code>expr1</code>. The returned * substring begins at the index specified by <code>expr2</code>. If <code>expr3</code> is not * specified, the substring will extend to the end of <code>expr1</code>. Otherwise it will extend * to the character at index <code>expr3 - 1</code>. * */ @Singleton @SoyPureFunction final class StrSubFunction implements SoyJavaFunction, SoyJsSrcFunction, SoyPySrcFunction { @Inject StrSubFunction() {} @Override public String getName() { return "strSub"; } @Override public Set<Integer> getValidArgsSizes() { return ImmutableSet.of(2, 3); } @Override public SoyValue computeForJava(List<SoyValue> args) { SoyValue arg0 = args.get(0); SoyValue arg1 = args.get(1); SoyValue arg2 = args.size() == 3 ? args.get(2) : null; Preconditions.checkArgument( arg0 instanceof StringData || arg0 instanceof SanitizedContent, "First argument to strSub() function is not StringData or SanitizedContent: %s", arg0); Preconditions.checkArgument( arg1 instanceof IntegerData, "Second argument to strSub() function is not IntegerData: %s", arg1); if (arg2 != null) { Preconditions.checkArgument( arg2 instanceof IntegerData, "Third argument to strSub() function is not IntegerData: %s", arg2); } String strArg0 = arg0.coerceToString(); int intArg1 = arg1.integerValue(); if (arg2 != null) { return StringData.forValue(strArg0.substring(intArg1, arg2.integerValue())); } else { return StringData.forValue(strArg0.substring(intArg1)); } } @Override public JsExpr computeForJsSrc(List<JsExpr> args) { // Coerce SanitizedContent args to strings. String arg0 = JsExprUtils.toString(args.get(0)).getText(); JsExpr arg1 = args.get(1); JsExpr arg2 = args.size() == 3 ? args.get(2) : null; return new JsExpr( "(" + arg0 + ").substring(" + arg1.getText() + (arg2 != null ? "," + arg2.getText() : "") + ")", Integer.MAX_VALUE); } @Override public PyExpr computeForPySrc(List<PyExpr> args) { // Coerce SanitizedContent args to strings. String base = args.get(0).toPyString().getText(); PyExpr start = args.get(1); PyExpr end = args.size() == 3 ? args.get(2) : null; return new PyStringExpr( "(" + base + ")[" + start.getText() + ":" + (end != null ? end.getText() : "") + "]"); } }