mirror of
git://git.sv.gnu.org/emacs.git
synced 2025-12-30 00:51:50 -08:00
* java/org/gnu/emacs/EmacsActivity.java (onCreate): * java/org/gnu/emacs/EmacsContextMenu.java: * java/org/gnu/emacs/EmacsDocumentsProvider.java (getMimeType): * java/org/gnu/emacs/EmacsDrawLine.java (perform): * java/org/gnu/emacs/EmacsDrawRectangle.java (perform): * java/org/gnu/emacs/EmacsFillPolygon.java: * java/org/gnu/emacs/EmacsFontDriver.java: * java/org/gnu/emacs/EmacsHandleObject.java: * java/org/gnu/emacs/EmacsInputConnection.java: * java/org/gnu/emacs/EmacsMultitaskActivity.java (EmacsMultitaskActivity): * java/org/gnu/emacs/EmacsNative.java: * java/org/gnu/emacs/EmacsNoninteractive.java (EmacsNoninteractive, main): * java/org/gnu/emacs/EmacsOpenActivity.java (EmacsOpenActivity) (startEmacsClient): * java/org/gnu/emacs/EmacsSdk7FontDriver.java: * java/org/gnu/emacs/EmacsSdk8Clipboard.java: * java/org/gnu/emacs/EmacsService.java (EmacsService, onCreate): * java/org/gnu/emacs/EmacsView.java (EmacsView, onLayout): * java/org/gnu/emacs/EmacsWindow.java (EmacsWindow): * java/org/gnu/emacs/EmacsWindowAttachmentManager.java (EmacsWindowAttachmentManager): Remove redundant includes. Reorganize some functions around, remove duplicate `getLibDir' functions, and remove unused local variables.
339 lines
8.5 KiB
Java
339 lines
8.5 KiB
Java
/* Communication module for Android terminals. -*- c-file-style: "GNU" -*-
|
|
|
|
Copyright (C) 2023 Free Software Foundation, Inc.
|
|
|
|
This file is part of GNU Emacs.
|
|
|
|
GNU Emacs is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or (at
|
|
your option) any later version.
|
|
|
|
GNU Emacs 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 General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with GNU Emacs. If not, see <https://www.gnu.org/licenses/>. */
|
|
|
|
package org.gnu.emacs;
|
|
|
|
import java.util.List;
|
|
import java.util.ArrayList;
|
|
|
|
import android.content.Context;
|
|
import android.content.Intent;
|
|
|
|
import android.os.Build;
|
|
|
|
import android.view.Menu;
|
|
import android.view.MenuItem;
|
|
import android.view.View;
|
|
import android.view.SubMenu;
|
|
|
|
import android.util.Log;
|
|
|
|
/* Context menu implementation. This object is built from JNI and
|
|
describes a menu hiearchy. Then, `inflate' can turn it into an
|
|
Android menu, which can be turned into a popup (or other kind of)
|
|
menu. */
|
|
|
|
public final class EmacsContextMenu
|
|
{
|
|
private static final String TAG = "EmacsContextMenu";
|
|
|
|
/* Whether or not an item was selected. */
|
|
public static boolean itemAlreadySelected;
|
|
|
|
/* Whether or not a submenu was selected. */
|
|
public static boolean wasSubmenuSelected;
|
|
|
|
private static class Item implements MenuItem.OnMenuItemClickListener
|
|
{
|
|
public int itemID;
|
|
public String itemName, tooltip;
|
|
public EmacsContextMenu subMenu;
|
|
public boolean isEnabled, isCheckable, isChecked;
|
|
public EmacsView inflatedView;
|
|
|
|
@Override
|
|
public boolean
|
|
onMenuItemClick (MenuItem item)
|
|
{
|
|
Log.d (TAG, "onMenuItemClick: " + itemName + " (" + itemID + ")");
|
|
|
|
if (subMenu != null)
|
|
{
|
|
/* Android 6.0 and earlier don't support nested submenus
|
|
properly, so display the submenu popup by hand. */
|
|
|
|
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.N)
|
|
{
|
|
Log.d (TAG, "onMenuItemClick: displaying submenu " + subMenu);
|
|
|
|
/* Still set wasSubmenuSelected -- if not set, the
|
|
dismissal of this context menu will result in a
|
|
context menu event being sent. */
|
|
wasSubmenuSelected = true;
|
|
|
|
/* Running a popup menu from inside a click handler
|
|
doesn't work, so make sure it is displayed
|
|
outside. */
|
|
|
|
inflatedView.post (new Runnable () {
|
|
@Override
|
|
public void
|
|
run ()
|
|
{
|
|
inflatedView.popupMenu (subMenu, 0, 0, true);
|
|
}
|
|
});
|
|
|
|
return true;
|
|
}
|
|
|
|
/* After opening a submenu within a submenu, Android will
|
|
send onContextMenuClosed for a ContextMenuBuilder. This
|
|
will normally confuse Emacs into thinking that the
|
|
context menu has been dismissed. Wrong!
|
|
|
|
Setting this flag makes EmacsActivity to only handle
|
|
SubMenuBuilder being closed, which always means the menu
|
|
has actually been dismissed. */
|
|
wasSubmenuSelected = true;
|
|
return false;
|
|
}
|
|
|
|
/* Send a context menu event. */
|
|
EmacsNative.sendContextMenu ((short) 0, itemID);
|
|
|
|
/* Say that an item has already been selected. */
|
|
itemAlreadySelected = true;
|
|
return true;
|
|
}
|
|
};
|
|
|
|
public List<Item> menuItems;
|
|
public String title;
|
|
private EmacsContextMenu parent;
|
|
|
|
/* Create a context menu with no items inside and the title TITLE,
|
|
which may be NULL. */
|
|
|
|
public static EmacsContextMenu
|
|
createContextMenu (String title)
|
|
{
|
|
EmacsContextMenu menu;
|
|
|
|
menu = new EmacsContextMenu ();
|
|
menu.menuItems = new ArrayList<Item> ();
|
|
menu.title = title;
|
|
|
|
return menu;
|
|
}
|
|
|
|
/* Add a normal menu item to the context menu with the id ITEMID and
|
|
the name ITEMNAME. Enable it if ISENABLED, else keep it
|
|
disabled.
|
|
|
|
If this is not a submenu and ISCHECKABLE is set, make the item
|
|
checkable. Likewise, if ISCHECKED is set, make the item
|
|
checked. */
|
|
|
|
public void
|
|
addItem (int itemID, String itemName, boolean isEnabled,
|
|
boolean isCheckable, boolean isChecked)
|
|
{
|
|
Item item;
|
|
|
|
item = new Item ();
|
|
item.itemID = itemID;
|
|
item.itemName = itemName;
|
|
item.isEnabled = isEnabled;
|
|
item.isCheckable = isCheckable;
|
|
item.isChecked = isChecked;
|
|
|
|
menuItems.add (item);
|
|
}
|
|
|
|
/* Create a disabled menu item with the name ITEMNAME. */
|
|
|
|
public void
|
|
addPane (String itemName)
|
|
{
|
|
Item item;
|
|
|
|
item = new Item ();
|
|
item.itemName = itemName;
|
|
|
|
menuItems.add (item);
|
|
}
|
|
|
|
/* Add a submenu to the context menu with the specified title and
|
|
item name. */
|
|
|
|
public EmacsContextMenu
|
|
addSubmenu (String itemName, String title, String tooltip)
|
|
{
|
|
EmacsContextMenu submenu;
|
|
Item item;
|
|
|
|
item = new Item ();
|
|
item.itemID = 0;
|
|
item.itemName = itemName;
|
|
item.tooltip = tooltip;
|
|
item.subMenu = createContextMenu (title);
|
|
item.subMenu.parent = this;
|
|
|
|
menuItems.add (item);
|
|
return item.subMenu;
|
|
}
|
|
|
|
/* Add the contents of this menu to MENU. Assume MENU will be
|
|
displayed in INFLATEDVIEW. */
|
|
|
|
private void
|
|
inflateMenuItems (Menu menu, EmacsView inflatedView)
|
|
{
|
|
Intent intent;
|
|
MenuItem menuItem;
|
|
SubMenu submenu;
|
|
|
|
for (Item item : menuItems)
|
|
{
|
|
if (item.subMenu != null)
|
|
{
|
|
/* This is a submenu. On versions of Android which
|
|
support doing so, create the submenu and add the
|
|
contents of the menu to it.
|
|
|
|
Note that Android 4.0 and later technically supports
|
|
having multiple layers of nested submenus, but if they
|
|
are used, onContextMenuClosed becomes unreliable. */
|
|
|
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N)
|
|
{
|
|
submenu = menu.addSubMenu (item.itemName);
|
|
item.subMenu.inflateMenuItems (submenu, inflatedView);
|
|
|
|
/* This is still needed to set wasSubmenuSelected. */
|
|
menuItem = submenu.getItem ();
|
|
}
|
|
else
|
|
menuItem = menu.add (item.itemName);
|
|
|
|
item.inflatedView = inflatedView;
|
|
menuItem.setOnMenuItemClickListener (item);
|
|
}
|
|
else
|
|
{
|
|
menuItem = menu.add (item.itemName);
|
|
menuItem.setOnMenuItemClickListener (item);
|
|
|
|
/* If the item ID is zero, then disable the item. */
|
|
if (item.itemID == 0 || !item.isEnabled)
|
|
menuItem.setEnabled (false);
|
|
|
|
/* Now make the menu item display a checkmark as
|
|
appropriate. */
|
|
|
|
if (item.isCheckable)
|
|
menuItem.setCheckable (true);
|
|
|
|
if (item.isChecked)
|
|
menuItem.setChecked (true);
|
|
|
|
/* If the tooltip text is set and the system is new enough
|
|
to support menu item tooltips, set it on the item. */
|
|
|
|
if (item.tooltip != null
|
|
&& Build.VERSION.SDK_INT >= Build.VERSION_CODES.O)
|
|
menuItem.setTooltipText (item.tooltip);
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Enter the items in this context menu to MENU.
|
|
Assume that MENU will be displayed in VIEW; this may lead to
|
|
popupMenu being called on VIEW if a submenu is selected. */
|
|
|
|
public void
|
|
expandTo (Menu menu, EmacsView view)
|
|
{
|
|
inflateMenuItems (menu, view);
|
|
}
|
|
|
|
/* Return the parent or NULL. */
|
|
|
|
public EmacsContextMenu
|
|
parent ()
|
|
{
|
|
return this.parent;
|
|
}
|
|
|
|
/* Like display, but does the actual work and runs in the main
|
|
thread. */
|
|
|
|
private boolean
|
|
display1 (EmacsWindow window, int xPosition, int yPosition)
|
|
{
|
|
/* Set this flag to false. It is used to decide whether or not to
|
|
send 0 in response to the context menu being closed. */
|
|
itemAlreadySelected = false;
|
|
|
|
/* No submenu has been selected yet. */
|
|
wasSubmenuSelected = false;
|
|
|
|
return window.view.popupMenu (this, xPosition, yPosition,
|
|
false);
|
|
}
|
|
|
|
/* Display this context menu on WINDOW, at xPosition and
|
|
yPosition. */
|
|
|
|
public boolean
|
|
display (final EmacsWindow window, final int xPosition,
|
|
final int yPosition)
|
|
{
|
|
Runnable runnable;
|
|
final Holder<Boolean> rc;
|
|
|
|
rc = new Holder<Boolean> ();
|
|
|
|
runnable = new Runnable () {
|
|
@Override
|
|
public void
|
|
run ()
|
|
{
|
|
synchronized (this)
|
|
{
|
|
rc.thing = display1 (window, xPosition, yPosition);
|
|
notify ();
|
|
}
|
|
}
|
|
};
|
|
|
|
EmacsService.syncRunnable (runnable);
|
|
return rc.thing;
|
|
}
|
|
|
|
/* Dismiss this context menu. WINDOW is the window where the
|
|
context menu is being displayed. */
|
|
|
|
public void
|
|
dismiss (final EmacsWindow window)
|
|
{
|
|
Runnable runnable;
|
|
|
|
EmacsService.SERVICE.runOnUiThread (new Runnable () {
|
|
@Override
|
|
public void
|
|
run ()
|
|
{
|
|
window.view.cancelPopupMenu ();
|
|
itemAlreadySelected = false;
|
|
}
|
|
});
|
|
}
|
|
};
|