【发布时间】:2011-08-30 11:12:12
【问题描述】:
我正在为 Android 制作一个应用小部件,由于它由图形等自定义元素组成,因此必须呈现为位图。
但是,我在这个过程中遇到了一些障碍。
1) 有没有办法找到应用小部件的最大可用空间? (或者:是否可以正确计算 WVGA(或类似宽度)情况下可用的最小空间的尺寸?
我不知道如何计算应用小部件的最大可用空间。使用传统的应用程序小部件,可以填充_parent,并且所有空间都将被使用。但是,当将小部件呈现为位图时,为了避免拉伸,必须计算正确的尺寸。该文档概述了如何计算最小尺寸,但对于 WVGA 等情况,横向模式下会有未使用的空间 - 导致小部件看起来比其他自然拉伸的小部件更短。
float density = getResources().getDisplayMetrics().density;
int cx = ((int)Math.ceil(appWidgetInfo.minWidth / density) + 2) / 74;
int cy = ((int)Math.ceil(appWidgetInfo.minHeight / density) + 2) / 74;
int portraitWidth = (int)Math.round((80.0f * cx - 2.0f) * density);
int portraitHeight = (int)Math.round((100.0f * cy - 2.0f) * density);
int landscapeWidth = (int)Math.round((106.0f * cx - 2.0f) * density);
int landscapeHeight = (int)Math.round((74.0f * cy - 2.0f) * density);
计算 cx 和 cy 给出水平和垂直单元格的数量。从计算的 dpi 中减去 - 2(例如 74 * cy - 2)是为了避免结果像素数被四舍五入的情况。 (例如在 Nexus One 的横向模式下,高度是 110,而不是 111 (74 * 1.5)。
2) 将位图分配给用作 RemoteView 的一部分以查看图像的 ImageView 时,有两种方法:
2.1) 通过使用 setImageViewUri,并将位图保存为 PNG 文件。然后使用 ContentProvider 中的 openFile() 实现来提供图像:
@Override
public ParcelFileDescriptor openFile(Uri uri, String mode) throws FileNotFoundException
// Code to set up imageFileName
File file = new File(getContext().getCacheDir(), imageFileName);
return ParcelFileDescriptor.open(file, ParcelFileDescriptor.MODE_READ_ONLY);
}
这行得通,这是我目前正在使用的方法。但是,如果我将 ImageView 的 scaleType 设置为“center”,文档应该是“使图像在视图中居中,但不执行缩放。”,图像被错误地缩放。将位图的密度设置为 DENSITY_NONE 或 getResources().getDisplayMetrics().densityDpi 在将位图保存为 PNG 时没有任何区别,当 ImageView 加载文件时,它似乎被忽略了。结果是由于某些 dpi 问题,图像被缩小了。这似乎描述了这种情况:
因为不可能使用 scaleType:center,所以我发现唯一可行的方法是将 ImageView 的 layout_width 和 layout_height 静态设置为给定的 dpi 数,然后将位图渲染为相同的 dpi。这需要使用 scaleType:fitXY。这种方法有效,但它是一个非常静态的设置 - 它不适用于可调整大小的 3.1 应用小部件(我尚未对此进行测试,但除非在每次调整大小时调用 onUpdate(),否则这是真的)。
有什么方法可以将图像加载到未缩放的 ImageView 中,还是由于框架中的错误而无法实现?
2.1) 直接使用 setImageViewBitmap。将此方法与位图上的 Bitmap.DENSITY_NONE 设置一起使用,可以在不正确缩放的情况下显示图像。这种方法的问题在于,通过 IPC 机制可以设置多大的图像存在限制:
(不允许更多链接)http://groups.google.com/group/android-developers/browse_thread/thread/b11550601e6b1dd3#4bef4fa8908f7e6a
我尝试了一些技巧来解决这个问题,方法是将小部件拆分为可以设置为 100x100 像素块的图像矩阵。这确实允许较大的小部件工作,但最终非常重并且在大型小部件 (4x4) 上失败。
很抱歉发了很长的帖子。在尝试使用位图呈现的应用程序小部件时,我试图解释一些不同的问题。如果有人尝试过同样的方法并找到更多解决这些问题的方法,或者有任何有用的 cmets,我们将不胜感激。
【问题讨论】: