org.pwsafe.passwordsafeswt.PasswordSafeJFace.java Source code

Java tutorial

Introduction

Here is the source code for org.pwsafe.passwordsafeswt.PasswordSafeJFace.java

Source

/*
 * $Id$
 * Copyright (c) 2008-2014 David Muller <roxon@users.sourceforge.net>.
 * All rights reserved. Use of the code is allowed under the
 * Artistic License 2.0 terms, as specified in the LICENSE file
 * distributed with this code, or available from
 * http://www.opensource.org/licenses/artistic-license-2.0.php
 */
package org.pwsafe.passwordsafeswt;

import static org.pwsafe.passwordsafeswt.preference.JpwPreferenceConstants.DISPLAY_AS_LIST_PREF;

import java.io.BufferedInputStream;
import java.io.ByteArrayOutputStream;
import java.io.DataOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
import java.io.Reader;
import java.security.NoSuchAlgorithmException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Comparator;
import java.util.Date;
import java.util.Iterator;
import java.util.List;
import java.util.Timer;
import java.util.TimerTask;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.eclipse.jface.action.IAction;
import org.eclipse.jface.action.MenuManager;
import org.eclipse.jface.action.Separator;
import org.eclipse.jface.action.StatusLineManager;
import org.eclipse.jface.action.ToolBarManager;
import org.eclipse.jface.preference.IPreferenceStore;
import org.eclipse.jface.preference.JFacePreferences;
import org.eclipse.jface.resource.JFaceResources;
import org.eclipse.jface.viewers.ColumnViewer;
import org.eclipse.jface.viewers.IElementComparer;
import org.eclipse.jface.viewers.StructuredSelection;
import org.eclipse.jface.viewers.TableViewer;
import org.eclipse.jface.viewers.TreeViewer;
import org.eclipse.jface.viewers.ViewerSorter;
import org.eclipse.jface.window.ApplicationWindow;
import org.eclipse.jface.window.Window;
import org.eclipse.swt.SWT;
import org.eclipse.swt.dnd.Clipboard;
import org.eclipse.swt.dnd.DND;
import org.eclipse.swt.dnd.TextTransfer;
import org.eclipse.swt.dnd.Transfer;
import org.eclipse.swt.events.ShellAdapter;
import org.eclipse.swt.events.ShellEvent;
import org.eclipse.swt.events.ShellListener;
import org.eclipse.swt.graphics.Image;
import org.eclipse.swt.graphics.Point;
import org.eclipse.swt.layout.FillLayout;
import org.eclipse.swt.widgets.Composite;
import org.eclipse.swt.widgets.Control;
import org.eclipse.swt.widgets.Display;
import org.eclipse.swt.widgets.Menu;
import org.eclipse.swt.widgets.MessageBox;
import org.eclipse.swt.widgets.Shell;
import org.eclipse.swt.widgets.Table;
import org.eclipse.swt.widgets.TableColumn;
import org.eclipse.swt.widgets.TableItem;
import org.eclipse.swt.widgets.Tree;
import org.eclipse.swt.widgets.TreeColumn;
import org.eclipse.swt.widgets.TreeItem;
import org.pwsafe.lib.datastore.PwsEntryBean;
import org.pwsafe.lib.datastore.PwsEntryStore;
import org.pwsafe.lib.exception.PasswordSafeException;
import org.pwsafe.lib.file.PwsFieldTypeV1;
import org.pwsafe.lib.file.PwsFieldTypeV2;
import org.pwsafe.lib.file.PwsFieldTypeV3;
import org.pwsafe.lib.file.PwsFile;
import org.pwsafe.lib.file.PwsFileFactory;
import org.pwsafe.lib.file.PwsFileStorage;
import org.pwsafe.lib.file.PwsRecord;
import org.pwsafe.lib.file.PwsRecordV1;
import org.pwsafe.lib.file.PwsRecordV2;
import org.pwsafe.passwordsafeswt.action.AboutAction;
import org.pwsafe.passwordsafeswt.action.AddRecordAction;
import org.pwsafe.passwordsafeswt.action.ChangeSafeCombinationAction;
import org.pwsafe.passwordsafeswt.action.ClearClipboardAction;
import org.pwsafe.passwordsafeswt.action.CopyPasswordAction;
import org.pwsafe.passwordsafeswt.action.CopyURLAction;
import org.pwsafe.passwordsafeswt.action.CopyUsernameAction;
import org.pwsafe.passwordsafeswt.action.DeleteRecordAction;
import org.pwsafe.passwordsafeswt.action.EditRecordAction;
import org.pwsafe.passwordsafeswt.action.ExitAppAction;
import org.pwsafe.passwordsafeswt.action.ExportToTextAction;
import org.pwsafe.passwordsafeswt.action.ExportToXMLAction;
import org.pwsafe.passwordsafeswt.action.FindRecordAction;
import org.pwsafe.passwordsafeswt.action.HelpAction;
import org.pwsafe.passwordsafeswt.action.ImportFromTextAction;
import org.pwsafe.passwordsafeswt.action.ImportFromXMLAction;
import org.pwsafe.passwordsafeswt.action.LockDbAction;
import org.pwsafe.passwordsafeswt.action.MRUFileAction;
import org.pwsafe.passwordsafeswt.action.NewFileAction;
import org.pwsafe.passwordsafeswt.action.OpenFileAction;
import org.pwsafe.passwordsafeswt.action.OpenFileReadWriteToggleAction;
import org.pwsafe.passwordsafeswt.action.OpenUrlAction;
import org.pwsafe.passwordsafeswt.action.OptionsAction;
import org.pwsafe.passwordsafeswt.action.SaveFileAction;
import org.pwsafe.passwordsafeswt.action.SaveFileAsAction;
import org.pwsafe.passwordsafeswt.action.UnlockDbAction;
import org.pwsafe.passwordsafeswt.action.ViewAsListAction;
import org.pwsafe.passwordsafeswt.action.ViewAsTreeAction;
import org.pwsafe.passwordsafeswt.action.VisitPasswordSafeWebsiteAction;
import org.pwsafe.passwordsafeswt.dialog.StartupDialog;
import org.pwsafe.passwordsafeswt.dnd.PwsEntryBeanTransfer;
import org.pwsafe.passwordsafeswt.dnd.TreeDragListener;
import org.pwsafe.passwordsafeswt.dnd.TreeDropper;
import org.pwsafe.passwordsafeswt.listener.TableColumnSelectionAdaptor;
import org.pwsafe.passwordsafeswt.listener.ViewerDoubleClickListener;
import org.pwsafe.passwordsafeswt.model.PasswordTableContentProvider;
import org.pwsafe.passwordsafeswt.model.PasswordTableLabelProvider;
import org.pwsafe.passwordsafeswt.model.PasswordTableSorter;
import org.pwsafe.passwordsafeswt.model.PasswordTreeContentProvider;
import org.pwsafe.passwordsafeswt.model.PasswordTreeLabelProvider;
import org.pwsafe.passwordsafeswt.preference.JpwPreferenceConstants;
import org.pwsafe.passwordsafeswt.preference.WidgetPreferences;
import org.pwsafe.passwordsafeswt.state.LockState;
import org.pwsafe.passwordsafeswt.util.IOUtils;
import org.pwsafe.passwordsafeswt.util.UserPreferences;
import org.pwsafe.passwordsafeswt.xml.XMLDataParser;

import au.com.bytecode.opencsv.CSVReader;
import au.com.bytecode.opencsv.CSVWriter;

/**
 * A port of PasswordSafe to JFace. This is the main class that is launched from
 * the commandline.
 * 
 * @author Glen Smith
 */
public class PasswordSafeJFace extends ApplicationWindow {

    private ImportFromXMLAction importFromXMLAction;
    private ExportToXMLAction exportToXMLAction;
    protected OptionsAction optionsAction;
    private TreeViewer treeViewer;
    private Tree tree;

    private TableViewer tableViewer;
    private Table table;

    /**
     * Is set to either Table- or Treeviewer, so it is not always needed to make
     * a difference.
     */
    private ColumnViewer viewer;

    private HelpAction helpAction;
    protected AboutAction aboutAction;
    private ChangeSafeCombinationAction changeSafeCombinationAction;
    private VisitPasswordSafeWebsiteAction visitPasswordSafeWebsiteAction;
    protected ExitAppAction exitAppAction;
    private NewFileAction newFileAction;
    private DeleteRecordAction deleteRecordAction;
    private EditRecordAction editRecordAction;
    private FindRecordAction findRecordAction;
    private AddRecordAction addRecordAction;
    private ClearClipboardAction clearClipboardAction;
    private CopyPasswordAction copyPasswordAction;
    private CopyUsernameAction copyUsernameAction;
    private CopyURLAction copyURLAction;
    private SaveFileAsAction saveFileAsAction;
    private SaveFileAction saveFileAction;
    private OpenFileAction openFileAction;
    private OpenFileReadWriteToggleAction openFileReadWriteToggleAction;
    private OpenUrlAction openUrlAction;
    private ViewAsListAction viewAsListAction;
    private ViewAsTreeAction viewAsTreeAction;
    private ExportToTextAction exportToTextAction;
    private ImportFromTextAction importFromTextAction;
    private LockDbAction lockDbAction;
    private UnlockDbAction unlockDbAction;

    private SysTray systemTray;
    private Composite composite;

    private final LockState lockState = new LockState();
    private boolean readOnly = false;
    private final Timer lockTimer = new Timer("SWTPassword lock timer", true); //$NON-NLS-1$
    private TimerTask lockTask;

    protected static final Log log = LogFactory.getLog(PasswordSafeJFace.class);

    public static final String APP_NAME = "PasswordSafeSWT"; //$NON-NLS-1$
    public static final String JPW_ICON = "jpwIcon";

    private static PasswordSafeJFace app;

    private PwsFile pwsFile;
    private PwsEntryStore dataStore;

    private static final String V1_GROUP_PLACEHOLDER = Messages.getString("PasswordSafeJFace.V1GroupPlaceholder"); //$NON-NLS-1$

    /**
     * Constructor.
     */
    public PasswordSafeJFace() {
        super(null);
        app = this;
        createActions();
        addToolBar(SWT.FLAT | SWT.WRAP);
        addMenuBar();
        addStatusLine();
        setEditMenusEnabled(false);
    }

    private void displayOpeningDialog() {
        final IPreferenceStore prefs = JFacePreferences.getPreferenceStore();
        final boolean openReadOnly = prefs.getBoolean(JpwPreferenceConstants.DEFAULT_OPEN_READ_ONLY);
        final StartupDialog sd = new StartupDialog(getShell(), UserPreferences.getInstance().getMRUFiles(),
                openReadOnly);
        boolean allDone = false;

        while (!allDone) {
            final String result = sd.open();
            if (result == StartupDialog.CANCEL || result == null) {
                // they cancelled or clicked the close button on the dialog
                exitApplication();
                return;
            } else if (result == StartupDialog.OPEN_FILE) {
                try {
                    openFile(sd.getFilename(), sd.getPassword());
                    this.setReadOnly(sd.getReadonly());

                    allDone = true;
                } catch (final FileNotFoundException anEx) {
                    log.warn("File + " + sd.getFilename() + " not found.");//$NON-NLS-1$
                    displayErrorDialog(Messages.getString("PasswordSafeJFace.OpenError.Title"), //$NON-NLS-1$
                            Messages.getString("PasswordSafeJFace.OpenError.NoFileFoundMessage") + sd.getFilename() //$NON-NLS-1$
                                    + "]", //$NON-NLS-1$
                            anEx);
                    UserPreferences.getInstance().removeMRUFile(sd.getFilename());
                    allDone = false;
                } catch (final Exception anEx) {
                    log.error("Exception on opening file + " + sd.getFilename(), anEx);//$NON-NLS-1$
                    UserPreferences.getInstance().setMostRecentFilename(sd.getFilename());
                    displayErrorDialog(Messages.getString("PasswordSafeJFace.OpenError.Title"), //$NON-NLS-1$
                            Messages.getString("PasswordSafeJFace.OpenError.Message"), anEx); //$NON-NLS-1$
                    allDone = false;
                }
            } else if (result == StartupDialog.NEW_FILE) {
                newFileAction.run();
                if (getPwsFile() != null)
                    allDone = true;
            } else if (result == StartupDialog.OPEN_OTHER) {
                openFileAction.run();
                allDone = true;
            }
        }
        setupStatusMessage();
    }

    private void startOpeningDialogThread(final Shell shell) {

        shell.getDisplay().asyncExec(new Runnable() {
            public void run() {
                displayOpeningDialog();
            }
        });
    }

    /**
     * Returns a singleton handle to the app for making business logic calls.
     * 
     * @return a handle to a singleton of this class
     */
    public static PasswordSafeJFace getApp() {
        return app;
    }

    /**
     * @see org.eclipse.jface.window.Window#createContents(org.eclipse.swt.widgets.Composite)
     */
    @Override
    protected Control createContents(final Composite parent) {
        final Composite container = new Composite(parent, SWT.NONE);
        container.setLayout(new FillLayout(SWT.VERTICAL));

        composite = new Composite(container, SWT.NONE);
        composite.setLayout(new FillLayout(SWT.VERTICAL));
        final IPreferenceStore thePrefs = JFacePreferences.getPreferenceStore();
        if (thePrefs.getBoolean(DISPLAY_AS_LIST_PREF)) {
            addTableView(composite);
            viewAsListAction.setChecked(true);
            viewAsTreeAction.setChecked(false);
        } else {
            addTreeView(composite);
            viewAsTreeAction.setChecked(true);
            viewAsListAction.setChecked(false);
        }

        final SysTray tray = new SysTray();
        final boolean isAvailable = tray.init(null);
        if (isAvailable) {
            systemTray = tray;
        }

        return container;
    }

    /**
     * Creates the JFace actions that will be used for menus and toolbars.
     */
    private void createActions() {

        newFileAction = new NewFileAction();
        openFileAction = new OpenFileAction();
        openFileReadWriteToggleAction = new OpenFileReadWriteToggleAction();
        saveFileAction = new SaveFileAction();
        saveFileAsAction = new SaveFileAsAction();
        exportToTextAction = new ExportToTextAction();
        importFromTextAction = new ImportFromTextAction();
        exitAppAction = new ExitAppAction();
        copyUsernameAction = new CopyUsernameAction();
        copyPasswordAction = new CopyPasswordAction();
        copyURLAction = new CopyURLAction();
        clearClipboardAction = new ClearClipboardAction();
        addRecordAction = new AddRecordAction();
        editRecordAction = new EditRecordAction();
        findRecordAction = new FindRecordAction();
        deleteRecordAction = new DeleteRecordAction();
        viewAsListAction = new ViewAsListAction();
        viewAsTreeAction = new ViewAsTreeAction();
        visitPasswordSafeWebsiteAction = new VisitPasswordSafeWebsiteAction();
        changeSafeCombinationAction = new ChangeSafeCombinationAction();
        aboutAction = new AboutAction();
        helpAction = new HelpAction();
        optionsAction = new OptionsAction();
        exportToXMLAction = new ExportToXMLAction();
        importFromXMLAction = new ImportFromXMLAction();
        lockDbAction = new LockDbAction();
        openUrlAction = new OpenUrlAction();
        unlockDbAction = new UnlockDbAction();
    }

    /**
     * @see org.eclipse.jface.window.ApplicationWindow#createMenuManager()
     */
    @Override
    protected MenuManager createMenuManager() {
        final MenuManager result = new MenuManager(Messages.getString("PasswordSafeJFace.Menu.Menu")); //$NON-NLS-1$

        final MenuManager menuManagerFile = new MenuManager(Messages.getString("PasswordSafeJFace.Menu.File")); //$NON-NLS-1$
        result.add(menuManagerFile);

        menuManagerFile.add(newFileAction);
        menuManagerFile.add(openFileAction);
        menuManagerFile.add(openFileReadWriteToggleAction);
        menuManagerFile.add(new Separator());

        final List<String> mruFiles = UserPreferences.getInstance().getMRUFiles();

        if (mruFiles != null && mruFiles.size() > 0) {
            for (int i = 0; i < mruFiles.size(); i++) {
                final String fileName = mruFiles.get(i);
                final String menuItem = "&" + (i + 1) + " " + new File(fileName).getName(); //$NON-NLS-1$ //$NON-NLS-2$
                final IAction nextMRUAction = new MRUFileAction(fileName, menuItem);
                menuManagerFile.add(nextMRUAction);
            }
            menuManagerFile.add(new Separator());
        }

        menuManagerFile.add(saveFileAction);
        menuManagerFile.add(saveFileAsAction);
        menuManagerFile.add(new Separator());

        final MenuManager menuManagerExportTo = new MenuManager(
                Messages.getString("PasswordSafeJFace.Menu.ExportTo")); //$NON-NLS-1$
        menuManagerFile.add(menuManagerExportTo);

        menuManagerExportTo.add(exportToTextAction);
        menuManagerExportTo.add(exportToXMLAction);

        final MenuManager menuManagerImportFrom = new MenuManager(
                Messages.getString("PasswordSafeJFace.Menu.ImportFrom")); //$NON-NLS-1$
        menuManagerFile.add(menuManagerImportFrom);

        menuManagerImportFrom.add(importFromTextAction);
        menuManagerImportFrom.add(importFromXMLAction);

        menuManagerFile.add(new Separator());
        menuManagerFile.add(exitAppAction);

        final MenuManager menuManagerEdit = new MenuManager(Messages.getString("PasswordSafeJFace.Menu.Edit")); //$NON-NLS-1$
        result.add(menuManagerEdit);

        menuManagerEdit.add(addRecordAction);
        menuManagerEdit.add(editRecordAction);
        menuManagerEdit.add(findRecordAction);
        menuManagerEdit.add(deleteRecordAction);
        menuManagerEdit.add(new Separator());
        menuManagerEdit.add(copyPasswordAction);
        menuManagerEdit.add(copyUsernameAction);
        menuManagerEdit.add(copyURLAction);
        menuManagerEdit.add(openUrlAction);
        menuManagerEdit.add(clearClipboardAction);

        final MenuManager menuManagerView = new MenuManager(Messages.getString("PasswordSafeJFace.Menu.View")); //$NON-NLS-1$
        result.add(menuManagerView);

        menuManagerView.add(viewAsListAction);
        menuManagerView.add(viewAsTreeAction);

        final MenuManager menuManagerManage = new MenuManager(Messages.getString("PasswordSafeJFace.Menu.Manage")); //$NON-NLS-1$
        result.add(menuManagerManage);

        menuManagerManage.add(changeSafeCombinationAction);
        menuManagerManage.add(new Separator());
        menuManagerManage.add(optionsAction);

        final MenuManager menuManagerHelp = new MenuManager(Messages.getString("PasswordSafeJFace.Menu.Help")); //$NON-NLS-1$
        result.add(menuManagerHelp);

        menuManagerHelp.add(helpAction);
        menuManagerHelp.add(new Separator());
        menuManagerHelp.add(visitPasswordSafeWebsiteAction);
        menuManagerHelp.add(new Separator());
        menuManagerHelp.add(aboutAction);

        return result;
    }

    /**
     * Creates the popup (right click) menu for the given control.
     * 
     * @param ctl the control to add the popup to
     * @return the created menu
     */
    protected Menu createPopupMenu(final Control ctl) {

        final MenuManager menuListPopup = new MenuManager(Messages.getString("PasswordSafeJFace.Menu.ListPopup")); //$NON-NLS-1$

        menuListPopup.createContextMenu(ctl);

        menuListPopup.add(copyPasswordAction);
        menuListPopup.add(copyUsernameAction);
        menuListPopup.add(copyURLAction);
        menuListPopup.add(openUrlAction);
        menuListPopup.add(new Separator());
        menuListPopup.add(addRecordAction);
        menuListPopup.add(editRecordAction);
        menuListPopup.add(deleteRecordAction);

        return menuListPopup.getMenu();
    }

    /**
     * @see org.eclipse.jface.window.ApplicationWindow#createToolBarManager(int)
     */
    @Override
    protected ToolBarManager createToolBarManager(final int style) {
        final ToolBarManager toolBarManager = new ToolBarManager(style);

        toolBarManager.add(newFileAction);
        toolBarManager.add(openFileAction);
        toolBarManager.add(saveFileAction);
        toolBarManager.add(new Separator());
        toolBarManager.add(copyUsernameAction);
        toolBarManager.add(copyPasswordAction);
        toolBarManager.add(copyURLAction);
        toolBarManager.add(openUrlAction);
        toolBarManager.add(clearClipboardAction);
        toolBarManager.add(new Separator());
        toolBarManager.add(addRecordAction);
        toolBarManager.add(editRecordAction);
        toolBarManager.add(deleteRecordAction);
        toolBarManager.add(new Separator());
        toolBarManager.add(helpAction);
        return toolBarManager;
    }

    /**
     * @see org.eclipse.jface.window.ApplicationWindow#createStatusLineManager()
     */
    @Override
    protected StatusLineManager createStatusLineManager() {
        final StatusLineManager statusLineManager = new StatusLineManager();
        statusLineManager.setMessage("http://jpwsafe.sf.net"); //$NON-NLS-1$
        return statusLineManager;
    }

    public void setupStatusMessage() {

        final PwsFile pwsf = getPwsFile();
        if (pwsf != null && pwsf.getRecordCount() > 0) {
            final IPreferenceStore thePrefs = JFacePreferences.getPreferenceStore();
            if (thePrefs.getBoolean(JpwPreferenceConstants.DOUBLE_CLICK_COPIES_TO_CLIPBOARD)) {
                setStatus(Messages.getString("PasswordSafeJFace.Status.DoubleClickToCopy")); //$NON-NLS-1$
            } else {
                setStatus(Messages.getString("PasswordSafeJFace.Status.DoubleClickToEdit")); //$NON-NLS-1$
            }
        } else {
            setStatus("http://jpwsafe.sf.net"); //$NON-NLS-1$
        }

    }

    /**
     * Main program entry point.
     * 
     * @param args commandline args that are passed in
     */
    public static void main(final String args[]) {
        log.info("PasswordSafe starting..."); //$NON-NLS-1$
        log.info("java.library.path is: [" + System.getProperty("java.library.path") + "]"); //$NON-NLS-1$ //$NON-NLS-2$ //$NON-NLS-3$
        log.info("log: " + log.getClass().getName()); //$NON-NLS-1$
        try {
            final PasswordSafeJFace window = new PasswordSafeJFace();
            window.setBlockOnOpen(true);
            window.open();
        } catch (final Throwable e) {
            log.error("Error Starting PasswordSafe", e); //$NON-NLS-1$
        } finally { // try to clean up in any case
            try {
                Display.getCurrent().dispose();
            } catch (final Exception anEx1) {
                log.warn("Cannot dispose app - ignoring", anEx1); //$NON-NLS-1$
            }

        }
        log.info("PasswordSafe terminating..."); //$NON-NLS-1$
    }

    /**
     * @see org.eclipse.jface.window.Window#configureShell(org.eclipse.swt.widgets.Shell)
     */
    @Override
    protected void configureShell(final Shell newShell) {
        super.configureShell(newShell);
        newShell.setText(PasswordSafeJFace.APP_NAME);
        final Image jpwIcon = IOUtils.getImage(PasswordSafeJFace.class,
                "/org/pwsafe/passwordsafeswt/images/cpane.ico"); //$NON-NLS-1$
        newShell.setImage(jpwIcon);
        // provide it for the rest of jpwsafe:
        JFaceResources.getImageRegistry().put(JPW_ICON, jpwIcon);
        Window.setDefaultImage(jpwIcon);
        WidgetPreferences.tuneShell(newShell, getClass());
        startOpeningDialogThread(newShell);
    }

    /**
     * @see org.eclipse.jface.window.Window#getInitialSize()
     */
    @Override
    protected Point getInitialSize() {
        return new Point(425, 375);
    }

    /**
     * Create a brand new empty safe.
     * 
     * @param password the password for the new safe
     */
    public void newFile(final StringBuilder password) {
        getShell().setText(PasswordSafeJFace.APP_NAME + Messages.getString("PasswordSafeJFace.AppTitle.Default")); //$NON-NLS-1$
        final PwsEntryStore newEntryStore = PwsFileFactory.newStore();
        newEntryStore.getPwsFile().setPassphrase(password);
        this.setPwsEntryStore(newEntryStore);
        this.setReadOnly(false);
    }

    /**
     * Opens a password safe from the file system. Readonly status stays
     * unchanged.
     * 
     * @param fileName
     * @param password
     * @throws Exception if bad things happen during open
     */
    public void openFile(final String fileName, final StringBuilder password) throws Exception {

        this.openFile(fileName, password, isReadOnly());
    }

    /**
     * Opens a password safe from the file system.
     * 
     * @param fileName
     * @param password
     * @param forReadOnly
     * @throws Exception if bad things happen during open
     */
    public void openFile(final String fileName, final StringBuilder password, final boolean forReadOnly)
            throws Exception {

        final PwsEntryStore file = PwsFileFactory.loadStore(fileName, password);
        getShell().setText(PasswordSafeJFace.APP_NAME + " - " + fileName); //$NON-NLS-1$
        setPwsEntryStore(file);
        setReadOnly(forReadOnly);
        if (true) // TODO (!openedFromMRU)
            UserPreferences.getInstance().setMostRecentFilename(fileName);
    }

    /**
     * Save the current safe.
     * 
     * @throws IOException if bad things happen during save
     * @throws NoSuchAlgorithmException if SHA-1 implementation not found
     */
    public void saveFile() throws IOException, NoSuchAlgorithmException {
        final PwsFile file = getPwsFile();
        if (file.getStorage() == null) {
            saveFileAsAction.run();
        } else {
            file.save();
        }
    }

    /**
     * Saves the current safe under a new filename.
     * 
     * @param newFilename the new name to save the file as
     * @throws IOException if something went wrong while trying to save the file
     * @throws NoSuchAlgorithmException if SHA-1 implementation not found
     */
    public void saveFileAs(final String newFilename) throws IOException, NoSuchAlgorithmException {
        final PwsFileStorage s = new PwsFileStorage(newFilename);
        getPwsFile().setStorage(s);
        getPwsFile().save();
        getShell().setText(PasswordSafeJFace.APP_NAME + " - " + newFilename); //$NON-NLS-1$
        UserPreferences.getInstance().setMostRecentFilename(newFilename);
    }

    /**
     * Copy field to clipboard.
     * 
     * @param cb handle to the clipboard
     * @param valueToCopy the value to copy to clipboard
     */
    public void copyToClipboard(final Clipboard cb, final String valueToCopy) {
        // TODO: check whether to create the clipboard here in order to reset it
        // later
        if (valueToCopy != null) {
            cb.setContents(new Object[] { valueToCopy }, new Transfer[] { TextTransfer.getInstance() });
            log.debug("Copied to clipboard");
        }
    }

    /**
     * Update the access time of an entry. This method should be moved to
     * PwsDatastore in V0.9.
     * 
     * @param anEntry to update the access time, may be sparse
     */
    public void updateAccessTime(PwsEntryBean anEntry) {
        if (anEntry != null) {
            // set access date
            if (!isReadOnly() && "3".equals(anEntry.getVersion())) {
                // fetch the real entry if sparse
                if (anEntry.isSparse()) {
                    anEntry = getPwsDataStore().getEntry(anEntry.getStoreIndex());
                }
                anEntry.setLastAccess(new Date());
                dataStore.updateEntry(anEntry);
            }
        }
    }

    /**
     * Locates the selected record in the underlying tree or table control.
     * 
     * @return the selected record
     */
    public PwsEntryBean getSelectedRecord() {

        PwsEntryBean recordToCopy = null;
        if (isTreeViewShowing()) {
            if (tree.getSelectionCount() == 1) {
                final TreeItem ti = tree.getSelection()[0];
                final Object treeData = ti.getData();
                if (treeData != null && treeData instanceof PwsEntryBean) { // must
                    // be
                    // a
                    // left,
                    // not
                    // a
                    // group
                    // entry
                    recordToCopy = (PwsEntryBean) treeData;
                }
            }
        } else {
            if (table != null && table.getSelectionCount() == 1) {
                final TableItem ti = table.getSelection()[0];
                recordToCopy = (PwsEntryBean) ti.getData();
            }
        }

        return recordToCopy;
    }

    /**
     * Updates an existing record.
     * 
     * @param newEntry the entry's new values
     */
    public void updateRecord(final PwsEntryBean newEntry) {
        if (log.isDebugEnabled())
            log.debug("Dialog has been edited, updating safe"); //$NON-NLS-1$
        getPwsDataStore().updateEntry(newEntry);
        if (isDirty()) {
            saveOnUpdateOrEditCheck();
        }
        // TODO: this should only be called if the the update changes visible
        // fields
        updateViewers();
    }

    /**
     * Adds a new record to the safe in a new dialog.
     * 
     * @param newEntry the entry to add
     */
    public void addRecord(final PwsEntryBean newEntry) {
        if (log.isDebugEnabled())
            log.debug("Dialog has created new record, updating safe"); //$NON-NLS-1$
        try {
            getPwsDataStore().addEntry(newEntry);
            saveOnUpdateOrEditCheck();
        } catch (final PasswordSafeException e) {
            displayErrorDialog(Messages.getString("PasswordSafeJFace.AddEntryError.Title"), //$NON-NLS-1$
                    Messages.getString("PasswordSafeJFace.AddEntryError.Message"), e); //$NON-NLS-1$
        }

        updateViewers();
    }

    /**
     * If the user has set "Save on Update or Edit", we save the file
     * immediately.
     * 
     */
    private void saveOnUpdateOrEditCheck() {
        final IPreferenceStore thePrefs = JFacePreferences.getPreferenceStore();
        if (thePrefs.getBoolean(JpwPreferenceConstants.SAVE_IMMEDIATELY_ON_EDIT)) {
            if (log.isDebugEnabled())
                log.debug("Save on Edit option active. Saving database."); //$NON-NLS-1$
            saveFileAction.run();
        }
    }

    /**
     * Prompts the user to save the current file, if it has been modified.
     * 
     * @return <code>true</code> if the user cancels the action which triggered
     *         the call to this method, <code>false</code> if the save was
     *         successful or ignored.
     */
    public boolean saveAppIfDirty() {
        boolean cancelled = false;
        if (isDirty()) {
            final int style = SWT.APPLICATION_MODAL | SWT.YES | SWT.NO | SWT.CANCEL;
            final MessageBox messageBox = new MessageBox(getShell(), style);
            messageBox.setText(Messages.getString("PasswordSafeJFace.SaveChanges.Title")); //$NON-NLS-1$
            messageBox.setMessage(Messages.getString("PasswordSafeJFace.SaveChanges.Message")); //$NON-NLS-1$
            final int result = messageBox.open();
            if (result == SWT.YES) {
                try {
                    saveFile();
                } catch (final IOException e1) {
                    displayErrorDialog(Messages.getString("PasswordSafeJFace.SaveSafeError.Title"), e1.getMessage(), //$NON-NLS-1$
                            e1);
                    cancelled = true;
                } catch (final NoSuchAlgorithmException e) {
                    displayErrorDialog(Messages.getString("PasswordSafeJFace.SaveSafeError.Title"), e.getMessage(), //$NON-NLS-1$
                            e);
                    cancelled = true;
                }
            } else if (result == SWT.CANCEL) {
                cancelled = true;
            }
        }
        return cancelled;
    }

    /**
     * Deletes the selected record in the tree or table.
     * 
     */
    public void deleteRecord() {
        final PwsEntryBean selectedRec = getSelectedRecord();
        getPwsDataStore().removeEntry(selectedRec);
        saveOnUpdateOrEditCheck();
        updateViewers();
    }

    /**
     * Has the current safe been modified.
     * 
     * @return true if the safe has been modified
     */
    public boolean isDirty() {
        if (getPwsFile() != null) {
            return getPwsFile().isModified();
        }
        return false;
    }

    /**
     * Is the Application in locked mode.
     * 
     * @return true if the application has been locked
     */
    public boolean isLocked() {
        return lockState.isLocked();
    }

    /**
     * Sets the Application in locked mode.
     * 
     * @param sets the locked mode
     */
    public void setLocked(final boolean isLocked) {
        lockState.setLocked(isLocked);
    }

    public LockState getLockStatus() {
        return lockState;
    }

    /**
     * Is the Application in read only mode.
     * 
     * @return true if the safe is opened read only
     */
    public boolean isReadOnly() {
        return readOnly;
    }

    /**
     * Sets the Application in read only mode.
     * 
     * @param sets the read only mode testtesttest@
     */
    public void setReadOnly(final boolean isReadOnly) {

        // TODO check modified flag in PwsFile?
        getPwsFile().setReadOnly(isReadOnly);

        setEditMenusEnabled(!isReadOnly);

        readOnly = isReadOnly;
    }

    /**
     * Returns the currently loaded pwsafe file.
     * 
     * @return Returns the pwsFile.
     */
    public PwsFile getPwsFile() {
        return pwsFile;
    }

    /**
     * Returns the currently loaded pwsafe file.
     * 
     * @return Returns the pwsFile.
     */
    public PwsEntryStore getPwsDataStore() {
        return dataStore;
    }

    /**
     * Sets the currently loaded pws entry store.
     * 
     * @param pwsEntryStore The pwsEntryStore to set.
     */
    private void setPwsEntryStore(final PwsEntryStore pwsEntryStore) {
        this.dataStore = pwsEntryStore;
        pwsFile = pwsEntryStore.getPwsFile();
        updateViewers();
    }

    /**
     * Clears the currently loaded store.
     * 
     */
    public void clearPwsStore() {
        this.pwsFile = null;
        this.dataStore = PwsFileFactory.getStore(pwsFile);
        updateViewers();
    }

    /**
     * Perform necessary shutdown operations, regardless of how the user exited
     * the application.
     * 
     */
    private void tidyUpOnExit() {
        final IPreferenceStore thePrefs = JFacePreferences.getPreferenceStore();

        if (thePrefs.getBoolean(JpwPreferenceConstants.CLEAR_CLIPBOARD_ON_MIN)) {
            clearClipboardAction.run();
        }
        thePrefs.setValue(DISPLAY_AS_LIST_PREF, !isTreeViewShowing());
        try {
            UserPreferences.getInstance().savePreferences();
        } catch (final IOException e) {
            displayErrorDialog(Messages.getString("PasswordSafeJFace.SavePrefsError.Title"), //$NON-NLS-1$
                    Messages.getString("PasswordSafeJFace.SavePrefsError.Message") + e.getMessage(), e); //$NON-NLS-1$
        }
    }

    /**
     * Exit the app.
     * 
     */
    public void exitApplication() {
        tidyUpOnExit();
        if (systemTray != null) {
            systemTray.dispose();
            systemTray = null;
        }
        getShell().close();
        if (getShell() != null && (!getShell().isDisposed()))
            getShell().dispose();
    }

    /**
     * Set the list view as the topmost view.
     */
    public void showListView() {
        if (table != null) // nothing to do
            return;
        if (tree != null) {
            tree.setVisible(false);
            tree.clearAll(false);
            tree.dispose();
            tree = null;
            treeViewer = null;
        }
        addTableView(composite);
        composite.layout();

    }

    protected void addTableView(final Composite aComposite) {
        tableViewer = new TableViewer(aComposite, SWT.FULL_SELECTION | SWT.BORDER);
        tableViewer.addDoubleClickListener(new ViewerDoubleClickListener());
        table = tableViewer.getTable();
        table.setHeaderVisible(true);
        table.setMenu(createPopupMenu(table));
        tableViewer.setContentProvider(new PasswordTableContentProvider());
        tableViewer.setLabelProvider(new PasswordTableLabelProvider());
        tableViewer.setInput(new Object());
        tableViewer.setSorter(new PasswordTableSorter());

        viewer = tableViewer;

        int column = 1;
        addTableColumn(column, "PasswordSafeJFace.Column.Title", "table/title"); //$NON-NLS-1$

        column++;
        addTableColumn(column, "PasswordSafeJFace.Column.UserName", "table/userName"); //$NON-NLS-1$

        final IPreferenceStore thePrefs = JFacePreferences.getPreferenceStore();
        if (thePrefs.getBoolean(JpwPreferenceConstants.SHOW_NOTES_IN_LIST)) {
            column++;
            addTableColumn(column, "PasswordSafeJFace.Column.Notes", "table/notes"); //$NON-NLS-1$
        }

        column++;
        addTableColumn(column, "PasswordSafeJFace.Column.LastChanged", "table/lastChange"); //$NON-NLS-1$

        // Sort on first column
        final PasswordTableSorter pts = (PasswordTableSorter) tableViewer.getSorter();
        pts.sortOnColumn(1);

    }

    private void addTableColumn(final int column, final String headerText, final String preferenceKey) {
        final TableColumn tableColumn = new TableColumn(table, SWT.NONE);
        tableColumn.setWidth(100);
        tableColumn.setText(Messages.getString(headerText));
        tableColumn.addSelectionListener(new TableColumnSelectionAdaptor(tableViewer, column));
        WidgetPreferences.tuneTableColumn(tableColumn, getClass(), preferenceKey);
    }

    /**
     * Set the table view as the topmost view.
     */
    public void showTreeView() {
        if (isTreeViewShowing()) // nothing to do
            return;
        if (table != null) {
            table.setVisible(false);
            table.clearAll();
            table.dispose();
            table = null;
            tableViewer = null;
        }
        addTreeView(composite);
        composite.layout();
    }

    protected void addTreeView(final Composite aComposite) {
        treeViewer = new TreeViewer(aComposite, SWT.BORDER);
        treeViewer.setLabelProvider(new PasswordTreeLabelProvider());
        treeViewer.setContentProvider(new PasswordTreeContentProvider());
        treeViewer.setSorter(new ViewerSorter());
        treeViewer.addDoubleClickListener(new ViewerDoubleClickListener());
        final int operations = DND.DROP_COPY | DND.DROP_MOVE;
        final Transfer[] transferTypes = new Transfer[] { PwsEntryBeanTransfer.getInstance(),
                TextTransfer.getInstance() };
        treeViewer.addDragSupport(operations, transferTypes, new TreeDragListener(treeViewer));
        treeViewer.addDropSupport(operations, transferTypes, new TreeDropper(treeViewer));
        treeViewer.setComparer(new IElementComparer() {
            public boolean equals(final Object a, final Object b) {
                if (a instanceof PwsEntryBean && b instanceof PwsEntryBean)
                    return ((PwsEntryBean) a).getStoreIndex() == (((PwsEntryBean) b).getStoreIndex());
                else
                    return a.equals(b);
            }

            public int hashCode(final Object element) {
                if (element instanceof PwsEntryBean)
                    return ((PwsEntryBean) element).getStoreIndex();
                else
                    return element.hashCode();
            }
        });
        tree = treeViewer.getTree();
        tree.setHeaderVisible(true);
        tree.setMenu(createPopupMenu(tree));

        treeViewer.setInput(new Object());
        viewer = treeViewer;

        int column = 1;
        addTreeColumn(column, "PasswordSafeJFace.Column.Title", "tree/title");//$NON-NLS-1$

        column++;
        addTreeColumn(column, "PasswordSafeJFace.Column.UserName", "tree/userName");//$NON-NLS-1$

        final IPreferenceStore thePrefs = JFacePreferences.getPreferenceStore();
        if (thePrefs.getBoolean(JpwPreferenceConstants.SHOW_NOTES_IN_LIST)) {
            column++;
            addTreeColumn(column, "PasswordSafeJFace.Column.Notes", "tree/notes");//$NON-NLS-1$
        }

        final TreeColumn[] columns = tree.getColumns();
        for (int i = 0; i < columns.length; i++) {
            // ps.getDefaultInt("bla");
            // columns[i].setWidth(100);
            columns[i].setMoveable(true);
        }
        // treeViewer.setExpandedState(arg0, arg1)

    }

    private void addTreeColumn(final int column, final String headerText, final String preferenceKey) {
        final TreeColumn treeColumn = new TreeColumn(tree, SWT.LEFT);
        treeColumn.setText(Messages.getString(headerText));
        treeColumn.setWidth(100);
        WidgetPreferences.tuneTreeColumn(treeColumn, getClass(), preferenceKey);
        // treeColumn.addSelectionListener(new
        // TreeColumnSelectionAdaptor(treeViewer, column));
    }

    /**
     * Returns whether the treeview is the active view.
     * 
     * @return true if the treeview is showing, false otherwise
     */
    public boolean isTreeViewShowing() {
        return (tree != null);
    }

    public String getSelectedTreeGroupPath() {
        final StringBuilder theGroupPath = new StringBuilder();
        if (isTreeViewShowing() && tree.getSelectionCount() == 1) {
            TreeItem ti = tree.getSelection()[0];

            if (ti.getItemCount() <= 0) {// leaf node
                ti = ti.getParentItem();
            }
            while (ti != null) {
                theGroupPath.insert(0, ti.getText());
                ti = ti.getParentItem();
                if (ti != null)
                    theGroupPath.insert(0, '.');

            }
            log.debug("Group path constructed *" + theGroupPath + "*"); //$NON-NLS-1$ //$NON-NLS-2$
        }
        return theGroupPath.toString();
    }

    public void highlightEntry(final PwsEntryBean entry) {
        if (isTreeViewShowing()) {
            // todo: save previous state? treeViewer.getExpandedElements()

            final String parent = ((PasswordTreeContentProvider) treeViewer.getContentProvider()).getParent(entry);
            final List<TreeItem> allItemsList = new ArrayList<TreeItem>();
            allItemsList.addAll(Arrays.asList(tree.getItems()));

            for (final String group : parent.split("\\.")) {
                for (final TreeItem item : allItemsList) {
                    // TreeItem.getData is a TreeGroup
                    if (group.equals(item.getData().toString())) {
                        item.setExpanded(true);
                        treeViewer.refresh();
                        allItemsList.clear();
                        allItemsList.addAll(Arrays.asList(item.getItems()));
                        break;
                    }
                }
            }

            final StructuredSelection selection = new StructuredSelection(entry);
            treeViewer.setSelection(selection, true);
            treeViewer.refresh(false);
        } else {
            tableViewer.setSelection(new StructuredSelection(entry), true);
            tableViewer.refresh(entry, false);
        }
    }

    private void expandToRoot(final TreeItem ti) {
        log.info("expandToRoot; ti=" + ti.toString());
        TreeItem tmpTI = ti.getParentItem();
        treeViewer.setExpandedState(ti, true);
        while (tmpTI.getParentItem() != null) {
            log.info("expanding; tmpTI=" + tmpTI.toString());
            tmpTI.setExpanded(true);
            treeViewer.refresh(tmpTI, false);
            tmpTI = tmpTI.getParentItem();
        }
    }

    /**
     * Refresh the tree and table.
     * 
     */
    public void updateViewers() {
        if (isTreeViewShowing()) {
            final Object[] currentExpansions = treeViewer.getExpandedElements();
            treeViewer.setInput(getPwsDataStore());
            treeViewer.refresh();
            treeViewer.setExpandedElements(currentExpansions);
        } else {
            tableViewer.setInput(getPwsDataStore());
            // tableViewer.refresh();
        }
    }

    /**
     * Turns on or off the edit record menus (add/edit/delete) and save/as
     * menus.
     * 
     * @param enabled true to enable the menus, false otherwise
     */
    private void setEditMenusEnabled(final boolean enabled) {
        addRecordAction.setEnabled(enabled);
        // TODO: maybe add a new action viewRecord, possibly with a dedicated
        // (read-only) dialog ?
        // for now edit is also needed to VIEW the records
        // editRecordAction.setEnabled(enabled);
        deleteRecordAction.setEnabled(enabled);
        findRecordAction.setEnabled(enabled);
        saveFileAction.setEnabled(enabled);
        saveFileAsAction.setEnabled(enabled);

        openFileReadWriteToggleAction.setOpenReadOnlyMode(enabled);
    }

    /**
     * TODO: Refactor this to one or several external Listeners. The URL copy
     * action should only be enabled when an URL is available.
     * 
     * @param enabled
     */
    public void setUrlCopyEnabled(final boolean enabled) {
        copyURLAction.setEnabled(enabled);
    }

    /**
     * Exports the contents of the password safe to tab separated file. TODO:
     * write unit test, then get rid of pwsrecord
     * 
     * @param filename full path to output file
     * @throws FileNotFoundException if the password file was not found
     */
    public void exportToText(final String filename) {
        FileWriter fw = null;
        try {
            fw = new FileWriter(filename);
            final Iterator<? extends PwsRecord> iter = getPwsFile().getRecords();
            final CSVWriter csvWriter = new CSVWriter(fw, '\t');
            while (iter.hasNext()) {
                final PwsRecord nextRecord = iter.next();
                final List<String> nextEntry = new ArrayList<String>();

                if (nextRecord instanceof PwsRecordV1) {
                    nextEntry.add(V1_GROUP_PLACEHOLDER);
                    nextEntry.add(PwsEntryBean.getSafeValue(nextRecord, PwsFieldTypeV1.TITLE));
                    nextEntry.add(PwsEntryBean.getSafeValue(nextRecord, PwsFieldTypeV1.USERNAME));
                    nextEntry.add(PwsEntryBean.getSafeValue(nextRecord, PwsFieldTypeV1.PASSWORD));
                } else if (nextRecord instanceof PwsRecordV2) {
                    nextEntry.add(PwsEntryBean.getSafeValue(nextRecord, PwsFieldTypeV2.GROUP));
                    nextEntry.add(PwsEntryBean.getSafeValue(nextRecord, PwsFieldTypeV2.TITLE));
                    nextEntry.add(PwsEntryBean.getSafeValue(nextRecord, PwsFieldTypeV2.USERNAME));
                    nextEntry.add(PwsEntryBean.getSafeValue(nextRecord, PwsFieldTypeV2.PASSWORD));
                } else {
                    nextEntry.add(PwsEntryBean.getSafeValue(nextRecord, PwsFieldTypeV3.GROUP));
                    nextEntry.add(PwsEntryBean.getSafeValue(nextRecord, PwsFieldTypeV3.TITLE));
                    nextEntry.add(PwsEntryBean.getSafeValue(nextRecord, PwsFieldTypeV3.USERNAME));
                    nextEntry.add(PwsEntryBean.getSafeValue(nextRecord, PwsFieldTypeV3.PASSWORD));
                }
                final String[] nextLine = nextEntry.toArray(new String[0]);
                csvWriter.writeNext(nextLine);

            }
        } catch (final IOException ioe) {
            displayErrorDialog(Messages.getString("PasswordSafeJFace.ExportError.Title"), //$NON-NLS-1$
                    Messages.getString("PasswordSafeJFace.ExportError.Message"), ioe); //$NON-NLS-1$
        } finally {
            if (fw != null) {
                try {
                    fw.close();
                } catch (final IOException e) {
                    log.warn("Could not close output text file", e);
                }
            }

        }

    }

    public void importFromText(final String filename) {

        final File theFile = new File(filename);

        if (!theFile.exists()) {
            throw new RuntimeException(
                    Messages.getString("PasswordSafeJFace.ImportCSV.NoFileFound") + filename + "]"); //$NON-NLS-1$ //$NON-NLS-2$
        }

        Reader fileReader = null;

        try {
            fileReader = new FileReader(theFile);

            final CSVReader csvReader = new CSVReader(fileReader, '\t');

            String[] nextLine;
            final PwsFile pwsFile = getPwsFile();
            while ((nextLine = csvReader.readNext()) != null) {
                if (nextLine.length < 2)
                    continue; // ignore blank lines
                final PwsEntryBean entry = PwsEntryBean.fromPwsRecord(getPwsFile().newRecord());
                entry.setSparse(false);
                if (!nextLine[0].equals(V1_GROUP_PLACEHOLDER)) {
                    entry.setGroup(nextLine[0]);
                }
                entry.setTitle(nextLine[1]);
                entry.setUsername(nextLine[2]);
                entry.setPassword(new StringBuilder(nextLine[3]));
                getPwsDataStore().addEntry(entry);
            }
        } catch (final Exception e) {

            throw new RuntimeException(
                    Messages.getString("PasswordSafeJFace.ImportCSV.ProcessingError") + filename + "]", e); //$NON-NLS-1$ //$NON-NLS-2$

        } finally {
            if (fileReader != null) {
                try {
                    fileReader.close();
                } catch (final IOException e) {
                    log.warn("Could not close import text file", e);
                }
            }
        }

        this.updateViewers();

    }

    /**
     * Import an XML file into the current passwordsafe.
     * 
     * @param filename the name of the file
     * @throws PasswordSafeException
     */
    public void importFromXML(final String filename) throws PasswordSafeException {

        final File theFile = new File(filename);

        if (!theFile.exists()) {
            throw new RuntimeException(
                    Messages.getString("PasswordSafeJFace.ImportXML.NoFileFound") + filename + "]"); //$NON-NLS-1$ //$NON-NLS-2$
        }
        BufferedInputStream bis;
        final ByteArrayOutputStream baos = new ByteArrayOutputStream();
        try {
            bis = new BufferedInputStream(new FileInputStream(filename));
        } catch (final FileNotFoundException e) {
            throw new RuntimeException(Messages.getString("PasswordSafeJFace.ImportXML.CantOpen"), e); //$NON-NLS-1$
        }

        String utf8String;
        try {
            int c;

            while ((c = bis.read()) != -1) {
                baos.write(c);
            }

            final byte[] utf8Bytes = baos.toByteArray();
            utf8String = new String(utf8Bytes, "UTF-8"); //$NON-NLS-1$
            final char header = utf8String.charAt(0);
            if (header > 255) {
                utf8String = utf8String.substring(1); // skip utf leading char
            }

        } catch (final Exception ioe) {
            throw new RuntimeException(Messages.getString("PasswordSafeJFace.ImportXML.IOException"), ioe); //$NON-NLS-1$
        } finally {
            IOUtils.closeQuietly(bis);
        }

        final XMLDataParser xdp = new XMLDataParser();
        final PwsEntryBean[] entries = xdp.parse(utf8String);
        if (entries != null && entries.length > 0) {
            final PwsFile pwsFile = getPwsFile();
            for (final PwsEntryBean entry : entries) {
                entry.setSparse(false);
                dataStore.addEntry(entry);
            }
            this.updateViewers();
        }

    }

    /**
     * Export the current safe to the name file in XML format.
     * 
     * @param filename the filename to export to
     * @throws IOException if there was a problem with the export to the XML
     *         file
     */
    public void exportToXML(final String filename) throws IOException {

        final PwsFile pwsFile = getPwsFile();
        if (pwsFile != null) {
            final List<PwsEntryBean> sparseEntries = dataStore.getSparseEntries();
            final List<PwsEntryBean> entryList = new ArrayList<PwsEntryBean>(sparseEntries.size());
            for (final PwsEntryBean sparseEntry : sparseEntries) {
                final PwsEntryBean nextDTO = dataStore.getEntry(sparseEntry.getStoreIndex());
                entryList.add(nextDTO);
            }
            final XMLDataParser xdp = new XMLDataParser();
            final String xmlOutput = xdp.convertToXML(entryList);
            final DataOutputStream dos = new DataOutputStream(new FileOutputStream(filename));
            // output the UTF-8 BOM byte by byte directly to the stream
            // http://tech.badpen.com/index.cfm?mode=entry&entry=21
            dos.write(239); // 0xEF
            dos.write(187); // 0xBB
            dos.write(191); // 0xBF
            dos.write(xmlOutput.getBytes("UTF-8"));
            dos.close();

            // TODO: Dispose entries !
            // for (PwsEntryBean entry : entryList) {
            // entry.dispose();
            // }
        }

    }

    /**
     * Display error dialog and log error.
     * 
     * @param title the title of the dialog
     * @param message the message to display
     * @param e the exception that caused the problem.
     */
    public void displayErrorDialog(final String title, final String message, final Exception e) {
        final MessageBox mb = new MessageBox(getShell(), SWT.ICON_ERROR);
        mb.setText(title);
        mb.setMessage(message);
        mb.open();
        log.error(message, e);
    }

    /**
     * Returns a shell listener. This shell listener gets registered with this
     * window's shell.
     * <p>
     * The default implementation of this framework method returns a new
     * listener that makes this window the active window for its window manager
     * (if it has one) when the shell is activated, and calls the framework
     * method <code>handleShellCloseEvent</code> when the shell is closed.
     * Subclasses may extend or reimplement.
     * </p>
     * 
     * @return a shell listener
     */
    @Override
    protected ShellListener getShellListener() {
        return new ShellAdapter() {
            volatile private boolean unlocking = false;

            @Override
            public void shellClosed(final ShellEvent event) {
                log.trace("PWSJface shell listener enter 'Closed'"); //$NON-NLS-1$
                event.doit = false; // don't close now

                final IPreferenceStore thePrefs = JFacePreferences.getPreferenceStore();
                if (thePrefs.getBoolean(JpwPreferenceConstants.SHOW_ICON_IN_SYSTEM_TRAY) && systemTray != null) {
                    // minimize on close when sys tray is present
                    getShell().setMinimized(true);
                    setReturnCode(OK);
                    return;
                }

                final boolean cancelled = saveAppIfDirty();
                if (cancelled) {
                    setReturnCode(OK);
                } else {
                    tidyUpOnExit();
                    handleShellCloseEvent(); // forwards to Window class
                }
                log.trace("PWSJface shell listener leave 'Closed'"); //$NON-NLS-1$
            }

            /**
             * Sent when a shell becomes the active window. The default behavior
             * is to do nothing.
             * 
             * @param e an event containing information about the activation
             */

            @Override
            public void shellActivated(final ShellEvent e) {
                // TODO: avoid recalling when aborting unlock
                log.trace("PWSJface shell listener enter 'Activated'"); //$NON-NLS-1$

                if (unlocking) {
                    log.info("PWSJface shell listener 'Activated' - aborted because unlocking"); //$NON-NLS-1$
                    e.doit = false;
                    return;
                }
                unlocking = true;
                try {
                    if (lockTask != null) {
                        lockTask.cancel();
                        lockTask = null;
                    }
                    if (lockState.isLocked()) {
                        log.info("PWSJface shell listener 'Activated' - unlocking"); //$NON-NLS-1$
                        // e.doit = false;
                        unlockDbAction.run();
                    }
                    // if (getShell().getMinimized()) {
                    // getShell().setMinimized(false);
                    // }
                } finally {
                    unlocking = false;
                }
                log.trace("PWSJface shell listener leave 'Activated'"); //$NON-NLS-1$
            }

            /**
             * Sent when a shell stops being the active window. The default
             * behavior is to do nothing.
             * 
             * @param e an event containing information about the deactivation
             */
            @Override
            public void shellDeactivated(final ShellEvent e) {
                log.trace("PWSJface shell listener enter 'Deactivated'"); //$NON-NLS-1$

                startLockTimer();
                log.trace("PWSJface shell listener leave 'Deactivated'"); //$NON-NLS-1$
            }

            /**
             * In <em>theory</em>, sent when a shell is un-minimized. On windows
             * sent when the shell is set visible. Do <strong>not</strong> call
             * setVisible(true) within this method, danger of endless loop.
             * 
             * @param e an event containing information about the
             *        un-minimization
             */

            @Override
            public void shellDeiconified(final ShellEvent e) {
                log.trace("PWSJface shell listener enter 'Deiconified'"); //$NON-NLS-1$

                if (unlocking) {
                    log.info("PWSJface shell listener deiconify - abort because of unlocking"); //$NON-NLS-1$
                    e.doit = false;
                    return;
                }
                if (lockTask != null) {
                    lockTask.cancel();
                    lockTask = null;
                }
                if (lockState.isLocked()) {
                    unlocking = true;
                    try {
                        log.info("PWSJface shell listener deiconify - unlocking"); //$NON-NLS-1$
                        e.doit = unlockDbAction.performUnlock();
                        // e.doit = false;
                        // Display.getDefault().asyncExec(unlockDbAction);
                    } finally {
                        unlocking = false;
                    }
                }
                log.trace("PWSJface shell listener leave 'Deiconified'"); //$NON-NLS-1$

            }

            /**
             * Sent when a shell is minimized. The default behavior is to do
             * nothing. <br>
             * On Mac Carbon, this is called the first time when disposing the
             * password dialogue when reactivating from Systray. The second time
             * it is not called. Still no good, therefore Systray on Mac is
             * blocked - see PasswordSafeMacOSX.main.
             * 
             * @param e an event containing information about the minimization
             */
            @Override
            public void shellIconified(final ShellEvent e) {
                log.trace("PWSJface shell listener enter 'Iconified'"); //$NON-NLS-1$

                final IPreferenceStore thePrefs = JFacePreferences.getPreferenceStore();
                if (thePrefs.getBoolean(JpwPreferenceConstants.CLEAR_CLIPBOARD_ON_MIN)) {
                    clearClipboardAction.run();
                }

                // handle Save and DB lock
                if (isDirty()) {
                    if (thePrefs.getBoolean(JpwPreferenceConstants.CONFIRM_SAVE_ON_MIN)) {
                        // TODO
                        saveFileAction.run();
                    } else {
                        saveFileAction.run();
                    }
                }

                // minimize to tray if available
                if (systemTray != null && thePrefs.getBoolean(JpwPreferenceConstants.SHOW_ICON_IN_SYSTEM_TRAY)) {
                    if (log.isDebugEnabled())
                        log.debug("Shrinking to tray");
                    getShell().setVisible(false);
                }

                // lock safe
                if (thePrefs.getBoolean(JpwPreferenceConstants.LOCK_DB_ON_MIN)) {
                    // important: lock in an async fashion
                    // otherwise LockState observers might be called
                    // before the shell has been minimized
                    getShell().getDisplay().asyncExec(lockDbAction);
                }
                startLockTimer();
                log.trace("PWSJface shell listener leave 'Iconified'"); //$NON-NLS-1$
            }

            private void startLockTimer() {
                final IPreferenceStore thePrefs = JFacePreferences.getPreferenceStore();
                if (pwsFile != null && thePrefs.getBoolean(JpwPreferenceConstants.LOCK_ON_IDLE)) {
                    if (lockTask != null) {// be on the safe side:
                        lockTask.cancel();
                    }
                    final String idleMins = thePrefs.getString(JpwPreferenceConstants.LOCK_ON_IDLE_MINS);
                    try {
                        final int mins = Integer.parseInt(idleMins);
                        lockTask = lockDbAction.createTaskTimer();
                        lockTimer.schedule(lockTask, mins * 60 * 1000);

                    } catch (final NumberFormatException anEx) {
                        log.warn("Unable to set lock timer to an amount of " + idleMins + " minutes"); //$NON-NLS-1$ //$NON-NLS-2$
                    }
                }
            }

        };
    }

    /**
     * Clears the current view of all data.
     */
    public void clearView() {
        if (isTreeViewShowing())
            tree.clearAll(true);
        else
            table.clearAll();
    }

    public Comparator getSorter() {
        return new Comparator() {
            public int compare(final Object o1, final Object o2) {
                return viewer.getSorter().compare(viewer, o1, o2);
            }
        };
    }
}