2 * Jexer - Java Text User Interface
4 * License: LGPLv3 or later
6 * This module is licensed under the GNU Lesser General Public License
7 * Version 3. Please see the file "COPYING" in this directory for more
8 * information about the GNU Lesser General Public License Version 3.
10 * Copyright (C) 2015 Kevin Lamonte
12 * This program is free software; you can redistribute it and/or
13 * modify it under the terms of the GNU Lesser General Public License
14 * as published by the Free Software Foundation; either version 3 of
15 * the License, or (at your option) any later version.
17 * This program is distributed in the hope that it will be useful, but
18 * WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
20 * General Public License for more details.
22 * You should have received a copy of the GNU Lesser General Public
23 * License along with this program; if not, see
24 * http://www.gnu.org/licenses/, or write to the Free Software
25 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
28 * @author Kevin Lamonte [kevin.lamonte@gmail.com]
33 import jexer
.TApplication
;
34 import jexer
.TKeypress
;
37 import jexer
.bits
.CellAttributes
;
38 import jexer
.bits
.GraphicsChars
;
39 import jexer
.bits
.MnemonicString
;
40 import jexer
.event
.TKeypressEvent
;
41 import jexer
.event
.TMouseEvent
;
42 import static jexer
.TKeypress
.*;
45 * TMenu is a top-level collection of TMenuItems.
47 public final class TMenu
extends TWindow
{
50 * If true, this is a sub-menu. Note package private access.
52 boolean isSubMenu
= false;
55 * The shortcut and title.
57 private MnemonicString mnemonic
;
60 * Get the mnemonic string.
62 * @return the full mnemonic string
64 public MnemonicString
getMnemonic() {
68 // Reserved menu item IDs
69 public static final int MID_UNUSED
= -1;
72 public static final int MID_EXIT
= 1;
73 public static final int MID_QUIT
= MID_EXIT
;
74 public static final int MID_OPEN_FILE
= 2;
75 public static final int MID_SHELL
= 3;
78 public static final int MID_CUT
= 10;
79 public static final int MID_COPY
= 11;
80 public static final int MID_PASTE
= 12;
81 public static final int MID_CLEAR
= 13;
84 public static final int MID_TILE
= 20;
85 public static final int MID_CASCADE
= 21;
86 public static final int MID_CLOSE_ALL
= 22;
87 public static final int MID_WINDOW_MOVE
= 23;
88 public static final int MID_WINDOW_ZOOM
= 24;
89 public static final int MID_WINDOW_NEXT
= 25;
90 public static final int MID_WINDOW_PREVIOUS
= 26;
91 public static final int MID_WINDOW_CLOSE
= 27;
96 * @param parent parent application
97 * @param x column relative to parent
98 * @param y row relative to parent
99 * @param label mnemonic menu title. Label must contain a keyboard
100 * shortcut (mnemonic), denoted by prefixing a letter with "&",
103 public TMenu(final TApplication parent
, final int x
, final int y
,
104 final String label
) {
106 super(parent
, label
, x
, y
, parent
.getScreen().getWidth(),
107 parent
.getScreen().getHeight());
109 // My parent constructor added me as a window, get rid of that
110 parent
.closeWindow(this);
112 // Setup the menu shortcut
113 mnemonic
= new MnemonicString(label
);
114 setTitle(mnemonic
.getRawLabel());
115 assert (mnemonic
.getShortcutIdx() >= 0);
117 // Recompute width and height to reflect an empty menu
118 setWidth(getTitle().length() + 4);
125 * Draw a top-level menu with title and menu items.
129 CellAttributes menuColor
;
130 CellAttributes background
= getTheme().getColor("tmenu");
132 if (getAbsoluteActive()) {
133 menuColor
= getTheme().getColor("tmenu.highlighted");
135 menuColor
= getTheme().getColor("tmenu");
138 assert (getAbsoluteActive());
140 // Fill in the interior background
141 for (int i
= 0; i
< getHeight(); i
++) {
142 hLineXY(0, i
, getWidth(), ' ', background
);
152 cTopLeft
= GraphicsChars
.ULCORNER
;
153 cTopRight
= GraphicsChars
.URCORNER
;
154 cBottomLeft
= GraphicsChars
.LLCORNER
;
155 cBottomRight
= GraphicsChars
.LRCORNER
;
156 cHSide
= GraphicsChars
.SINGLE_BAR
;
158 // Place the corner characters
159 putCharXY(1, 0, cTopLeft
, background
);
160 putCharXY(getWidth() - 2, 0, cTopRight
, background
);
161 putCharXY(1, getHeight() - 1, cBottomLeft
, background
);
162 putCharXY(getWidth() - 2, getHeight() - 1, cBottomRight
, background
);
164 // Draw the box lines
165 hLineXY(1 + 1, 0, getWidth() - 4, cHSide
, background
);
166 hLineXY(1 + 1, getHeight() - 1, getWidth() - 4, cHSide
, background
);
169 getScreen().drawBoxShadow(0, 0, getWidth(), getHeight());
173 * Handle mouse button presses.
175 * @param mouse mouse button event
178 public void onMouseDown(final TMouseEvent mouse
) {
183 for (TWidget widget
: getChildren()) {
184 if (widget
.mouseWouldHit(mouse
)) {
185 // Dispatch to this child, also activate it
188 // Set x and y relative to the child's coordinates
189 mouse
.setX(mouse
.getAbsoluteX() - widget
.getAbsoluteX());
190 mouse
.setY(mouse
.getAbsoluteY() - widget
.getAbsoluteY());
191 widget
.handleEvent(mouse
);
198 * Handle mouse button releases.
200 * @param mouse mouse button release event
203 public void onMouseUp(final TMouseEvent mouse
) {
208 for (TWidget widget
: getChildren()) {
209 if (widget
.mouseWouldHit(mouse
)) {
210 // Dispatch to this child, also activate it
213 // Set x and y relative to the child's coordinates
214 mouse
.setX(mouse
.getAbsoluteX() - widget
.getAbsoluteX());
215 mouse
.setY(mouse
.getAbsoluteY() - widget
.getAbsoluteY());
216 widget
.handleEvent(mouse
);
223 * Handle mouse movements.
225 * @param mouse mouse motion event
228 public void onMouseMotion(final TMouseEvent mouse
) {
232 // See if we should activate a different menu item
233 for (TWidget widget
: getChildren()) {
234 if ((mouse
.getMouse1())
235 && (widget
.mouseWouldHit(mouse
))
237 // Activate this menu item
239 if (widget
instanceof TSubMenu
) {
240 ((TSubMenu
) widget
).dispatch();
250 * @param keypress keystroke event
253 public void onKeypress(final TKeypressEvent keypress
) {
254 if (getActiveChild() != null) {
255 if (getActiveChild() instanceof TSubMenu
) {
256 getActiveChild().onKeypress(keypress
);
261 if (keypress
.equals(kbEsc
)) {
262 getApplication().closeMenu();
265 if (keypress
.equals(kbDown
)) {
269 if (keypress
.equals(kbUp
)) {
273 if (keypress
.equals(kbRight
)) {
275 getApplication().switchMenu(true);
279 if (keypress
.equals(kbLeft
)) {
281 getApplication().closeSubMenu();
283 getApplication().switchMenu(false);
288 // Switch to a menuItem if it has an mnemonic
289 if (!keypress
.getKey().getIsKey()
290 && !keypress
.getKey().getAlt()
291 && !keypress
.getKey().getCtrl()) {
292 for (TWidget widget
: getChildren()) {
293 TMenuItem item
= (TMenuItem
) widget
;
294 if ((item
.getMnemonic() != null)
295 && (Character
.toLowerCase(item
.getMnemonic().getShortcut())
296 == Character
.toLowerCase(keypress
.getKey().getCh()))
298 // Send an enter keystroke to it
300 item
.handleEvent(new TKeypressEvent(kbEnter
));
306 // Dispatch the keypress to an active widget
307 for (TWidget widget
: getChildren()) {
308 if (widget
.getActive()) {
310 widget
.handleEvent(keypress
);
317 * Convenience function to add a custom menu item.
319 * @param id menu item ID. Must be greater than 1024.
320 * @param label menu item label
321 * @param key global keyboard accelerator
322 * @return the new menu item
324 public final TMenuItem
addItem(final int id
, final String label
,
325 final TKeypress key
) {
328 return addItemInternal(id
, label
, key
);
332 * Convenience function to add a custom menu item.
334 * @param id menu item ID. Must be greater than 1024.
335 * @param label menu item label
336 * @param key global keyboard accelerator
337 * @return the new menu item
339 private TMenuItem
addItemInternal(final int id
, final String label
,
340 final TKeypress key
) {
342 int newY
= getChildren().size() + 1;
343 assert (newY
< getHeight());
345 TMenuItem menuItem
= new TMenuItem(this, id
, 1, newY
, label
);
346 menuItem
.setKey(key
);
347 setHeight(getHeight() + 1);
348 if (menuItem
.getWidth() + 2 > getWidth()) {
349 setWidth(menuItem
.getWidth() + 2);
351 for (TWidget widget
: getChildren()) {
352 widget
.setWidth(getWidth() - 2);
354 getApplication().addAccelerator(menuItem
, toLower(key
));
355 getApplication().recomputeMenuX();
361 * Convenience function to add a menu item.
363 * @param id menu item ID. Must be greater than 1024.
364 * @param label menu item label
365 * @return the new menu item
367 public final TMenuItem
addItem(final int id
, final String label
) {
369 return addItemInternal(id
, label
);
373 * Convenience function to add a menu item.
375 * @param id menu item ID
376 * @param label menu item label
377 * @return the new menu item
379 private TMenuItem
addItemInternal(final int id
, final String label
) {
380 int newY
= getChildren().size() + 1;
381 assert (newY
< getHeight());
383 TMenuItem menuItem
= new TMenuItem(this, id
, 1, newY
, label
);
384 setHeight(getHeight() + 1);
385 if (menuItem
.getWidth() + 2 > getWidth()) {
386 setWidth(menuItem
.getWidth() + 2);
388 for (TWidget widget
: getChildren()) {
389 widget
.setWidth(getWidth() - 2);
391 getApplication().recomputeMenuX();
397 * Convenience function to add one of the default menu items.
399 * @param id menu item ID. Must be between 0 (inclusive) and 1023
401 * @return the new menu item
403 public final TMenuItem
addDefaultItem(final int id
) {
408 TKeypress key
= null;
409 boolean hasKey
= true;
454 label
= "Cl&ose All";
457 case MID_WINDOW_MOVE
:
458 label
= "&Size/Move";
461 case MID_WINDOW_ZOOM
:
465 case MID_WINDOW_NEXT
:
469 case MID_WINDOW_PREVIOUS
:
473 case MID_WINDOW_CLOSE
:
479 throw new IllegalArgumentException("Invalid menu ID: " + id
);
483 return addItemInternal(id
, label
, key
);
485 return addItemInternal(id
, label
);
489 * Convenience function to add a menu separator.
491 public final void addSeparator() {
492 int newY
= getChildren().size() + 1;
493 assert (newY
< getHeight());
495 TMenuItem menuItem
= new TMenuSeparator(this, 1, newY
);
496 setHeight(getHeight() + 1);
500 * Convenience function to add a sub-menu.
502 * @param title menu title. Title must contain a keyboard shortcut,
503 * denoted by prefixing a letter with "&", e.g. "&File"
504 * @return the new sub-menu
506 public final TSubMenu
addSubMenu(final String title
) {
507 int newY
= getChildren().size() + 1;
508 assert (newY
< getHeight());
510 TSubMenu subMenu
= new TSubMenu(this, title
, 1, newY
);
511 setHeight(getHeight() + 1);
512 if (subMenu
.getWidth() + 2 > getWidth()) {
513 setWidth(subMenu
.getWidth() + 2);
515 for (TWidget widget
: getChildren()) {
516 widget
.setWidth(getWidth() - 2);
518 getApplication().recomputeMenuX();
520 subMenu
.menu
.setX(getX() + getWidth() - 2);