com.vaadin.polymer.Polymer Maven / Gradle / Ivy
package com.vaadin.polymer;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
import com.google.gwt.core.client.GWT;
import com.google.gwt.dom.client.Document;
import com.google.gwt.dom.client.Element;
import com.google.gwt.user.client.DOM;
import com.google.gwt.user.client.Timer;
import com.vaadin.polymer.elemental.Function;
import com.vaadin.polymer.elemental.HTMLElement;
public abstract class Polymer {
private static Set urlImported = new HashSet<>();
private static HashMap> pending = new HashMap>();
/**
* Inserts the appropriate <import> of a component given by url.
*
* @param href either an absolute url or a path relative to bower_components folder.
*/
public static void importHref(String href) {
importHref(href, null, null);
}
/**
* Inserts the appropriate <import> of a component given by url.
*
* @param href either an absolute url or a path relative to bower_components folder.
* @param ok callback to run in case of success
*/
public static void importHref(String href, Function ok) {
importHref(href, ok, null);
}
/**
* Inserts the appropriate <import> of a component given by url.
*
* @param hrefOrTag it can be an absolute url, a relative path or a tag name.
* - if it is a relative path, we prefix it with bower_components
* in case it is not already prefixed.
* - if it is a tag, we compose the relative url:
* bower_components/tagName/tagName.html
* @param ok callback to run in case of success
* @param err callback to run in case of failure
*/
public static void importHref(String hrefOrTag, final Function ok, Function err) {
if (!hrefOrTag.startsWith("http")) {
// It's a tag
if (hrefOrTag.matches("[\\w-]+")) {
hrefOrTag = hrefOrTag + "/" + hrefOrTag + ".html";
}
// It's not prefixed with the bower_components convention
if (!hrefOrTag.startsWith("bower_components")) {
hrefOrTag = "bower_components/" + hrefOrTag;
}
hrefOrTag = GWT.getModuleBaseForStaticFiles() + hrefOrTag;
}
final String href = hrefOrTag;
boolean loading = pending.containsKey(href);
if (loading || !urlImported.contains(href)) {
if (!loading) {
pending.put(href, new ArrayList());
}
final List oks = pending.get(href);
if (ok != null) {
oks.add(ok);
}
if (!loading) {
importHrefImpl(href, new Function() {
public Object call(Object arg) {
urlImported.add(href);
for (Function ok : oks) {
ok.call(arg);
}
pending.remove(href);
return null;
}
}, err);
}
} else if (ok != null) {
ok.call(href);
}
}
/**
* Inserts the appropriate <import> of a list of components
*
* @param hrefs a list of absolute urls or relative paths to load.
* @param ok callback to run in case of all import success
* @param err callback to run in case of failure
*/
public static void importHref(final List hrefs, final Function ok) {
importHref(hrefs, ok, null);
}
/**
* Inserts the appropriate <import> of a list of components
*
* @param hrefs a list of absolute urls or relative paths to load.
* @param ok callback to run in case of all import success
* @param err callback to run in case of failure
*/
public static void importHref(final List hrefs, final Function ok, Function err) {
Function allOk = new Function() {
int count = hrefs.size();
public Object call(Object arg) {
if (--count == 0) {
ok.call(arg);
}
return null;
}
};
for (String href : hrefs) {
importHref(href, allOk, err);
}
}
/**
* Returns a new instance of the Element. It loads the web component
* from the bower_components/src url if it was not loaded yet.
*/
public static T createElement(final String tagName, final String... imports) {
@SuppressWarnings("unchecked")
final T e = (T)Document.get().createElement(tagName);
if (imports.length > 0) {
ensureCustomElement(e, imports);
} else {
ensureCustomElement(e, tagName);
}
return e;
}
public static void ensureCustomElement(final T elem, String... imports) {
if (isRegisteredElement(elem)) {
return;
}
// Delay this so as the developer gets an early version of the element and
// can assign properties soon.
new Timer() {
public void run() {
// We need to remove ownProperties from the element when it's not
// registered because a bug in Polymer 1.0.x
// https://github.com/Polymer/polymer/issues/1882
saveProperties((Element)elem);
}
}.schedule(0);
new Timer() {
public void run() {
// Restore saved ownProperties
restoreProperties((Element)elem);
}
}.schedule(5);
for (String src : imports) {
importHref(src, null, null);
}
}
/**
* Returns a new instance of the Element. It loads the web component
* from the bower_components/tagName/tagName.html url if not loaded yet.
*/
public static T createElement(String tagName) {
return createElement(tagName, new String[]{});
}
/**
* Returns the JsInterop instance of Document
*/
public static com.vaadin.polymer.elemental.Document getDocument() {
return (com.vaadin.polymer.elemental.Document)Document.get();
}
/**
* Check whether a certain custom element has been registered.
*/
private native static boolean isRegisteredElement(Object e)
/*-{
return !!e && e.constructor !== $wnd.HTMLElement && e.constructor != $wnd.HTMLUnknownElement;
}-*/;
/**
* Dynamically import a link and monitors when it has been loaded.
*
* This could be done via Polymer importHref, but the method needs a custom element
* instance to be run.
*/
private native static Element importHrefImpl(String href, Function onload, Function onerror)
/*-{
console.log("gwt-polymer loading: ", href.replace(/^.*components\//,''));
// TODO(manolo): use Polymer.Base.importHref when it works in FF
var l = $doc.createElement('link');
l.rel = 'import';
l.href = href;
if (onload) {
l.onload = function() {
[email protected]::call(*)(href);
}
}
if (onerror) {
l.onerror = function() {
[email protected]::call(*)(href);
}
}
$doc.head.appendChild(l);
return l;
}-*/;
public static void ready(HTMLElement e, Function f) {
onReady((Element)e, f);
}
public static void ready(Element e, Function f) {
onReady(e, f);
}
/**
* Restore all properties saved previously to the element was
* registered.
*
* Hack for: https://github.com/Polymer/polymer/issues/1882
*/
private static native void restoreProperties(Element e)
/*-{
if (e && e.__o) {
@com.vaadin.polymer.Polymer::onReady(*)(e, function(){
for (i in e.__o) {
e[i] = e.__o[i];
}
delete e.__o;
});
}
}-*/;
/**
* If an element is not ready, loops until it gets ready, then
* run a Function (JsFunction or JavaFunction)
*/
private static native void onReady(Element e, Object f)
/*-{
function isReady(f) {
if (@com.vaadin.polymer.Polymer::isRegisteredElement(*)(e)) {
if (typeof f == 'function')
f(e);
else
[email protected]::call(*)(e);
return true;
}
}
if (!isReady(f)) {
var id = setInterval(function() {
if (isReady(f))
clearInterval(id);
}, 0);
}
}-*/;
/**
* Read all element properties and save in a JS object in the element,
* so as we can restore then once the element is registered.
*
* We consider all ownProperties but those beginning or ending with '_'
* which is the symbol used by webcomponentjs to store private info.
*
* Hack for: https://github.com/Polymer/polymer/issues/1882
*
* TODO: this is a temporary workaround, and if the issue is not fixed in
* polymer we could eventually implement the fix based on a generated proxy
* per component to store for a while any method call.
*/
private static native boolean saveProperties(Element e)
/*-{
if ([email protected]::isRegisteredElement(*)(e)) {
var o = {};
for (i in e) {
if (e.hasOwnProperty(i) && !/(^_|_$)/.test(i)) {
o[i] = e[i];
delete(e[i]);
e.__o = o;
}
}
}
}-*/;
/**
* Utility method to show a loading element if there is no one in
* hosting page.
*/
public static void startLoading() {
Element l = DOM.getElementById("loading");
if (l == null) {
l = DOM.createDiv();
l.setAttribute("style", "position:fixed;top:0px;left:0px;width:100%;text-align:center;font-family:arial;font-size:24px;color:#4285f4;");
l.setId("loading");
l.setInnerText("loading" + "...");
Document.get().getBody().appendChild(l);
}
}
public static void endLoading(final Element container, Element webcomponent) {
endLoading(container, webcomponent, null);
}
/**
* Utility method to remove a loading message and show a container when a
* web component becomes available.
*
* @param container : The container to show when the component is available
* @param webcomponent : Web component to monitor
* @param callback : Calback function
*/
public static void endLoading(final Element container, Element webcomponent, final Function func) {
container.getStyle().setOpacity(0);
container.getStyle().setProperty("transition", "opacity 1.1s");
ready(webcomponent, new Function() {
public Object call(Object arg) {
reFlow();
container.getStyle().setOpacity(1);
DOM.getElementById("loading").getStyle().setOpacity(0);
return func != null ? func.call(arg) : null;
}
});
}
/**
* Force a browser re-flow. For some reason in Chrome we need to force
* it the very first time we initialize the UI. It seems it happens with
* widgets and no with elements but not 100% positive. To test it try
* to reload the app in SDM and do not move the mouse, moving or clicking
* mouse on body also makes the UI re-draw.
*
*/
private static native void reFlow()
/*-{
if (navigator.userAgent.toLowerCase().indexOf('chrome') > -1) {
var c = 0;
var id = setInterval(function() {
// Using $doc.body.offsetWidth in an if, otherwise closure
// compiler prunes it.
if (c++ >= 20 && $doc.body.offsetWidth > 0)
clearInterval(id);
}, 50);
}
}-*/;
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy