Java tutorial
/* * SonarLint for Eclipse * Copyright (C) 2015-2016 SonarSource SA * sonarlint@sonarsource.com * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 3 of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with this program; if not, write to the Free Software Foundation, * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ package org.sonarlint.eclipse.ui.internal.popup; import org.eclipse.core.runtime.IProgressMonitor; import org.eclipse.core.runtime.IStatus; import org.eclipse.core.runtime.Status; import org.eclipse.core.runtime.jobs.Job; import org.eclipse.jface.resource.JFaceResources; import org.eclipse.jface.resource.LocalResourceManager; import org.eclipse.jface.window.Window; import org.eclipse.swt.SWT; import org.eclipse.swt.events.ControlAdapter; import org.eclipse.swt.events.ControlEvent; import org.eclipse.swt.events.MouseAdapter; import org.eclipse.swt.events.MouseEvent; import org.eclipse.swt.events.MouseTrackAdapter; import org.eclipse.swt.graphics.Color; import org.eclipse.swt.graphics.GC; import org.eclipse.swt.graphics.Image; import org.eclipse.swt.graphics.Point; import org.eclipse.swt.graphics.Rectangle; import org.eclipse.swt.graphics.Region; import org.eclipse.swt.layout.GridData; import org.eclipse.swt.layout.GridLayout; import org.eclipse.swt.widgets.Composite; import org.eclipse.swt.widgets.Control; import org.eclipse.swt.widgets.Display; import org.eclipse.swt.widgets.Label; import org.eclipse.swt.widgets.Monitor; import org.eclipse.swt.widgets.Shell; import org.sonarlint.eclipse.ui.internal.SonarLintImages; import org.sonarlint.eclipse.ui.internal.popup.AnimationUtil.FadeJob; import org.sonarlint.eclipse.ui.internal.popup.AnimationUtil.IFadeListener; /** * A popup window with a title bar and message area for displaying notifications. * * Copied from Mylyn waiting for an inclusion in standard Eclipse */ public abstract class AbstractNotificationPopup extends Window { private static final int TITLE_HEIGHT = 24; private static final String LABEL_JOB_CLOSE = "Close Notification Job"; private static final int MAX_WIDTH = 400; private static final int MIN_HEIGHT = 100; private static final long DEFAULT_DELAY_CLOSE = 8 * (long) 1000; private static final int PADDING_EDGE = 5; private long delayClose = DEFAULT_DELAY_CLOSE; protected LocalResourceManager resources; private GradientColors color; private final Display display; private Shell shell; private Region lastUsedRegion; private Image lastUsedBgImage; private final Job closeJob = new Job(LABEL_JOB_CLOSE) { @Override protected IStatus run(IProgressMonitor monitor) { if (!display.isDisposed()) { display.asyncExec(new Runnable() { @Override public void run() { Shell shell = AbstractNotificationPopup.this.getShell(); if (shell == null || shell.isDisposed()) { return; } if (isMouseOver(shell)) { scheduleAutoClose(); return; } AbstractNotificationPopup.this.closeFade(); } }); } if (monitor.isCanceled()) { return Status.CANCEL_STATUS; } return Status.OK_STATUS; } }; private final boolean respectDisplayBounds = true; private final boolean respectMonitorBounds = true; private FadeJob fadeJob; private boolean fadingEnabled; public AbstractNotificationPopup(Display display) { this(display, SWT.NO_TRIM | SWT.ON_TOP | SWT.NO_FOCUS | SWT.TOOL); } public AbstractNotificationPopup(Display display, int style) { super(new Shell(display)); setShellStyle(style); this.display = display; resources = new LocalResourceManager(JFaceResources.getResources()); initResources(); closeJob.setSystem(true); } public boolean isFadingEnabled() { return fadingEnabled; } public void setFadingEnabled(boolean fadingEnabled) { this.fadingEnabled = fadingEnabled; } protected abstract String getPopupShellTitle(); protected Image getPopupShellImage(int maximumHeight) { return null; } /** * Override to populate with notifications. * * @param parent */ protected void createContentArea(Composite parent) { // empty by default } /** * Override to customize the title bar */ protected void createTitleArea(Composite parent) { ((GridData) parent.getLayoutData()).heightHint = TITLE_HEIGHT; Label titleImageLabel = new Label(parent, SWT.NONE); titleImageLabel.setImage(getPopupShellImage(TITLE_HEIGHT)); Label titleTextLabel = new Label(parent, SWT.NONE); titleTextLabel.setText(getPopupShellTitle()); titleTextLabel.setFont(JFaceResources.getFontRegistry().getBold(JFaceResources.DEFAULT_FONT)); titleTextLabel.setForeground(getTitleForeground()); titleTextLabel.setLayoutData(new GridData(SWT.FILL, SWT.CENTER, true, true)); titleTextLabel.setCursor(parent.getDisplay().getSystemCursor(SWT.CURSOR_HAND)); final Label button = new Label(parent, SWT.NONE); button.setImage(SonarLintImages.NOTIFICATION_CLOSE); button.addMouseTrackListener(new MouseTrackAdapter() { @Override public void mouseEnter(MouseEvent e) { button.setImage(SonarLintImages.NOTIFICATION_CLOSE_HOVER); } @Override public void mouseExit(MouseEvent e) { button.setImage(SonarLintImages.NOTIFICATION_CLOSE); } }); button.addMouseListener(new MouseAdapter() { @Override public void mouseUp(MouseEvent e) { close(); setReturnCode(CANCEL); } }); } protected Color getTitleForeground() { return color.getTitleText(); } private void initResources() { color = new GradientColors(display, resources); } @Override protected void configureShell(Shell newShell) { super.configureShell(newShell); shell = newShell; newShell.setBackground(color.getBorder()); } @Override public void create() { super.create(); addRegion(shell); } private void addRegion(Shell shell) { Region region = new Region(); Point s = shell.getSize(); /* Add entire Shell */ region.add(0, 0, s.x, s.y); /* Subtract Top-Left Corner */ region.subtract(0, 0, 5, 1); region.subtract(0, 1, 3, 1); region.subtract(0, 2, 2, 1); region.subtract(0, 3, 1, 1); region.subtract(0, 4, 1, 1); /* Subtract Top-Right Corner */ region.subtract(s.x - 5, 0, 5, 1); region.subtract(s.x - 3, 1, 3, 1); region.subtract(s.x - 2, 2, 2, 1); region.subtract(s.x - 1, 3, 1, 1); region.subtract(s.x - 1, 4, 1, 1); /* Subtract Bottom-Left Corner */ region.subtract(0, s.y, 5, 1); region.subtract(0, s.y - 1, 3, 1); region.subtract(0, s.y - 2, 2, 1); region.subtract(0, s.y - 3, 1, 1); region.subtract(0, s.y - 4, 1, 1); /* Subtract Bottom-Right Corner */ region.subtract(s.x - 5, s.y - 0, 5, 1); region.subtract(s.x - 3, s.y - 1, 3, 1); region.subtract(s.x - 2, s.y - 2, 2, 1); region.subtract(s.x - 1, s.y - 3, 1, 1); region.subtract(s.x - 1, s.y - 4, 1, 1); /* Dispose old first */ if (shell.getRegion() != null) { shell.getRegion().dispose(); } /* Apply Region */ shell.setRegion(region); /* Remember to dispose later */ lastUsedRegion = region; } private boolean isMouseOver(Shell shell) { if (display.isDisposed()) { return false; } return shell.getBounds().contains(display.getCursorLocation()); } @Override public int open() { if (shell == null || shell.isDisposed()) { shell = null; create(); } constrainShellSize(); shell.setLocation(fixupDisplayBounds(shell.getSize(), shell.getLocation())); if (isFadingEnabled()) { shell.setAlpha(0); } shell.setVisible(true); fadeJob = AnimationUtil.fadeIn(shell, new IFadeListener() { @Override public void faded(Shell shell, int alpha) { if (shell.isDisposed()) { return; } if (alpha == 255) { scheduleAutoClose(); } } }); return Window.OK; } protected void scheduleAutoClose() { if (delayClose > 0) { closeJob.schedule(delayClose); } } @Override protected Control createContents(Composite parent) { ((GridLayout) parent.getLayout()).marginWidth = 1; ((GridLayout) parent.getLayout()).marginHeight = 1; /* Outer Composite holding the controls */ final Composite outerCircle = new Composite(parent, SWT.NO_FOCUS); outerCircle.setLayoutData(new GridData(SWT.FILL, SWT.FILL, true, true)); outerCircle.setBackgroundMode(SWT.INHERIT_FORCE); outerCircle.addControlListener(new ControlAdapter() { @Override public void controlResized(ControlEvent e) { Rectangle clArea = outerCircle.getClientArea(); lastUsedBgImage = new Image(outerCircle.getDisplay(), clArea.width, clArea.height); GC gc = new GC(lastUsedBgImage); /* Gradient */ drawGradient(gc, clArea); /* Fix Region Shape */ fixRegion(gc, clArea); gc.dispose(); Image oldBGImage = outerCircle.getBackgroundImage(); outerCircle.setBackgroundImage(lastUsedBgImage); if (oldBGImage != null) { oldBGImage.dispose(); } } private void drawGradient(GC gc, Rectangle clArea) { gc.setForeground(color.getGradientBegin()); gc.setBackground(color.getGradientEnd()); gc.fillGradientRectangle(clArea.x, clArea.y, clArea.width, clArea.height, true); } private void fixRegion(GC gc, Rectangle clArea) { gc.setForeground(color.getBorder()); /* Fill Top Left */ gc.drawPoint(2, 0); gc.drawPoint(3, 0); gc.drawPoint(1, 1); gc.drawPoint(0, 2); gc.drawPoint(0, 3); /* Fill Top Right */ gc.drawPoint(clArea.width - 4, 0); gc.drawPoint(clArea.width - 3, 0); gc.drawPoint(clArea.width - 2, 1); gc.drawPoint(clArea.width - 1, 2); gc.drawPoint(clArea.width - 1, 3); /* Fill Bottom Left */ gc.drawPoint(2, clArea.height - 0); gc.drawPoint(3, clArea.height - 0); gc.drawPoint(1, clArea.height - 1); gc.drawPoint(0, clArea.height - 2); gc.drawPoint(0, clArea.height - 3); /* Fill Bottom Right */ gc.drawPoint(clArea.width - 4, clArea.height - 0); gc.drawPoint(clArea.width - 3, clArea.height - 0); gc.drawPoint(clArea.width - 2, clArea.height - 1); gc.drawPoint(clArea.width - 1, clArea.height - 2); gc.drawPoint(clArea.width - 1, clArea.height - 3); } }); GridLayout layout = new GridLayout(1, false); layout.marginWidth = 0; layout.marginHeight = 0; layout.verticalSpacing = 0; outerCircle.setLayout(layout); /* Title area containing label and close button */ final Composite titleCircle = new Composite(outerCircle, SWT.NO_FOCUS); titleCircle.setLayoutData(new GridData(SWT.FILL, SWT.CENTER, true, false)); titleCircle.setBackgroundMode(SWT.INHERIT_FORCE); layout = new GridLayout(4, false); layout.marginWidth = 3; layout.marginHeight = 0; layout.verticalSpacing = 5; layout.horizontalSpacing = 3; titleCircle.setLayout(layout); /* Create Title Area */ createTitleArea(titleCircle); /* Outer composite to hold content controlls */ Composite outerContentCircle = new Composite(outerCircle, SWT.NONE); outerContentCircle.setBackgroundMode(SWT.INHERIT_FORCE); layout = new GridLayout(1, false); layout.marginWidth = 0; layout.marginHeight = 0; outerContentCircle.setLayout(layout); outerContentCircle.setLayoutData(new GridData(SWT.FILL, SWT.FILL, true, true)); outerContentCircle.setBackground(outerCircle.getBackground()); /* Middle composite to show a 1px black line around the content controls */ Composite middleContentCircle = new Composite(outerContentCircle, SWT.NO_FOCUS); middleContentCircle.setBackgroundMode(SWT.INHERIT_FORCE); layout = new GridLayout(1, false); layout.marginWidth = 0; layout.marginHeight = 0; layout.marginTop = 1; middleContentCircle.setLayout(layout); middleContentCircle.setLayoutData(new GridData(SWT.FILL, SWT.FILL, true, true)); middleContentCircle.setBackground(color.getBorder()); /* Inner composite containing the content controls */ Composite innerContent = new Composite(middleContentCircle, SWT.NO_FOCUS); innerContent.setLayoutData(new GridData(SWT.FILL, SWT.FILL, true, true)); innerContent.setBackgroundMode(SWT.INHERIT_FORCE); layout = new GridLayout(1, false); layout.marginWidth = 0; layout.marginHeight = 5; layout.marginLeft = 5; layout.marginRight = 5; innerContent.setLayout(layout); innerContent.setBackground(shell.getDisplay().getSystemColor(SWT.COLOR_WHITE)); /* Content Area */ createContentArea(innerContent); setNullBackground(outerCircle); return outerCircle; } private static void setNullBackground(final Composite outerCircle) { for (Control c : outerCircle.getChildren()) { c.setBackground(null); if (c instanceof Composite) { setNullBackground((Composite) c); } } } @Override protected void initializeBounds() { Rectangle clArea = getPrimaryClientArea(); Point initialSize = shell.computeSize(SWT.DEFAULT, SWT.DEFAULT); int height = Math.max(initialSize.y, MIN_HEIGHT); int width = Math.min(initialSize.x, MAX_WIDTH); Point size = new Point(width, height); shell.setLocation(clArea.width + clArea.x - size.x - PADDING_EDGE, clArea.height + clArea.y - size.y - PADDING_EDGE); shell.setSize(size); } private Rectangle getPrimaryClientArea() { Monitor primaryMonitor = shell.getDisplay().getPrimaryMonitor(); return (primaryMonitor != null) ? primaryMonitor.getClientArea() : shell.getDisplay().getClientArea(); } public void closeFade() { if (fadeJob != null) { fadeJob.cancelAndWait(false); } fadeJob = AnimationUtil.fadeOut(getShell(), new IFadeListener() { @Override public void faded(Shell shell, int alpha) { if (!shell.isDisposed()) { if (alpha == 0) { shell.close(); } else if (isMouseOver(shell)) { if (fadeJob != null) { fadeJob.cancelAndWait(false); } fadeJob = AnimationUtil.fastFadeIn(shell, new IFadeListener() { @Override public void faded(Shell shell, int alpha) { if (shell.isDisposed()) { return; } if (alpha == 255) { scheduleAutoClose(); } } }); } } } }); } @Override public boolean close() { resources.dispose(); if (lastUsedRegion != null) { lastUsedRegion.dispose(); } if (lastUsedBgImage != null && !lastUsedBgImage.isDisposed()) { lastUsedBgImage.dispose(); } return super.close(); } public long getDelayClose() { return delayClose; } public void setDelayClose(long delayClose) { this.delayClose = delayClose; } private Point fixupDisplayBounds(Point tipSize, Point location) { if (respectDisplayBounds) { Rectangle bounds; Point rightBounds = new Point(tipSize.x + location.x, tipSize.y + location.y); if (respectMonitorBounds) { bounds = shell.getDisplay().getPrimaryMonitor().getBounds(); } else { bounds = getPrimaryClientArea(); } if (!(bounds.contains(location) && bounds.contains(rightBounds))) { if (rightBounds.x > bounds.x + bounds.width) { location.x -= rightBounds.x - (bounds.x + bounds.width); } if (rightBounds.y > bounds.y + bounds.height) { location.y -= rightBounds.y - (bounds.y + bounds.height); } if (location.x < bounds.x) { location.x = bounds.x; } if (location.y < bounds.y) { location.y = bounds.y; } } } return location; } }