Usando JNA para obter / identificador de aplicação conjunto
Pergunta
No seguimento minha pergunta anterior sobre a barra de tarefas do Windows 7 , eu gostaria de diagnosticar por que o Windows não está reconhecendo que a minha candidatura é independente da javaw.exe
. I atualmente tem o seguinte código JNA para obter a AppUserModelID
:
public class AppIdTest {
public static void main(String[] args) {
NativeLibrary lib;
try {
lib = NativeLibrary.getInstance("shell32");
} catch (Error e) {
System.err.println("Could not load Shell32 library.");
return;
}
Object[] functionArgs = new Object[1];
String functionName = null;
Function function;
try {
functionArgs[0] = new String("Vendor.MyJavaApplication")
.getBytes("UTF-16");
functionName = "GetCurrentProcessExplicitAppUserModelID";
function = lib.getFunction(functionName);
// Output the current AppId
System.out.println("1: " + function.getString(0));
functionName = "SetCurrentProcessExplicitAppUserModelID";
function = lib.getFunction(functionName);
// Set the new AppId
int ret = function.invokeInt(functionArgs);
if (ret != 0) {
Logger.out.error(function.getName() + " returned error code "
+ ret + ".");
}
functionName = "GetCurrentProcessExplicitAppUserModelID";
function = lib.getFunction(functionName);
// Output the current AppId
System.out.println("2: " + function.getString(0));
// Output the current AppID, converted from UTF-16
System.out.println("3: "
+ new String(function.getByteArray(0, 255), "UTF-16"));
} catch (UnsupportedEncodingException e) {
System.err.println("System does not support UTF-16 encoding.");
} catch (UnsatisfiedLinkError e) {
System.err.println(functionName + " was not found in "
+ lib.getFile().getName() + ".");
}
}
}
A saída da aplicação é aparentemente sem sentido:
1: ‹ÿU‹ìƒìL¡¬Ÿv3ʼnEüSV‹uƒ&
2: ‹ÿU‹ìƒìL¡¬Ÿv3ʼnEüSV‹uƒ&
3: ????????????????P???????????
Ser ciente do facto de que a saída pode ser UTF-16, em (3) I tentativa para converter um array de bytes de UTF-16. Em toda a honestidade que eu não sei se a minha abordagem aqui é certo que (a) Não sei o tamanho de uma PWSTR
e (b) Eu não sei se GetCurrentProcessExplicitAppUserModelID
é de fato retornando um array de bytes ou string.
Estou ciente de que JSmooth irá executar o processo de GUI em um invólucro que simula este efeito. reivindicações launch4j a fazer o mesmo, mas não parece trabalho. Eu estou olhando para ter o conjunto AppUserModelID
independentemente do Java invólucro .
O que está errado aqui?
Solução
Eu não vi a sua pergunta antes de outra forma eu teria dado uma chance, mesmo sem uma recompensa.
Aqui está o que eu vim acima com. Por favor note, como indicado no próprio código, eu não implementar memória a limpeza adequada com a função CoTaskMemFree
(de Ole32.dll
). Então eu sugiro que você tome apenas a implementação para SetCurrentProcessExplicitAppUserModelID()
package com.stackoverflow.AppIdTest;
import com.sun.jna.Native;
import com.sun.jna.NativeLong;
import com.sun.jna.Pointer;
import com.sun.jna.WString;
import com.sun.jna.ptr.PointerByReference;
public class AppIdTest
{
public static void main(String[] args) throws Exception
{
setCurrentProcessExplicitAppUserModelID(AppIdTest.class.getName());
System.out.println(getCurrentProcessExplicitAppUserModelID());
}
// DO NOT DO THIS, IT'S JUST FOR TESTING PURPOSE AS I'M NOT FREEING THE MEMORY
// AS REQUESTED BY THE DOCUMENTATION:
//
// http://msdn.microsoft.com/en-us/library/dd378419%28VS.85%29.aspx
//
// "The caller is responsible for freeing this string with CoTaskMemFree when
// it is no longer needed"
public static String getCurrentProcessExplicitAppUserModelID()
{
final PointerByReference r = new PointerByReference();
if (GetCurrentProcessExplicitAppUserModelID(r).longValue() == 0)
{
final Pointer p = r.getValue();
return p.getString(0, true); // here we leak native memory by lazyness
}
return "N/A";
}
public static void setCurrentProcessExplicitAppUserModelID(final String appID)
{
if (SetCurrentProcessExplicitAppUserModelID(new WString(appID)).longValue() != 0)
throw new RuntimeException("unable to set current process explicit AppUserModelID to: " + appID);
}
private static native NativeLong GetCurrentProcessExplicitAppUserModelID(PointerByReference appID);
private static native NativeLong SetCurrentProcessExplicitAppUserModelID(WString appID);
static
{
Native.register("shell32");
}
}
Será que funciona para você?
Pelo menos aqui corretamente imprime volta:
com.stackoverflow.AppIdTest.AppIdTest
Outras dicas
Se você só precisa definir o AppUserModelId em seguida, o código acima JNA é suficiente. No entanto, se você quiser tirar proveito dos novos recursos do Windows 7 em seu aplicativo Java, então confira J7Goodies uma biblioteca Java fornecer ao Windows 7 extensões da barra de tarefas.
Editar : mais informações de J7Goodies Guia do Programador
4.2. Definir AppUserModelID
Para usar qualquer um dos Windows 7 apresenta um aplicativo deve explicitamente definir seu identificador de processo - Aplicativo de Usuário Modelo ID (
AppUserModelID
). Ele não pode ter mais de 128 caracteres e não pode conter espaços. Cada seção deve ser camel-encaixotado, por exemplo:CompanyName.ProductName.SubProduct.VersionInformation
Este identificador deve ser ajustado antes de qualquer GUI (janela) está mostrado. Você configurá-lo chamando:
// Remember to set AppUserModelID before creating any UI
AppUserModelId.setCurrentProcessId("StrixCode.J7Goodies.Appname");
4.3. Definindo propriedades da janela
Um aplicativo Java não pode ser preso à barra de tarefas do Windows 7, a menos que suas propriedades de janela são definidos. o propriedades consistem em quatro campos:
- AppUserModelID - o mesmo que passou para
AppUserModelId.setCurrentProcessId(String)
- RelaunchDisplayName - nome do aplicativo
- RelaunchCommand - o comando completo usado para iniciar a aplicação. No caso de um programa Java será:
<path to javaw.exe> -jar <path to application jar>
- RelaunchIcon - caminho para a aplicação é ícone
Importante :
RelaunchCommand
eRelaunchDisplayName
deve ser sempre ajustada juntos. Para definir essas propriedades usar a classe WindowProperties simples.
WindowProperties props = new WindowProperties(myFrame);
props.setRelaunchCommand("<full path to javaw.exe –arguments>");
props.setRelaunchDisplayName("My Java Application");
props.setRelaunchIcon("<full path to an .ico or .exe file>");
props.setAppUserModelID("StrixCode.J7Goodies.Appname");
props.save();
Aqui está uma forma mais simples exemplo de como chamar SetCurrentProcessExplicitAppUserModelID
via JNA
:
import com.sun.jna.*;
import com.sun.jna.win32.*;
interface Shell32 extends StdCallLibrary {
Shell32 INSTANCE = (Shell32) Native.loadLibrary("shell32", Shell32.class, W32APIOptions.DEFAULT_OPTIONS);
NativeLong SetCurrentProcessExplicitAppUserModelID(WString appID);
}