Commit | Line | Data |
---|---|---|
333f0e7b NR |
1 | package be.nikiroo.fanfix.reader; |
2 | ||
3 | import java.awt.BorderLayout; | |
4 | import java.awt.Color; | |
d3c84ac3 | 5 | import java.awt.Graphics; |
333f0e7b | 6 | import java.awt.Graphics2D; |
b4dc6ab5 | 7 | import java.awt.Polygon; |
d3c84ac3 | 8 | import java.awt.Rectangle; |
333f0e7b NR |
9 | import java.awt.event.MouseEvent; |
10 | import java.awt.event.MouseListener; | |
11 | import java.awt.image.BufferedImage; | |
57f02339 NR |
12 | import java.io.ByteArrayInputStream; |
13 | import java.io.ByteArrayOutputStream; | |
14 | import java.io.IOException; | |
15 | import java.io.InputStream; | |
16 | import java.net.MalformedURLException; | |
333f0e7b NR |
17 | import java.util.ArrayList; |
18 | import java.util.Date; | |
19 | import java.util.EventListener; | |
20 | import java.util.List; | |
21 | ||
57f02339 | 22 | import javax.imageio.ImageIO; |
333f0e7b NR |
23 | import javax.swing.ImageIcon; |
24 | import javax.swing.JLabel; | |
25 | import javax.swing.JPanel; | |
26 | ||
57f02339 | 27 | import be.nikiroo.fanfix.Instance; |
333f0e7b | 28 | import be.nikiroo.fanfix.data.MetaData; |
4d205683 | 29 | import be.nikiroo.fanfix.data.Story; |
326093dc | 30 | import be.nikiroo.utils.ImageUtils; |
c39a3e30 | 31 | import be.nikiroo.utils.ui.UIUtils; |
333f0e7b NR |
32 | |
33 | /** | |
5dd985cf | 34 | * A book item presented in a {@link GuiReaderFrame}. |
333f0e7b NR |
35 | * |
36 | * @author niki | |
37 | */ | |
5dd985cf | 38 | class GuiReaderBook extends JPanel { |
333f0e7b NR |
39 | /** |
40 | * Action on a book item. | |
41 | * | |
42 | * @author niki | |
43 | */ | |
92fb0719 | 44 | interface BookActionListener extends EventListener { |
333f0e7b NR |
45 | /** |
46 | * The book was selected (single click). | |
47 | * | |
48 | * @param book | |
5dd985cf | 49 | * the {@link GuiReaderBook} itself |
333f0e7b | 50 | */ |
5dd985cf | 51 | public void select(GuiReaderBook book); |
333f0e7b NR |
52 | |
53 | /** | |
54 | * The book was double-clicked. | |
55 | * | |
56 | * @param book | |
5dd985cf | 57 | * the {@link GuiReaderBook} itself |
333f0e7b | 58 | */ |
5dd985cf | 59 | public void action(GuiReaderBook book); |
9843a5e5 NR |
60 | |
61 | /** | |
5dd985cf | 62 | * A popup menu was requested for this {@link GuiReaderBook}. |
9843a5e5 NR |
63 | * |
64 | * @param book | |
5dd985cf | 65 | * the {@link GuiReaderBook} itself |
9843a5e5 NR |
66 | * @param e |
67 | * the {@link MouseEvent} that generated this call | |
68 | */ | |
5dd985cf | 69 | public void popupRequested(GuiReaderBook book, MouseEvent e); |
333f0e7b NR |
70 | } |
71 | ||
4d205683 NR |
72 | private static final long serialVersionUID = 1L; |
73 | ||
74 | // TODO: export some of the configuration options? | |
b4dc6ab5 NR |
75 | private static final int COVER_WIDTH = 100; |
76 | private static final int COVER_HEIGHT = 150; | |
77 | private static final int SPINE_WIDTH = 5; | |
78 | private static final int SPINE_HEIGHT = 5; | |
79 | private static final int HOFFSET = 20; | |
80 | private static final Color SPINE_COLOR_BOTTOM = new Color(180, 180, 180); | |
81 | private static final Color SPINE_COLOR_RIGHT = new Color(100, 100, 100); | |
82 | private static final int TEXT_WIDTH = COVER_WIDTH + 40; | |
83 | private static final int TEXT_HEIGHT = 50; | |
84 | private static final String AUTHOR_COLOR = "#888888"; | |
4d205683 NR |
85 | private static final Color BORDER = Color.black; |
86 | private static final long doubleClickDelay = 200; // in ms | |
87 | // | |
3b2b638f | 88 | |
333f0e7b | 89 | private JLabel icon; |
3b2b638f | 90 | private JLabel title; |
333f0e7b NR |
91 | private boolean selected; |
92 | private boolean hovered; | |
93 | private Date lastClick; | |
4d205683 | 94 | |
92fb0719 | 95 | private List<BookActionListener> listeners; |
e42573a0 | 96 | private Reader reader; |
22848428 | 97 | private MetaData meta; |
10d558d2 NR |
98 | private boolean cached; |
99 | ||
4d205683 | 100 | /** |
5dd985cf | 101 | * Create a new {@link GuiReaderBook} item for the given {@link Story}. |
4d205683 | 102 | * |
e42573a0 NR |
103 | * @param reader |
104 | * the associated reader | |
4d205683 | 105 | * @param meta |
14b57448 | 106 | * the story {@link MetaData} or source (if no LUID) |
4d205683 NR |
107 | * @param cached |
108 | * TRUE if it is locally cached | |
5dd985cf | 109 | * @param seeWordCount |
793f1071 | 110 | * TRUE to see word counts, FALSE to see authors |
4d205683 | 111 | */ |
e42573a0 NR |
112 | public GuiReaderBook(Reader reader, MetaData meta, boolean cached, |
113 | boolean seeWordCount) { | |
114 | this.reader = reader; | |
10d558d2 | 115 | this.cached = cached; |
22848428 | 116 | this.meta = meta; |
333f0e7b | 117 | |
793f1071 NR |
118 | String optSecondary = meta.getAuthor(); |
119 | if (seeWordCount) { | |
120 | if (meta.getWords() >= 4000) { | |
121 | optSecondary = (meta.getWords() / 1000) + "k words"; | |
122 | } else if (meta.getWords() > 0) { | |
123 | optSecondary = meta.getWords() + " words"; | |
124 | } else { | |
a3641e4b | 125 | optSecondary = ""; |
793f1071 NR |
126 | } |
127 | } | |
128 | ||
129 | if (optSecondary != null && !optSecondary.isEmpty()) { | |
130 | optSecondary = "(" + optSecondary + ")"; | |
14b57448 NR |
131 | } else { |
132 | optSecondary = ""; | |
b4dc6ab5 | 133 | } |
4d205683 | 134 | |
57f02339 | 135 | icon = new JLabel(generateCoverIcon(meta)); |
3b2b638f | 136 | title = new JLabel( |
b4dc6ab5 NR |
137 | String.format( |
138 | "<html>" | |
139 | + "<body style='width: %d px; height: %d px; text-align: center'>" | |
140 | + "%s" + "<br>" + "<span style='color: %s;'>" | |
141 | + "%s" + "</span>" + "</body>" + "</html>", | |
142 | TEXT_WIDTH, TEXT_HEIGHT, meta.getTitle(), AUTHOR_COLOR, | |
793f1071 | 143 | optSecondary)); |
b4dc6ab5 | 144 | |
edd46289 NR |
145 | setLayout(new BorderLayout(10, 10)); |
146 | add(icon, BorderLayout.CENTER); | |
147 | add(title, BorderLayout.SOUTH); | |
333f0e7b NR |
148 | |
149 | setupListeners(); | |
333f0e7b NR |
150 | } |
151 | ||
152 | /** | |
153 | * The book current selection state. | |
154 | * | |
4d205683 | 155 | * @return the selection state |
333f0e7b NR |
156 | */ |
157 | public boolean isSelected() { | |
158 | return selected; | |
159 | } | |
160 | ||
161 | /** | |
162 | * The book current selection state. | |
163 | * | |
164 | * @param selected | |
4d205683 | 165 | * TRUE if it is selected |
333f0e7b NR |
166 | */ |
167 | public void setSelected(boolean selected) { | |
edd46289 NR |
168 | if (this.selected != selected) { |
169 | this.selected = selected; | |
170 | repaint(); | |
171 | } | |
333f0e7b NR |
172 | } |
173 | ||
4d205683 NR |
174 | /** |
175 | * The item mouse-hover state. | |
176 | * | |
177 | * @param hovered | |
178 | * TRUE if it is mouse-hovered | |
179 | */ | |
333f0e7b | 180 | private void setHovered(boolean hovered) { |
edd46289 NR |
181 | if (this.hovered != hovered) { |
182 | this.hovered = hovered; | |
183 | repaint(); | |
184 | } | |
333f0e7b NR |
185 | } |
186 | ||
4d205683 NR |
187 | /** |
188 | * Setup the mouse listener that will activate {@link BookActionListener} | |
189 | * events. | |
190 | */ | |
333f0e7b | 191 | private void setupListeners() { |
5dd985cf | 192 | listeners = new ArrayList<GuiReaderBook.BookActionListener>(); |
333f0e7b | 193 | addMouseListener(new MouseListener() { |
211f7ddb | 194 | @Override |
333f0e7b | 195 | public void mouseReleased(MouseEvent e) { |
9843a5e5 NR |
196 | if (e.isPopupTrigger()) { |
197 | popup(e); | |
198 | } | |
333f0e7b NR |
199 | } |
200 | ||
211f7ddb | 201 | @Override |
333f0e7b | 202 | public void mousePressed(MouseEvent e) { |
9843a5e5 NR |
203 | if (e.isPopupTrigger()) { |
204 | popup(e); | |
205 | } | |
333f0e7b NR |
206 | } |
207 | ||
211f7ddb | 208 | @Override |
333f0e7b NR |
209 | public void mouseExited(MouseEvent e) { |
210 | setHovered(false); | |
211 | } | |
212 | ||
211f7ddb | 213 | @Override |
333f0e7b NR |
214 | public void mouseEntered(MouseEvent e) { |
215 | setHovered(true); | |
216 | } | |
217 | ||
211f7ddb | 218 | @Override |
333f0e7b | 219 | public void mouseClicked(MouseEvent e) { |
3b2b638f NR |
220 | if (isEnabled()) { |
221 | Date now = new Date(); | |
222 | if (lastClick != null | |
223 | && now.getTime() - lastClick.getTime() < doubleClickDelay) { | |
224 | click(true); | |
225 | } else { | |
226 | click(false); | |
227 | } | |
9843a5e5 | 228 | |
3b2b638f | 229 | lastClick = now; |
333f0e7b | 230 | } |
333f0e7b | 231 | } |
333f0e7b | 232 | |
9843a5e5 NR |
233 | private void click(boolean doubleClick) { |
234 | for (BookActionListener listener : listeners) { | |
235 | if (doubleClick) { | |
5dd985cf | 236 | listener.action(GuiReaderBook.this); |
9843a5e5 | 237 | } else { |
5dd985cf | 238 | listener.select(GuiReaderBook.this); |
9843a5e5 NR |
239 | } |
240 | } | |
333f0e7b | 241 | } |
9843a5e5 NR |
242 | |
243 | private void popup(MouseEvent e) { | |
244 | for (BookActionListener listener : listeners) { | |
5dd985cf NR |
245 | listener.select((GuiReaderBook.this)); |
246 | listener.popupRequested(GuiReaderBook.this, e); | |
9843a5e5 NR |
247 | } |
248 | } | |
249 | }); | |
333f0e7b NR |
250 | } |
251 | ||
4d205683 NR |
252 | /** |
253 | * Add a new {@link BookActionListener} on this item. | |
254 | * | |
255 | * @param listener | |
256 | * the listener | |
257 | */ | |
92fb0719 | 258 | public void addActionListener(BookActionListener listener) { |
333f0e7b NR |
259 | listeners.add(listener); |
260 | } | |
d3c84ac3 | 261 | |
4d205683 | 262 | /** |
5dd985cf | 263 | * The Library {@link MetaData} of the book represented by this item. |
4d205683 | 264 | * |
22848428 | 265 | * @return the meta |
4d205683 | 266 | */ |
22848428 NR |
267 | public MetaData getMeta() { |
268 | return meta; | |
10d558d2 NR |
269 | } |
270 | ||
271 | /** | |
5dd985cf | 272 | * This item {@link GuiReader} library cache state. |
10d558d2 | 273 | * |
5dd985cf | 274 | * @return TRUE if it is present in the {@link GuiReader} cache |
10d558d2 NR |
275 | */ |
276 | public boolean isCached() { | |
277 | return cached; | |
278 | } | |
279 | ||
280 | /** | |
5dd985cf | 281 | * This item {@link GuiReader} library cache state. |
10d558d2 NR |
282 | * |
283 | * @param cached | |
5dd985cf | 284 | * TRUE if it is present in the {@link GuiReader} cache |
10d558d2 NR |
285 | */ |
286 | public void setCached(boolean cached) { | |
f977d05b NR |
287 | if (this.cached != cached) { |
288 | this.cached = cached; | |
edd46289 | 289 | repaint(); |
f977d05b | 290 | } |
10d558d2 NR |
291 | } |
292 | ||
4d205683 | 293 | /** |
5dd985cf | 294 | * Paint the item, then call {@link GuiReaderBook#paintOverlay(Graphics)}. |
4d205683 | 295 | */ |
d3c84ac3 NR |
296 | @Override |
297 | public void paint(Graphics g) { | |
298 | super.paint(g); | |
edd46289 NR |
299 | paintOverlay(g); |
300 | } | |
d3c84ac3 | 301 | |
edd46289 NR |
302 | /** |
303 | * Draw a partially transparent overlay if needed depending upon the | |
304 | * selection and mouse-hover states on top of the normal component, as well | |
305 | * as a possible "cached" icon if the item is cached. | |
5dd985cf NR |
306 | * |
307 | * @param g | |
308 | * the {@link Graphics} to paint onto | |
edd46289 NR |
309 | */ |
310 | public void paintOverlay(Graphics g) { | |
4d205683 NR |
311 | Rectangle clip = g.getClipBounds(); |
312 | if (clip.getWidth() <= 0 || clip.getHeight() <= 0) { | |
313 | return; | |
314 | } | |
315 | ||
b4dc6ab5 NR |
316 | int h = COVER_HEIGHT; |
317 | int w = COVER_WIDTH; | |
4d205683 NR |
318 | int xOffset = (TEXT_WIDTH - COVER_WIDTH) - 1; |
319 | int yOffset = HOFFSET; | |
320 | ||
321 | if (BORDER != null) { | |
322 | if (BORDER != null) { | |
323 | g.setColor(BORDER); | |
324 | g.drawRect(xOffset, yOffset, COVER_WIDTH, COVER_HEIGHT); | |
325 | } | |
326 | ||
327 | xOffset++; | |
328 | yOffset++; | |
329 | } | |
b4dc6ab5 NR |
330 | |
331 | int[] xs = new int[] { xOffset, xOffset + SPINE_WIDTH, | |
332 | xOffset + w + SPINE_WIDTH, xOffset + w }; | |
4d205683 NR |
333 | int[] ys = new int[] { yOffset + h, yOffset + h + SPINE_HEIGHT, |
334 | yOffset + h + SPINE_HEIGHT, yOffset + h }; | |
b4dc6ab5 NR |
335 | g.setColor(SPINE_COLOR_BOTTOM); |
336 | g.fillPolygon(new Polygon(xs, ys, xs.length)); | |
337 | xs = new int[] { xOffset + w, xOffset + w + SPINE_WIDTH, | |
338 | xOffset + w + SPINE_WIDTH, xOffset + w }; | |
4d205683 NR |
339 | ys = new int[] { yOffset, yOffset + SPINE_HEIGHT, |
340 | yOffset + h + SPINE_HEIGHT, yOffset + h }; | |
b4dc6ab5 NR |
341 | g.setColor(SPINE_COLOR_RIGHT); |
342 | g.fillPolygon(new Polygon(xs, ys, xs.length)); | |
343 | ||
d3c84ac3 | 344 | Color color = new Color(255, 255, 255, 0); |
3b2b638f NR |
345 | if (!isEnabled()) { |
346 | } else if (selected && !hovered) { | |
d3c84ac3 NR |
347 | color = new Color(80, 80, 100, 40); |
348 | } else if (!selected && hovered) { | |
349 | color = new Color(230, 230, 255, 100); | |
350 | } else if (selected && hovered) { | |
351 | color = new Color(200, 200, 255, 100); | |
352 | } | |
353 | ||
4d205683 NR |
354 | g.setColor(color); |
355 | g.fillRect(clip.x, clip.y, clip.width, clip.height); | |
356 | ||
10d558d2 | 357 | if (cached) { |
c39a3e30 NR |
358 | UIUtils.drawEllipse3D(g, Color.green.darker(), COVER_WIDTH |
359 | + HOFFSET + 30, 10, 20, 20); | |
10d558d2 | 360 | } |
d3c84ac3 | 361 | } |
edd46289 NR |
362 | |
363 | /** | |
57f02339 | 364 | * Generate a cover icon based upon the given {@link MetaData}. |
edd46289 | 365 | * |
57f02339 | 366 | * @param meta |
14b57448 NR |
367 | * the {@link MetaData} about the target {@link Story} or source |
368 | * (if no LUID) | |
edd46289 NR |
369 | * |
370 | * @return the icon | |
371 | */ | |
57f02339 | 372 | private ImageIcon generateCoverIcon(MetaData meta) { |
57f02339 | 373 | BufferedImage resizedImage = null; |
14b57448 NR |
374 | String id = null; |
375 | ||
376 | if (meta.getLuid() != null) { | |
377 | id = meta.getUuid() + ".thumb_" + SPINE_WIDTH + "x" + COVER_WIDTH | |
378 | + "+" + SPINE_HEIGHT + "+" + COVER_HEIGHT + "@" + HOFFSET; | |
379 | InputStream in = Instance.getCache().getFromCache(id); | |
380 | if (in != null) { | |
381 | try { | |
382 | resizedImage = ImageUtils.fromStream(in); | |
383 | in.close(); | |
384 | in = null; | |
385 | } catch (IOException e) { | |
386 | Instance.syserr(e); | |
387 | } | |
57f02339 NR |
388 | } |
389 | } | |
390 | ||
391 | if (resizedImage == null) { | |
392 | try { | |
14b57448 NR |
393 | BufferedImage cover = null; |
394 | if (meta.getLuid() == null) { | |
395 | cover = reader.getLibrary() | |
396 | .getSourceCover(meta.getSource()); | |
397 | } else { | |
398 | cover = reader.getLibrary().getCover(meta.getLuid()); | |
399 | } | |
57f02339 NR |
400 | |
401 | resizedImage = new BufferedImage(SPINE_WIDTH + COVER_WIDTH, | |
402 | SPINE_HEIGHT + COVER_HEIGHT + HOFFSET, | |
403 | BufferedImage.TYPE_4BYTE_ABGR); | |
404 | Graphics2D g = resizedImage.createGraphics(); | |
405 | g.setColor(Color.white); | |
406 | g.fillRect(0, HOFFSET, COVER_WIDTH, COVER_HEIGHT); | |
407 | if (cover != null) { | |
408 | g.drawImage(cover, 0, HOFFSET, COVER_WIDTH, COVER_HEIGHT, | |
409 | null); | |
410 | } else { | |
411 | g.setColor(Color.black); | |
412 | g.drawLine(0, HOFFSET, COVER_WIDTH, HOFFSET + COVER_HEIGHT); | |
413 | g.drawLine(COVER_WIDTH, HOFFSET, 0, HOFFSET + COVER_HEIGHT); | |
414 | } | |
415 | g.dispose(); | |
416 | ||
14b57448 NR |
417 | if (id != null) { |
418 | ByteArrayOutputStream out = new ByteArrayOutputStream(); | |
419 | ImageIO.write(resizedImage, "png", out); | |
420 | byte[] imageBytes = out.toByteArray(); | |
421 | InputStream in = new ByteArrayInputStream(imageBytes); | |
422 | Instance.getCache().addToCache(in, id); | |
423 | in.close(); | |
424 | in = null; | |
425 | } | |
57f02339 NR |
426 | } catch (MalformedURLException e) { |
427 | Instance.syserr(e); | |
428 | } catch (IOException e) { | |
429 | Instance.syserr(e); | |
430 | } | |
edd46289 | 431 | } |
edd46289 NR |
432 | |
433 | return new ImageIcon(resizedImage); | |
434 | } | |
333f0e7b | 435 | } |