| /******************************************************************************* |
| * 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. |