Eclipse 中的平台特定代码
Platform specific code in Eclipse
我们正在开发带有图形布局组件的 java desktop/Eclipse RCP 应用程序。我们在 windows 的较新(8、8.1、10)版本上遇到了 GUI 缩放问题,似乎只能通过 win32 api 调用检索缩放因子,因此我们添加了一个小 DLL这样做并通过 JNI 调用它。在 windows 上工作正常,在 linux/osx 上在 Maven 中工作正常,因为 class 从未在那里实例化。
问题是要获得比例因子,我们需要 window 句柄,我们像这样检索它:
public float getScale(GC gc) {
return getScale(gc.getGCData().hwnd);
}
其中 GC
是一个 org.eclipse.swt.graphics.GC
,下面的调用是对 dll 的调用。但是,在 linux 上调试时,那几行给我一个编译错误,因为该对象没有 hwnd
变量。我如何以更干净的方式解决这个问题而不出现编译错误?
您可以使用插件片段,如Greg sugested. A fragment can have a platform filter,当当前平台与过滤器表达式不匹配时,它会阻止加载它。
Windws 32 位的平台过滤器如下所示:
Eclipse-PlatformFilter: (& (osgi.ws=win32) (osgi.os=win32) (osgi.arch=x86))
片段还必须指定主机包。在运行时,片段资源合并到主机包classspace
Fragment-Host: the.parent.bundle
为避免编译错误,您将在主机包中有一个接口或抽象 class,以及两个具有不同实现的片段 Windows 和 Linux。
Windows 实现可以安全地访问 hwnd
字段,Linux 实现可以 return 表示缩放不是 available/fixed 的值。
例如,主机包会有 ScaleProvider
:
class abstract ScaleProvider {
ScaleProvider getInstance() {
String className = ScaleProvider.class.getName() + "Impl";
try {
Class<?> implClass = ScaleProvider.class.getClassLoader().loadClass( className );
return ( ScaleProvider )implClass.newInstance();
} catch( ClassNotFoundException | InstantiationException | IllegalAccessException e ) {
throw new RuntimeException( e );
}
}
abstract float getScale( Gc gc );
}
为简洁起见,实现加载器和接口描述是一种类型。 getInstance()
代码需要一个名为 ScaleProviderImpl
的 class 出现在同一个包中。
ScaleProviderImpl
class 由匹配当前平台的片段提供,并且会有 class
class ScaleProviderImpl extends ScaleProvider {
float getScale( Gc gc ) {
// platform-specific implementation
}
}
此示例假定始终存在匹配的平台特定片段。但是如果找不到 ScaleProviderImpl
,您当然可以将 getInstance()
代码更改为 return 默认实现。
我们正在开发带有图形布局组件的 java desktop/Eclipse RCP 应用程序。我们在 windows 的较新(8、8.1、10)版本上遇到了 GUI 缩放问题,似乎只能通过 win32 api 调用检索缩放因子,因此我们添加了一个小 DLL这样做并通过 JNI 调用它。在 windows 上工作正常,在 linux/osx 上在 Maven 中工作正常,因为 class 从未在那里实例化。
问题是要获得比例因子,我们需要 window 句柄,我们像这样检索它:
public float getScale(GC gc) {
return getScale(gc.getGCData().hwnd);
}
其中 GC
是一个 org.eclipse.swt.graphics.GC
,下面的调用是对 dll 的调用。但是,在 linux 上调试时,那几行给我一个编译错误,因为该对象没有 hwnd
变量。我如何以更干净的方式解决这个问题而不出现编译错误?
您可以使用插件片段,如Greg sugested. A fragment can have a platform filter,当当前平台与过滤器表达式不匹配时,它会阻止加载它。
Windws 32 位的平台过滤器如下所示:
Eclipse-PlatformFilter: (& (osgi.ws=win32) (osgi.os=win32) (osgi.arch=x86))
片段还必须指定主机包。在运行时,片段资源合并到主机包classspace
Fragment-Host: the.parent.bundle
为避免编译错误,您将在主机包中有一个接口或抽象 class,以及两个具有不同实现的片段 Windows 和 Linux。
Windows 实现可以安全地访问 hwnd
字段,Linux 实现可以 return 表示缩放不是 available/fixed 的值。
例如,主机包会有 ScaleProvider
:
class abstract ScaleProvider {
ScaleProvider getInstance() {
String className = ScaleProvider.class.getName() + "Impl";
try {
Class<?> implClass = ScaleProvider.class.getClassLoader().loadClass( className );
return ( ScaleProvider )implClass.newInstance();
} catch( ClassNotFoundException | InstantiationException | IllegalAccessException e ) {
throw new RuntimeException( e );
}
}
abstract float getScale( Gc gc );
}
为简洁起见,实现加载器和接口描述是一种类型。 getInstance()
代码需要一个名为 ScaleProviderImpl
的 class 出现在同一个包中。
ScaleProviderImpl
class 由匹配当前平台的片段提供,并且会有 class
class ScaleProviderImpl extends ScaleProvider {
float getScale( Gc gc ) {
// platform-specific implementation
}
}
此示例假定始终存在匹配的平台特定片段。但是如果找不到 ScaleProviderImpl
,您当然可以将 getInstance()
代码更改为 return 默认实现。