/******************************************************************************* * Copyright (c) 2006, 2015 Wind River Systems, Inc. and others. * All rights reserved. This program and the accompanying materials * are made available under the terms of the Eclipse Public License v1.0 * which accompanies this distribution, and is available at * http://www.eclipse.org/legal/epl-v10.html * * 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.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; 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; public 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. Fully * equivalent to <code>readTaggedComment(getClass(), tag)</code> * @since 4.0 */ protected String readTaggedComment(final String tag) throws IOException { return TestSourceReader.readTaggedComment(CTestPlugin.getDefault().getBundle(), "ui", getClass(), tag); } /** * Reads multiple sections in comments from the source of the given class. * @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<Control>(); 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.