Как я могу обрезать пробелы по Velocity - программирование
Подтвердить что ты не робот

Как я могу обрезать пробелы по 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'))
4b9b3361

Ответ 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>