419 lines
16 KiB
Java
419 lines
16 KiB
Java
package org.simplericity.macify.eawt;
|
|
|
|
/*
|
|
* Copyright 2007 Eirik Bjorsnos.
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
|
|
import javax.imageio.ImageIO;
|
|
import java.awt.*;
|
|
import java.awt.image.BufferedImage;
|
|
import java.io.*;
|
|
import java.lang.reflect.*;
|
|
import java.net.URL;
|
|
import java.net.URLClassLoader;
|
|
import java.net.MalformedURLException;
|
|
import java.util.Collections;
|
|
import java.util.HashMap;
|
|
import java.util.Map;
|
|
|
|
|
|
/**
|
|
* Implements Application by calling the Mac OS X API through reflection.
|
|
* If this class is used on a non-OS X platform the operations will have no effect or they will simulate
|
|
* what the Apple API would do for those who manipulate state. ({@link #setEnabledAboutMenu(boolean)} etc.)
|
|
*/
|
|
@SuppressWarnings("unchecked")
|
|
public class DefaultApplication implements Application {
|
|
|
|
private Object application;
|
|
private Class applicationListenerClass;
|
|
|
|
Map listenerMap = Collections.synchronizedMap(new HashMap<Object, Object>());
|
|
private boolean enabledAboutMenu = true;
|
|
private boolean enabledPreferencesMenu;
|
|
private boolean aboutMenuItemPresent = true;
|
|
private boolean preferencesMenuItemPresent;
|
|
private ClassLoader classLoader;
|
|
|
|
public DefaultApplication() {
|
|
try {
|
|
final File file = new File("/System/Library/Java");
|
|
if (file.exists()) {
|
|
ClassLoader scl = ClassLoader.getSystemClassLoader();
|
|
Class clc = scl.getClass();
|
|
if (URLClassLoader.class.isAssignableFrom(clc)) {
|
|
Method addUrl = URLClassLoader.class.getDeclaredMethod("addURL", new Class[]{URL.class});
|
|
addUrl.setAccessible(true);
|
|
addUrl.invoke(scl, new Object[]{file.toURI().toURL()});
|
|
}
|
|
}
|
|
|
|
Class appClass = Class.forName("com.apple.eawt.Application");
|
|
application = appClass.getMethod("getApplication", new Class[0]).invoke(null, new Object[0]);
|
|
applicationListenerClass = Class.forName("com.apple.eawt.ApplicationListener");
|
|
} catch (ClassNotFoundException e) {
|
|
application = null;
|
|
} catch (IllegalAccessException e) {
|
|
throw new RuntimeException(e);
|
|
} catch (NoSuchMethodException e) {
|
|
throw new RuntimeException(e);
|
|
} catch (InvocationTargetException e) {
|
|
throw new RuntimeException(e);
|
|
} catch (MalformedURLException e) {
|
|
throw new RuntimeException(e);
|
|
}
|
|
|
|
}
|
|
|
|
public boolean isMac() {
|
|
return application != null;
|
|
}
|
|
|
|
public void addAboutMenuItem() {
|
|
if (isMac()) {
|
|
callMethod(application, "addAboutMenuItem");
|
|
} else {
|
|
this.aboutMenuItemPresent = true;
|
|
}
|
|
}
|
|
|
|
public void addApplicationListener(ApplicationListener applicationListener) {
|
|
|
|
if (!Modifier.isPublic(applicationListener.getClass().getModifiers())) {
|
|
throw new IllegalArgumentException("ApplicationListener must be a public class");
|
|
}
|
|
if (isMac()) {
|
|
Object listener = Proxy.newProxyInstance(getClass().getClassLoader(),
|
|
new Class[]{applicationListenerClass},
|
|
new ApplicationListenerInvocationHandler(applicationListener));
|
|
|
|
callMethod(application, "addApplicationListener", new Class[]{applicationListenerClass}, new Object[]{listener});
|
|
listenerMap.put(applicationListener, listener);
|
|
} else {
|
|
listenerMap.put(applicationListener, applicationListener);
|
|
}
|
|
}
|
|
|
|
public void addPreferencesMenuItem() {
|
|
if (isMac()) {
|
|
callMethod("addPreferencesMenuItem");
|
|
} else {
|
|
this.preferencesMenuItemPresent = true;
|
|
}
|
|
}
|
|
|
|
public boolean getEnabledAboutMenu() {
|
|
if (isMac()) {
|
|
return callMethod("getEnabledAboutMenu").equals(Boolean.TRUE);
|
|
} else {
|
|
return enabledAboutMenu;
|
|
}
|
|
}
|
|
|
|
public boolean getEnabledPreferencesMenu() {
|
|
if (isMac()) {
|
|
Object result = callMethod("getEnabledPreferencesMenu");
|
|
return result.equals(Boolean.TRUE);
|
|
} else {
|
|
return enabledPreferencesMenu;
|
|
}
|
|
}
|
|
|
|
public Point getMouseLocationOnScreen() {
|
|
if (isMac()) {
|
|
try {
|
|
Method method = application.getClass().getMethod("getMouseLocationOnScreen", new Class[0]);
|
|
return (Point) method.invoke(null, new Object[0]);
|
|
} catch (NoSuchMethodException e) {
|
|
throw new RuntimeException(e);
|
|
} catch (IllegalAccessException e) {
|
|
throw new RuntimeException(e);
|
|
} catch (InvocationTargetException e) {
|
|
throw new RuntimeException(e);
|
|
}
|
|
} else {
|
|
return new Point(0, 0);
|
|
}
|
|
}
|
|
|
|
public boolean isAboutMenuItemPresent() {
|
|
if (isMac()) {
|
|
return callMethod("isAboutMenuItemPresent").equals(Boolean.TRUE);
|
|
} else {
|
|
return aboutMenuItemPresent;
|
|
}
|
|
}
|
|
|
|
public boolean isPreferencesMenuItemPresent() {
|
|
if (isMac()) {
|
|
return callMethod("isPreferencesMenuItemPresent").equals(Boolean.TRUE);
|
|
} else {
|
|
return this.preferencesMenuItemPresent;
|
|
}
|
|
}
|
|
|
|
public void removeAboutMenuItem() {
|
|
if (isMac()) {
|
|
callMethod("removeAboutMenuItem");
|
|
} else {
|
|
this.aboutMenuItemPresent = false;
|
|
}
|
|
}
|
|
|
|
public synchronized void removeApplicationListener(ApplicationListener applicationListener) {
|
|
if (isMac()) {
|
|
Object listener = listenerMap.get(applicationListener);
|
|
callMethod(application, "removeApplicationListener", new Class[]{applicationListenerClass}, new Object[]{listener});
|
|
|
|
}
|
|
listenerMap.remove(applicationListener);
|
|
}
|
|
|
|
public void removePreferencesMenuItem() {
|
|
if (isMac()) {
|
|
callMethod("removeAboutMenuItem");
|
|
} else {
|
|
this.preferencesMenuItemPresent = false;
|
|
}
|
|
}
|
|
|
|
public void setEnabledAboutMenu(boolean enabled) {
|
|
if (isMac()) {
|
|
callMethod(application, "setEnabledAboutMenu", new Class[]{Boolean.TYPE}, new Object[]{Boolean.valueOf(enabled)});
|
|
} else {
|
|
this.enabledAboutMenu = enabled;
|
|
}
|
|
}
|
|
|
|
public void setEnabledPreferencesMenu(boolean enabled) {
|
|
if (isMac()) {
|
|
callMethod(application, "setEnabledPreferencesMenu", new Class[]{Boolean.TYPE}, new Object[]{Boolean.valueOf(enabled)});
|
|
} else {
|
|
this.enabledPreferencesMenu = enabled;
|
|
}
|
|
|
|
}
|
|
|
|
public int requestUserAttention(int type) {
|
|
if (type != REQUEST_USER_ATTENTION_TYPE_CRITICAL && type != REQUEST_USER_ATTENTION_TYPE_INFORMATIONAL) {
|
|
throw new IllegalArgumentException("Requested user attention type is not allowed: " + type);
|
|
}
|
|
try {
|
|
Object application = getNSApplication();
|
|
Field critical = application.getClass().getField("UserAttentionRequestCritical");
|
|
Field informational = application.getClass().getField("UserAttentionRequestInformational");
|
|
Field actual = type == REQUEST_USER_ATTENTION_TYPE_CRITICAL ? critical : informational;
|
|
|
|
return ((Integer) application.getClass().getMethod("requestUserAttention", new Class[]{Integer.TYPE}).invoke(application, new Object[]{actual.get(null)})).intValue();
|
|
|
|
} catch (ClassNotFoundException e) {
|
|
return -1;
|
|
} catch (NoSuchMethodException e) {
|
|
throw new RuntimeException(e);
|
|
} catch (IllegalAccessException e) {
|
|
throw new RuntimeException(e);
|
|
} catch (InvocationTargetException e) {
|
|
throw new RuntimeException(e);
|
|
} catch (NoSuchFieldException e) {
|
|
throw new RuntimeException(e);
|
|
}
|
|
}
|
|
|
|
public void cancelUserAttentionRequest(int request) {
|
|
try {
|
|
Object application = getNSApplication();
|
|
application.getClass().getMethod("cancelUserAttentionRequest", new Class[]{Integer.TYPE}).invoke(application, new Object[]{new Integer(request)});
|
|
} catch (ClassNotFoundException e) {
|
|
// Nada
|
|
} catch (NoSuchMethodException e) {
|
|
throw new RuntimeException(e);
|
|
} catch (IllegalAccessException e) {
|
|
throw new RuntimeException(e);
|
|
} catch (InvocationTargetException e) {
|
|
throw new RuntimeException(e);
|
|
}
|
|
}
|
|
|
|
private Object getNSApplication() throws ClassNotFoundException {
|
|
try {
|
|
Class applicationClass = Class.forName("com.apple.cocoa.application.NSApplication");
|
|
return applicationClass.getMethod("sharedApplication", new Class[0]).invoke(null, new Object[0]);
|
|
} catch (IllegalAccessException e) {
|
|
throw new RuntimeException(e);
|
|
} catch (InvocationTargetException e) {
|
|
throw new RuntimeException(e);
|
|
} catch (NoSuchMethodException e) {
|
|
throw new RuntimeException(e);
|
|
}
|
|
}
|
|
|
|
public void setApplicationIconImage(BufferedImage image) {
|
|
if (isMac()) {
|
|
try {
|
|
Method setDockIconImage = application.getClass().getMethod("setDockIconImage", Image.class);
|
|
|
|
try {
|
|
setDockIconImage.invoke(application, image);
|
|
} catch (IllegalAccessException e) {
|
|
|
|
} catch (InvocationTargetException e) {
|
|
|
|
}
|
|
} catch (NoSuchMethodException mnfe) {
|
|
|
|
|
|
ByteArrayOutputStream stream = new ByteArrayOutputStream();
|
|
try {
|
|
ImageIO.write(image, "png", stream);
|
|
} catch (IOException e) {
|
|
throw new RuntimeException(e);
|
|
}
|
|
|
|
try {
|
|
Class nsDataClass = Class.forName("com.apple.cocoa.foundation.NSData");
|
|
Constructor constructor = nsDataClass.getConstructor(new Class[]{new byte[0].getClass()});
|
|
|
|
Object nsData = constructor.newInstance(new Object[]{stream.toByteArray()});
|
|
|
|
Class nsImageClass = Class.forName("com.apple.cocoa.application.NSImage");
|
|
Object nsImage = nsImageClass.getConstructor(new Class[]{nsDataClass}).newInstance(new Object[]{nsData});
|
|
|
|
Object application = getNSApplication();
|
|
|
|
application.getClass().getMethod("setApplicationIconImage", new Class[]{nsImageClass}).invoke(application, new Object[]{nsImage});
|
|
|
|
} catch (ClassNotFoundException e) {
|
|
|
|
} catch (NoSuchMethodException e) {
|
|
throw new RuntimeException(e);
|
|
} catch (IllegalAccessException e) {
|
|
throw new RuntimeException(e);
|
|
} catch (InvocationTargetException e) {
|
|
throw new RuntimeException(e);
|
|
} catch (InstantiationException e) {
|
|
throw new RuntimeException(e);
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
public BufferedImage getApplicationIconImage() {
|
|
if (isMac()) {
|
|
|
|
try {
|
|
Method getDockIconImage = application.getClass().getMethod("getDockIconImage");
|
|
try {
|
|
return (BufferedImage) getDockIconImage.invoke(application);
|
|
} catch (IllegalAccessException e) {
|
|
|
|
} catch (InvocationTargetException e) {
|
|
|
|
}
|
|
} catch (NoSuchMethodException nsme) {
|
|
|
|
try {
|
|
Class nsDataClass = Class.forName("com.apple.cocoa.foundation.NSData");
|
|
Class nsImageClass = Class.forName("com.apple.cocoa.application.NSImage");
|
|
Object application = getNSApplication();
|
|
Object nsImage = application.getClass().getMethod("applicationIconImage", new Class[0]).invoke(application, new Object[0]);
|
|
|
|
Object nsData = nsImageClass.getMethod("TIFFRepresentation", new Class[0]).invoke(nsImage, new Object[0]);
|
|
|
|
Integer length = (Integer) nsDataClass.getMethod("length", new Class[0]).invoke(nsData, new Object[0]);
|
|
byte[] bytes = (byte[]) nsDataClass.getMethod("bytes", new Class[]{Integer.TYPE, Integer.TYPE}).invoke(nsData, new Object[]{Integer.valueOf(0), length});
|
|
|
|
BufferedImage image = ImageIO.read(new ByteArrayInputStream(bytes));
|
|
return image;
|
|
|
|
} catch (ClassNotFoundException e) {
|
|
e.printStackTrace();
|
|
} catch (NoSuchMethodException e) {
|
|
throw new RuntimeException(e);
|
|
} catch (IllegalAccessException e) {
|
|
throw new RuntimeException(e);
|
|
} catch (InvocationTargetException e) {
|
|
throw new RuntimeException(e);
|
|
} catch (IOException e) {
|
|
throw new RuntimeException(e);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
private Object callMethod(String methodname) {
|
|
return callMethod(application, methodname, new Class[0], new Object[0]);
|
|
}
|
|
|
|
private Object callMethod(Object object, String methodname) {
|
|
return callMethod(object, methodname, new Class[0], new Object[0]);
|
|
}
|
|
|
|
private Object callMethod(Object object, String methodname, Class[] classes, Object[] arguments) {
|
|
try {
|
|
if (classes == null) {
|
|
classes = new Class[arguments.length];
|
|
for (int i = 0; i < classes.length; i++) {
|
|
classes[i] = arguments[i].getClass();
|
|
|
|
}
|
|
}
|
|
Method addListnerMethod = object.getClass().getMethod(methodname, classes);
|
|
return addListnerMethod.invoke(object, arguments);
|
|
} catch (NoSuchMethodException e) {
|
|
throw new RuntimeException(e);
|
|
} catch (IllegalAccessException e) {
|
|
throw new RuntimeException(e);
|
|
} catch (InvocationTargetException e) {
|
|
throw new RuntimeException(e);
|
|
}
|
|
}
|
|
|
|
class ApplicationListenerInvocationHandler implements InvocationHandler {
|
|
private ApplicationListener applicationListener;
|
|
|
|
ApplicationListenerInvocationHandler(ApplicationListener applicationListener) {
|
|
this.applicationListener = applicationListener;
|
|
}
|
|
|
|
public Object invoke(Object object, Method appleMethod, Object[] objects) throws Throwable {
|
|
|
|
ApplicationEvent event = createApplicationEvent(objects[0]);
|
|
try {
|
|
Method method = applicationListener.getClass().getMethod(appleMethod.getName(), new Class[]{ApplicationEvent.class});
|
|
return method.invoke(applicationListener, new Object[]{event});
|
|
} catch (NoSuchMethodException e) {
|
|
if (appleMethod.getName().equals("equals") && objects.length == 1) {
|
|
return Boolean.valueOf(object == objects[0]);
|
|
}
|
|
return null;
|
|
}
|
|
}
|
|
}
|
|
|
|
private ApplicationEvent createApplicationEvent(final Object appleApplicationEvent) {
|
|
return (ApplicationEvent) Proxy.newProxyInstance(getClass().getClassLoader(), new Class[]{ApplicationEvent.class}, new InvocationHandler() {
|
|
public Object invoke(Object o, Method method, Object[] objects) throws Throwable {
|
|
return appleApplicationEvent.getClass().getMethod(method.getName(), method.getParameterTypes()).invoke(appleApplicationEvent, objects);
|
|
}
|
|
});
|
|
}
|
|
}
|