Přeskočit na hlavní obsah

A všichni si rozumíme 2 - klávesové zkratky

Psali jste někdy lokalizované aplikace ve Swingu? A jak jste řešili klávesové zkratky (Mnemonics) pro různé jazyky? Já jsem se při programování JSignPdf rozhodl použít přístup, který znám z Delphi. V resource řetězcích vložím ampersand (&) před písmeno, které má být použito jako klávesová zkratka. Když chci zobrazit znak ampersand, tak napíšu dva za sebe.

A jak to funguje v kódu? Mám wrapper pro ResourceBundle, který nabízí mimo metod pro přístup k lokalizovaným stringům i metodu, která vrací MnemonicIndex (pozici znaku, který je použit jako klávesová zkratka).

/**
* Returns message for given key from active ResourceBundle
* @param aKey name of key in resource bundle
* @return message for given key
*/
public String get(final String aKey) {
  String tmpMessage = bundle.getString(aKey);
  if (tmpMessage == null) {
    tmpMessage = aKey;
  } else {
    tmpMessage = tmpMessage.replaceAll("&([^&])", "$1");
  }
  return tmpMessage;
}

/**
* Returns index of character which should be used as a mnemonic.
* It returns -1 if such an character doesn't exist.
* @param aKey resource key
* @return index (position) of character in translated message
*/
public int getMnemonicIndex(final String aKey) {
  String tmpMessage = bundle.getString(aKey);
  int tmpResult = -1;
  if (tmpMessage != null) {
    int searchFrom = 0;
    int tmpDoubles = 0;
    int tmpPos;
    final int tmpLen = tmpMessage.length();
    do {
      tmpPos = tmpMessage.indexOf('&', searchFrom);
      if (tmpPos == tmpLen-1) tmpPos = -1;
      if (tmpPos>-1) {
        if (tmpMessage.charAt(tmpPos+1) != '&') {
          tmpResult = tmpPos - tmpDoubles;
        } else {
          searchFrom = tmpPos + 2;
          tmpDoubles++;
        }
      }
    } while (tmpPos!=-1 && tmpResult==-1 && searchFrom<tmpLen);
  }
  return tmpResult;
}

/**
* Returns message for given key from active ResourceBundle and replaces
* parameters with values given in array.
* @param aKey key in resource bundle
* @param anArgs array of parameters to replace in message
* @return message for given key with given arguments
*/
public String get(String aKey, String anArgs[]) {
  String tmpMessage = get(aKey);
  if (aKey==tmpMessage || anArgs == null || anArgs.length == 0) {
    return tmpMessage;
  }
  final MessageFormat tmpFormat = new MessageFormat(tmpMessage);
  return tmpFormat.format(anArgs);
}

použití uvedeného wrapperu potom vypadá následovně (res je jméno instance wrapperu):

/**
* Application translations.
*/
private void translateLabels() {
  setTitle(res.get("gui.title", new String[] {Constants.VERSION}));
  setLabelAndMnemonic(lblKeystoreType, "gui.keystoreType.label");
  setLabelAndMnemonic(chkbAdvanced, "gui.advancedView.checkbox");
  setLabelAndMnemonic(btnSignIt,"gui.signIt.button");
}

/**
* Sets translations and mnemonics for labels and different kind of buttons
* @param aComponent component in which should be label set
* @param aKey message key
*/
private void setLabelAndMnemonic(final JComponent aComponent, final String aKey) {
  final String tmpLabelText = res.get(aKey);
  final int tmpMnemIndex = res.getMnemonicIndex(aKey);
  if (aComponent instanceof JLabel) {
    final JLabel tmpLabel = (JLabel) aComponent;
    tmpLabel.setText(tmpLabelText);
    if (tmpMnemIndex>-1) {
      tmpLabel.setDisplayedMnemonic(tmpLabelText.toLowerCase().charAt(tmpMnemIndex));
      tmpLabel.setDisplayedMnemonicIndex(tmpMnemIndex);
    }
  } else if (aComponent instanceof AbstractButton) {
    //handles Buttons, Checkboxes and Radiobuttons
    final AbstractButton tmpBtn = (AbstractButton) aComponent;
    tmpBtn.setText(tmpLabelText);
    if (tmpMnemIndex>-1) {
      tmpBtn.setMnemonic(tmpLabelText.toLowerCase().charAt(tmpMnemIndex));
    }
  } else {
    throw new IllegalArgumentException();
  }
}

Resource fajl potom obsahuje takovéto záznamy:

gui.advancedView.checkbox=A&dvanced view
gui.alias.label=Key &alias

Komentáře

Populární příspěvky z tohoto blogu

Three ways to redirect HTTP requests to HTTPs in WildFly and JBoss EAP

WildFly application server (and JBoss EAP) supports several simple ways how to redirect the communication from plain HTTP to TLS protected HTTPs. This article presents 3 ways. Two are on the application level and the last one is on the server level valid for requests to all deployments. 1. Request confidentiality in the deployment descriptor The first way is based on the Servlet specification. You need to specify which URLs should be protected in the web.xml deployment descriptor. It's the same approach as the one used for specifying which URLs require authentication/authorization. Just instead of requesting an assigned role, you request a transport-guarantee . Sample content of the WEB-INF/web.xml <web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd" version="3.1...

Acegi - logujeme loginy

Používáte-li pro správu přístupu k vaší webové aplikaci framework Acegi, možná se vám bude hodit zaznamenávat uživatelské přístupy (platné loginy) někam do databáze. Zde je jeden ze způsobů jak se s tímto problémem vypořádat. Následující přiklad používá Hibernate a databázi Oracle. Nejdříve si připravím vlastní metodu pro zápis do databáze v DAO . Umístím ji do třídy cz.mujpackage.dao.UserDao , která rozšiřuje org.springframework.orm.hibernate3.support.HibernateDaoSupport a poskytuje metody pro správu uživatelů, rolí, apod. Pro zvýšení výkonu použiji v Hibernate SQLQuery namísto vytváření instance třídy modelu a jejího ukládání pomocí metody save(...) . /** * Adds log entry to table AUTH_LOG (Oracle database form - pk_sequence has to be configured) * @param aName username * @param aRemoteAddress remote address of request */ public void logAuthenticationSuccess(final String aName, final String aRemoteAddress) { final HibernateCallback callback = new HibernateCallback() { publ...

Jak na Excel 2 - Automation a JNI

První metoda pro práci s Excelem v Javě, kterou si v seriálu ukážeme je využití MS Automation. To znamená že budeme s Excelem pracovat stejným způsobem jako při psaní skriptů ve windows (viz Windows Scripting ). To sice přináší největší funkcionalitu, ale na druhé straně spoustu omezení spočívající v předpokladech, které musí aplikace splnit. Tuto metodu doporučuji pouze v případě, že chcete používat funkcionalitu, které se nedá docílit použitím jiných metod (viz Jak na Excel 1) - například spouštění maker. Co budeme muset splnit pro využití této metody? naše Java aplikace musí běžet na stroji s Windows, kde musí být nainstalovaný Excel musíme mít Java-COM bridge, t.j. nástroj který nám umožní v Javě pracovat s COM objekty, většinou nějakou knihovnu volající přes JNI funkce z MS Windows Existuje několik open source knihoven implementujících Java-COM bridge. Tyto knihovny pracují v některém ze dvou základních režimů (některé zvládají oba). První typ přístupu využívá vygenerovan...