All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.badlogic.gdx.backends.gwt.widgets.ResizableWidgetCollection Maven / Gradle / Ivy

/*
 * Copyright 2008 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.badlogic.gdx.backends.gwt.widgets;

import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;

import com.google.gwt.event.logical.shared.ResizeEvent;
import com.google.gwt.event.logical.shared.ResizeHandler;
import com.google.gwt.event.shared.HandlerRegistration;
import com.google.gwt.user.client.Timer;
import com.google.gwt.user.client.Window;

/** A collection of {@link ResizableWidget} that periodically checks the outer dimensions of a widget and redraws it as necessary.
 * Every {@link ResizableWidgetCollection} uses a timer, so consider the cost when adding one.
 * 
 * Typically, a {@link ResizableWidgetCollection} is only needed if you expect your widgets to resize based on window resizing or
 * other events. Fixed sized Widgets do not need to be added to a {@link ResizableWidgetCollection} as they cannot be resized. */
public class ResizableWidgetCollection implements ResizeHandler, Iterable {
	/** Information about a widgets size. */
	static class ResizableWidgetInfo {

		private ResizableWidget widget;
		private int curOffsetHeight = 0;
		private int curOffsetWidth = 0;
		private int curClientHeight = 0;
		private int curClientWidth = 0;

		/** Constructor.
		 * 
		 * @param widget the widget that will be monitored */
		public ResizableWidgetInfo (ResizableWidget widget) {
			this.widget = widget;
			updateSizes();
		}

		public int getClientHeight () {
			return curClientHeight;
		}

		public int getClientWidth () {
			return curClientWidth;
		}

		public int getOffsetHeight () {
			return curOffsetHeight;
		}

		public int getOffsetWidth () {
			return curOffsetWidth;
		}

		/** Update the current sizes.
		 * 
		 * @return true if the sizes changed, false if not. */
		public boolean updateSizes () {
			int offsetWidth = widget.getElement().getOffsetWidth();
			int offsetHeight = widget.getElement().getOffsetHeight();
			int clientWidth = widget.getElement().getClientWidth();
			int clientHeight = widget.getElement().getClientHeight();
			if (offsetWidth != curOffsetWidth || offsetHeight != curOffsetHeight || clientWidth != curClientWidth
				|| clientHeight != curClientHeight) {
				this.curOffsetWidth = offsetWidth;
				this.curOffsetHeight = offsetHeight;
				this.curClientWidth = clientWidth;
				this.curClientHeight = clientHeight;
				return true;
			}

			return false;
		}
	}

	/** The default delay between resize checks in milliseconds. */
	private static final int DEFAULT_RESIZE_CHECK_DELAY = 400;

	/** A static {@link ResizableWidgetCollection} that can be used in most cases. */
	private static ResizableWidgetCollection staticCollection = null;

	/** Get the globally accessible {@link ResizableWidgetCollection}. In most cases, the global collection can be used for all
	 * {@link ResizableWidget}s.
	 * 
	 * @return the global {@link ResizableWidgetCollection} */
	public static ResizableWidgetCollection get () {
		if (staticCollection == null) {
			staticCollection = new ResizableWidgetCollection();
		}
		return staticCollection;
	}

	/** The timer used to periodically compare the dimensions of elements to their old dimensions. */
	private Timer resizeCheckTimer = new Timer() {
		@Override
		public void run () {
			// Ignore changes that result from window resize events
			if (windowHeight != Window.getClientHeight() || windowWidth != Window.getClientWidth()) {
				windowHeight = Window.getClientHeight();
				windowWidth = Window.getClientWidth();
				schedule(resizeCheckDelay);
				return;
			}

			// Look for elements that have new dimensions
			checkWidgetSize();

			// Start checking again
			if (resizeCheckingEnabled) {
				schedule(resizeCheckDelay);
			}
		}
	};

	/** A hash map of the resizable widgets this collection is checking. */
	private final Map widgets = new HashMap<>();

	/** The current window height. */
	int windowHeight = 0;

	/** The current window width. */
	int windowWidth = 0;

	/** The hook used to remove the window handler. */
	private HandlerRegistration windowHandler;

	/** The delay between resize checks. */
	int resizeCheckDelay = DEFAULT_RESIZE_CHECK_DELAY;

	/** A boolean indicating that resize checking should run. */
	boolean resizeCheckingEnabled;

	/** Create a ResizableWidget. */
	public ResizableWidgetCollection () {
		this(DEFAULT_RESIZE_CHECK_DELAY);
	}

	/** Constructor.
	 * 
	 * @param resizeCheckingEnabled false to disable resize checking */
	public ResizableWidgetCollection (boolean resizeCheckingEnabled) {
		this(DEFAULT_RESIZE_CHECK_DELAY, resizeCheckingEnabled);
	}

	/** Constructor.
	 * 
	 * @param resizeCheckDelay the delay between checks in milliseconds */
	public ResizableWidgetCollection (int resizeCheckDelay) {
		this(resizeCheckDelay, true);
	}

	/** Constructor. */
	protected ResizableWidgetCollection (int resizeCheckDelay, boolean resizeCheckingEnabled) {
		setResizeCheckDelay(resizeCheckDelay);
		setResizeCheckingEnabled(resizeCheckingEnabled);
	}

	/** Add a resizable widget to the collection.
	 * 
	 * @param widget the resizable widget to add */
	public void add (ResizableWidget widget) {
		widgets.put(widget, new ResizableWidgetInfo(widget));
	}

	/** Check to see if any Widgets have been resized and call their handlers appropriately. */
	public void checkWidgetSize () {
		for (Map.Entry entry : widgets.entrySet()) {
			ResizableWidget widget = entry.getKey();
			ResizableWidgetInfo info = entry.getValue();

			// Call the onResize method only if the widget is attached
			if (info.updateSizes()) {
				// Check that the offset width and height are greater than 0.
				if (info.getOffsetWidth() > 0 && info.getOffsetHeight() > 0 && widget.isAttached()) {
					// Send the client dimensions, which is the space available for
					// rendering.
					widget.onResize(info.getOffsetWidth(), info.getOffsetHeight());
				}
			}
		}
	}

	/** Get the delay between resize checks in milliseconds.
	 * 
	 * @return the resize check delay */
	public int getResizeCheckDelay () {
		return resizeCheckDelay;
	}

	/** Check whether or not resize checking is enabled.
	 * 
	 * @return true is resize checking is enabled */
	public boolean isResizeCheckingEnabled () {
		return resizeCheckingEnabled;
	}

	public Iterator iterator () {
		return widgets.keySet().iterator();
	}

	/** Remove a {@link ResizableWidget} from the collection.
	 * 
	 * @param widget the widget to remove */
	public void remove (ResizableWidget widget) {
		widgets.remove(widget);
	}

	/** Set the delay between resize checks in milliseconds.
	 * 
	 * @param resizeCheckDelay the new delay */
	public void setResizeCheckDelay (int resizeCheckDelay) {
		this.resizeCheckDelay = resizeCheckDelay;
	}

	/** Set whether resize checking is enabled. If disabled, elements will still be resized on window events, but the timer will
	 * not check their dimensions periodically.
	 *
	 * @param enabled true to enable the resize checking timer */
	public void setResizeCheckingEnabled (boolean enabled) {
		if (enabled && !resizeCheckingEnabled) {
			resizeCheckingEnabled = true;
			if (windowHandler == null) {
				windowHandler = Window.addResizeHandler(this);
			}
			resizeCheckTimer.schedule(resizeCheckDelay);
		} else if (!enabled && resizeCheckingEnabled) {
			resizeCheckingEnabled = false;
			if (windowHandler != null) {
				windowHandler.removeHandler();
				windowHandler = null;
			}
			resizeCheckTimer.cancel();
		}
	}

	/** Inform the {@link ResizableWidgetCollection} that the size of a widget has changed and already been redrawn. This will
	 * prevent the widget from being redrawn on the next loop.
	 * 
	 * @param widget the widget's size that changed */
	public void updateWidgetSize (ResizableWidget widget) {
		if (!widget.isAttached()) {
			return;
		}

		ResizableWidgetInfo info = widgets.get(widget);
		if (info != null) {
			info.updateSizes();
		}
	}

	/** Called when the browser window is resized. */
	@Override
	public void onResize (ResizeEvent event) {
		checkWidgetSize();
	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy