You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

RendererVisitor.java 6.3KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153
  1. /*
  2. * Copyright 2000-2018 Vaadin Ltd.
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License"); you may not
  5. * use this file except in compliance with the License. You may obtain a copy of
  6. * the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
  12. * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
  13. * License for the specific language governing permissions and limitations under
  14. * the License.
  15. */
  16. package com.vaadin.server.widgetsetutils.metadata;
  17. import com.google.gwt.core.ext.TreeLogger;
  18. import com.google.gwt.core.ext.TreeLogger.Type;
  19. import com.google.gwt.core.ext.UnableToCompleteException;
  20. import com.google.gwt.core.ext.typeinfo.JClassType;
  21. import com.google.gwt.core.ext.typeinfo.JMethod;
  22. import com.google.gwt.core.ext.typeinfo.JParameterizedType;
  23. import com.google.gwt.core.ext.typeinfo.JType;
  24. import com.google.gwt.core.ext.typeinfo.NotFoundException;
  25. import com.vaadin.client.connectors.AbstractRendererConnector;
  26. import com.vaadin.client.connectors.grid.AbstractGridRendererConnector;
  27. import elemental.json.JsonValue;
  28. /**
  29. * Generates type data for renderer connectors.
  30. * <ul>
  31. * <li>Stores the return type of the overridden
  32. * {@link AbstractGridRendererConnector#getRenderer() getRenderer} method to
  33. * enable automatic creation of an instance of the proper renderer type.
  34. * <li>Stores the presentation type of the connector to enable the
  35. * {@link AbstractGridRendererConnector#decode(elemental.json.JsonValue) decode}
  36. * method to work without having to implement a "getPresentationType" method.
  37. * </ul>
  38. *
  39. * @see WidgetInitVisitor
  40. *
  41. * @since 7.4
  42. * @author Vaadin Ltd
  43. */
  44. public class RendererVisitor extends TypeVisitor {
  45. @Override
  46. public void visitConnector(TreeLogger logger, JClassType type,
  47. ConnectorBundle bundle) throws UnableToCompleteException {
  48. if (ConnectorBundle.isConnectedRendererConnector(type)) {
  49. doRendererType(logger, type, bundle);
  50. doPresentationType(logger, type, bundle);
  51. }
  52. }
  53. private static void doRendererType(TreeLogger logger, JClassType type,
  54. ConnectorBundle bundle) throws UnableToCompleteException {
  55. // The class in which createRenderer is implemented
  56. JClassType createRendererClass = ConnectorBundle
  57. .findInheritedMethod(type, "createRenderer").getEnclosingType();
  58. // Needs GWT constructor if createRenderer is not overridden
  59. String connectorSrcName = createRendererClass.getQualifiedSourceName();
  60. if (isAbstractRendererConnector(connectorSrcName)) {
  61. // createRenderer not overridden
  62. JMethod getRenderer = ConnectorBundle.findInheritedMethod(type,
  63. "getRenderer");
  64. String rendererSrcName = getRenderer.getEnclosingType()
  65. .getQualifiedSourceName();
  66. if (isAbstractRendererConnector(rendererSrcName)) {
  67. // getRenderer not overridden
  68. logger.log(Type.ERROR, type.getQualifiedSourceName()
  69. + " must override either createRenderer or getRenderer");
  70. throw new UnableToCompleteException();
  71. }
  72. JClassType rendererType = getRenderer.getReturnType().isClass();
  73. bundle.setNeedsGwtConstructor(rendererType);
  74. // Also needs renderer type to find the right GWT constructor
  75. bundle.setNeedsReturnType(type, getRenderer);
  76. logger.log(Type.DEBUG,
  77. "Renderer type of " + type + " is " + rendererType);
  78. }
  79. }
  80. private static void doPresentationType(TreeLogger logger, JClassType type,
  81. ConnectorBundle bundle) throws UnableToCompleteException {
  82. JType presentationType = getPresentationType(type, logger);
  83. bundle.setPresentationType(type, presentationType);
  84. if (!hasCustomDecodeMethod(type, logger)) {
  85. bundle.setNeedsSerialize(presentationType);
  86. }
  87. logger.log(Type.DEBUG,
  88. "Presentation type of " + type + " is " + presentationType);
  89. }
  90. private static boolean hasCustomDecodeMethod(JClassType type,
  91. TreeLogger logger) throws UnableToCompleteException {
  92. try {
  93. JMethod decodeMethod = ConnectorBundle.findInheritedMethod(type,
  94. "decode",
  95. type.getOracle().getType(JsonValue.class.getName()));
  96. if (decodeMethod == null) {
  97. throw new NotFoundException();
  98. }
  99. String decodeSrcName = decodeMethod.getEnclosingType()
  100. .getQualifiedSourceName();
  101. return !isAbstractRendererConnector(decodeSrcName);
  102. } catch (NotFoundException e) {
  103. logger.log(Type.ERROR,
  104. "Can't find decode method for renderer " + type, e);
  105. throw new UnableToCompleteException();
  106. }
  107. }
  108. private static JType getPresentationType(JClassType type, TreeLogger logger)
  109. throws UnableToCompleteException {
  110. JClassType originalType = type;
  111. while (type != null) {
  112. String typeBinName = type.getQualifiedBinaryName();
  113. if (isAbstractRendererConnector(typeBinName)) {
  114. JParameterizedType parameterized = type.isParameterized();
  115. if (parameterized == null) {
  116. logger.log(Type.ERROR, type.getQualifiedSourceName()
  117. + " must define the generic parameter of the inherited "
  118. + AbstractRendererConnector.class.getSimpleName());
  119. throw new UnableToCompleteException();
  120. }
  121. return parameterized.getTypeArgs()[0];
  122. }
  123. type = type.getSuperclass();
  124. }
  125. throw new IllegalArgumentException("The type "
  126. + originalType.getQualifiedSourceName() + " does not extend "
  127. + AbstractRendererConnector.class.getName());
  128. }
  129. private static boolean isAbstractRendererConnector(
  130. String connectorSrcName) {
  131. return connectorSrcName
  132. .equals(AbstractRendererConnector.class.getName())
  133. || connectorSrcName
  134. .equals(ConnectorBundle.OLD_RENDERER_CONNECTOR_NAME);
  135. }
  136. }