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

org.assertj.swing.hierarchy.NewHierarchy Maven / Gradle / Ivy

There is a newer version: 3.17.1
Show newest version
/*
 * Created on Oct 31, 2007
 * 
 * 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 @2007-2013 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. */ public static @Nonnull NewHierarchy ignoreExistingComponents() { return new NewHierarchy(true); } /** * Creates a new {@link NewHierarchy} which contains existing AWT or Swing {@code Component}s. * * @return the created hierarchy. */ public static @Nonnull 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 public @Nonnull 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 public @Nonnull Collection roots() { Collection roots = super.roots(); roots.removeAll(filter.filtered()); return roots; } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy