org.assertj.swing.hierarchy.NewHierarchy Maven / Gradle / Ivy
Show all versions of assertj-swing Show documentation
/**
* 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.
*
* Copyright 2012-2015 the original author or authors.
*/
package org.assertj.swing.hierarchy;
import static java.awt.AWTEvent.COMPONENT_EVENT_MASK;
import static java.awt.AWTEvent.WINDOW_EVENT_MASK;
import static org.assertj.core.util.Lists.emptyList;
import static org.assertj.swing.listener.WeakEventListener.attachAsWeakEventListener;
import java.awt.Component;
import java.awt.Container;
import java.awt.Toolkit;
import java.awt.Window;
import java.util.Collection;
import javax.annotation.Nonnull;
import org.assertj.core.util.VisibleForTesting;
import org.assertj.swing.annotation.RunsInCurrentThread;
import org.assertj.swing.util.ToolkitProvider;
/**
*
* Isolates a {@link ComponentHierarchy} to limit to only those components created during the lifetime of this
* hierarchy. Existing AWT or Swing {@code Component}s (and any subsequently generated sub-windows) are ignored by
* default.
*
*
*
* Implicitly auto-filters {@code Window}s which are disposed (i.e. generates a {@code WindowEvent#WINDOW_CLOSED}
* event), but also implicitly un-filters them if they should be shown again. Any {@code Window} explicitly disposed by
* the calling {@link ComponentHierarchy#dispose(java.awt.Window)} will be ignored permanently.
*
*
* @author Alex Ruiz
*/
public class NewHierarchy extends ExistingHierarchy {
private final WindowFilter filter;
private final TransientWindowListener transientWindowListener;
/**
* Creates a new {@link NewHierarchy} which does not contain any existing AWT or Swing {@code Component}s.
*
* @return the created hierarchy.
*/
@Nonnull public static NewHierarchy ignoreExistingComponents() {
return new NewHierarchy(true);
}
/**
* Creates a new {@link NewHierarchy} which contains existing AWT or Swing {@code Component}s.
*
* @return the created hierarchy.
*/
@Nonnull public static NewHierarchy includeExistingComponents() {
return new NewHierarchy(false);
}
private NewHierarchy(boolean ignoreExisting) {
this(ToolkitProvider.instance().defaultToolkit(), ignoreExisting);
}
private NewHierarchy(@Nonnull Toolkit toolkit, boolean ignoreExisting) {
this.filter = new WindowFilter(parentFinder(), childrenFinder());
transientWindowListener = new TransientWindowListener(filter);
setUp(toolkit, ignoreExisting);
}
@VisibleForTesting
NewHierarchy(@Nonnull Toolkit toolkit, @Nonnull WindowFilter filter, boolean ignoreExisting) {
this.filter = filter;
transientWindowListener = new TransientWindowListener(filter);
setUp(toolkit, ignoreExisting);
}
@RunsInCurrentThread
private void setUp(@Nonnull Toolkit toolkit, boolean ignoreExisting) {
if (ignoreExisting) {
ignoreExisting();
}
attachAsWeakEventListener(toolkit, transientWindowListener, WINDOW_EVENT_MASK | COMPONENT_EVENT_MASK);
}
/**
*
* Makes all currently existing AWT and Swing {@code Component} invisible to this hierarchy, without affecting their
* current state.
*
*
*
* Note: This method is accessed in the current executing thread. Such thread may or may not be the event
* dispatch thread (EDT). Client code must call this method from the EDT.
*
*/
@RunsInCurrentThread
public void ignoreExisting() {
for (Container c : roots()) {
if (c != null) {
filter.ignore(c);
}
}
}
/**
*
* Make the given AWT or Swing {@code Component} visible to this hierarchy.
*
*
*
* Note: This method is accessed in the current executing thread. Such thread may or may not be the event
* dispatch thread (EDT). Client code must call this method from the EDT.
*
*
* @param c the given {@code Component}.
*/
@RunsInCurrentThread
public void recognize(@Nonnull Component c) {
filter.recognize(c);
}
/**
*
* Returns all the children of the given AWT or Swing {@code Component}, omitting those which are currently filtered.
*
*
*
* Note: This method is accessed in the current executing thread. Such thread may or may not be the event
* dispatch thread (EDT). Client code must call this method from the EDT.
*
*
* @param c the given {@code Component}.
* @return all the children of the given {@code Component}, omitting those which are currently filtered.
*/
@RunsInCurrentThread
@Override
@Nonnull public Collection childrenOf(@Nonnull Component c) {
if (filter.isIgnored(c)) {
return emptyList();
}
Collection children = super.childrenOf(c);
// this only removes those components which are directly filtered, not necessarily those which have a filtered
// ancestor.
children.removeAll(filter.filtered());
return children;
}
/**
*
* Returns {@code true} if the given AWT or Swing {@code Component} is not ignored.
*
*
*
* Note: This method is accessed in the current executing thread. Such thread may or may not be the event
* dispatch thread (EDT). Client code must call this method from the EDT.
*
*
* @param c the given {@code Component}.
* @return {@code true} if the given {@code Component} is not ignored, {@code false} otherwise.
*/
@RunsInCurrentThread
@Override
public boolean contains(@Nonnull Component c) {
return super.contains(c) && !filter.isIgnored(c);
}
/**
*
* Disposes the given {@code Window}, but only if it currently exists within the hierarchy. It will no longer appear
* in this hierarchy or be reachable in a hierarchy walk.
*
*
*
* Note: This method is accessed in the current executing thread. Such thread may or may not be the event
* dispatch thread (EDT). Client code must call this method from the EDT.
*
*
* @param w the {@code Window} to dispose.
*/
@RunsInCurrentThread
@Override
public void dispose(@Nonnull Window w) {
if (!contains(w)) {
return;
}
super.dispose(w);
filter.ignore(w);
}
/**
* @return all available root containers, excluding those which have been filtered.
*/
@Override
@Nonnull public Collection roots() {
Collection roots = super.roots();
roots.removeAll(filter.filtered());
return roots;
}
}