Извлечь изображение курсора в Java
Мне было интересно, если есть способ извлечь объект Image из объекта Cursor в Java.
Использование для этого будет, например:
Image img = extractCursorImage(Cursor.getDefaultCursor());
Который вы затем можете нарисовать на кнопке панели инструментов (это цель, для которой я хочу это).
2 ответа
Класс Cursor довольно абстрактный - все важные вещи делегированы нативному коду, поэтому вы не можете просто рисовать его в графическом контексте. Не существует очевидного способа обойти необходимость предварительно определять значки или делать это в собственном коде.
не могли бы вы помочь мне использовать ту функцию, которую вы упомянули?
Ниже приведен код для рисования встроенных курсоров Windows с использованием библиотеки JNA. Если вы можете использовать JNA, вы можете избежать компиляторов C++.
Я, вероятно, делаю слишком много нативных звонков, но стоимость создания значков одним выстрелом невелика.
http://f.imagehost.org/0709/hand.png
Код для отображения курсора в виде изображения Java:
public class LoadCursor {
public static void draw(BufferedImage image, int cursor,
int diFlags) {
int width = image.getWidth();
int height = image.getHeight();
User32 user32 = User32.INSTANCE;
Gdi32 gdi32 = Gdi32.INSTANCE;
Pointer hIcon = user32
.LoadCursorW(Pointer.NULL, cursor);
Pointer hdc = gdi32.CreateCompatibleDC(Pointer.NULL);
Pointer bitmap = gdi32.CreateCompatibleBitmap(hdc,
width, height);
gdi32.SelectObject(hdc, bitmap);
user32.DrawIconEx(hdc, 0, 0, hIcon, width, height, 0,
Pointer.NULL, diFlags);
for (int x = 0; x < width; x++) {
for (int y = 0; y < width; y++) {
int rgb = gdi32.GetPixel(hdc, x, y);
image.setRGB(x, y, rgb);
}
}
gdi32.DeleteObject(bitmap);
gdi32.DeleteDC(hdc);
}
public static void main(String[] args) {
final int width = 128;
final int height = 128;
BufferedImage image = new BufferedImage(width, height,
BufferedImage.TYPE_INT_ARGB);
draw(image, User32.IDC_HAND, User32.DI_NORMAL);
BufferedImage mask = new BufferedImage(width, height,
BufferedImage.TYPE_INT_RGB);
draw(mask, User32.IDC_HAND, User32.DI_MASK);
applyMask(image, mask);
JLabel icon = new JLabel();
icon.setIcon(new ImageIcon(image));
JFrame frame = new JFrame();
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.setContentPane(icon);
frame.pack();
frame.setVisible(true);
}
private static void applyMask(BufferedImage image,
BufferedImage mask) {
int width = image.getWidth();
int height = image.getHeight();
for (int x = 0; x < width; x++) {
for (int y = 0; y < height; y++) {
int masked = mask.getRGB(x, y);
if ((masked & 0x00FFFFFF) == 0) {
int rgb = image.getRGB(x, y);
rgb = 0xFF000000 | rgb;
image.setRGB(x, y, rgb);
}
}
}
}
}
Интерфейс User32.dll:
public interface User32 extends Library {
public static User32 INSTANCE = (User32) Native
.loadLibrary("User32", User32.class);
/** @see #LoadCursorW(Pointer, int) */
public static final int IDC_ARROW = 32512;
/** @see #LoadCursorW(Pointer, int) */
public static final int IDC_IBEAM = 32513;
/** @see #LoadCursorW(Pointer, int) */
public static final int IDC_WAIT = 32514;
/** @see #LoadCursorW(Pointer, int) */
public static final int IDC_CROSS = 32515;
/** @see #LoadCursorW(Pointer, int) */
public static final int IDC_UPARROW = 32516;
/** @see #LoadCursorW(Pointer, int) */
public static final int IDC_SIZENWSE = 32642;
/** @see #LoadCursorW(Pointer, int) */
public static final int IDC_SIZENESW = 32643;
/** @see #LoadCursorW(Pointer, int) */
public static final int IDC_SIZEWE = 32644;
/** @see #LoadCursorW(Pointer, int) */
public static final int IDC_SIZENS = 32645;
/** @see #LoadCursorW(Pointer, int) */
public static final int IDC_SIZEALL = 32646;
/** @see #LoadCursorW(Pointer, int) */
public static final int IDC_NO = 32648;
/** @see #LoadCursorW(Pointer, int) */
public static final int IDC_HAND = 32649;
/** @see #LoadCursorW(Pointer, int) */
public static final int IDC_APPSTARTING = 32650;
/** @see #LoadCursorW(Pointer, int) */
public static final int IDC_HELP = 32651;
/** @see #LoadCursorW(Pointer, int) */
public static final int IDC_ICON = 32641;
/** @see #LoadCursorW(Pointer, int) */
public static final int IDC_SIZE = 32640;
/** @see #DrawIconEx(Pointer, int, int, Pointer, int, int, int, Pointer, int) */
public static final int DI_COMPAT = 4;
/** @see #DrawIconEx(Pointer, int, int, Pointer, int, int, int, Pointer, int) */
public static final int DI_DEFAULTSIZE = 8;
/** @see #DrawIconEx(Pointer, int, int, Pointer, int, int, int, Pointer, int) */
public static final int DI_IMAGE = 2;
/** @see #DrawIconEx(Pointer, int, int, Pointer, int, int, int, Pointer, int) */
public static final int DI_MASK = 1;
/** @see #DrawIconEx(Pointer, int, int, Pointer, int, int, int, Pointer, int) */
public static final int DI_NORMAL = 3;
/** @see #DrawIconEx(Pointer, int, int, Pointer, int, int, int, Pointer, int) */
public static final int DI_APPBANDING = 1;
/** http://msdn.microsoft.com/en-us/library/ms648391(VS.85).aspx */
public Pointer LoadCursorW(Pointer hInstance,
int lpCursorName);
/** http://msdn.microsoft.com/en-us/library/ms648065(VS.85).aspx */
public boolean DrawIconEx(Pointer hdc, int xLeft,
int yTop, Pointer hIcon, int cxWidth, int cyWidth,
int istepIfAniCur, Pointer hbrFlickerFreeDraw,
int diFlags);
}
Интерфейс Gdi32.dll:
public interface Gdi32 extends Library {
public static Gdi32 INSTANCE = (Gdi32) Native
.loadLibrary("Gdi32", Gdi32.class);
/** http://msdn.microsoft.com/en-us/library/dd183489(VS.85).aspx */
public Pointer CreateCompatibleDC(Pointer hdc);
/** http://msdn.microsoft.com/en-us/library/dd183488(VS.85).aspx */
public Pointer CreateCompatibleBitmap(Pointer hdc,
int nWidth, int nHeight);
/** http://msdn.microsoft.com/en-us/library/dd162957(VS.85).aspx */
public Pointer SelectObject(Pointer hdc, Pointer hgdiobj);
/** http://msdn.microsoft.com/en-us/library/dd145078(VS.85).aspx */
public int SetPixel(Pointer hdc, int X, int Y, int crColor);
/** http://msdn.microsoft.com/en-us/library/dd144909(VS.85).aspx */
public int GetPixel(Pointer hdc, int nXPos, int nYPos);
/** http://msdn.microsoft.com/en-us/library/dd183539(VS.85).aspx */
public boolean DeleteObject(Pointer hObject);
/** http://msdn.microsoft.com/en-us/library/dd183533(VS.85).aspx */
public boolean DeleteDC(Pointer hdc);
}
Решение для Linux:
private BufferedImage getCursorImage(){
X11 x11 = X11.INSTANCE;
Xfixes xfixes = Xfixes.INSTANCE;
X11.Display display = x11.XOpenDisplay(null);
Xfixes.XFixesCursorImage cursorImage = xfixes.XFixesGetCursorImage(display);
ByteBuffer buf = cursorImage.pixels.getPointer().getByteBuffer(0,
cursorImage.width * cursorImage.height * NativeLong.SIZE);
buf.order(ByteOrder.LITTLE_ENDIAN);
BufferedImage bim = new BufferedImage(cursorImage.width, cursorImage.height, BufferedImage.TYPE_INT_ARGB);
WritableRaster raster = bim.getRaster();
for (int y = 0; y < cursorImage.height; y++) {
for (int x = 0; x < cursorImage.width; x++) {
long z = NativeLong.SIZE == 8 ? buf.getLong() : buf.getInt();
int b = (int) ((z >> 24) & 0xFF);
int a = (int) ((z >> 16) & 0xFF);
int g = (int) ((z >> 8) & 0xFF);
int r = (int) (z & 0xFF);
raster.setPixel(x, y, new int[] { a, r, g, b });
}
}
x11.XCloseDisplay(display);
return bim;
}
Интерфейс JNA:
public interface Xfixes extends Library {
Xfixes INSTANCE = Native.load("Xfixes", Xfixes.class);
@Structure.FieldOrder({ "x", "y", "width", "height", "xhot", "yhot", "cursor_serial", "pixels", "atom", "name"})
class XFixesCursorImage extends Structure {
public short x;
public short y;
public short width;
public short height;
public short xhot;
public short yhot;
public NativeLong cursor_serial;
public NativeLongByReference pixels;
public NativeLong atom;
public Pointer name;
public XFixesCursorImage() {
super();
}
}
XFixesCursorImage XFixesGetCursorImage(X11.Display dpy);
}
Может быть возможно найти изображения системного курсора из JRE. Смотреть на исходный код тоже может быть интересно.