Libraryless. Click here for Pure Java version (20393L/126K).
1 | sclass ImageSurface extends Surface { |
2 | BufferedImage image; |
3 | double zoomX = 1, zoomY = 1, zoomFactor = 1.5; |
4 | private Rectangle selection; |
5 | new L<AutoCloseable> tools; |
6 | |
7 | // use overlays now |
8 | O overlay; // voidfunc(Graphics2D) |
9 | L<G2Drawable> overlays = syncL(); |
10 | |
11 | Runnable onSelectionChange; |
12 | settable bool verbose; |
13 | bool noMinimumSize = true; |
14 | S titleForUpload; |
15 | O onZoom; |
16 | bool specialPurposed; // true = don't show image changing commands in popup menu |
17 | settable bool allowPaste; |
18 | settable bool zoomable = true; |
19 | bool noAlpha; // set to true to speed up drawing if you don't use alpha |
20 | O interpolationMode = RenderingHints.VALUE_INTERPOLATION_BILINEAR; |
21 | O onNewImage; // use imageChanged event instead |
22 | BufferedImage imageToDraw; // if you want to draw a different image |
23 | File file; // where image was loaded from |
24 | bool autoZoomToDisplay; // only works 100% when not in scrollpane |
25 | settable bool repaintInThread; // after setImage, repaint in same thread |
26 | BoolVar showingVar; |
27 | |
28 | Pt mousePosition; |
29 | event mousePositionChanged; |
30 | event imageChanged; |
31 | event userModifiedImage(BufferedImage image); |
32 | |
33 | public ImageSurface() { |
34 | this(dummyImage()); |
35 | } |
36 | |
37 | static BufferedImage dummyImage() { |
38 | ret whiteImage(1); |
39 | } |
40 | |
41 | *(File file) { |
42 | setImage(file); |
43 | } |
44 | |
45 | *(MakesBufferedImage image) { |
46 | this(image != null ? image.getBufferedImage() : dummyImage()); |
47 | } |
48 | |
49 | *(BufferedImage image) { |
50 | setImage(image); |
51 | clearSurface = false; |
52 | |
53 | // perform auto-zoom when shown, resized or when parent resized |
54 | bindToComponent(this, l0 performAutoZoom, null); |
55 | onResize(this, l0 performAutoZoom); |
56 | onEnclosingScrollPaneResize(this, l0 performAutoZoom); |
57 | |
58 | componentPopupMenu2(this, ImageSurface_popupMenuMaker()); |
59 | new ImageSurfaceSelector(this); |
60 | |
61 | jHandleFileDrop(this, voidfunc(File f) { setImage(loadBufferedImage(f)) }); |
62 | |
63 | imageSurfaceOnHover(this, p -> { |
64 | mousePosition = p; |
65 | mousePositionChanged(); |
66 | }); |
67 | } |
68 | |
69 | public ImageSurface(RGBImage image, double zoom) { |
70 | this(image); |
71 | setZoom(zoom); |
72 | } |
73 | |
74 | // point is already in image coordinates |
75 | protected void fillPopupMenu(JPopupMenu menu, Point point) { |
76 | if (zoomable) { |
77 | JMenuItem miZoomReset = new JMenuItem("Zoom 100%"); |
78 | miZoomReset.addActionListener(new ActionListener() { |
79 | public void actionPerformed(ActionEvent evt) { |
80 | setZoom(1.0); |
81 | centerPoint(point); |
82 | } |
83 | }); |
84 | menu.add(miZoomReset); |
85 | |
86 | JMenuItem miZoomIn = new JMenuItem("Zoom in"); |
87 | miZoomIn.addActionListener(new ActionListener() { |
88 | public void actionPerformed(ActionEvent evt) { |
89 | zoomIn(zoomFactor); |
90 | centerPoint(point); |
91 | } |
92 | }); |
93 | menu.add(miZoomIn); |
94 | |
95 | JMenuItem miZoomOut = new JMenuItem("Zoom out"); |
96 | miZoomOut.addActionListener(new ActionListener() { |
97 | public void actionPerformed(ActionEvent evt) { |
98 | zoomOut(zoomFactor); |
99 | centerPoint(point); |
100 | } |
101 | }); |
102 | menu.add(miZoomOut); |
103 | |
104 | menu.add(jMenuItemStayCheckedOnClick("Zoom to window", |
105 | -> autoZoomToDisplay, |
106 | -> setAutoZoomToDisplay(true))); |
107 | |
108 | addMenuItem(menu, "Show full screen", r { showFullScreen() }); |
109 | |
110 | S col = ""; |
111 | |
112 | if (hasImage()) pcall { |
113 | int rgb = getPixel(getImage(), point.x, point.y); |
114 | col = " #" + takeLast(6, intToHex(rgb)); |
115 | } |
116 | |
117 | addMenuItem(menu, "Point: " + point.x + "," + point.y + col + " (image: " + w() + "*" + h() + ")", null); |
118 | |
119 | menu.addSeparator(); |
120 | } |
121 | |
122 | if (!specialPurposed) |
123 | addMenuItem(menu, "Load image...", r { selectFile("Load image", |
124 | voidfunc(File f) { setImage(loadImage2(f)) }) }); |
125 | addMenuItem(menu, "Save image...", r { saveImage() }); |
126 | ifdef ImageSurface_AllowUpload |
127 | addMenuItem(menu, "Upload image...", r { uploadTheImage() }); |
128 | endifdef |
129 | addMenuItem(menu, "Copy image to clipboard", r { copyImageToClipboard(getImage()) }); |
130 | if (!specialPurposed || allowPaste) |
131 | addMenuItem(menu, "Paste image from clipboard", r { loadFromClipboard() }); |
132 | |
133 | if (!specialPurposed) |
134 | addMenuItem(menu, "Load image snippet...", r { |
135 | selectImageSnippet(voidfunc(S imageID) { |
136 | setImage(loadImage2(imageID)) |
137 | }); |
138 | }); |
139 | if (selection != null) |
140 | addMenuItem(menu, "Crop", r { crop() }); |
141 | if (!specialPurposed) |
142 | addMenuItem(menu, "No image", r { noImage() }); |
143 | } |
144 | |
145 | void noImage() { setImage((BufferedImage) null); } |
146 | |
147 | void crop() { |
148 | if (selection == null) ret; |
149 | BufferedImage img = cloneClipBufferedImage(getImage(), selection); |
150 | selection = null; |
151 | setUserModifiedImage(img); |
152 | } |
153 | |
154 | void setUserModifiedImage(BufferedImage img) { |
155 | setImage(img); |
156 | userModifiedImage(img); |
157 | } |
158 | |
159 | void loadFromClipboard() { |
160 | BufferedImage img = getImageFromClipboard(); |
161 | if (img != null) |
162 | setUserModifiedImage(img); |
163 | } |
164 | |
165 | swappable File defaultImageDir() { ret getProgramDir(); } |
166 | |
167 | void saveImage() { |
168 | var image = getImage(); |
169 | JFileChooser fileChooser = new JFileChooser(defaultImageDir()); |
170 | if (fileChooser.showSaveDialog(this) == JFileChooser.APPROVE_OPTION) { |
171 | try { |
172 | main saveImage(image, file = fileChooser.getSelectedFile()); |
173 | } catch e { |
174 | popup(e); |
175 | } |
176 | } |
177 | } |
178 | |
179 | void drawImageItself(int w, int h, Graphics2D g) { |
180 | int iw = getZoomedWidth(), ih = getZoomedHeight(); |
181 | if (interpolationMode == RenderingHints.VALUE_INTERPOLATION_NEAREST_NEIGHBOR || zoomX >= 1 || zoomY >= 1) { |
182 | // faster |
183 | g.drawImage(image, 0, 0, iw, ih, null); |
184 | } else |
185 | g.drawImage(resizeImage(image, iw, ih), 0, 0, null); // smoother |
186 | } |
187 | |
188 | swappable void drawBackground(int w, int h, Graphics2D g) { |
189 | g.setColor(or(getBackground(), Color.white)); |
190 | g.fillRect(0, 0, w, h); |
191 | } |
192 | |
193 | public void render(int w, int h, Graphics2D g) { |
194 | if (verbose) _print("render"); |
195 | g.setRenderingHint(RenderingHints.KEY_INTERPOLATION, interpolationMode); |
196 | drawBackground(w, h, g); |
197 | BufferedImage image = or(imageToDraw, this.image); |
198 | if (!hasImage()) |
199 | drawBackground(w, h, g); |
200 | else { |
201 | bool alpha = !noAlpha && hasTransparency(image); |
202 | if (alpha) |
203 | drawBackground(w, h, g); |
204 | |
205 | drawImageItself(w, h, g); |
206 | |
207 | int iw = getZoomedWidth(), ih = getZoomedHeight(); |
208 | if (!alpha) { |
209 | g.fillRect(iw, 0, w-iw, h); |
210 | g.fillRect(0, ih, iw, h-ih); |
211 | } |
212 | } |
213 | |
214 | if (overlay != null) { |
215 | if (verbose) _print("render overlay"); |
216 | pcallF(overlay, g); |
217 | } |
218 | |
219 | for (var overlay : cloneList(overlays)) pcall { |
220 | overlay.drawOn(cloneGraphics(g)); |
221 | } |
222 | |
223 | if (selection != null) { |
224 | if (verbose) _print("render selection"); |
225 | // drawRect is inclusive, selection is exclusive, so... whatever, tests show it's cool. |
226 | drawSelectionRect(g, selection, Color.green, Color.white); |
227 | } |
228 | } |
229 | |
230 | public void drawSelectionRect(Graphics2D g, Rectangle selection, Color green, Color white) { |
231 | drawSelectionRect(g, selection, green, white, zoomX, zoomY); |
232 | } |
233 | |
234 | public void drawSelectionRect(Graphics2D g, Rectangle selection, Color green, Color white, double zoomX, double zoomY) { |
235 | g.setColor(green); |
236 | int top = (int) (selection.y * zoomY); |
237 | int bottom = (int) ((selection.y+selection.height) * zoomY); |
238 | int left = (int) (selection.x * zoomX); |
239 | int right = (int) ((selection.x+selection.width) * zoomX); |
240 | g.drawRect(left-1, top-1, right-left+1, bottom-top+1); |
241 | g.setColor(white); |
242 | g.drawRect(left - 2, top - 2, right - left + 3, bottom - top + 3); |
243 | } |
244 | |
245 | public ImageSurface setZoom(double zoom) { |
246 | setZoom(zoom, zoom); |
247 | this; |
248 | } |
249 | |
250 | public void setZoom(double zoomX, double zoomY) { |
251 | autoZoomToDisplay = false; |
252 | setZoom_dontChangeAutoZoom(zoomX, zoomY); |
253 | } |
254 | |
255 | public void setZoom_dontChangeAutoZoom(double zoomX, double zoomY default zoomX) { |
256 | if (this.zoomX == zoomX && this.zoomY == zoomY) ret; |
257 | if (verbose) _print("Setting zoom"); |
258 | this.zoomX = zoomX; |
259 | this.zoomY = zoomY; |
260 | revalidateMe(); |
261 | repaint(); |
262 | centerPoint(new Point(getImage().getWidth()/2, getImage().getHeight()/2)); |
263 | |
264 | pcallF(onZoom); |
265 | } |
266 | |
267 | public scaffolded Dimension getMinimumSize() { |
268 | if (noMinimumSize) ret new Dimension(1, 1); |
269 | int w = getZoomedWidth(); |
270 | int h = getZoomedHeight(); |
271 | Dimension min = super.getMinimumSize(); |
272 | ret printIfScaffoldingEnabled(this, new Dimension(Math.max(w, min.width), Math.max(h, min.height))); |
273 | } |
274 | |
275 | int getZoomedHeight() { |
276 | return (int) (h() * zoomY); |
277 | } |
278 | |
279 | int getZoomedWidth() { |
280 | return (int) (w() * zoomX); |
281 | } |
282 | |
283 | bool isShowing_quick() { |
284 | if (showingVar == null) swing { |
285 | showingVar if null = componentShowingVar(ImageSurface.this); |
286 | } |
287 | |
288 | ret showingVar!; |
289 | } |
290 | |
291 | public void setImageIfShowing_thisThread(BufferedImage etc image) { |
292 | if (isShowing_quick()) |
293 | setImage_thisThread(image); |
294 | } |
295 | |
296 | void setImage(File file) { |
297 | setFile(file); |
298 | setImage(loadImage2(file)); |
299 | } |
300 | |
301 | public void setImage(MakesBufferedImage image) swing { |
302 | setImage_thisThread(image); |
303 | } |
304 | |
305 | public void setImage(BufferedImage img) swing { |
306 | setImage_thisThread(img); |
307 | } |
308 | |
309 | public void setImage_thisThread(BufferedImage etc img) { |
310 | BufferedImage newImage = img != null ? img : dummyImage(); |
311 | BufferedImage oldImage = image; |
312 | image = newImage; |
313 | if (verbose) print("Old image size:" + imageSize(oldImage) + ", new image size: " + imageSize(newImage)); |
314 | bool sameSize = imagesHaveSameSize(oldImage, newImage); |
315 | if (!sameSize) { |
316 | if (verbose) _print("New image size"); |
317 | revalidateMe(); |
318 | } |
319 | quickRepaint(); |
320 | pcallF(onNewImage); |
321 | if (!sameSize && autoZoomToDisplay) zoomToDisplaySize(); |
322 | imageChanged(); |
323 | } |
324 | |
325 | void setImageAndZoomToDisplay(BufferedImage img) { |
326 | setImage(img); |
327 | zoomToDisplaySize(); |
328 | } |
329 | |
330 | public BufferedImage getImage() { |
331 | return image; |
332 | } |
333 | |
334 | public double getZoomX() { |
335 | return zoomX; |
336 | } |
337 | |
338 | public double getZoomY() { |
339 | return zoomY; |
340 | } |
341 | |
342 | public scaffolded Dimension getPreferredSize() { |
343 | ret printIfScaffoldingEnabled(this, new Dimension(getZoomedWidth(), getZoomedHeight())); |
344 | } |
345 | |
346 | /** returns a scrollpane with the scroll-mode prevent-garbage-drawing fix applied */ |
347 | public JScrollPane makeScrollPane() { |
348 | JScrollPane scrollPane = new JScrollPane(this); |
349 | scrollPane.getViewport().setScrollMode(JViewport.BACKINGSTORE_SCROLL_MODE); |
350 | return scrollPane; |
351 | } |
352 | |
353 | public void zoomToWindow() { zoomToDisplaySize(); } |
354 | public void zoomToDisplaySize() swing { |
355 | if (!hasImage()) return; |
356 | Dimension display = getDisplaySize(); |
357 | if (display.width == 0 || display.height == 0) ret; |
358 | int w = w(), h = h(); |
359 | double xRatio = (display.width-5)/(double) w; |
360 | double yRatio = (display.height-5)/(double) h; |
361 | if (scaffoldingEnabled(this)) |
362 | printVars zoomToDisplaySize(+display, +w, +h, +xRatio, +yRatio); |
363 | setZoom_dontChangeAutoZoom(min(xRatio, yRatio)); |
364 | revalidateMe(); |
365 | } |
366 | |
367 | /** tricky magic to get parent scroll pane */ |
368 | private scaffolded Dimension getDisplaySize() { |
369 | Container c = getParent(); |
370 | while (c != null) { |
371 | if (c instanceof JScrollPane) |
372 | return c.getSize(); |
373 | c = c.getParent(); |
374 | } |
375 | return getSize(); |
376 | } |
377 | |
378 | public void setSelection(Rect r) { |
379 | setSelection(toRectangle(r)); |
380 | } |
381 | |
382 | public void setSelection(Rectangle r) { |
383 | if (neq(selection, r)) { |
384 | selection = r; |
385 | pcallF(onSelectionChange); |
386 | quickRepaint(); |
387 | } |
388 | } |
389 | |
390 | public Rectangle getSelection() { |
391 | return selection; |
392 | } |
393 | |
394 | public RGBImage getRGBImage() { |
395 | return new RGBImage(getImage()); |
396 | } |
397 | |
398 | // p is in image coordinates |
399 | void centerPoint(Point p) { |
400 | JScrollPane sp = enclosingScrollPane(this); |
401 | if (sp == null) ret; |
402 | |
403 | p = new Point((int) (p.x*getZoomX()), (int) (p.y*getZoomY())); |
404 | final JViewport viewport = sp.getViewport(); |
405 | Dimension viewSize = viewport.getExtentSize(); |
406 | |
407 | //_print("centerPoint " + p); |
408 | int x = max(0, p.x-viewSize.width/2); |
409 | int y = max(0, p.y-viewSize.height/2); |
410 | |
411 | //_print("centerPoint " + p + " => " + x + "/" + y); |
412 | p = new Point(x,y); |
413 | //_print("centerPoint " + p); |
414 | final Point _p = p; |
415 | awtLater(r { |
416 | viewport.setViewPosition(_p); |
417 | }); |
418 | } |
419 | |
420 | Pt pointFromEvent(MouseEvent e) { |
421 | ret pointFromComponentCoordinates(new Pt(e.getX(), e.getY())); |
422 | } |
423 | |
424 | Pt pointFromComponentCoordinates(Pt p) { |
425 | ret new Pt((int) (p.x/zoomX), (int) (p.y/zoomY)); |
426 | } |
427 | |
428 | Pt pointToComponentCoordinates(double x, double y) { |
429 | ret new Pt((int) (x*zoomX), (int) (y*zoomY)); |
430 | } |
431 | |
432 | void uploadTheImage { |
433 | //call(hotwire(/*#1007313*/#1016427), "go", getImage(), titleForUpload); |
434 | ifdef ImageSurface_AllowUpload |
435 | var img = getImage(); |
436 | JTextField tf = jTextField(titleForUpload); |
437 | showFormTitled("Upload Image (PNG)", |
438 | "Image title (optional)", tf, func { |
439 | disableSubmitButton(getFrame(tf)); |
440 | thread "Upload Image" { |
441 | try { |
442 | messageBox("Image uploaded as " + uploadPNGToImageServer(img, titleForUpload = getTextTrim(tf))); |
443 | disposeFrame(tf); |
444 | } catch e { |
445 | enableSubmitButton(getFrame(tf)); |
446 | messageBox(e); |
447 | } |
448 | } |
449 | false; |
450 | }); |
451 | endifdef |
452 | } |
453 | |
454 | void showFullScreen() { |
455 | showFullScreenImageSurface(getImage()); |
456 | } |
457 | |
458 | void zoomIn(double f) { setZoom(getZoomX()*f, getZoomY()*f); } |
459 | void zoomOut(double f) { setZoom(getZoomX()/f, getZoomY()/f); } |
460 | |
461 | selfType setFile(File f) { file = f; this; } |
462 | |
463 | void setOverlay(IVF1<Graphics2D> overlay) { |
464 | this.overlay = overlay; |
465 | } |
466 | |
467 | bool hasImage() { ret image != null; } |
468 | public int w() { ret image.getWidth(); } |
469 | public int h() { ret image.getHeight(); } |
470 | |
471 | void setPixelated aka pixelate(bool b) { |
472 | assertTrue(b); |
473 | imageSurface_pixelated(this); |
474 | } |
475 | |
476 | selfType setAutoZoomToDisplay aka autoZoomToDisplay(bool b) { |
477 | if (autoZoomToDisplay = b) |
478 | zoomToDisplaySize(); |
479 | this; |
480 | } |
481 | |
482 | void quickRepaint() { |
483 | if (repaintInThread) |
484 | paintImmediately(0, 0, getWidth(), getHeight()); |
485 | else |
486 | repaint(); |
487 | } |
488 | |
489 | void setTool(ImageSurfaceMouseHandler tool) swing { |
490 | removeAllTools(); |
491 | addTool(tool); |
492 | } |
493 | |
494 | bool hasTool(AutoCloseable tool) { |
495 | ret swing(-> tools.contains(tool)); |
496 | } |
497 | |
498 | void addTool(ImageSurfaceMouseHandler tool) swing { |
499 | if (!tools.contains(tool)) |
500 | tool.register(this); |
501 | } |
502 | |
503 | void removeTool(AutoCloseable tool) swing { |
504 | if (tools.contains(tool)) { |
505 | close(tool); |
506 | tools.remove(tool); |
507 | } |
508 | } |
509 | |
510 | void removeAllTools aka clearTools() { |
511 | closeAllAndClear(tools); |
512 | } |
513 | |
514 | void performAutoZoom { |
515 | if (autoZoomToDisplay) zoomToDisplaySize(); |
516 | } |
517 | |
518 | void revalidateMe() { |
519 | revalidateIncludingFullCenterContainer(this); |
520 | } |
521 | |
522 | void addOverlay(G2Drawable overlay) { |
523 | overlays.add(overlay); |
524 | repaint(); |
525 | } |
526 | |
527 | void clearOverlays() { |
528 | if (nempty(overlays)) { |
529 | overlays.clear(); |
530 | repaint(); |
531 | } |
532 | } |
533 | |
534 | void setOverlay(G2Drawable overlay) { |
535 | clearOverlays(); |
536 | if (overlay != null) addOverlay(overlay); |
537 | } |
538 | |
539 | void loadImage(File f) { |
540 | setImage(loadImage2(f)); |
541 | } |
542 | |
543 | JComponent visualize() { |
544 | ret jscroll_center_borderless(this); |
545 | } |
546 | |
547 | void standardZoom() { |
548 | setZoom(1.0); |
549 | } |
550 | |
551 | !include #1034645 // print |
552 | } // end of ImageSurface |
553 | |
554 | // static function allows garbage collection |
555 | static VF2<ImageSurface, JPopupMenu> ImageSurface_popupMenuMaker() { |
556 | ret voidfunc(ImageSurface is, JPopupMenu menu) { |
557 | Point p = is.pointFromEvent(componentPopupMenu_mouseEvent.get()).getPoint(); |
558 | is.fillPopupMenu(menu, p); |
559 | }; |
560 | } |
download show line numbers debug dex old transpilations
Travelled to 20 computer(s): aoiabmzegqzx, bhatertpkbcr, cbybwowwnfue, cfunsshuasjs, ddnzoavkxhuk, gwrvuhgaqvyk, irmadwmeruwu, ishqpsrjomds, jtubtzbbkimh, lpdgvwnxivlt, mqqgnosmbjvj, onxytkatvevr, ppjhyzlbdabe, pyentgdyhuwx, pzhvpgtvlbxg, sawdedvomwva, tslmcundralx, tvejysmllsmz, vouqrxazstgt, xrpafgyirdlv
No comments. add comment
Snippet ID: | #1004553 |
Snippet name: | ImageSurface |
Eternal ID of this version: | #1004553/166 |
Text MD5: | a8fda1b0ec37bb45eada4c195f7d096e |
Transpilation MD5: | 3a0e63e26f98a8ed746c71124d8ce927 |
Author: | stefan |
Category: | javax / gui |
Type: | JavaX fragment (include) |
Public (visible to everyone): | Yes |
Archived (hidden from active list): | No |
Created/modified: | 2024-08-18 17:16:29 |
Source code size: | 16313 bytes / 560 lines |
Pitched / IR pitched: | No / No |
Views / Downloads: | 1511 / 7681 |
Version history: | 165 change(s) |
Referenced in: | [show references] |