blob: ec32bd1a6f2cfb903e64b6dcfcc5879052fb124e [file] [log] [blame]
/*
* Copyright 2000-2014 JetBrains s.r.o.
*
* 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.
*/
package com.intellij.openapi.wm.impl;
import com.intellij.ide.DataManager;
import com.intellij.ide.IdeEventQueue;
import com.intellij.ide.impl.DataManagerImpl;
import com.intellij.ide.ui.UISettings;
import com.intellij.ide.ui.UISettingsListener;
import com.intellij.ide.ui.customization.CustomActionsSchema;
import com.intellij.openapi.Disposable;
import com.intellij.openapi.actionSystem.*;
import com.intellij.openapi.actionSystem.ex.ActionManagerEx;
import com.intellij.openapi.actionSystem.impl.ActionMenu;
import com.intellij.openapi.actionSystem.impl.MenuItemPresentationFactory;
import com.intellij.openapi.actionSystem.impl.WeakTimerListener;
import com.intellij.openapi.application.ApplicationManager;
import com.intellij.openapi.application.ModalityState;
import com.intellij.openapi.util.Disposer;
import com.intellij.openapi.util.SystemInfo;
import com.intellij.openapi.util.registry.Registry;
import com.intellij.openapi.wm.ex.IdeFrameEx;
import com.intellij.openapi.wm.impl.status.ClockPanel;
import com.intellij.ui.ColorUtil;
import com.intellij.ui.Gray;
import com.intellij.ui.ScreenUtil;
import com.intellij.ui.border.CustomLineBorder;
import com.intellij.util.ui.Animator;
import com.intellij.util.ui.UIUtil;
import org.java.ayatana.ApplicationMenu;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
import javax.swing.*;
import javax.swing.border.Border;
import javax.swing.border.EmptyBorder;
import java.awt.*;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import java.awt.geom.AffineTransform;
import java.awt.geom.GeneralPath;
import java.awt.geom.RoundRectangle2D;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.util.ArrayList;
import java.util.List;
/**
* @author Anton Katilin
* @author Vladimir Kondratyev
*/
public class IdeMenuBar extends JMenuBar implements IdeEventQueue.EventDispatcher {
private static final int COLLAPSED_HEIGHT = 2;
private enum State {
EXPANDED, COLLAPSING, COLLAPSED, EXPANDING, TEMPORARY_EXPANDED;
boolean isInProgress() {
return this == COLLAPSING || this == EXPANDING;
}
}
private final MyTimerListener myTimerListener;
private List<AnAction> myVisibleActions;
private List<AnAction> myNewVisibleActions;
private final MenuItemPresentationFactory myPresentationFactory;
private final DataManager myDataManager;
private final ActionManagerEx myActionManager;
private final Disposable myDisposable = Disposer.newDisposable();
private boolean myDisabled = false;
@Nullable private final ClockPanel myClockPanel;
@Nullable private final MyExitFullScreenButton myButton;
@Nullable private final Animator myAnimator;
@Nullable private final Timer myActivationWatcher;
@NotNull private State myState = State.EXPANDED;
private double myProgress = 0;
private boolean myActivated = false;
public IdeMenuBar(ActionManagerEx actionManager, DataManager dataManager) {
myActionManager = actionManager;
myTimerListener = new MyTimerListener();
myVisibleActions = new ArrayList<AnAction>();
myNewVisibleActions = new ArrayList<AnAction>();
myPresentationFactory = new MenuItemPresentationFactory();
myDataManager = dataManager;
if (WindowManagerImpl.isFloatingMenuBarSupported()) {
myAnimator = new MyAnimator();
myActivationWatcher = new Timer(100, new MyActionListener());
myClockPanel = new ClockPanel();
myButton = new MyExitFullScreenButton();
add(myClockPanel);
add(myButton);
addPropertyChangeListener(WindowManagerImpl.FULL_SCREEN, new PropertyChangeListener() {
@Override public void propertyChange(PropertyChangeEvent evt) {
updateState();
}
});
addMouseListener(new MyMouseListener());
}
else {
myAnimator = null;
myActivationWatcher = null;
myClockPanel = null;
myButton = null;
}
}
@Override
public Border getBorder() {
//avoid moving lines
if (myState == State.EXPANDING || myState == State.COLLAPSING) {
return new EmptyBorder(0,0,0,0);
}
//fix for Darcula double border
if (myState == State.TEMPORARY_EXPANDED && UIUtil.isUnderDarcula()) {
return new CustomLineBorder(Gray._75, 0, 0, 1, 0);
}
//save 1px for mouse handler
if (myState == State.COLLAPSED) {
return new EmptyBorder(0, 0, 1, 0);
}
return UISettings.getInstance().SHOW_MAIN_TOOLBAR || UISettings.getInstance().SHOW_NAVIGATION_BAR ? super.getBorder() : null;
}
@Override
public void paint(Graphics g) {
//otherwise, there will be 1px line on top
if (myState == State.COLLAPSED) {
return;
}
super.paint(g);
}
@Override
public void doLayout() {
super.doLayout();
if (myClockPanel != null && myButton != null) {
if (myState != State.EXPANDED) {
myClockPanel.setVisible(true);
myButton.setVisible(true);
Dimension preferredSize = myButton.getPreferredSize();
myButton.setBounds(getBounds().width - preferredSize.width, 0, preferredSize.width, preferredSize.height);
preferredSize = myClockPanel.getPreferredSize();
myClockPanel.setBounds(getBounds().width - preferredSize.width - myButton.getWidth(), 0, preferredSize.width, preferredSize.height);
}
else {
myClockPanel.setVisible(false);
myButton.setVisible(false);
}
}
}
@Override
public void menuSelectionChanged(boolean isIncluded) {
if (!isIncluded && myState == State.TEMPORARY_EXPANDED) {
myActivated = false;
setState(State.COLLAPSING);
restartAnimator();
return;
}
if (isIncluded && myState == State.COLLAPSED) {
myActivated = true;
setState(State.TEMPORARY_EXPANDED);
revalidate();
repaint();
//noinspection SSBasedInspection
SwingUtilities.invokeLater(new Runnable() {
@Override
public void run() {
JMenu menu = getMenu(getSelectionModel().getSelectedIndex());
if (menu.isPopupMenuVisible()) {
menu.setPopupMenuVisible(false);
menu.setPopupMenuVisible(true);
}
}
});
}
super.menuSelectionChanged(isIncluded);
}
private boolean isActivated() {
int index = getSelectionModel().getSelectedIndex();
if (index == -1) {
return false;
}
return getMenu(index).isPopupMenuVisible();
}
private void updateState() {
if (myAnimator != null) {
Window window = SwingUtilities.getWindowAncestor(this);
if (window instanceof IdeFrameEx) {
boolean fullScreen = ((IdeFrameEx)window).isInFullScreen();
if (fullScreen) {
setState(State.COLLAPSING);
restartAnimator();
}
else {
myAnimator.suspend();
setState(State.EXPANDED);
if (myClockPanel != null) {
myClockPanel.setVisible(false);
myButton.setVisible(false);
}
}
}
}
}
private void setState(@NotNull State state) {
myState = state;
if (myState == State.EXPANDING && myActivationWatcher != null && !myActivationWatcher.isRunning()) {
myActivationWatcher.start();
} else if (myActivationWatcher != null && myActivationWatcher.isRunning()) {
if (state == State.EXPANDED || state == State.COLLAPSED) {
myActivationWatcher.stop();
}
}
}
@Override
public Dimension getPreferredSize() {
Dimension dimension = super.getPreferredSize();
if (myState.isInProgress()) {
dimension.height =
COLLAPSED_HEIGHT + (int)((myState == State.COLLAPSING ? (1 - myProgress) : myProgress) * (dimension.height - COLLAPSED_HEIGHT));
}
else if (myState == State.COLLAPSED) {
dimension.height = COLLAPSED_HEIGHT;
}
return dimension;
}
private void restartAnimator() {
if (myAnimator != null) {
myAnimator.reset();
myAnimator.resume();
}
}
@Override
public void addNotify() {
super.addNotify();
updateMenuActions();
// Add updater for menus
myActionManager.addTimerListener(1000, new WeakTimerListener(myActionManager, myTimerListener));
UISettingsListener UISettingsListener = new UISettingsListener() {
public void uiSettingsChanged(final UISettings source) {
updateMnemonicsVisibility();
myPresentationFactory.reset();
}
};
UISettings.getInstance().addUISettingsListener(UISettingsListener, myDisposable);
Disposer.register(ApplicationManager.getApplication(), myDisposable);
IdeEventQueue.getInstance().addDispatcher(this, myDisposable);
}
@Override
public void removeNotify() {
if (ScreenUtil.isStandardAddRemoveNotify(this)) {
if (myAnimator != null) {
myAnimator.suspend();
}
Disposer.dispose(myDisposable);
}
super.removeNotify();
}
@Override
public boolean dispatch(AWTEvent e) {
if (e instanceof MouseEvent) {
MouseEvent mouseEvent = (MouseEvent)e;
Component component = findActualComponent(mouseEvent);
if (myState != State.EXPANDED /*&& !myState.isInProgress()*/) {
boolean mouseInside = myActivated || UIUtil.isDescendingFrom(component, this);
if (e.getID() == MouseEvent.MOUSE_EXITED && e.getSource() == SwingUtilities.windowForComponent(this) && !myActivated) mouseInside = false;
if (mouseInside && myState == State.COLLAPSED) {
setState(State.EXPANDING);
restartAnimator();
}
else if (!mouseInside && myState != State.COLLAPSING && myState != State.COLLAPSED) {
setState(State.COLLAPSING);
restartAnimator();
}
}
}
return false;
}
private Component findActualComponent(MouseEvent mouseEvent) {
Component component = mouseEvent.getComponent();
Component deepestComponent;
if (myState != State.EXPANDED &&
!myState.isInProgress() &&
contains(SwingUtilities.convertPoint(component, mouseEvent.getPoint(), this))) {
deepestComponent = this;
}
else {
deepestComponent = SwingUtilities.getDeepestComponentAt(mouseEvent.getComponent(), mouseEvent.getX(), mouseEvent.getY());
}
if (deepestComponent != null) {
component = deepestComponent;
}
return component;
}
void updateMenuActions() {
myNewVisibleActions.clear();
if (!myDisabled) {
DataContext dataContext = ((DataManagerImpl)myDataManager).getDataContextTest(this);
expandActionGroup(dataContext, myNewVisibleActions, myActionManager);
}
if (!myNewVisibleActions.equals(myVisibleActions)) {
// should rebuild UI
final boolean changeBarVisibility = myNewVisibleActions.isEmpty() || myVisibleActions.isEmpty();
final List<AnAction> temp = myVisibleActions;
myVisibleActions = myNewVisibleActions;
myNewVisibleActions = temp;
removeAll();
final boolean enableMnemonics = !UISettings.getInstance().DISABLE_MNEMONICS;
for (final AnAction action : myVisibleActions) {
add(new ActionMenu(null, ActionPlaces.MAIN_MENU, (ActionGroup)action, myPresentationFactory, enableMnemonics, true));
}
fixMenuBackground();
updateMnemonicsVisibility();
if (myClockPanel != null) {
add(myClockPanel);
add(myButton);
}
validate();
if (changeBarVisibility) {
invalidate();
final JFrame frame = (JFrame)SwingUtilities.getAncestorOfClass(JFrame.class, this);
if (frame != null) {
frame.validate();
}
}
}
}
@Override
public void updateUI() {
super.updateUI();
fixMenuBackground();
}
@Override
protected void paintComponent(Graphics g) {
super.paintComponent(g);
if (UIUtil.isUnderDarcula() || UIUtil.isUnderIntelliJLaF()) {
g.setColor(UIManager.getColor("MenuItem.background"));
g.fillRect(0, 0, getWidth(), getHeight());
}
}
@Override
protected void paintChildren(Graphics g) {
if (myState.isInProgress()) {
Graphics2D g2 = (Graphics2D)g;
AffineTransform oldTransform = g2.getTransform();
AffineTransform newTransform = oldTransform != null ? new AffineTransform(oldTransform) : new AffineTransform();
newTransform.concatenate(AffineTransform.getTranslateInstance(0, getHeight() - super.getPreferredSize().height));
g2.setTransform(newTransform);
super.paintChildren(g2);
g2.setTransform(oldTransform);
}
else if (myState != State.COLLAPSED) {
super.paintChildren(g);
}
}
/**
* Hacks a problem under Alloy LaF which draws menu bar in different background menu items are drawn in.
*/
private void fixMenuBackground() {
if (UIUtil.isUnderAlloyLookAndFeel() && getMenuCount() > 0) {
final JMenu menu = getMenu(0);
if (menu != null) { // hack for Substance LAF compatibility
menu.updateUI();
setBackground(menu.getBackground());
}
}
}
private void expandActionGroup(final DataContext context, final List<AnAction> newVisibleActions, ActionManager actionManager) {
final ActionGroup mainActionGroup = (ActionGroup)CustomActionsSchema.getInstance().getCorrectedAction(IdeActions.GROUP_MAIN_MENU);
if (mainActionGroup == null) return;
final AnAction[] children = mainActionGroup.getChildren(null);
for (final AnAction action : children) {
if (!(action instanceof ActionGroup)) {
continue;
}
final Presentation presentation = myPresentationFactory.getPresentation(action);
final AnActionEvent e = new AnActionEvent(null, context, ActionPlaces.MAIN_MENU, presentation, actionManager, 0);
e.setInjectedContext(action.isInInjectedContext());
action.update(e);
if (presentation.isVisible()) { // add only visible items
newVisibleActions.add(action);
}
}
}
@Override
public int getMenuCount() {
int menuCount = super.getMenuCount();
return myClockPanel != null ? menuCount - 2: menuCount;
}
private void updateMnemonicsVisibility() {
final boolean enabled = !UISettings.getInstance().DISABLE_MNEMONICS;
for (int i = 0; i < getMenuCount(); i++) {
((ActionMenu)getMenu(i)).setMnemonicEnabled(enabled);
}
}
public void disableUpdates() {
myDisabled = true;
updateMenuActions();
}
public void enableUpdates() {
myDisabled = false;
updateMenuActions();
}
private final class MyTimerListener implements TimerListener {
@Override
public ModalityState getModalityState() {
return ModalityState.stateForComponent(IdeMenuBar.this);
}
@Override
public void run() {
if (!isShowing()) {
return;
}
final Window myWindow = SwingUtilities.windowForComponent(IdeMenuBar.this);
if (myWindow != null && !myWindow.isActive()) return;
// do not update when a popup menu is shown (if popup menu contains action which is also in the menu bar, it should not be enabled/disabled)
final MenuSelectionManager menuSelectionManager = MenuSelectionManager.defaultManager();
final MenuElement[] selectedPath = menuSelectionManager.getSelectedPath();
if (selectedPath.length > 0) {
return;
}
// don't update toolbar if there is currently active modal dialog
final Window window = KeyboardFocusManager.getCurrentKeyboardFocusManager().getFocusedWindow();
if (window instanceof Dialog) {
if (((Dialog)window).isModal()) {
return;
}
}
updateMenuActions();
if (UIUtil.isWinLafOnVista()) {
repaint();
}
}
}
private class MyAnimator extends Animator {
public MyAnimator() {
super("MenuBarAnimator", 16, 300, false);
}
@Override
public void paintNow(int frame, int totalFrames, int cycle) {
myProgress = (1 - Math.cos(Math.PI * ((float)frame / totalFrames))) / 2;
revalidate();
repaint();
}
@Override
protected void paintCycleEnd() {
myProgress = 1;
switch (myState) {
case COLLAPSING:
setState(State.COLLAPSED);
break;
case EXPANDING:
setState(State.TEMPORARY_EXPANDED);
break;
default:
}
revalidate();
if (myState == State.COLLAPSED) {
//we should repaint parent, to clear 1px on top when menu is collapsed
getParent().repaint();
} else {
repaint();
}
}
}
private class MyActionListener implements ActionListener {
@Override
public void actionPerformed(ActionEvent e) {
if (myState == State.EXPANDED || myState == State.EXPANDING) {
return;
}
boolean activated = isActivated();
if (myActivated && !activated && myState == State.TEMPORARY_EXPANDED) {
myActivated = false;
setState(State.COLLAPSING);
restartAnimator();
}
if (activated) {
myActivated = true;
}
}
}
private static class MyMouseListener extends MouseAdapter {
@Override
public void mousePressed(MouseEvent e) {
Component c = e.getComponent();
if (c instanceof IdeMenuBar) {
Dimension size = c.getSize();
Insets insets = ((IdeMenuBar)c).getInsets();
Point p = e.getPoint();
if (p.y < insets.top || p.y >= size.height - insets.bottom) {
Component item = ((IdeMenuBar)c).findComponentAt(p.x, size.height / 2);
if (item instanceof JMenuItem) {
// re-target border clicks as a menu item ones
item.dispatchEvent(
new MouseEvent(item, e.getID(), e.getWhen(), e.getModifiers(), 1, 1, e.getClickCount(), e.isPopupTrigger(), e.getButton()));
e.consume();
return;
}
}
}
super.mouseClicked(e);
}
}
public static void installAppMenuIfNeeded(@NotNull final JFrame frame) {
if (SystemInfo.isLinux && Registry.is("linux.native.menu") && "Unity".equals(System.getenv("XDG_CURRENT_DESKTOP"))) {
//noinspection SSBasedInspection
SwingUtilities.invokeLater(new Runnable() {
public void run() {
ApplicationMenu.tryInstall(frame);
}
});
}
}
private static class MyExitFullScreenButton extends JButton {
private MyExitFullScreenButton() {
setFocusable(false);
addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
Window window = SwingUtilities.getWindowAncestor(MyExitFullScreenButton.this);
if (window instanceof IdeFrameEx) {
((IdeFrameEx)window).toggleFullScreen(false);
}
}
});
addMouseListener(new MouseAdapter() {
@Override
public void mouseEntered(MouseEvent e) {
model.setRollover(true);
}
@Override
public void mouseExited(MouseEvent e) {
model.setRollover(false);
}
});
}
@Override
public Dimension getPreferredSize() {
int height;
Container parent = getParent();
if (isVisible() && parent != null) {
height = parent.getSize().height - parent.getInsets().top - parent.getInsets().bottom;
}
else {
height = super.getPreferredSize().height;
}
return new Dimension(height, height);
}
@Override
public Dimension getMaximumSize() {
return getPreferredSize();
}
@Override
public void paint(Graphics g) {
Graphics2D g2d = (Graphics2D)g.create();
try {
g2d.setColor(UIManager.getColor("Label.background"));
g2d.fillRect(0, 0, getWidth()+1, getHeight()+1);
g2d.setRenderingHint(RenderingHints.KEY_FRACTIONALMETRICS, RenderingHints.VALUE_FRACTIONALMETRICS_ON);
g2d.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
g2d.setRenderingHint(RenderingHints.KEY_ALPHA_INTERPOLATION, RenderingHints.VALUE_ALPHA_INTERPOLATION_QUALITY);
double s = (double)getHeight() / 13;
g2d.translate(s, s);
Shape plate = new RoundRectangle2D.Double(0, 0, s * 11, s * 11, s, s);
Color color = UIManager.getColor("Label.foreground");
boolean hover = model.isRollover() || model.isPressed();
g2d.setColor(ColorUtil.withAlpha(color, hover? .25: .18));
g2d.fill(plate);
g2d.setColor(ColorUtil.withAlpha(color, hover? .4 : .33));
g2d.draw(plate);
g2d.setColor(ColorUtil.withAlpha(color, hover ? .7 : .66));
GeneralPath path = new GeneralPath();
path.moveTo(s * 2, s * 6);
path.lineTo(s * 5, s * 6);
path.lineTo(s * 5, s * 9);
path.lineTo(s * 4, s * 8);
path.lineTo(s * 2, s * 10);
path.quadTo(s * 2 - s/ Math.sqrt(2), s * 9 + s/Math.sqrt(2), s, s * 9);
path.lineTo(s * 3, s * 7);
path.lineTo(s * 2, s * 6);
path.closePath();
g2d.fill(path);
g2d.draw(path);
path = new GeneralPath();
path.moveTo(s * 6, s * 2);
path.lineTo(s * 6, s * 5);
path.lineTo(s * 9, s * 5);
path.lineTo(s * 8, s * 4);
path.lineTo(s * 10, s * 2);
path.quadTo(s * 9 + s/ Math.sqrt(2), s * 2 - s/Math.sqrt(2), s * 9 , s);
path.lineTo(s * 7, s * 3);
path.lineTo(s * 6, s * 2);
path.closePath();
g2d.fill(path);
g2d.draw(path);
} finally {
g2d.dispose();
}
}
}
}