Как изменить URL GWT Place с «:» по умолчанию на «/»?

#gwt #tokenize

#gwt #токенизировать

Вопрос:

По умолчанию URL GWT Place состоит из простого имени класса Place (например, «HelloPlace»), за которым следует двоеточие (:), и токена, возвращаемого PlaceTokenizer.

Мой вопрос в том, как я могу изменить «:» на «/»?

Ответ №1:

Я только что создал свой собственный PlaceHistoryMapper, который напрямую реализует интерфейс вместо использования AbstractPlaceHistoryMapper:

 public class AppPlaceHistoryMapper implements PlaceHistoryMapper
{
    String delimiter = "/";

    @Override
    public Place getPlace(String token)
    {

        String[] tokens = token.split(delimiter, 2); 

            if (tokens[0].equals("HelloPlace"))
                 ...
    }

    @Override
    public String getToken(Place place)
    {
        if (place instanceof HelloPlace)
        {
            return "HelloPlace"   delimiter   whatever;
        }
        else ...
    }
}
  

Это, конечно, требует дополнительного кода для написания, но вы можете управлять своей структурой URL в одном месте и использовать косые черты вместо двоеточий!

Комментарии:

1. Приятно, я об этом не подумал. В этом случае @WithTokenizers аннотация перестает работать (верно?), Но это уже не так важно, если вы настроите все в своем собственном Mapper.

2. Это также дает вам дополнительную гибкость при использовании URL-адресов в определенном порядке и сложных условиях — в моем приложении, например, если первая часть URL-адреса содержит допустимое название курса, я могу перенаправить логику в один набор мест, но если это не так, я могу перенаправить в другой. Я думаю, что это проще писать и понятнее читать, но с другой стороны, у меня есть некоторое недоверие к сгенерированному коду. Забавно для пользователя GWT, я знаю 😉

3. Спасибо, Райли, я заметил, что это невозможно, и ваше обходное решение после отправки этой темы сюда: code.google.com/p/google-web-toolkit/issues/detail?id=5899 . Хотя я бы предпочел не вдаваться в такие сложности из-за синтаксиса URL, я думаю, что на данный момент это единственное решение.

4. Просто чтобы убедиться, что я правильно понимаю PlaceHistoryMapper: предположим, у меня есть два места: PlaceA abd PlaceB — getPlace с возвратом нового экземпляра PlaceA или PlaceB, если его можно найти в строке токена, и getToken вернет полный фрагмент URL (без имени хоста) для этого места. Я прав?

5. Да, это верно! PlaceHistoryMapper соединяет только URL и Places , и это единственное, что это делает.

Ответ №2:

Вот как настроить разделитель при использовании стандартных мест GWT. (PlaceHistoryMapper)

Больше ничего менять не нужно; это работает со стандартным способом использования мест и маркеров.

Вставить в gwt.xml

  <generate-with
  class="com.google.gwt.place.rebind.CustomPlaceHistoryMapperGenerator">
  <when-type-assignable class="com.google.gwt.place.shared.PlaceHistoryMapper" />
 </generate-with>
  

Добавьте CustomAbstractPlaceHistoryMapper

 package com.siderakis.client.mvp;

import com.google.gwt.place.impl.AbstractPlaceHistoryMapper;
import com.google.gwt.place.shared.Place;
import com.google.gwt.place.shared.PlaceTokenizer;

public abstract class CustomAbstractPlaceHistoryMapper extends AbstractPlaceHistoryMapper {
 public final static String DELIMITER = "/";

 public static class CustomPrefixAndToken extends PrefixAndToken {
  public CustomPrefixAndToken(String prefix, String token) {
   super(prefix, token);
   assert prefix != null amp;amp; !prefix.contains(DELIMITER);
  }

  @Override
  public String toString() {
   return (prefix.length() == 0) ? token : prefix   DELIMITER   token;
  }

 }

 @Override
 public Place getPlace(String token) {
  int colonAt = token.indexOf(DELIMITER);
  String initial;
  String rest;
  if (colonAt >= 0) {
   initial = token.substring(0, colonAt);
   rest = token.substring(colonAt   1);
  } else {
   initial = "";
   rest = token;
  }
  PlaceTokenizer tokenizer = getTokenizer(initial);
  if (tokenizer != null) {
   return tokenizer.getPlace(rest);
  }
  return null;
 }

}
  

Добавьте CustomPlaceHistoryMapperGenerator

 /*
 * Copyright 2010 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.gwt.place.rebind;

import java.io.PrintWriter;

import com.siderakis.client.mvp.CustomAbstractPlaceHistoryMapper;
import com.siderakis.client.mvp.CustomAbstractPlaceHistoryMapper.CustomPrefixAndToken;

import com.google.gwt.core.client.GWT;
import com.google.gwt.core.ext.Generator;
import com.google.gwt.core.ext.GeneratorContext;
import com.google.gwt.core.ext.TreeLogger;
import com.google.gwt.core.ext.UnableToCompleteException;
import com.google.gwt.core.ext.typeinfo.JClassType;
import com.google.gwt.core.ext.typeinfo.JMethod;
import com.google.gwt.place.shared.Place;
import com.google.gwt.place.shared.PlaceTokenizer;
import com.google.gwt.user.rebind.ClassSourceFileComposerFactory;
import com.google.gwt.user.rebind.SourceWriter;

/**
 * Generates implementations of
 * {@link com.google.gwt.place.shared.PlaceHistoryMapper PlaceHistoryMapper}.
 */
public class CustomPlaceHistoryMapperGenerator extends Generator {
  private PlaceHistoryGeneratorContext context;

  @Override
  public String generate(TreeLogger logger, GeneratorContext generatorContext,
      String interfaceName) throws UnableToCompleteException {

    context = PlaceHistoryGeneratorContext.create(logger,
        generatorContext.getTypeOracle(), interfaceName);

    if (context == null) {
      return null;
    }

    PrintWriter out = generatorContext.tryCreate(logger, context.packageName,
        context.implName);

    if (out != null) {
      generateOnce(generatorContext, context, out);
    }

    return context.packageName   "."   context.implName;
  }

  private void generateOnce(GeneratorContext generatorContext, PlaceHistoryGeneratorContext context,
      PrintWriter out) throws UnableToCompleteException {

    TreeLogger logger = context.logger.branch(TreeLogger.DEBUG, String.format(
        "Generating implementation of %s", context.interfaceType.getName()));
    ClassSourceFileComposerFactory f = new ClassSourceFileComposerFactory(
        context.packageName, context.implName);

    String superClassName = String.format("%s<%s>",
        CustomAbstractPlaceHistoryMapper.class.getSimpleName(),
        context.factoryType == null ? "Void" : context.factoryType.getName());
    f.setSuperclass(superClassName);
    f.addImplementedInterface(context.interfaceType.getName());

    f.addImport(CustomAbstractPlaceHistoryMapper.class.getName());
    f.addImport(context.interfaceType.getQualifiedSourceName());

    f.addImport(CustomAbstractPlaceHistoryMapper.class.getCanonicalName());
    if (context.factoryType != null) {
      f.addImport(context.factoryType.getQualifiedSourceName());
    }

    f.addImport(Place.class.getCanonicalName());
    f.addImport(PlaceTokenizer.class.getCanonicalName());
    f.addImport(CustomPrefixAndToken.class.getCanonicalName());

    f.addImport(GWT.class.getCanonicalName());

    SourceWriter sw = f.createSourceWriter(generatorContext, out);
    sw.println();

    writeGetPrefixAndToken(context, sw);
    sw.println();

    writeGetTokenizer(context, sw);
    sw.println();

    sw.outdent();
    sw.println("}");
    generatorContext.commit(logger, out);
  }

  private void writeGetPrefixAndToken(PlaceHistoryGeneratorContext context,
      SourceWriter sw) throws UnableToCompleteException {
    sw.println("protected CustomPrefixAndToken getPrefixAndToken(Place newPlace) {");
    sw.indent();
    for (JClassType placeType : context.getPlaceTypes()) {
      String placeTypeName = placeType.getQualifiedSourceName();
      String prefix = context.getPrefix(placeType);

      sw.println("if (newPlace instanceof "   placeTypeName   ") {");
      sw.indent();
      sw.println(placeTypeName   " place = ("   placeTypeName   ") newPlace;");

      JMethod getter = context.getTokenizerGetter(prefix);
      if (getter != null) {
        sw.println(String.format("return new CustomPrefixAndToken("%s", "
              "factory.%s().getToken(place));", escape(prefix),
            getter.getName()));
      } else {
        sw.println(String.format(
            "PlaceTokenizer<%s> t = GWT.create(%s.class);", placeTypeName,
            context.getTokenizerType(prefix).getQualifiedSourceName()));
        sw.println(String.format("return new CustomPrefixAndToken("%s", "
              "t.getToken((%s) place));", escape(prefix), placeTypeName));
      }

      sw.outdent();
      sw.println("}");
    }

    sw.println("return null;");
    sw.outdent();
    sw.println("}");
  }

  private void writeGetTokenizer(PlaceHistoryGeneratorContext context,
      SourceWriter sw) throws UnableToCompleteException {
    sw.println("protected PlaceTokenizer getTokenizer(String prefix) {");
    sw.indent();

    for (String prefix : context.getPrefixes()) {
      JMethod getter = context.getTokenizerGetter(prefix);

      sw.println("if (""   escape(prefix)   "".equals(prefix)) {");
      sw.indent();

      if (getter != null) {
        sw.println("return factory."   getter.getName()   "();");
      } else {
        sw.println(String.format("return GWT.create(%s.class);",
            context.getTokenizerType(prefix).getQualifiedSourceName()));
      }

      sw.outdent();
      sw.println("}");
    }

    sw.println("return null;");
    sw.outdent();
    sw.println("}");
    sw.outdent();
  }

}
  

Ответ №3:

Хороший вопрос. Проблема в том, что это жестко запрограммировано в AbstractPlaceHistoryMapper :

AbstractPlaceHistoryMapper.PrefixAndToken.toString():

 return (prefix.length() == 0) ? token : prefix   ":"   token;
  

AbstractPlaceHistoryMapper.getPlace(строковый токен):

 int colonAt = token.indexOf(':');
...
  

И AbstractPlaceHistoryMapper жестко запрограммирован в PlaceHistoryMapperGenerator .

Вероятно, можно было бы заменить генератор, предоставив свой собственный XML-файл модуля и перенастроив привязку, но в целом я бы счел это «в принципе не настраиваемым». (Но смотрите Ответ Райли о хорошей альтернативе без декларативной конфигурации токенизатора!)

Ответ №4:

)

Я действительно оценил ответы, спасибо, что помогли мне оптимизировать мое время (я следил за отладчиком, где был вызван мой метод getToken (), и он проходит через слушателей, обработчиков, фокусников и тому подобные забавные вещи 🙂

Итак, идея HistoryMapper идеальна, но решение, если вы хотите добавить поисковик, намного проще, поскольку вам просто нужно добавить дополнительный ‘!’ после хэша закладки #!

Итак, достаточно просто украсить исходный результат дополнительным символом.

 public class PlaceHistoryMapperDecorator implements PlaceHistoryMapper {
private static final String CRAWLER_PREFIX = "!";
protected PlaceHistoryMapper delegateHistoryMapper;

public PlaceHistoryMapperDecorator(PlaceHistoryMapper delegateHistoryMapper) {
    this.delegateHistoryMapper = delegateHistoryMapper;
}

@Override
public Place getPlace(String token) {
    String cleanToken = token;
    if (token.startsWith(CRAWLER_PREFIX))
        cleanToken = token.substring(CRAWLER_PREFIX.length());
    else {
        if (token.length() > 0)
            System.err.println("there might be an error: can't find crawler prefix in "   token);
    }
    return delegateHistoryMapper.getPlace(cleanToken);
}

@Override
public String getToken(Place place) {
    return CRAWLER_PREFIX   delegateHistoryMapper.getToken(place);
}
  

}

Затем вы передаете этот новый экземпляр в свой PlaceHistoryHandler и все

         PlaceHistoryMapperDecorator historyMapperDecorator = new PlaceHistoryMapperDecorator((PlaceHistoryMapper) GWT.create(AppPlaceHistoryMapper.class));
    PlaceHistoryHandler historyHandler = new PlaceHistoryHandler(historyMapperDecorator);
  

Я протестировал его перед отправкой этого сообщения, он отлично работает 🙂