1 /*
   2  * Copyright (c) 2003, 2007, Oracle and/or its affiliates. All rights reserved.
   3  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
   4  *
   5  * This code is free software; you can redistribute it and/or modify it
   6  * under the terms of the GNU General Public License version 2 only, as
   7  * published by the Free Software Foundation.  Oracle designates this
   8  * particular file as subject to the "Classpath" exception as provided
   9  * by Oracle in the LICENSE file that accompanied this code.
  10  *
  11  * This code is distributed in the hope that it will be useful, but WITHOUT
  12  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
  13  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
  14  * version 2 for more details (a copy is included in the LICENSE file that
  15  * accompanied this code).
  16  *
  17  * You should have received a copy of the GNU General Public License version
  18  * 2 along with this work; if not, write to the Free Software Foundation,
  19  * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
  20  *
  21  * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
  22  * or visit www.oracle.com if you need additional information or have any
  23  * questions.
  24  */
  25 
  26 package sun.awt.im;
  27 
  28 import java.awt.AWTException;
  29 import java.awt.CheckboxMenuItem;
  30 import java.awt.Component;
  31 import java.awt.Container;
  32 import java.awt.PopupMenu;
  33 import java.awt.Menu;
  34 import java.awt.MenuItem;
  35 import java.awt.Toolkit;
  36 import java.awt.event.ActionEvent;
  37 import java.awt.event.ActionListener;
  38 import java.awt.im.spi.InputMethodDescriptor;
  39 import java.util.Locale;
  40 import javax.swing.JCheckBoxMenuItem;
  41 import javax.swing.JComponent;
  42 import javax.swing.JDialog;
  43 import javax.swing.JFrame;
  44 import javax.swing.JPopupMenu;
  45 import javax.swing.JMenu;
  46 import javax.swing.JMenuItem;
  47 
  48 /**
  49  * {@code InputMethodPopupMenu} provides the popup selection menu
  50  */
  51 
  52 abstract class InputMethodPopupMenu implements ActionListener {
  53 
  54     // Factory method to provide the menu, depending on the client, i.e.,
  55     // provide Swing popup menu if client is a swing app, otherwise AWT popup
  56     // is created.
  57     static InputMethodPopupMenu getInstance(Component client, String title) {
  58         if ((client instanceof JFrame) ||
  59             (client instanceof JDialog)) {
  60                 return new JInputMethodPopupMenu(title);
  61         } else {
  62             return new AWTInputMethodPopupMenu(title);
  63         }
  64     }
  65 
  66     abstract void show(Component c, int x, int y);
  67 
  68     abstract void removeAll();
  69 
  70     abstract void addSeparator();
  71 
  72     abstract void addToComponent(Component c);
  73 
  74     abstract Object createSubmenu(String label);
  75 
  76     abstract void add(Object menuItem);
  77 
  78     abstract void addMenuItem(String label, String command, String currentSelection);
  79 
  80     abstract void addMenuItem(Object targetMenu, String label, String command,
  81                               String currentSelection);
  82 
  83     void addOneInputMethodToMenu(InputMethodLocator locator, String currentSelection) {
  84         InputMethodDescriptor descriptor = locator.getDescriptor();
  85         String label = descriptor.getInputMethodDisplayName(null, Locale.getDefault());
  86         String command = locator.getActionCommandString();
  87         Locale[] locales = null;
  88         int localeCount;
  89         try {
  90             locales = descriptor.getAvailableLocales();
  91             localeCount = locales.length;
  92         } catch (AWTException e) {
  93             // ??? should have better error handling -
  94             // tell user what happened, then remove this input method from the list.
  95             // For the time being, just show it disabled.
  96             localeCount = 0;
  97         }
  98         if (localeCount == 0) {
  99             // could be IIIMP adapter which has lost its connection
 100             addMenuItem(label, null, currentSelection);
 101         } else if (localeCount == 1) {
 102             if (descriptor.hasDynamicLocaleList()) {
 103                 // try to make sure that what the user sees and what
 104                 // we eventually select is consistent even if the locale
 105                 // list changes in the meantime
 106                 label = descriptor.getInputMethodDisplayName(locales[0], Locale.getDefault());
 107                 command = locator.deriveLocator(locales[0]).getActionCommandString();
 108             }
 109             addMenuItem(label, command, currentSelection);
 110         } else {
 111             Object submenu = createSubmenu(label);
 112             add(submenu);
 113             for (int j = 0; j < localeCount; j++) {
 114                 Locale locale = locales[j];
 115                 String subLabel = getLocaleName(locale);
 116                 String subCommand = locator.deriveLocator(locale).getActionCommandString();
 117                 addMenuItem(submenu, subLabel, subCommand, currentSelection);
 118             }
 119         }
 120     }
 121 
 122     /**
 123      * Returns whether command indicates the same input method as currentSelection,
 124      * taking into account that command may not specify a locale where currentSelection does.
 125      */
 126     static boolean isSelected(String command, String currentSelection) {
 127         if (command == null || currentSelection == null) {
 128             return false;
 129         }
 130         if (command.equals(currentSelection)) {
 131             return true;
 132         }
 133         // currentSelection may indicate a locale where command does not
 134         int index = currentSelection.indexOf('\n');
 135         if (index != -1 && currentSelection.substring(0, index).equals(command)) {
 136             return true;
 137         }
 138         return false;
 139     }
 140 
 141     /**
 142      * Returns a localized locale name for input methods with the
 143      * given locale. It falls back to Locale.getDisplayName() and
 144      * then to Locale.toString() if no localized locale name is found.
 145      *
 146      * @param locale Locale for which localized locale name is obtained
 147      */
 148     String getLocaleName(Locale locale) {
 149         String localeString = locale.toString();
 150         String localeName = Toolkit.getProperty("AWT.InputMethodLanguage." + localeString, null);
 151         if (localeName == null) {
 152             localeName = locale.getDisplayName();
 153             if (localeName == null || localeName.length() == 0)
 154                 localeName = localeString;
 155         }
 156         return localeName;
 157     }
 158 
 159     // ActionListener implementation
 160     public void actionPerformed(ActionEvent event) {
 161         String choice = event.getActionCommand();
 162         ((ExecutableInputMethodManager)InputMethodManager.getInstance()).changeInputMethod(choice);
 163     }
 164 
 165 }
 166 
 167 class JInputMethodPopupMenu extends InputMethodPopupMenu {
 168     static JPopupMenu delegate = null;
 169 
 170     JInputMethodPopupMenu(String title) {
 171         synchronized (this) {
 172             if (delegate == null) {
 173                 delegate = new JPopupMenu(title);
 174             }
 175         }
 176     }
 177 
 178     void show(Component c, int x, int y) {
 179         delegate.show(c, x, y);
 180     }
 181 
 182     void removeAll() {
 183         delegate.removeAll();
 184     }
 185 
 186     void addSeparator() {
 187         delegate.addSeparator();
 188     }
 189 
 190     void addToComponent(Component c) {
 191     }
 192 
 193     Object createSubmenu(String label) {
 194         return new JMenu(label);
 195     }
 196 
 197     void add(Object menuItem) {
 198         delegate.add((JMenuItem)menuItem);
 199     }
 200 
 201     void addMenuItem(String label, String command, String currentSelection) {
 202         addMenuItem(delegate, label, command, currentSelection);
 203     }
 204 
 205     void addMenuItem(Object targetMenu, String label, String command, String currentSelection) {
 206         JMenuItem menuItem;
 207         if (isSelected(command, currentSelection)) {
 208             menuItem = new JCheckBoxMenuItem(label, true);
 209         } else {
 210             menuItem = new JMenuItem(label);
 211         }
 212         menuItem.setActionCommand(command);
 213         menuItem.addActionListener(this);
 214         menuItem.setEnabled(command != null);
 215         if (targetMenu instanceof JMenu) {
 216             ((JMenu)targetMenu).add(menuItem);
 217         } else {
 218             ((JPopupMenu)targetMenu).add(menuItem);
 219         }
 220     }
 221 
 222 }
 223 
 224 class AWTInputMethodPopupMenu extends InputMethodPopupMenu {
 225     static PopupMenu delegate = null;
 226 
 227     AWTInputMethodPopupMenu(String title) {
 228         synchronized (this) {
 229             if (delegate == null) {
 230                 delegate = new PopupMenu(title);
 231             }
 232         }
 233     }
 234 
 235     void show(Component c, int x, int y) {
 236         delegate.show(c, x, y);
 237     }
 238 
 239     void removeAll() {
 240         delegate.removeAll();
 241     }
 242 
 243     void addSeparator() {
 244         delegate.addSeparator();
 245     }
 246 
 247     void addToComponent(Component c) {
 248         c.add(delegate);
 249     }
 250 
 251     Object createSubmenu(String label) {
 252         return new Menu(label);
 253     }
 254 
 255     void add(Object menuItem) {
 256         delegate.add((MenuItem)menuItem);
 257     }
 258 
 259     void addMenuItem(String label, String command, String currentSelection) {
 260         addMenuItem(delegate, label, command, currentSelection);
 261     }
 262 
 263     void addMenuItem(Object targetMenu, String label, String command, String currentSelection) {
 264         MenuItem menuItem;
 265         if (isSelected(command, currentSelection)) {
 266             menuItem = new CheckboxMenuItem(label, true);
 267         } else {
 268             menuItem = new MenuItem(label);
 269         }
 270         menuItem.setActionCommand(command);
 271         menuItem.addActionListener(this);
 272         menuItem.setEnabled(command != null);
 273         ((Menu)targetMenu).add(menuItem);
 274     }
 275 }