Fix cover not deleted, add new UI option "Move to"
[fanfix.git] / src / be / nikiroo / fanfix / reader / LocalReaderFrame.java
index 966869f502c8f9a65d9fe2ef6bd55ee78d698bad..03df7fbdf5cf133d86ac5d11634392e7edb61632 100644 (file)
@@ -2,46 +2,81 @@ package be.nikiroo.fanfix.reader;
 
 import java.awt.BorderLayout;
 import java.awt.Color;
-import java.awt.Desktop;
+import java.awt.Frame;
+import java.awt.Toolkit;
+import java.awt.datatransfer.DataFlavor;
 import java.awt.event.ActionEvent;
 import java.awt.event.ActionListener;
 import java.awt.event.KeyEvent;
+import java.awt.event.MouseEvent;
 import java.awt.event.WindowEvent;
 import java.io.File;
 import java.io.IOException;
+import java.net.URL;
 import java.util.ArrayList;
+import java.util.HashMap;
 import java.util.List;
+import java.util.Map;
+import java.util.Map.Entry;
 
+import javax.swing.BoxLayout;
+import javax.swing.JFileChooser;
 import javax.swing.JFrame;
 import javax.swing.JMenu;
 import javax.swing.JMenuBar;
 import javax.swing.JMenuItem;
 import javax.swing.JOptionPane;
 import javax.swing.JPanel;
+import javax.swing.JPopupMenu;
 import javax.swing.JScrollPane;
 import javax.swing.SwingUtilities;
+import javax.swing.filechooser.FileFilter;
+import javax.swing.filechooser.FileNameExtensionFilter;
 
 import be.nikiroo.fanfix.Instance;
+import be.nikiroo.fanfix.Library;
+import be.nikiroo.fanfix.bundles.Config;
 import be.nikiroo.fanfix.bundles.UiConfig;
 import be.nikiroo.fanfix.data.MetaData;
+import be.nikiroo.fanfix.data.Story;
+import be.nikiroo.fanfix.output.BasicOutput.OutputType;
 import be.nikiroo.fanfix.reader.LocalReaderBook.BookActionListener;
 import be.nikiroo.utils.Progress;
+import be.nikiroo.utils.Version;
+import be.nikiroo.utils.ui.ConfigEditor;
 import be.nikiroo.utils.ui.ProgressBar;
-import be.nikiroo.utils.ui.WrapLayout;
 
+/**
+ * A {@link Frame} that will show a {@link LocalReaderBook} item for each
+ * {@link Story} in the main cache ({@link Instance#getCache()}), and offer a
+ * way to copy them to the {@link LocalReader} cache ({@link LocalReader#lib}),
+ * read them, delete them...
+ * 
+ * @author niki
+ */
 class LocalReaderFrame extends JFrame {
        private static final long serialVersionUID = 1L;
        private LocalReader reader;
-       private List<MetaData> stories;
-       private List<LocalReaderBook> books;
-       private JPanel bookPane;
-       private String type;
+       private Map<LocalReaderGroup, String> booksByType;
+       private Map<LocalReaderGroup, String> booksByAuthor;
+       private JPanel pane;
        private Color color;
        private ProgressBar pgBar;
        private JMenuBar bar;
-
+       private LocalReaderBook selectedBook;
+       private boolean words; // words or authors (secondary info on books)
+
+       /**
+        * Create a new {@link LocalReaderFrame}.
+        * 
+        * @param reader
+        *            the associated {@link LocalReader} to forward some commands
+        *            and access its {@link Library}
+        * @param type
+        *            the type of {@link Story} to load, or NULL for all types
+        */
        public LocalReaderFrame(LocalReader reader, String type) {
-               super("Fanfix Library");
+               super(String.format("Fanfix %s Library", Version.getCurrentVersion()));
 
                this.reader = reader;
 
@@ -49,111 +84,442 @@ class LocalReaderFrame extends JFrame {
                setSize(800, 600);
                setLayout(new BorderLayout());
 
-               books = new ArrayList<LocalReaderBook>();
-               bookPane = new JPanel(new WrapLayout(WrapLayout.LEADING, 5, 5));
+               pane = new JPanel();
+               pane.setLayout(new BoxLayout(pane, BoxLayout.PAGE_AXIS));
 
                color = Instance.getUiConfig().getColor(UiConfig.BACKGROUND_COLOR);
-
                if (color != null) {
                        setBackground(color);
-                       bookPane.setBackground(color);
+                       pane.setBackground(color);
                }
 
-               JScrollPane scroll = new JScrollPane(bookPane);
+               JScrollPane scroll = new JScrollPane(pane);
                scroll.getVerticalScrollBar().setUnitIncrement(16);
                add(scroll, BorderLayout.CENTER);
 
                pgBar = new ProgressBar();
                add(pgBar, BorderLayout.SOUTH);
 
-               refreshBooks(type);
-               setJMenuBar(createMenu());
+               pgBar.addActionListener(new ActionListener() {
+                       public void actionPerformed(ActionEvent e) {
+                               invalidate();
+                               pgBar.setProgress(null);
+                               validate();
+                               setEnabled(true);
+                       }
+               });
 
-               setVisible(true);
-       }
+               pgBar.addUpdateListener(new ActionListener() {
+                       public void actionPerformed(ActionEvent e) {
+                               invalidate();
+                               validate();
+                               repaint();
+                       }
+               });
+
+               booksByType = new HashMap<LocalReaderGroup, String>();
+               booksByAuthor = new HashMap<LocalReaderGroup, String>();
 
-       private void refreshBooks(String type) {
-               this.type = type;
-               stories = Instance.getLibrary().getList(type);
-               books.clear();
-               bookPane.removeAll();
-               for (MetaData meta : stories) {
-                       LocalReaderBook book = new LocalReaderBook(meta);
-                       if (color != null) {
-                               book.setBackground(color);
+               pane.setVisible(false);
+               final Progress pg = new Progress();
+               final String typeF = type;
+               outOfUi(pg, new Runnable() {
+                       public void run() {
+                               Instance.getLibrary().refresh(pg);
+                               invalidate();
+                               setJMenuBar(createMenu());
+                               addBookPane(typeF, true);
+                               refreshBooks();
+                               validate();
+                               pane.setVisible(true);
                        }
+               });
+
+               setVisible(true);
+       }
 
-                       books.add(book);
-                       final String luid = meta.getLuid();
-                       book.addActionListener(new BookActionListener() {
-                               public void select(LocalReaderBook book) {
-                                       for (LocalReaderBook abook : books) {
-                                               abook.setSelected(abook == book);
+       /**
+        * Add a new {@link LocalReaderGroup} on the frame to display the books of
+        * the selected type or author.
+        * 
+        * @param value
+        *            the author or the type, or NULL to get all the
+        *            authors-or-types
+        * @param type
+        *            TRUE for type, FALSE for author
+        */
+       private void addBookPane(String value, boolean type) {
+               if (value == null) {
+                       if (type) {
+                               for (String tt : Instance.getLibrary().getTypes()) {
+                                       if (tt != null) {
+                                               addBookPane(tt, type);
                                        }
                                }
-
-                               public void action(LocalReaderBook book) {
-                                       final Progress pg = new Progress();
-                                       outOfUi(pg, new Runnable() {
-                                               public void run() {
-                                                       try {
-                                                               File target = LocalReaderFrame.this.reader
-                                                                               .getTarget(luid, pg);
-                                                               Desktop.getDesktop().browse(target.toURI());
-                                                       } catch (IOException e) {
-                                                               Instance.syserr(e);
-                                                       }
-                                               }
-                                       });
+                       } else {
+                               for (String tt : Instance.getLibrary().getAuthors()) {
+                                       if (tt != null) {
+                                               addBookPane(tt, type);
+                                       }
                                }
-                       });
+                       }
 
-                       bookPane.add(book);
+                       return;
                }
 
-               bookPane.validate();
-               bookPane.repaint();
+               LocalReaderGroup bookPane = new LocalReaderGroup(reader, value, color);
+               if (type) {
+                       booksByType.put(bookPane, value);
+               } else {
+                       booksByAuthor.put(bookPane, value);
+               }
+
+               this.invalidate();
+               pane.invalidate();
+               pane.add(bookPane);
+               pane.validate();
+               this.validate();
+
+               bookPane.setActionListener(new BookActionListener() {
+                       public void select(LocalReaderBook book) {
+                               selectedBook = book;
+                       }
+
+                       public void popupRequested(LocalReaderBook book, MouseEvent e) {
+                               JPopupMenu popup = new JPopupMenu();
+                               popup.add(createMenuItemOpenBook());
+                               popup.addSeparator();
+                               popup.add(createMenuItemExport());
+                               popup.add(createMenuItemMove());
+                               popup.add(createMenuItemClearCache());
+                               popup.add(createMenuItemRedownload());
+                               popup.addSeparator();
+                               popup.add(createMenuItemDelete());
+                               popup.show(e.getComponent(), e.getX(), e.getY());
+                       }
+
+                       public void action(final LocalReaderBook book) {
+                               openBook(book);
+                       }
+               });
+       }
+
+       private void removeBookPanes() {
+               booksByType.clear();
+               booksByAuthor.clear();
+               pane.invalidate();
+               this.invalidate();
+               pane.removeAll();
+               pane.validate();
+               this.validate();
        }
 
+       /**
+        * Refresh the list of {@link LocalReaderBook}s from disk.
+        * 
+        */
+       private void refreshBooks() {
+               for (LocalReaderGroup group : booksByType.keySet()) {
+                       List<MetaData> stories = Instance.getLibrary().getListByType(
+                                       booksByType.get(group));
+                       group.refreshBooks(stories, words);
+               }
+
+               for (LocalReaderGroup group : booksByAuthor.keySet()) {
+                       List<MetaData> stories = Instance.getLibrary().getListByAuthor(
+                                       booksByAuthor.get(group));
+                       group.refreshBooks(stories, words);
+               }
+
+               pane.repaint();
+               this.repaint();
+       }
+
+       /**
+        * Create the main menu bar.
+        * 
+        * @return the bar
+        */
        private JMenuBar createMenu() {
                bar = new JMenuBar();
 
                JMenu file = new JMenu("File");
+               file.setMnemonic(KeyEvent.VK_F);
 
-               JMenuItem imprt = new JMenuItem("Import", KeyEvent.VK_I);
+               JMenuItem imprt = new JMenuItem("Import URL...", KeyEvent.VK_U);
                imprt.addActionListener(new ActionListener() {
                        public void actionPerformed(ActionEvent e) {
-                               final String url = JOptionPane.showInputDialog(
-                                               LocalReaderFrame.this, "url of the story to import?",
-                                               "Importing from URL", JOptionPane.QUESTION_MESSAGE);
-                               if (url != null && !url.isEmpty()) {
-                                       final Progress pg = new Progress("Importing " + url);
-                                       outOfUi(pg, new Runnable() {
-                                               public void run() {
-                                                       Exception ex = null;
-                                                       try {
-                                                               Instance.getLibrary().imprt(
-                                                                               BasicReader.getUrl(url), pg);
-                                                       } catch (IOException e) {
-                                                               ex = e;
+                               imprt(true);
+                       }
+               });
+               JMenuItem imprtF = new JMenuItem("Import File...", KeyEvent.VK_F);
+               imprtF.addActionListener(new ActionListener() {
+                       public void actionPerformed(ActionEvent e) {
+                               imprt(false);
+                       }
+               });
+               JMenuItem exit = new JMenuItem("Exit", KeyEvent.VK_X);
+               exit.addActionListener(new ActionListener() {
+                       public void actionPerformed(ActionEvent e) {
+                               LocalReaderFrame.this.dispatchEvent(new WindowEvent(
+                                               LocalReaderFrame.this, WindowEvent.WINDOW_CLOSING));
+                       }
+               });
+
+               file.add(createMenuItemOpenBook());
+               file.add(createMenuItemExport());
+               file.add(createMenuItemMove());
+               file.addSeparator();
+               file.add(imprt);
+               file.add(imprtF);
+               file.addSeparator();
+               file.add(exit);
+
+               bar.add(file);
+
+               JMenu edit = new JMenu("Edit");
+               edit.setMnemonic(KeyEvent.VK_E);
+
+               edit.add(createMenuItemClearCache());
+               edit.add(createMenuItemRedownload());
+               edit.addSeparator();
+               edit.add(createMenuItemDelete());
+
+               bar.add(edit);
+
+               JMenu view = new JMenu("View");
+               view.setMnemonic(KeyEvent.VK_V);
+               JMenuItem vauthors = new JMenuItem("Author");
+               vauthors.setMnemonic(KeyEvent.VK_A);
+               vauthors.addActionListener(new ActionListener() {
+                       public void actionPerformed(ActionEvent e) {
+                               words = false;
+                               refreshBooks();
+                       }
+               });
+               view.add(vauthors);
+               JMenuItem vwords = new JMenuItem("Word count");
+               vwords.setMnemonic(KeyEvent.VK_W);
+               vwords.addActionListener(new ActionListener() {
+                       public void actionPerformed(ActionEvent e) {
+                               words = true;
+                               refreshBooks();
+                       }
+               });
+               view.add(vwords);
+               bar.add(view);
+
+               JMenu sources = new JMenu("Sources");
+               sources.setMnemonic(KeyEvent.VK_S);
+
+               List<String> tt = Instance.getLibrary().getTypes();
+               tt.add(0, null);
+               for (final String type : tt) {
+                       JMenuItem item = new JMenuItem(type == null ? "All" : type);
+                       item.addActionListener(new ActionListener() {
+                               public void actionPerformed(ActionEvent e) {
+                                       removeBookPanes();
+                                       addBookPane(type, true);
+                                       refreshBooks();
+                               }
+                       });
+                       sources.add(item);
+
+                       if (type == null) {
+                               sources.addSeparator();
+                       }
+               }
+
+               bar.add(sources);
+
+               JMenu authors = new JMenu("Authors");
+               authors.setMnemonic(KeyEvent.VK_A);
+
+               List<String> aa = Instance.getLibrary().getAuthors();
+               aa.add(0, null);
+               for (final String author : aa) {
+                       JMenuItem item = new JMenuItem(author == null ? "All"
+                                       : author.isEmpty() ? "[unknown]" : author);
+                       item.addActionListener(new ActionListener() {
+                               public void actionPerformed(ActionEvent e) {
+                                       removeBookPanes();
+                                       addBookPane(author, false);
+                                       refreshBooks();
+                               }
+                       });
+                       authors.add(item);
+
+                       if (author == null || author.isEmpty()) {
+                               authors.addSeparator();
+                       }
+               }
+
+               bar.add(authors);
+
+               JMenu options = new JMenu("Options");
+               options.setMnemonic(KeyEvent.VK_O);
+               options.add(createMenuItemConfig());
+               options.add(createMenuItemUiConfig());
+               bar.add(options);
+
+               return bar;
+       }
+
+       /**
+        * Create the Fanfix Configuration menu item.
+        * 
+        * @return the item
+        */
+       private JMenuItem createMenuItemConfig() {
+               final String title = "Fanfix Configuration";
+               JMenuItem item = new JMenuItem(title);
+               item.setMnemonic(KeyEvent.VK_F);
+
+               item.addActionListener(new ActionListener() {
+                       public void actionPerformed(ActionEvent e) {
+                               ConfigEditor<Config> ed = new ConfigEditor<Config>(
+                                               Config.class, Instance.getConfig(),
+                                               "This is where you configure the options of the program.");
+                               JFrame frame = new JFrame(title);
+                               frame.add(ed);
+                               frame.setSize(800, 600);
+                               frame.setVisible(true);
+                       }
+               });
+
+               return item;
+       }
+
+       /**
+        * Create the UI Configuration menu item.
+        * 
+        * @return the item
+        */
+       private JMenuItem createMenuItemUiConfig() {
+               final String title = "UI Configuration";
+               JMenuItem item = new JMenuItem(title);
+               item.setMnemonic(KeyEvent.VK_U);
+
+               item.addActionListener(new ActionListener() {
+                       public void actionPerformed(ActionEvent e) {
+                               ConfigEditor<UiConfig> ed = new ConfigEditor<UiConfig>(
+                                               UiConfig.class, Instance.getUiConfig(),
+                                               "This is where you configure the graphical appearence of the program.");
+                               JFrame frame = new JFrame(title);
+                               frame.add(ed);
+                               frame.setSize(800, 600);
+                               frame.setVisible(true);
+                       }
+               });
+
+               return item;
+       }
+
+       /**
+        * Create the export menu item.
+        * 
+        * @return the item
+        */
+       private JMenuItem createMenuItemExport() {
+               final JFileChooser fc = new JFileChooser();
+               fc.setAcceptAllFileFilterUsed(false);
+
+               final Map<FileFilter, OutputType> filters = new HashMap<FileFilter, OutputType>();
+               for (OutputType type : OutputType.values()) {
+                       String ext = type.getDefaultExtension(false);
+                       String desc = type.getDesc(false);
+
+                       if (ext == null || ext.isEmpty()) {
+                               filters.put(createAllFilter(desc), type);
+                       } else {
+                               filters.put(new FileNameExtensionFilter(desc, ext), type);
+                       }
+               }
+
+               // First the "ALL" filters, then, the extension filters
+               for (Entry<FileFilter, OutputType> entry : filters.entrySet()) {
+                       if (!(entry.getKey() instanceof FileNameExtensionFilter)) {
+                               fc.addChoosableFileFilter(entry.getKey());
+                       }
+               }
+               for (Entry<FileFilter, OutputType> entry : filters.entrySet()) {
+                       if (entry.getKey() instanceof FileNameExtensionFilter) {
+                               fc.addChoosableFileFilter(entry.getKey());
+                       }
+               }
+               //
+
+               JMenuItem export = new JMenuItem("Save as...", KeyEvent.VK_S);
+               export.addActionListener(new ActionListener() {
+                       public void actionPerformed(ActionEvent e) {
+                               if (selectedBook != null) {
+                                       fc.showDialog(LocalReaderFrame.this, "Save");
+                                       if (fc.getSelectedFile() != null) {
+                                               final OutputType type = filters.get(fc.getFileFilter());
+                                               final String path = fc.getSelectedFile()
+                                                               .getAbsolutePath()
+                                                               + type.getDefaultExtension(false);
+                                               final Progress pg = new Progress();
+                                               outOfUi(pg, new Runnable() {
+                                                       public void run() {
+                                                               try {
+                                                                       Instance.getLibrary().export(
+                                                                                       selectedBook.getMeta().getLuid(),
+                                                                                       type, path, pg);
+                                                               } catch (IOException e) {
+                                                                       Instance.syserr(e);
+                                                               }
                                                        }
+                                               });
+                                       }
+                               }
+                       }
+               });
+
+               return export;
+       }
 
-                                                       final Exception e = ex;
+       /**
+        * Create a {@link FileFilter} that accepts all files and return the given
+        * description.
+        * 
+        * @param desc
+        *            the description
+        * 
+        * @return the filter
+        */
+       private FileFilter createAllFilter(final String desc) {
+               return new FileFilter() {
+                       @Override
+                       public String getDescription() {
+                               return desc;
+                       }
+
+                       @Override
+                       public boolean accept(File f) {
+                               return true;
+                       }
+               };
+       }
 
-                                                       final boolean ok = (e == null);
+       /**
+        * Create the refresh (delete cache) menu item.
+        * 
+        * @return the item
+        */
+       private JMenuItem createMenuItemClearCache() {
+               JMenuItem refresh = new JMenuItem("Clear cache", KeyEvent.VK_C);
+               refresh.addActionListener(new ActionListener() {
+                       public void actionPerformed(ActionEvent e) {
+                               if (selectedBook != null) {
+                                       outOfUi(null, new Runnable() {
+                                               public void run() {
+                                                       reader.clearLocalReaderCache(selectedBook.getMeta()
+                                                                       .getLuid());
+                                                       selectedBook.setCached(false);
                                                        SwingUtilities.invokeLater(new Runnable() {
                                                                public void run() {
-                                                                       if (!ok) {
-                                                                               JOptionPane.showMessageDialog(
-                                                                                               LocalReaderFrame.this,
-                                                                                               "Cannot import: " + url,
-                                                                                               e.getMessage(),
-                                                                                               JOptionPane.ERROR_MESSAGE);
-
-                                                                               setAllEnabled(true);
-                                                                       } else {
-                                                                               refreshBooks(type);
-                                                                       }
+                                                                       selectedBook.repaint();
                                                                }
                                                        });
                                                }
@@ -161,73 +527,319 @@ class LocalReaderFrame extends JFrame {
                                }
                        }
                });
-               JMenu types = new JMenu("Type");
-               List<String> tt = Instance.getLibrary().getTypes();
-               tt.add(0, null);
-               for (final String type : tt) {
-                       JMenuItem item = new JMenuItem(type == null ? "[all]" : type);
+
+               return refresh;
+       }
+
+       /**
+        * Create the delete menu item.
+        * 
+        * @return the item
+        */
+       private JMenuItem createMenuItemMove() {
+               JMenu moveTo = new JMenu("Move to...");
+               moveTo.setMnemonic(KeyEvent.VK_M);
+
+               List<String> types = new ArrayList<String>();
+               types.add(null);
+               types.addAll(Instance.getLibrary().getTypes());
+
+               for (String type : types) {
+                       JMenuItem item = new JMenuItem(type == null ? "New type..." : type);
+
+                       moveTo.add(item);
+                       if (type == null) {
+                               moveTo.addSeparator();
+                       }
+
+                       final String ftype = type;
                        item.addActionListener(new ActionListener() {
                                public void actionPerformed(ActionEvent e) {
-                                       refreshBooks(type);
+                                       if (selectedBook != null) {
+                                               String type = ftype;
+                                               if (type == null) {
+                                                       Object rep = JOptionPane.showInputDialog(
+                                                                       LocalReaderFrame.this, "Move to:",
+                                                                       "Moving story",
+                                                                       JOptionPane.QUESTION_MESSAGE, null, null,
+                                                                       selectedBook.getMeta().getSource());
+                                                       if (rep == null) {
+                                                               return;
+                                                       } else {
+                                                               type = rep.toString();
+                                                       }
+                                               }
+
+                                               final String ftype = type;
+                                               outOfUi(null, new Runnable() {
+                                                       public void run() {
+                                                               reader.changeType(selectedBook.getMeta()
+                                                                               .getLuid(), ftype);
+
+                                                               selectedBook = null;
+
+                                                               SwingUtilities.invokeLater(new Runnable() {
+                                                                       public void run() {
+                                                                               setJMenuBar(createMenu());
+                                                                       }
+                                                               });
+                                                       }
+                                               });
+                                       }
                                }
                        });
-                       types.add(item);
                }
-               JMenuItem exit = new JMenuItem("Exit", KeyEvent.VK_X);
-               exit.addActionListener(new ActionListener() {
+
+               return moveTo;
+       }
+
+       /**
+        * Create the redownload (then delete original) menu item.
+        * 
+        * @return the item
+        */
+       private JMenuItem createMenuItemRedownload() {
+               JMenuItem refresh = new JMenuItem("Redownload", KeyEvent.VK_R);
+               refresh.addActionListener(new ActionListener() {
                        public void actionPerformed(ActionEvent e) {
-                               LocalReaderFrame.this.dispatchEvent(new WindowEvent(
-                                               LocalReaderFrame.this, WindowEvent.WINDOW_CLOSING));
+                               if (selectedBook != null) {
+                                       final MetaData meta = selectedBook.getMeta();
+                                       imprt(meta.getUrl(), new Runnable() {
+                                               public void run() {
+                                                       reader.delete(meta.getLuid());
+                                                       LocalReaderFrame.this.selectedBook = null;
+                                               }
+                                       }, "Removing old copy");
+                               }
                        }
                });
 
-               file.add(imprt);
-               file.add(types);
-               file.addSeparator();
-               file.add(exit);
+               return refresh;
+       }
 
-               bar.add(file);
+       /**
+        * Create the delete menu item.
+        * 
+        * @return the item
+        */
+       private JMenuItem createMenuItemDelete() {
+               JMenuItem delete = new JMenuItem("Delete", KeyEvent.VK_D);
+               delete.addActionListener(new ActionListener() {
+                       public void actionPerformed(ActionEvent e) {
+                               if (selectedBook != null) {
+                                       outOfUi(null, new Runnable() {
+                                               public void run() {
+                                                       reader.delete(selectedBook.getMeta().getLuid());
+                                                       selectedBook = null;
+                                               }
+                                       });
+                               }
+                       }
+               });
 
-               return bar;
+               return delete;
        }
 
-       private void outOfUi(final Progress pg, final Runnable run) {
-               pgBar.setProgress(pg);
+       /**
+        * Create the open menu item.
+        * 
+        * @return the item
+        */
+       private JMenuItem createMenuItemOpenBook() {
+               JMenuItem open = new JMenuItem("Open", KeyEvent.VK_O);
+               open.addActionListener(new ActionListener() {
+                       public void actionPerformed(ActionEvent e) {
+                               if (selectedBook != null) {
+                                       openBook(selectedBook);
+                               }
+                       }
+               });
 
-               SwingUtilities.invokeLater(new Runnable() {
+               return open;
+       }
+
+       /**
+        * Open a {@link LocalReaderBook} item.
+        * 
+        * @param book
+        *            the {@link LocalReaderBook} to open
+        */
+       private void openBook(final LocalReaderBook book) {
+               final Progress pg = new Progress();
+               outOfUi(pg, new Runnable() {
                        public void run() {
-                               setAllEnabled(false);
-                               pgBar.addActioListener(new ActionListener() {
-                                       public void actionPerformed(ActionEvent e) {
-                                               pgBar.setProgress(null);
-                                               setAllEnabled(true);
-                                       }
-                               });
+                               try {
+                                       reader.open(book.getMeta().getLuid(), pg);
+                                       SwingUtilities.invokeLater(new Runnable() {
+                                               public void run() {
+                                                       book.setCached(true);
+                                               }
+                                       });
+                               } catch (IOException e) {
+                                       // TODO: error message?
+                                       Instance.syserr(e);
+                               }
                        }
                });
+       }
+
+       /**
+        * Process the given action out of the Swing UI thread and link the given
+        * {@link ProgressBar} to the action.
+        * <p>
+        * The code will make sure that the {@link ProgressBar} (if not NULL) is set
+        * to done when the action is done.
+        * 
+        * @param pg
+        *            the {@link ProgressBar} or NULL
+        * @param run
+        *            the action to run
+        */
+       private void outOfUi(Progress progress, final Runnable run) {
+               final Progress pg = new Progress();
+               final Progress reload = new Progress("Reload books");
+               if (progress == null) {
+                       progress = new Progress();
+               }
+
+               pg.addProgress(progress, 90);
+               pg.addProgress(reload, 10);
+
+               invalidate();
+               pgBar.setProgress(pg);
+               validate();
+               setEnabled(false);
 
                new Thread(new Runnable() {
                        public void run() {
                                run.run();
+                               refreshBooks();
+                               reload.done();
                                if (!pg.isDone()) {
-                                       pg.setProgress(pg.getMax());
+                                       // will trigger pgBar ActionListener:
+                                       pg.done();
                                }
                        }
-               }).start();
+               }, "outOfUi thread").start();
        }
 
-       public void setAllEnabled(boolean enabled) {
-               for (LocalReaderBook book : books) {
-                       book.setEnabled(enabled);
-                       book.validate();
-                       book.repaint();
+       /**
+        * Import a {@link Story} into the main {@link Library}.
+        * <p>
+        * Should be called inside the UI thread.
+        * 
+        * @param askUrl
+        *            TRUE for an {@link URL}, false for a {@link File}
+        */
+       private void imprt(boolean askUrl) {
+               JFileChooser fc = new JFileChooser();
+
+               Object url;
+               if (askUrl) {
+                       String clipboard = "";
+                       try {
+                               clipboard = ("" + Toolkit.getDefaultToolkit()
+                                               .getSystemClipboard().getData(DataFlavor.stringFlavor))
+                                               .trim();
+                       } catch (Exception e) {
+                               // No data will be handled
+                       }
+
+                       if (clipboard == null || !clipboard.startsWith("http")) {
+                               clipboard = "";
+                       }
+
+                       url = JOptionPane.showInputDialog(LocalReaderFrame.this,
+                                       "url of the story to import?", "Importing from URL",
+                                       JOptionPane.QUESTION_MESSAGE, null, null, clipboard);
+               } else if (fc.showOpenDialog(this) != JFileChooser.CANCEL_OPTION) {
+                       url = fc.getSelectedFile().getAbsolutePath();
+               } else {
+                       url = null;
                }
-               bar.setEnabled(enabled);
-               bookPane.setEnabled(enabled);
-               bookPane.validate();
-               bookPane.repaint();
-               setEnabled(enabled);
-               validate();
+
+               if (url != null && !url.toString().isEmpty()) {
+                       imprt(url.toString(), null, null);
+               }
+       }
+
+       /**
+        * Actually import the {@link Story} into the main {@link Library}.
+        * <p>
+        * Should be called inside the UI thread.
+        * 
+        * @param url
+        *            the {@link Story} to import by {@link URL}
+        * @param onSuccess
+        *            Action to execute on success
+        */
+       private void imprt(final String url, final Runnable onSuccess,
+                       String onSuccessPgName) {
+               final Progress pg = new Progress();
+               final Progress pgImprt = new Progress();
+               final Progress pgOnSuccess = new Progress(onSuccessPgName);
+               pg.addProgress(pgImprt, 95);
+               pg.addProgress(pgOnSuccess, 5);
+
+               outOfUi(pg, new Runnable() {
+                       public void run() {
+                               Exception ex = null;
+                               try {
+                                       Instance.getLibrary().imprt(BasicReader.getUrl(url),
+                                                       pgImprt);
+                               } catch (IOException e) {
+                                       ex = e;
+                               }
+
+                               final Exception e = ex;
+
+                               final boolean ok = (e == null);
+
+                               pgOnSuccess.setProgress(0);
+                               if (!ok) {
+                                       Instance.syserr(e);
+                                       SwingUtilities.invokeLater(new Runnable() {
+                                               public void run() {
+                                                       JOptionPane.showMessageDialog(
+                                                                       LocalReaderFrame.this, "Cannot import: "
+                                                                                       + url, e.getMessage(),
+                                                                       JOptionPane.ERROR_MESSAGE);
+                                               }
+                                       });
+                               } else {
+                                       if (onSuccess != null) {
+                                               onSuccess.run();
+                                       }
+                               }
+                               pgOnSuccess.done();
+                       }
+               });
+       }
+
+       /**
+        * Enables or disables this component, depending on the value of the
+        * parameter <code>b</code>. An enabled component can respond to user input
+        * and generate events. Components are enabled initially by default.
+        * <p>
+        * Disabling this component will also affect its children.
+        * 
+        * @param b
+        *            If <code>true</code>, this component is enabled; otherwise
+        *            this component is disabled
+        */
+       @Override
+       public void setEnabled(boolean b) {
+               if (bar != null) {
+                       bar.setEnabled(b);
+               }
+
+               for (LocalReaderGroup group : booksByType.keySet()) {
+                       group.setEnabled(b);
+               }
+               for (LocalReaderGroup group : booksByAuthor.keySet()) {
+                       group.setEnabled(b);
+               }
+               super.setEnabled(b);
                repaint();
        }
 }