Blob Blame Raw


#include "filebrowser.h"
#include "thumbnail.h"
#include "thumbnailviewer.h"
#include "tw/treeview.h"
#include "tw/colors.h"
#include "tw/button.h"
#include "tw/scrollbar.h"
#include "tw/mainshell.h"
#include "tfilepath.h"
#include "tsystem.h"
#include "tenv.h"

#include "tw/message.h"

using namespace TwConsts;

//===================================================================

namespace {

#include "filebrowser_icons.h"

//===================================================================

class DirectoryBrowser : public ThumbnailViewer {
  TFilePath m_filepath;
  vector<string> m_fileTypes;

  GenericFileBrowserAction *m_fileSelChangeAction;
  GenericFileBrowserAction *m_fileDblClickAction;

public:
  DirectoryBrowser(TWidget *parent, const vector<string> &fileTypes)
      : ThumbnailViewer(parent)
      , m_fileSelChangeAction(0)
      , m_fileDblClickAction(0)
      , m_fileTypes(fileTypes) {}

  ~DirectoryBrowser() {
    if (m_fileSelChangeAction) delete m_fileSelChangeAction;

    if (m_fileDblClickAction) delete m_fileDblClickAction;
  }

  void setPath(const TFilePath &filepath);
  void setFilter(const vector<string> &fileTypes);

  TFilePath getPath() const { return m_filepath; }

  void onDoubleClick(int index);
  void onSelect(int index);

  void setFileSelChangeAction(GenericFileBrowserAction *action) {
    m_fileSelChangeAction = action;
  }

  void setFileDblClickAction(GenericFileBrowserAction *action) {
    m_fileDblClickAction = action;
  }
};

//-------------------------------------------------------------------

void DirectoryBrowser::setPath(const TFilePath &filepath) {
  m_filepath = filepath;
  loadDirectory(filepath, m_fileTypes);
}

//-------------------------------------------------------------------

void DirectoryBrowser::setFilter(const vector<string> &fileTypes) {
  m_fileTypes = fileTypes;
  if (!m_filepath.isEmpty()) loadDirectory(m_filepath, m_fileTypes);
}

//-------------------------------------------------------------------

void DirectoryBrowser::onDoubleClick(int index) {
  Thumbnail *thumbnail = getItem(index);
  if (thumbnail->getType() != Thumbnail::SCENE) return;
  TFilePath path =
      thumbnail->getPath().getParentDir() + (thumbnail->getName() + "_files");
  setPath(path);
  invalidate();
}

//-------------------------------------------------------------------

void DirectoryBrowser::onSelect(int index) {
  if (index != -1) {
    Thumbnail *thumbnail = getItem(index);
    if (m_fileSelChangeAction)
      m_fileSelChangeAction->sendCommand(thumbnail->getPath());
  }
}

//===================================================================

class MyTreeViewItem : public TTreeViewItem {
public:
  MyTreeViewItem(TTreeViewItemParent *parent) : TTreeViewItem(parent) {}

  TDimension getIconSize() const { return TDimension(26, 26); }

  void drawIcon(TTreeView *w, const TPoint &origin) {
    TRect rect(origin, getIconSize());
    w->setColor(TWidget::Black);
    w->drawRect(rect);
    rect = rect.enlarge(-4);
    w->setColor(Red);
    w->drawRect(rect);
    w->setColor(Blue);
    w->fillRect(rect.enlarge(-1));
  }

  void drawName(TTreeView *w, const TPoint &origin) {
    string name         = toString(getName());
    TDimension textSize = w->getTextSize(name);
    TDimension iconSize = getIconSize();

    TPoint pos = origin;
    pos.x += iconSize.lx;

    if (isSelected()) {
      w->setColor(ToonzHighlightColor);
      w->fillRect(pos.x + 2, pos.y + 2, pos.x + textSize.lx + 8,
                  pos.y + iconSize.ly - 1 - 2);
    }
    w->setColor(TWidget::Black);
    w->drawText(pos + TPoint(5, iconSize.ly / 2 - textSize.ly / 2), name);
  }

  /*
void draw(TTreeView *w, const TPoint &origin) {
drawIcon(w,origin);
drawName(w,origin);
}
*/
};

//===================================================================

class MyTreeView : public TTreeView {
public:
  MyTreeView(TWidget *parent, string name = "MyTreeView")
      : TTreeView(parent, name) {
    setBackgroundColor(White);
  }

  void drawButton(const TPoint &p, bool open) {
    int r1 = 3, r2 = 1;
    TRect rect(p.x - r1, p.y - r1, p.x + r1, p.y + r1);
    setColor(getBackgroundColor());
    fillRect(rect.enlarge(-1));
    setColor(Gray210);
    drawRect(rect);
    setColor(Black);
    drawLine(p.x - r2, p.y, p.x + r2, p.y);
    if (!open) drawLine(p.x, p.y - r2, p.x, p.y + r2);
  }
};

//===================================================================

class SimpleFileItem : public MyTreeViewItem {
  TFilePath m_path;

public:
  SimpleFileItem(TTreeViewItemParent *parent, const TFilePath &path)
      : MyTreeViewItem(parent), m_path(path) {
    setIsLeaf(true);
  }

  wstring getName() const { return m_path.withoutParentDir().getWideString(); }

  TFilePath getPath() const { return m_path; }
};

//-------------------------------------------------------------------

bool caseInsensitiveLessThan(const TFilePath &fp1, const TFilePath &fp2) {
  return (stricmp(toString(fp1.getWideString()).c_str(),
                  toString(fp2.getWideString()).c_str()) < 0);
}

//-------------------------------------------------------------------

class FileFolderItem : public MyTreeViewItem {
  TFilePath m_path;

public:
  FileFolderItem(TTreeViewItemParent *parent, const TFilePath &path)
      : MyTreeViewItem(parent), m_path(path) {}
  wstring getName() const {
    return m_path.isRoot() ? m_path.getWideString()
                           : m_path.withoutParentDir().getWideString();
  }

  TDimension getIconSize() const { return TDimension(16, 16); }

  void onOpen() {
    clearItems();
    try {
      TFilePathSet fps = TSystem::readDirectory(m_path);
      fps.sort(caseInsensitiveLessThan);
      for (TFilePathSet::iterator it = fps.begin(); it != fps.end(); it++) {
        if (TFileStatus(*it).isDirectory()) new FileFolderItem(this, *it);
        // else
        //  new SimpleFileItem(this, *it);
      }
    } catch (TException &e) {
      TMessage::error(toString(e.getMessage()));
    }
  }
  TFilePath getPath() const { return m_path; }

  void drawIcon(TTreeView *w, const TPoint &origin) {
    w->rectwrite(isOpen() ? openfolder : closedfolder, origin);
  }
};

//-------------------------------------------------------------------

class MyComputerFolder : public MyTreeViewItem {
public:
  MyComputerFolder(TTreeViewItemParent *parent) : MyTreeViewItem(parent) {}

  wstring getName() const { return toWideString("My Computer"); }

  void onOpen() {
    clearItems();

    TFilePathSet fps = TSystem::getDisks();
    for (TFilePathSet::iterator it = fps.begin(); it != fps.end(); it++) {
      TTreeViewItem *item = new FileFolderItem(this, *it);
    }
  }
  void drawIcon(TTreeView *w, const TPoint &origin) {
    w->rectwrite(mycomputer, origin);
  }
};

//-------------------------------------------------------------------

class HistoryFolder : public MyTreeViewItem {
public:
  HistoryFolder(TTreeViewItemParent *parent) : MyTreeViewItem(parent) {}

  wstring getName() const { return toWideString("History"); }

  void onOpen() { clearItems(); }
  void drawIcon(TTreeView *w, const TPoint &origin) {
    w->rectwrite(history, origin);
  }
};

//-------------------------------------------------------------------

class LibraryFolder : public MyTreeViewItem {
public:
  LibraryFolder(TTreeViewItemParent *parent) : MyTreeViewItem(parent) {}

  wstring getName() const { return toWideString("Library"); }

  void onOpen() {
    clearItems();
    TFilePath root = TEnv::getRootDir();
    new FileFolderItem(this, root + "patterns");
    new FileFolderItem(this, root + "images");
    new FileFolderItem(this, root + "scenes");
  }
  void drawIcon(TTreeView *w, const TPoint &origin) {
    w->rectwrite(library, origin);
  }
};

//-------------------------------------------------------------------

class WorkFolder : public FileFolderItem {
public:
  WorkFolder(TTreeViewItemParent *parent)
      : FileFolderItem(parent, TEnv::getRootDir() + "work") {}

  TDimension getIconSize() const { return TDimension(24, 24); }

  wstring getName() const { return toWideString("Work"); }

  void drawIcon(TTreeView *w, const TPoint &origin) {
    w->rectwrite(workinprogress, origin);
  }
};
//-------------------------------------------------------------------

class FileFolderTreeView : public MyTreeView {
  DirectoryBrowser *m_viewer;
  TTreeViewItem *m_myComputer;

public:
  FileFolderTreeView(TWidget *parent, DirectoryBrowser *viewer)
      : MyTreeView(parent), m_viewer(viewer) {
    m_myComputer = new MyComputerFolder(this);
    m_myComputer->open();

    /*
new HistoryFolder(this);
new LibraryFolder(this);
new WorkFolder(this);
*/
  }

  void leftButtonDoubleClick(const TMouseEvent &e);

  void onSelect(TTreeViewItem *item);
  void setCurrentDir(const TFilePath &dirPath);
};

//===================================================================

void FileFolderTreeView::leftButtonDoubleClick(const TMouseEvent &e) {
  TTreeViewItem *item = getSelectedItem();
  if (item) {
    if (!item->isOpen())
      item->open();
    else
      item->close();

    updateVisibleItems();
    invalidate();
  }
}

//-------------------------------------------------------------------

void FileFolderTreeView::onSelect(TTreeViewItem *item) {
  FileFolderItem *folder = dynamic_cast<FileFolderItem *>(item);
  if (folder && folder->getPath() != m_viewer->getPath()) {
    TFilePath path = folder->getPath();
    m_viewer->setPath(path);
    m_viewer->invalidate();
  }
}

//-------------------------------------------------------------------

TTreeViewItem *openChild(TTreeViewItem *folder, const TFilePath &childPath) {
  int count = folder->getItemCount();
  for (int i = 0; i < count; ++i) {
    FileFolderItem *item = dynamic_cast<FileFolderItem *>(folder->getItem(i));

    if (item && item->getPath() == childPath) {
      item->open();
      return item;
    }
  }

  return 0;
}

//-------------------------------------------------------------------

void FileFolderTreeView::setCurrentDir(const TFilePath &dirPath) {
  assert(TFileStatus(dirPath).isDirectory());

  list<TFilePath> ancestors;
  ancestors.push_back(dirPath);

  if (!dirPath.isRoot()) {
    TFilePath fp = dirPath;
    while (!(fp = fp.getParentDir()).isRoot()) ancestors.push_front(fp);
    ancestors.push_front(fp);
  }

  TTreeViewItem *folder = m_myComputer;
  while (!ancestors.empty() && folder) {
    TFilePath fp = ancestors.front();
    folder       = openChild(folder, fp);
    ancestors.pop_front();
  }

  if (folder) {
    updateVisibleItems();
    select(folder);

    TRect rect;
    getPlacement(m_selectedItemIndex, rect);
    int y = rect.getP00().y;
    if (y <= 0)
      y = m_yoffset - y;
    else if (y > getLy())
      y = m_yoffset - (rect.y1 - getLy());
    else
      y = m_yoffset;

    scrollToY(y);

    updateScrollbars();
  }
}

};  // anonymous namespace

//==============================================================================
//==============================================================================

class FileBrowser::Data {
public:
  Data() {}
  ~Data() {}

  FileFolderTreeView *m_tree;
  DirectoryBrowser *m_panel;
  TScrollbar *m_panelScb, *m_treeVscb, *m_treeHscb;

  TIconButton *m_upButton, *m_createFolderButton, *m_deleteFileButton;

  vector<string> m_fileTypes;
};

//-------------------------------------------------------------------

FileBrowser::FileBrowser(TWidget *parent, string name,
                         const vector<string> &fileTypes)
    : TWidget(parent, name), m_data(new Data()) {
  m_data->m_fileTypes = fileTypes;

  m_data->m_panel = new DirectoryBrowser(this, fileTypes);
  m_data->m_tree  = new FileFolderTreeView(this, m_data->m_panel);

  m_data->m_panelScb = new TScrollbar(this);
  m_data->m_treeHscb = new TScrollbar(this);
  m_data->m_treeVscb = new TScrollbar(this);

  m_data->m_panel->setScrollbars(0, m_data->m_panelScb);
  m_data->m_tree->setScrollbars(m_data->m_treeHscb, m_data->m_treeVscb);

  setBackgroundColor(Gray210);

  m_data->m_upButton = new TIconButton(this, parentfolder);
  tconnect(*m_data->m_upButton, this, &FileBrowser::selectParentDirectory);

  m_data->m_createFolderButton = new TIconButton(this, newfolder);
  m_data->m_deleteFileButton   = new TIconButton(this, trash);
}

//-------------------------------------------------------------------

FileBrowser::~FileBrowser() { delete m_data; }

//-------------------------------------------------------------------

void FileBrowser::setFilter(const vector<string> &fileTypes) {
  m_data->m_fileTypes = fileTypes;
  m_data->m_panel->setFilter(fileTypes);
}

//-------------------------------------------------------------------

void FileBrowser::draw() {
  setColor(White);
  fillRect(0, 0, 4, getLy() - 1);

  TPoint p;
  p = m_data->m_treeVscb->getGeometry().getP11();

  setColor(Gray150);
  drawLine(p.x + 1, 1, p.x + 1, p.y);

  p = m_data->m_panel->getGeometry().getP01();
  drawLine(p.x - 1, 1, p.x - 1, p.y);
  drawLine(p.x - 1, p.y + 1, getLx() - 2, p.y + 1);
}

//-------------------------------------------------------------------

void FileBrowser::configureNotify(const TDimension &d) {
  /*
TRect rect = getBounds();
int x = 200;
int w = 15;
int xa = x-w-4;
TDimension scbSize(w, rect.getLy());
m_data->m_tree->setGeometry(rect.x0,rect.y0,xa-1,rect.y1);
m_data->m_treeVscb->setGeometry(TPoint(xa,rect.y0),scbSize);

m_data->m_treeHscb->setGeometry(rect.x0, rect.y0,
x-1-scbSize,rect.y0+scbSize-1);

int buttonBarHeight = 30;

rect.y1 -= buttonBarHeight;
scbSize.ly -= buttonBarHeight;

m_data->m_panel->setGeometry(x,rect.y0,rect.x1-w,rect.y1);
m_data->m_panelScb->setGeometry(TPoint(rect.x1-w+1,rect.y0),scbSize);

int y = rect.y1+4;

m_data->m_upButton->setPosition(x-1, y);
m_data->m_createFolderButton->setPosition(x+25, y);
m_data->m_deleteFileButton->setPosition(rect.x1-26, y);

invalidate();

*/

  const int scbSize = 15;

  TRect rect = getBounds();
  int x      = 196;

  m_data->m_tree->setGeometry(rect.x0, rect.y0 + scbSize, x - 1 - scbSize,
                              rect.y1);
  m_data->m_treeVscb->setGeometry(x - scbSize, rect.y0 + scbSize, x - 1,
                                  rect.y1);
  m_data->m_treeHscb->setGeometry(rect.x0, rect.y0, x - 1 - scbSize,
                                  rect.y0 + scbSize - 1);

  int buttonBarHeight = 29;

  rect.y1 -= buttonBarHeight;

  m_data->m_panel->setGeometry(x + 4, rect.y0, rect.x1 - scbSize, rect.y1);
  m_data->m_panelScb->setGeometry(rect.x1 - scbSize + 1, rect.y0, rect.x1,
                                  rect.y1);

  int y = rect.y1 + 4;
  x += 4;
  m_data->m_upButton->setPosition(x - 1, y);
  m_data->m_createFolderButton->setPosition(x + 25, y);
  m_data->m_deleteFileButton->setPosition(rect.x1 - 26, y);

  m_data->m_upButton->invalidate();
  m_data->m_createFolderButton->invalidate();
  m_data->m_deleteFileButton->invalidate();
  m_data->m_treeVscb->invalidate();
  m_data->m_treeHscb->invalidate();
  m_data->m_tree->invalidate();
  invalidate();
}

//-------------------------------------------------------------------

void FileBrowser::setBackgroundColor(const TGuiColor &color) {
  TWidget::setBackgroundColor(color);
  m_data->m_panel->setBackgroundColor(color);
  m_data->m_tree->setBackgroundColor(color);
}

//-------------------------------------------------------------------

void FileBrowser::setFileSelChangeAction(GenericFileBrowserAction *action) {
  m_data->m_panel->setFileSelChangeAction(action);
}

//-------------------------------------------------------------------

void FileBrowser::setFileDblClickAction(GenericFileBrowserAction *action) {
  m_data->m_panel->setFileDblClickAction(action);
}

//-------------------------------------------------------------------

TFilePath FileBrowser::getCurrentDir() const {
  FileFolderItem *item =
      dynamic_cast<FileFolderItem *>(m_data->m_tree->getSelectedItem());
  if (item)
    return item->getPath();
  else
    return TFilePath();
}

//-------------------------------------------------------------------

void FileBrowser::setCurrentDir(const TFilePath &dirPath) {
  m_data->m_tree->setCurrentDir(dirPath);
}

//-------------------------------------------------------------------

void FileBrowser::selectParentDirectory() { m_data->m_tree->selectParent(); }