blob: 5fec6d0cd014e6717f398f0e658310152d7c547d [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2006, 2015 Wind River Systems, Inc. and others.
*
* This program and the accompanying materials
* are made available under the terms of the Eclipse Public License 2.0
* which accompanies this distribution, and is available at
* https://www.eclipse.org/legal/epl-2.0/
*
* SPDX-License-Identifier: EPL-2.0
*
* Contributors:
* Markus Schorn - initial API and implementation
*******************************************************************************/
package org.eclipse.cdt.ui.tests;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
import org.eclipse.cdt.core.dom.ast.IASTTranslationUnit;
import org.eclipse.cdt.core.index.IIndex;
import org.eclipse.cdt.core.model.CModelException;
import org.eclipse.cdt.core.model.ICProject;
import org.eclipse.cdt.core.testplugin.util.BaseTestCase;
import org.eclipse.cdt.core.testplugin.util.TestSourceReader;
import org.eclipse.cdt.ui.testplugin.CTestPlugin;
import org.eclipse.cdt.ui.testplugin.util.StringAsserts;
import org.eclipse.core.commands.ExecutionException;
import org.eclipse.core.commands.NotEnabledException;
import org.eclipse.core.commands.NotHandledException;
import org.eclipse.core.commands.common.NotDefinedException;
import org.eclipse.core.resources.IContainer;
import org.eclipse.core.resources.IFile;
import org.eclipse.core.runtime.CoreException;
import org.eclipse.core.runtime.Path;
import org.eclipse.swt.SWT;
import org.eclipse.swt.SWTException;
import org.eclipse.swt.widgets.Composite;
import org.eclipse.swt.widgets.Control;
import org.eclipse.swt.widgets.Display;
import org.eclipse.swt.widgets.Event;
import org.eclipse.swt.widgets.Tree;
import org.eclipse.swt.widgets.TreeItem;
import org.eclipse.ui.IEditorPart;
import org.eclipse.ui.IEditorReference;
import org.eclipse.ui.IViewPart;
import org.eclipse.ui.IViewReference;
import org.eclipse.ui.IWorkbenchPage;
import org.eclipse.ui.IWorkbenchPartSite;
import org.eclipse.ui.IWorkbenchWindow;
import org.eclipse.ui.PartInitException;
import org.eclipse.ui.PlatformUI;
import org.eclipse.ui.WorkbenchException;
import org.eclipse.ui.handlers.IHandlerService;
import org.eclipse.ui.internal.WorkbenchPartReference;
public abstract class BaseUITestCase extends BaseTestCase {
public BaseUITestCase() {
super();
}
public BaseUITestCase(String name) {
super(name);
}
@Override
protected void setUp() throws Exception {
super.setUp();
final IWorkbenchPage activePage = PlatformUI.getWorkbench().getActiveWorkbenchWindow().getActivePage();
IViewPart view = activePage.findView("org.eclipse.cdt.ui.tests.DOMAST.DOMAST");
if (view != null) {
activePage.hideView(view);
}
}
@Override
protected void tearDown() throws Exception {
runEventQueue(0);
super.tearDown();
}
/**
* Reads a section in comments form the source of the given class.
*/
protected String readTaggedComment(Class clazz, final String tag) throws IOException {
return TestSourceReader.readTaggedComment(CTestPlugin.getDefault().getBundle(), "ui", clazz, tag);
}
/**
* Reads a section in comments form the source of the given class. Fully
* equivalent to <code>readTaggedComment(getClass(), tag)</code>
* @since 4.0
*/
protected String readTaggedComment(final String tag) throws IOException {
return readTaggedComment(getClass(), tag);
}
/**
* Reads multiple sections in comments from the source of the given class.
*
* Trailing whitespace can be removed by editor/clean-up actions. To enforce whitespace
* at end of line, use ${whitspace_eol}, which will be removed, but cause the
* whitespace to the left of it to be preserved.
*
* @since 4.0
*/
public StringBuilder[] getContentsForTest(int sections) throws IOException {
return TestSourceReader.getContentsForTest(CTestPlugin.getDefault().getBundle(), "ui", getClass(), getName(),
sections);
}
public String getAboveComment() throws IOException {
return getContentsForTest(1)[0].toString();
}
protected IFile createFile(IContainer container, String fileName, String contents) throws Exception {
return TestSourceReader.createFile(container, new Path(fileName), contents);
}
protected IASTTranslationUnit createIndexBasedAST(IIndex index, ICProject project, IFile file)
throws CModelException, CoreException {
return TestSourceReader.createIndexBasedAST(index, project, file);
}
protected void runEventQueue(int time) {
final long endTime = System.currentTimeMillis() + time;
while (true) {
while (Display.getCurrent().readAndDispatch()) {
}
long diff = endTime - System.currentTimeMillis();
if (diff <= 0) {
break;
}
try {
Thread.sleep(Math.min(20, diff));
} catch (InterruptedException e) {
return;
}
}
}
protected void expandTreeItem(Tree tree, int idx) {
expandTreeItem(tree, new int[] { idx });
}
protected void expandTreeItem(Tree tree, int idx1, int idx2) {
expandTreeItem(tree, new int[] { idx1, idx2 });
}
protected void expandTreeItem(Tree tree, int[] idxs) {
TreeItem item = tree.getItem(idxs[0]);
assertNotNull(item);
expandTreeItem(item);
for (int i = 1; i < idxs.length; i++) {
item = item.getItem(idxs[i]);
assertNotNull(item);
expandTreeItem(item);
}
}
protected void expandTreeItem(TreeItem item) {
Event event = new Event();
event.item = item;
item.getParent().notifyListeners(SWT.Expand, event);
item.setExpanded(true);
runEventQueue(0);
}
protected void selectTreeItem(Tree tree, int idx) {
selectTreeItem(tree, new int[] { idx });
}
protected void selectTreeItem(Tree tree, int idx1, int idx2) {
selectTreeItem(tree, new int[] { idx1, idx2 });
}
protected void selectTreeItem(Tree tree, int[] idxs) {
TreeItem item = tree.getItem(idxs[0]);
assertNotNull(item);
for (int i = 1; i < idxs.length; i++) {
item = item.getItem(idxs[i]);
assertNotNull(item);
}
tree.setSelection(item);
Event event = new Event();
event.item = item;
item.getParent().notifyListeners(SWT.Selection, event);
runEventQueue(0);
}
protected void closeEditor(IEditorPart editor) {
IWorkbenchPartSite site;
IWorkbenchPage page;
if (editor != null && (site = editor.getSite()) != null && (page = site.getPage()) != null) {
page.closeEditor(editor, false);
}
}
protected void closeAllEditors() {
IWorkbenchWindow[] windows = PlatformUI.getWorkbench().getWorkbenchWindows();
for (IWorkbenchWindow window : windows) {
IWorkbenchPage[] pages = window.getPages();
for (IWorkbenchPage page : pages) {
page.closeAllEditors(false);
}
}
}
protected void restoreAllParts() throws WorkbenchException {
IWorkbenchPage page = PlatformUI.getWorkbench().getActiveWorkbenchWindow().getActivePage();
page.zoomOut();
runEventQueue(0);
IViewReference[] viewRefs = page.getViewReferences();
for (IViewReference ref : viewRefs) {
page.setPartState(ref, IWorkbenchPage.STATE_RESTORED);
}
IEditorReference[] editorRefs = page.getEditorReferences();
for (IEditorReference ref : editorRefs) {
page.setPartState(ref, IWorkbenchPage.STATE_RESTORED);
}
runEventQueue(0);
}
protected IViewPart activateView(String id) throws PartInitException {
IViewPart view = PlatformUI.getWorkbench().getActiveWorkbenchWindow().getActivePage().showView(id);
assertNotNull(view);
runEventQueue(0);
return view;
}
protected void executeCommand(IViewPart viewPart, String commandID)
throws ExecutionException, NotDefinedException, NotEnabledException, NotHandledException {
IHandlerService hs = viewPart.getSite().getService(IHandlerService.class);
assertNotNull(hs);
hs.executeCommand(commandID, null);
}
private Control[] findControls(Control w, Class<?> clazz) {
ArrayList<Control> result = new ArrayList<>();
findControls(w, clazz, result);
return result.toArray(new Control[result.size()]);
}
private void findControls(Control w, Class<?> clazz, List<Control> result) {
if (clazz.isInstance(w)) {
result.add(w);
}
if (w instanceof Composite) {
Composite comp = (Composite) w;
Control[] children = comp.getChildren();
for (Control element : children) {
findControls(element, clazz, result);
}
}
}
final protected TreeItem checkTreeNode(IViewPart part, int i0, String label) {
assertNotNull(label); // we don't handle testing for a base node to not appear; can be added if/when needed
IViewReference viewRef = part.getViewSite().getPage().findViewReference(part.getViewSite().getId());
Control viewControl = ((WorkbenchPartReference) viewRef).getPane().getControl();
Tree tree = null;
TreeItem root = null;
StringBuilder cands = new StringBuilder();
for (int i = 0; i < 400; i++) {
cands.setLength(0);
Control[] trees = findControls(viewControl, Tree.class);
for (int j = 0; j < trees.length; j++) {
try {
tree = (Tree) trees[j];
root = tree.getItem(i0);
if (label.equals(root.getText())) {
return root;
}
if (j > 0) {
cands.append('|');
}
cands.append(root.getText());
} catch (SWTException e) {
// in case widget was disposed, item may be replaced
} catch (IllegalArgumentException e) {
// item does not yet exist.
}
}
runEventQueue(10);
}
assertNotNull("No tree in viewpart", tree);
assertNotNull("Tree node " + label + "{" + i0 + "} does not exist!", root);
assertEquals(label, cands.toString());
return root;
}
final protected TreeItem checkTreeNode(Tree tree, int i0, String label) {
assertNotNull(label); // we don't handle testing for a base node to not appear; can be added if/when needed
TreeItem root = null;
for (int millis = 0; millis < 5000; millis = millis == 0 ? 1 : millis * 2) {
runEventQueue(millis);
try {
root = tree.getItem(i0);
if (label.equals(root.getText())) {
return root;
}
} catch (SWTException e) {
// in case widget was disposed, item may be replaced
} catch (IllegalArgumentException e) {
// item does not yet exist.
}
}
fail("Tree node " + label + "{" + i0 + "} does not exist!");
return null;
}
/**
* Pass label=null to test that the {i0,i1} node doesn't exist
*/
final protected TreeItem checkTreeNode(Tree tree, int i0, int i1, String label) {
String firstItemText = null;
int timeout = (label == null) ? 1000 : 5000; // see footnote[0]
// If {i0,i1} exists, whether or not it matches label (when label != null)
boolean nodePresent = false;
for (int millis = 0; millis < timeout; millis = millis == 0 ? 1 : millis * 2) {
nodePresent = false;
runEventQueue(millis);
TreeItem i0Node = tree.getItem(i0);
if (!i0Node.getExpanded()) {
expandTreeItem(i0Node);
}
try {
TreeItem firstItem = i0Node.getItem(0);
firstItemText = firstItem.getText();
if (!firstItemText.isEmpty() && !firstItemText.equals("...")) {
TreeItem item = i0Node.getItem(i1);
nodePresent = true;
if (label != null && label.equals(item.getText())) {
return item;
}
}
} catch (SWTException e) {
// in case widget was disposed, item may be replaced
} catch (IllegalArgumentException e) {
// item does not yet exist.
}
}
if (label == null) {
assertFalse("Tree node {" + i0 + "," + i1 + "} exists but shouldn't!", nodePresent);
} else {
fail("Tree node " + label + "{" + i0 + "," + i1 + "} does not exist!");
}
return null;
}
public static void assertEqualString(String actual, String expected) {
StringAsserts.assertEqualString(actual, expected);
}
}
// Footnotes
// [0] Waiting for something to appear is very efficient; waiting for it to not
// appear is very inefficient. In the former case, regardless of how much time
// is alloted, we stop waiting as soon as the item appears, whereas in the
// latter we have to wait the entire timeout. In test suites with thousands of
// tests, efficiency is critical. Thus, in testing that a tree node doesn't have
// an Nth child, we shoot for efficiency and accept the risk of a false
// negative. More specifically, we wait only one second for the item TO NOT
// appear, whereas we give an item up to five seconds TO appear. This compromise
// is better than not having that sort of test at all, which some would argue is
// the better approach. In practice, it takes about 60-150 ms for the item to
// appear (on my machine), but we give it up to five seconds. Waiting one second
// for it to not appear should be more than adequate.