Как я могу обрезать пробелы по Velocity
У меня есть метод, называемый render_something, который может создавать много пробелов, например:
<a href="#">#render_something('xxx')</a>
Результат может быть:
<a href="#">
something that generate from redner_something
</a>
На самом деле я хочу, чтобы это было так:
<a href="#">something that generate from redner_something</a>
Имеет ли скорость что-то вроде этого?
#trim(#render_something('xxx'))
Ответы
Ответ 1
Я просто прочитал эту статью о Velocity Whitespace Gobbling, который предлагает несколько обходов, включая Пробел Velocity Whitespace Truncated By Line Comment.
Это в основном предлагает комментировать разрывы строк, помещая комментарии в конце каждой строки. Он также предлагает не отступать от кода в ваших макросах, чтобы предотвратить появление лишних (одного из моих любимых слов) пробелов.
TBH это не отличное решение, но может удовлетворить ваши потребности. Просто поставьте ##
в конце каждой строки вашего макроса, и это сделает вещи немного приятнее... вроде
Ответ 2
Кажется, работает только java native trim().
$someValue.trim()
работает для меня
Ответ 3
Решение
В классе, где вы создаете VelocityEngine, добавьте метод следующим образом
public String trim(String str) {
return str.trim()/*.replace("\n", "").replace("\r", "")*/;
}
затем добавьте следующее к создаваемому VelocityContext:
context.put("trimmer", this);
и, наконец, в шаблоне скорости сделайте следующее
<a href="#">$trimmer.trim("#render_something('xxx')")</a>
Почему это работает?
Хотя поведение Velocity четко определено, может быть немного сложно понять, как это работает. Отдельный метод trim() необходим для получения char -последовательности из шаблона в Java-метод, где вы можете вызвать фактическую trim() в String. Насколько я знаю, в Velocity нет обрезки, но вы всегда можете перезвонить на Java с помощью трюков, подобных этому.
Двойные кавычки необходимы, потому что #render_something - это просто макрос, а не вызов функции, это означает, что результаты операторов макроса передаются дословно в точку, где выполняется макрос.
Ответ 4
Я изо всех сил пытался найти прямое решение для протирания пробелов, поэтому вот тот, который я, наконец, придумал. Это вдохновлено, и Вадзим отвечает, и эта страница http://wiki.apache.org/velocity/StructuredGlobbingResourceLoader
StructuredGlobbingResourceLoader, который мы можем найти на веб-сайте, имеет сложное поведение и не избавляется от каких-либо пробелов, поэтому я изменил его, чтобы получить простое поведение: "Удалите все пробелы в начале строк и добавьте комментарий в конце каждой строки" (что предотвращает оценку линии). Фильтр применяется во входном потоке во время загрузки.
Этот тип шаблона скорости
#if($value)
the value is $value
#end
преобразуется в
#if($value)##
the value is $value##
#end##
Затем, если вы хотите иметь разрывы строк или начало пробелов в строке, вам придется поместить ($ br, "\n" ) и поместить ($ sp, "") в ваш контекст, например, Vadzim объяснил и явно использовал их в вашем шаблоне. Этот способ позволит вам сохранить отступы с максимальным контролем.
возьмите класс с этой страницы http://wiki.apache.org/velocity/StructuredGlobbingResourceLoader
измените расширенный класс на тип загрузчика, который вам нужен (этот использует загрузчик webapp)
замените метод read() кодом, который я предоставляю
используйте класс в качестве своего ресурсного загрузчика в своих свойствах. Пример для загрузчика webapp: webapp.resource.loader.class=... StructuredGlobbingResourceLoader
public int read() throws IOException {
int ch;
switch(state){
case bol: //beginning of line, read until non-indentation character
while(true){
ch = in.read();
if (ch!=(int)' ' && ch!=(int)'\t'){
state = State.content;
return processChar(ch);
}
}
case content:
ch = in.read();
return processChar(ch);
//eol states replace all "\n" by "##\n"
case eol1:
state = State.eol2;
return (int)'#';
case eol2:
state = State.bol;
return (int)'\n';
case eof:
return -1;
}
return -1;
}
//Return the normal character if not end of file or \n
private int processChar(int ch){
switch(ch){
case -1:
state = State.eof;
return -1;
case (int)'\n':
state = State.eol1;
return (int)'#';
default:
return ch;
}
}
Любые отзывы о моей реализации приветствуются
Ответ 5
Вот мое альтернативное решение для скорости whitespace gobbling, которое позволяет создавать структуру шаблонов табуляции.
Каждый текст шаблона предварительно обрабатывается при первой загрузке в пользовательском ResourceLoader
:
private String enhanceTemplate(String body) {
if (!body.startsWith("##preserveWhitespace")) {
body = body.replaceAll("(##.*)?[ \\t\\r]*\\n+[ \\t\\r]*", Matcher.quoteReplacement("##\n"));
body = body.trim();
}
return body;
}
Это заменяет все новые строки и корректирующие пробелы только одной прокомментированной новой линией.
Разрывы строк и пространства хвоста могут быть введены явно с переменными $br и $sp из контекста контекста:
private static final VelocityContext DEFAULT_CONTEXT = new VelocityContext(new HashMap<String, String>() {{
put("sp", " ");
put("br", "\n");
}});
Ответ 6
В некоторых случаях мне пришлось существенно свести к минимуму мой script, как и js или css. Он работает хорошо, хотя людям не так просто читать. Еще один вариант устранения лишнего пространства:
<ul class="tabs">#foreach($par in $bodypars)#set( $parLen = ${_MathTool.toInteger($bodypars.size())} )#set( $parLn = $parLen - 1 )#set( $thClass = 'tb'+${parLn} )#set( $thaClass = '' )#if( $foreach.index == 1 )#set( $thClass = ${thClass}+' selected' )#set( $thaClass = ' selected' )#end#if($foreach.index != 0 && $parLen <= $maxTabs)#set ( $btitle = $_XPathTool.selectSingleNode($par,'item-subtitle') )<li class="${thClass}">#if($!btitle && $btitle != '')<a href="#" class="#cleanString($btitle.value.toLowerCase())${thaClass}">$_SerializerTool.serialize($btitle, true)</a>#end</li>#end#end</ul>