2016-05-01 00:00:14 +02:00

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);
}
});
}
}