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.

EmbeddedConnector.java 9.8KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267
  1. /*
  2. * Copyright 2000-2021 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.client.ui.embedded;
  17. import java.util.Map;
  18. import java.util.logging.Logger;
  19. import com.google.gwt.dom.client.Document;
  20. import com.google.gwt.dom.client.Element;
  21. import com.google.gwt.dom.client.NativeEvent;
  22. import com.google.gwt.dom.client.Node;
  23. import com.google.gwt.dom.client.NodeList;
  24. import com.google.gwt.dom.client.ObjectElement;
  25. import com.google.gwt.dom.client.Style;
  26. import com.google.gwt.user.client.DOM;
  27. import com.google.gwt.user.client.Event;
  28. import com.vaadin.client.VTooltip;
  29. import com.vaadin.client.communication.StateChangeEvent;
  30. import com.vaadin.client.ui.AbstractComponentConnector;
  31. import com.vaadin.client.ui.ClickEventHandler;
  32. import com.vaadin.client.ui.VEmbedded;
  33. import com.vaadin.shared.MouseEventDetails;
  34. import com.vaadin.shared.ui.Connect;
  35. import com.vaadin.shared.ui.embedded.EmbeddedServerRpc;
  36. import com.vaadin.shared.ui.embedded.EmbeddedState;
  37. import com.vaadin.ui.Embedded;
  38. /**
  39. * A connector class for the Embedded component.
  40. *
  41. * @author Vaadin Ltd
  42. */
  43. @Connect(Embedded.class)
  44. public class EmbeddedConnector extends AbstractComponentConnector {
  45. private Element resourceElement;
  46. private ObjectElement objectElement;
  47. private String resourceUrl;
  48. @SuppressWarnings("deprecation")
  49. @Override
  50. public void onStateChanged(StateChangeEvent stateChangeEvent) {
  51. super.onStateChanged(stateChangeEvent);
  52. // if theme has changed the resourceUrl may need to be updated
  53. updateResourceIfNecessary();
  54. VEmbedded widget = getWidget();
  55. // Save details
  56. widget.client = getConnection();
  57. boolean clearBrowserElement = true;
  58. clickEventHandler.handleEventHandlerRegistration();
  59. final EmbeddedState state = getState();
  60. if (state.type != Embedded.TYPE_OBJECT) {
  61. // remove old style name related to type
  62. if (widget.type != null) {
  63. widget.removeStyleName(VEmbedded.CLASSNAME + "-" + widget.type);
  64. }
  65. // remove old style name related to mime type
  66. if (widget.mimetype != null) {
  67. widget.removeStyleName(
  68. VEmbedded.CLASSNAME + "-" + widget.mimetype);
  69. }
  70. widget.type = state.type == Embedded.TYPE_IMAGE ? "image"
  71. : "browser";
  72. if (widget.type.equals("image")) {
  73. widget.addStyleName(VEmbedded.CLASSNAME + "-image");
  74. Element el = null;
  75. boolean created = false;
  76. NodeList<Node> nodes = widget.getElement().getChildNodes();
  77. if (nodes != null && nodes.getLength() == 1) {
  78. Node n = nodes.getItem(0);
  79. if (n.getNodeType() == Node.ELEMENT_NODE) {
  80. Element e = (Element) n;
  81. if (e.getTagName().equals("IMG")) {
  82. el = e;
  83. }
  84. }
  85. }
  86. if (el == null) {
  87. widget.setHTML("");
  88. el = DOM.createImg();
  89. created = true;
  90. DOM.sinkEvents(el, Event.ONLOAD);
  91. }
  92. // Set attributes
  93. Style style = el.getStyle();
  94. style.setProperty("width", state.width);
  95. style.setProperty("height", state.height);
  96. resourceElement = el;
  97. objectElement = null;
  98. setResourceUrl(getResourceUrl("src"));
  99. if (state.altText != null) {
  100. el.setPropertyString("alt", state.altText);
  101. }
  102. if (created) {
  103. // insert in dom late
  104. widget.getElement().appendChild(el);
  105. }
  106. /*
  107. * Sink tooltip events so tooltip is displayed when hovering the
  108. * image.
  109. */
  110. widget.sinkEvents(VTooltip.TOOLTIP_EVENTS);
  111. } else if (widget.type.equals("browser")) {
  112. widget.addStyleName(VEmbedded.CLASSNAME + "-browser");
  113. if (widget.browserElement == null) {
  114. widget.setHTML(
  115. "<iframe width=\"100%\" height=\"100%\" frameborder=\"0\""
  116. + " allowTransparency=\"true\" src=\"\""
  117. + " name=\"" + getConnectorId()
  118. + "\"></iframe>");
  119. widget.browserElement = DOM
  120. .getFirstChild(widget.getElement());
  121. }
  122. resourceElement = widget.browserElement;
  123. objectElement = null;
  124. setResourceUrl(getResourceUrl("src"));
  125. clearBrowserElement = false;
  126. } else {
  127. getLogger()
  128. .severe("Unknown Embedded type '" + widget.type + "'");
  129. }
  130. } else if (state.mimeType != null) {
  131. // remove old style name related to type
  132. if (widget.type != null) {
  133. widget.removeStyleName(VEmbedded.CLASSNAME + "-" + widget.type);
  134. }
  135. // remove old style name related to mime type
  136. if (widget.mimetype != null) {
  137. widget.removeStyleName(
  138. VEmbedded.CLASSNAME + "-" + widget.mimetype);
  139. }
  140. final String mime = state.mimeType;
  141. if (mime.equals("application/x-shockwave-flash")) {
  142. widget.mimetype = "flash";
  143. // Handle embedding of Flash
  144. widget.addStyleName(VEmbedded.CLASSNAME + "-flash");
  145. widget.setHTML(
  146. widget.createFlashEmbed(state, getResourceUrl("src")));
  147. } else if (mime.equals("image/svg+xml")) {
  148. widget.mimetype = "svg";
  149. widget.addStyleName(VEmbedded.CLASSNAME + "-svg");
  150. String data;
  151. Map<String, String> parameters = state.parameters;
  152. ObjectElement obj = Document.get().createObjectElement();
  153. resourceElement = null;
  154. if (parameters.get("data") == null) {
  155. objectElement = obj;
  156. data = getResourceUrl("src");
  157. setResourceUrl(data);
  158. } else {
  159. objectElement = null;
  160. data = "data:image/svg+xml," + parameters.get("data");
  161. obj.setData(data);
  162. }
  163. widget.setHTML("");
  164. obj.setType(mime);
  165. if (!isUndefinedWidth()) {
  166. obj.getStyle().setProperty("width", "100%");
  167. }
  168. if (!isUndefinedHeight()) {
  169. obj.getStyle().setProperty("height", "100%");
  170. }
  171. if (state.classId != null) {
  172. obj.setAttribute("classid", state.classId);
  173. }
  174. if (state.codebase != null) {
  175. obj.setAttribute("codebase", state.codebase);
  176. }
  177. if (state.codetype != null) {
  178. obj.setAttribute("codetype", state.codetype);
  179. }
  180. if (state.archive != null) {
  181. obj.setAttribute("archive", state.archive);
  182. }
  183. if (state.standby != null) {
  184. obj.setAttribute("standby", state.standby);
  185. }
  186. widget.getElement().appendChild(obj);
  187. if (state.altText != null) {
  188. obj.setInnerText(state.altText);
  189. }
  190. } else {
  191. getLogger().severe("Unknown Embedded mimetype '" + mime + "'");
  192. }
  193. } else {
  194. getLogger()
  195. .severe("Unknown Embedded; no type or mimetype attribute");
  196. }
  197. if (clearBrowserElement) {
  198. widget.browserElement = null;
  199. }
  200. }
  201. private void updateResourceIfNecessary() {
  202. if (resourceElement != null || objectElement != null) {
  203. String src = getResourceUrl("src");
  204. if (src != null && !src.isEmpty()) {
  205. if (!src.equals(resourceUrl)) {
  206. setResourceUrl(src);
  207. }
  208. } else if (resourceUrl != null && !resourceUrl.isEmpty()) {
  209. setResourceUrl("");
  210. }
  211. }
  212. }
  213. private void setResourceUrl(String src) {
  214. resourceUrl = src;
  215. if (resourceElement != null) {
  216. resourceElement.setAttribute("src", src);
  217. } else if (objectElement != null) {
  218. objectElement.setData(src);
  219. }
  220. }
  221. @Override
  222. public VEmbedded getWidget() {
  223. return (VEmbedded) super.getWidget();
  224. }
  225. @Override
  226. public EmbeddedState getState() {
  227. return (EmbeddedState) super.getState();
  228. }
  229. /** Click event handler for sending click data to the server. */
  230. protected final ClickEventHandler clickEventHandler = new ClickEventHandler(
  231. this) {
  232. @Override
  233. protected void fireClick(NativeEvent event,
  234. MouseEventDetails mouseDetails) {
  235. getRpcProxy(EmbeddedServerRpc.class).click(mouseDetails);
  236. }
  237. };
  238. private static Logger getLogger() {
  239. return Logger.getLogger(EmbeddedConnector.class.getName());
  240. }
  241. }