Blob Blame Raw


#include "thumbnailviewer.h"
#include "thumbnail.h"
#include "tfilepath.h"
#include "tsystem.h"

#include "tw/textfield.h"
#include "tw/colors.h"
#include "tw/event.h"
#include "tw/dragdrop.h"
#include "tw/keycodes.h"
#include "tw/mainshell.h"

#include "tw/message.h"

using namespace TwConsts;

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

class ThumbnailViewer::NameField : public TTextField {
  Thumbnail *m_thumbnail;

public:
  NameField(TWidget *parent) : TTextField(parent), m_thumbnail(0) {}
  void onFocusChange(bool on) {
    if (!on) close();
  }

  void close() {
    if (m_thumbnail) {
      try {
        m_thumbnail->setName(toString(getText()));
      } catch (...) {
        TMessage::error("unable to change name");
      }
    }
    hide();
  }

  void keyDown(int key, unsigned long flags, const TPoint &p) {
    if (key == TK_Return)
      close();
    else
      TTextField::keyDown(key, flags, p);
  }

  void setThumbnail(Thumbnail *thumbnail) {
    m_thumbnail = thumbnail;
    setText(m_thumbnail->getName());
    selectAll();
  }
};

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

ThumbnailViewer::ThumbnailViewer(TWidget *parent, string name)
    : TScrollView(parent, name)
    , m_selectedItemIndex(-1)
    , m_itemSize(94, 100)
    , m_itemSpace(10, 10)
    , m_margins(10, 10)
    , m_playButtonBox(TPoint(85, 2), TDimension(10, 10))
    , m_iconBox(TPoint(2 + 4 - 3, 20 + 4), TDimension(96 - 8, 78 - 8))
    , m_textBox(TPoint(2 + 4 + 2 + 10, 2 + 2), TDimension(65, 18))
    , m_flag(false)
    , m_playing(false)
    , m_loading(false)
    , m_timerActive(false)
    , m_dragDropArmed(false)
    , m_oldPos(0, 0) {
  setBackgroundColor(White);
  m_nameField = new NameField(this);
  m_nameField->hide();
}

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

ThumbnailViewer::~ThumbnailViewer() { clearPointerContainer(m_items); }

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

int ThumbnailViewer::getColumnCount() const {
  int columnWidth    = m_itemSize.lx + m_itemSpace.lx;
  int availableWidth = getLx() - m_margins.lx * 2 + m_itemSpace.lx;
  return tmax(1, (availableWidth) / columnWidth);
}

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

TPoint ThumbnailViewer::getItemPos(int index) {
  int columnCount = getColumnCount();
  int row         = index / columnCount;
  int col         = index % columnCount;
  int x           = m_margins.lx + col * (m_itemSize.lx + m_itemSpace.lx);
  int y           = getLy() - m_margins.ly - m_itemSize.ly -
          row * (m_itemSize.ly + m_itemSpace.ly);
  return TPoint(x, y);
}

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

TRect ThumbnailViewer::getItemBounds(int index) {
  return TRect(getItemPos(index), m_itemSize);
}

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

int ThumbnailViewer::findItem(const TPoint &pos) {
  int columnCount = getColumnCount();
  int col         = (pos.x - m_margins.lx) / (m_itemSize.lx + m_itemSpace.lx);
  int row         = (getLy() - m_margins.ly - m_itemSpace.ly - pos.y) /
            (m_itemSize.ly + m_itemSpace.ly);
  int index = row * columnCount + col;
  if (0 <= index && index < (int)m_items.size() &&
      getItemBounds(index).contains(pos))
    return index;
  else
    return -1;
}

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

int ThumbnailViewer::getItemCount() const { return m_items.size(); }

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

Thumbnail *ThumbnailViewer::getItem(int index) const {
  assert(0 <= index && index < (int)m_items.size());
  return m_items[index];
}

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

void ThumbnailViewer::configureNotify(const TDimension &d) {
  TScrollView::configureNotify(d);
  // m_nameField->setGeometry(TPoint(10,10), TDimension(100,30));
  updateContentSize();
  invalidate();
}

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

void ThumbnailViewer::drawFrame(int index) {
  TPoint base = getItemPos(index);
  assert(0 <= index && index < (int)m_items.size());
  assert(m_items[index]->getRaster());
  assert(m_iconBox.getSize() == m_items[index]->getRaster()->getSize());
  rectwrite(m_items[index]->getRaster(), base + m_iconBox.getP00());
}

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

void ThumbnailViewer::drawItem(int index) {
  TPoint base = getItemPos(index);
  if (index < 0 || index >= (int)m_items.size()) {
    setColor(getBackgroundColor());
    fillRect(TRect(base, m_itemSize));
    return;
  }

  // setColor(Black);
  // fillRect(TRect(base, m_itemSize));

  Thumbnail *item = m_items[index];
  bool selected   = item != 0 && index == m_selectedItemIndex;

  TRect r0(base, m_itemSize);
  TRect r1 = m_iconBox + base;
  TRect r2 = m_textBox + base;

  setColor(selected ? Cyan : getBackgroundColor());
  fillRect(r0.x0, r1.y0, r1.x0 - 1, r1.y1);  // sinistra
  fillRect(r1.x1 + 1, r1.y0, r0.x1, r1.y1);  // destra
  fillRect(r0.x0, r0.y0, r0.x1, r1.y0 - 1);  // sotto
  // fillRect(r1.x0,r1.y1+1,r1.x1,r0.y1); // sopra
  TRect topLine(r0.x0, r1.y1 + 1, r0.x1, r0.y1);
  TRect sceneLine = topLine;
  sceneLine.y0    = sceneLine.y1 - 3;
  topLine.y1      = sceneLine.y0 - 1;
  fillRect(topLine);

  if (item->getType() == Thumbnail::SCENE) {
    setColor(Black);
    drawRect(sceneLine);
    setColor(White);
    fillRect(sceneLine.enlarge(-1));
    int x;
    setColor(Black);
    for (x = sceneLine.x0 + 5; x + 5 < sceneLine.x1; x += 10) {
      int y = sceneLine.y0 + 1;
      drawLine(x, y, x + 4, y);
      y++;
      x++;
      drawLine(x, y, x + 4, y);
    }
  } else {
    setColor(White);
    fillRect(sceneLine);
  }

  string name   = item->getName();
  int maxWidth  = r2.getLx() - 4;
  TDimension ts = getTextSize(name);
  if (ts.lx > maxWidth) {
    do {
      name.pop_back();
      ts = getTextSize(name);
    } while (name.size() > 1 && ts.lx > maxWidth);
  }

  TPoint textOrigin = r2.getP00() + TPoint(2 - 10, (r2.getLy() - ts.ly) / 2);

  setColor(Black);
  string fname          = name;
  string type           = item->getPath().getType();
  if (type != "") fname = fname + "." + type;

  fname = item->getPath().getLevelName();
  drawText(textOrigin, fname);

  if (selected) {
    TRect r = m_playButtonBox + base;
    setColor(Blue);
    if (m_playing) {
      int x0 = r.x0, x3 = r.x1;
      int x1 = (x0 + x3) / 2 - 1;
      int x2 = x3 - (x1 - x0);
      drawRect(x0, r.y0, x1, r.y1);
      drawRect(x2, r.y0, x3, r.y1);
    } else {
      vector<TPoint> pts;
      int d = (r.y1 - r.y0 + 1) / 2;
      int x = r.x0;
      int y = r.y0;
      pts.push_back(TPoint(x, y));
      pts.push_back(TPoint(x + d, y + d));
      pts.push_back(TPoint(x, y + d * 2));
      fillPolygon(Cyan, Blue, &pts[0], 3);
    }
  }
  assert(0 <= index && index < (int)m_items.size());
  assert(m_items[index]->getRaster());
  assert(m_iconBox.getSize() == m_items[index]->getRaster()->getSize());
  rectwrite(m_items[index]->getRaster(), base + m_iconBox.getP00());
}

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

void ThumbnailViewer::drawBackground() {
  int itemCount   = m_items.size();
  int columnCount = getColumnCount();
  int rowCount    = (itemCount + columnCount - 1) / columnCount;

  TRect clipRect;
  getClipRect(clipRect);  // coordinate window
  clipRect -= TPoint(m_xoff, m_yoff);
  TRect rect;
  rect.x0 = m_margins.lx;
  rect.x1 = m_margins.lx + (m_itemSize.lx + m_itemSpace.lx) * columnCount -
            m_itemSpace.lx - 1;
  rect.y1 = getLy() - m_margins.ly - 1;
  rect.y0 = rect.y1 - (m_itemSize.ly + m_itemSpace.ly) * rowCount +
            m_itemSpace.ly + 1;

  setColor(getBackgroundColor());

  // alto
  if (rect.y1 + 1 <= clipRect.y1)
    fillRect(clipRect.x0, rect.y1 + 1, clipRect.x1, clipRect.y1);

  // basso
  if (clipRect.y0 <= rect.y0 - 1)
    fillRect(clipRect.x0, clipRect.y0, clipRect.x1, rect.y0 - 1);

  int y0 = tmax(clipRect.y0, rect.y0);
  int y1 = tmin(clipRect.y1, rect.y1);
  if (y0 <= y1) {
    // sinistra
    if (clipRect.x0 <= rect.x0 - 1) fillRect(clipRect.x0, y0, rect.x0 - 1, y1);

    // destra
    if (rect.x1 + 1 <= clipRect.x1) fillRect(rect.x1 + 1, y0, clipRect.x1, y1);
  }

  // interlinee verticali
  for (int r = 0; r < rowCount; r++) {
    int y1 = rect.y1 - (m_itemSize.ly + m_itemSpace.ly) * r + m_itemSpace.ly;
    int y0 = y1 - m_itemSpace.ly + 1;
    fillRect(clipRect.x0, y0, clipRect.x1, y1);
  }
  for (int c = 0; c < columnCount - 1; c++) {
    int x0 =
        m_margins.lx + (m_itemSize.lx + m_itemSpace.lx) * c + m_itemSize.lx;
    int x1 = x0 + m_itemSpace.lx - 1;
    fillRect(x0, clipRect.y0, x1, clipRect.y1);
  }

  // ultima riga
  c = itemCount % columnCount;
  if (c != 0) {
    TRect box = getItemBounds(itemCount - 1);
    if (box.y0 <= clipRect.y1 && box.y1 >= clipRect.y0 &&
        box.x1 + 1 <= clipRect.x1)
      fillRect(box.x1 + 1, box.y0, clipRect.x1, box.y1);
  }
}

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

void ThumbnailViewer::repaint() {
  if (m_flag) {
    if (m_selectedItemIndex) drawFrame(m_selectedItemIndex);
  } else {
    // m_loadingDirectory = false;
    // setColor(Black);
    // fillRect(getBounds());

    drawBackground();

    TRect clipRect;
    getClipRect(clipRect);  // coordinate window
    clipRect -= TPoint(m_xoff, m_yoff);

    bool loading = false;
    for (int i = 0; i < (int)m_items.size(); i++) {
      TRect bounds = getItemBounds(i);
      if (bounds.overlaps(clipRect)) {
        drawItem(i);
        if (m_items[i]->isIconLoaded() == false) loading = true;
      }
    }
    if (loading) {
      m_loading = true;
      if (!m_timerActive) {
        m_timerActive = true;
        startTimer(200);
      }
    }
  }
}

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

void ThumbnailViewer::select(int index) {
  assert(index == -1 || 0 <= index && index < (int)m_items.size());
  m_playing = false;
  if (m_timerActive) {
    m_timerActive = false;
    stopTimer();
  }
  Thumbnail *thumbnail;
  if (m_selectedItemIndex >= 0) {
    thumbnail = m_items[m_selectedItemIndex];
    thumbnail->setPlaying(false);
  }
  m_selectedItemIndex = index;
  onSelect(m_selectedItemIndex);
}

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

void ThumbnailViewer::startPlaying() {
  if (m_playing) return;
  if (m_selectedItemIndex < 0) return;
  m_playing = true;
  assert(0 <= m_selectedItemIndex && m_selectedItemIndex < (int)m_items.size());
  m_items[m_selectedItemIndex]->setPlaying(true);
  if (!m_timerActive) {
    m_timerActive = true;
    startTimer(200);
  }
}

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

void ThumbnailViewer::stopPlaying() {
  if (!m_playing) return;
  m_playing = false;
  if (m_selectedItemIndex >= 0) {
    assert(0 <= m_selectedItemIndex &&
           m_selectedItemIndex < (int)m_items.size());
    m_items[m_selectedItemIndex]->gotoFrame(0);
    m_items[m_selectedItemIndex]->setPlaying(false);
  }
  if (m_timerActive) {
    m_timerActive = false;
    stopTimer();
  }
}

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

void ThumbnailViewer::leftButtonDown(const TMouseEvent &e) {
  TPoint pos      = e.m_pos - TPoint(m_xoff, m_yoff);
  m_oldPos        = e.m_pos;
  m_dragDropArmed = false;

  m_nameField->hide();
  //  if(setGeometry(m_textBox + bounds.getP00());
  //           m_nameField->setText(m_items[index]->getName());
  //           m_nameField->show();

  int index = findItem(pos);
  if (index >= 0) {
    if (m_selectedItemIndex == index) {
      TRect bounds = getItemBounds(index);
      pos -= bounds.getP00();
      if (m_playButtonBox.contains(pos)) {
        if (!m_playing)
          startPlaying();
        else
          stopPlaying();
        invalidate();
      } else if (m_textBox.contains(pos)) {
        stopPlaying();
        m_nameField->setGeometry(m_textBox + bounds.getP00() + TPoint(-10, 0));
        m_nameField->setThumbnail(m_items[index]);
        m_nameField->show();
        setFocusOwner(m_nameField);
      } else {
        m_dragDropArmed = true;
      }
    } else {
      select(index);
      m_dragDropArmed = true;
    }
  } else
    select(-1);
  invalidate();
}

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

void ThumbnailViewer::leftButtonDrag(const TMouseEvent &e) {
  if (!m_dragDropArmed || m_selectedItemIndex < 0) return;
  if (norm2(e.m_pos - m_oldPos) < 100) return;
  m_dragDropArmed = false;

  Thumbnail *item = m_items[m_selectedItemIndex];
  TFilePath path  = item->getPath();
  if (path.getWideString() != toWideString("")) {
    TDropSource dropSource;
    std::vector<std::string> v;
    v.push_back(toString(path.getWideString()));
    TDataObject data(v);
    dropSource.doDragDrop(data);
  }
}

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

void ThumbnailViewer::leftButtonUp(const TMouseEvent &e) {
  m_dragDropArmed = false;
}

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

void ThumbnailViewer::leftButtonDoubleClick(const TMouseEvent &e) {
  TPoint pos = e.m_pos - TPoint(m_xoff, m_yoff);
  m_nameField->hide();
  int index = findItem(pos);
  if (index >= 0) onDoubleClick(index);
}

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

void ThumbnailViewer::onTimer(int) {
  stopTimer();
  m_timerActive = false;
  if (m_playing && m_selectedItemIndex >= 0) {
    Thumbnail *thumbnail = m_items[m_selectedItemIndex];
    bool ret             = thumbnail->nextFrame();
    if (ret) {
      m_timerActive = true;
      startTimer(100);
    } else {
      thumbnail->gotoFrame(0);
      thumbnail->setPlaying(false);
      m_playing = false;
    }
    invalidate();
  } else if (m_loading) {
    for (int i = 0; i < (int)m_items.size(); i++)
      if (!m_items[i]->isIconLoaded()) {
        m_items[i]->loadIcon();
        invalidate();
        return;
      }
  }
}

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

void ThumbnailViewer::addItem(Thumbnail *item) { m_items.push_back(item); }

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

void ThumbnailViewer::removeItem(Thumbnail *item) {
  std::vector<Thumbnail *>::iterator it =
      std::find(m_items.begin(), m_items.end(), item);
  if (it != m_items.end()) m_items.erase(it);

  m_selectedItemIndex = -1;
}

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

void ThumbnailViewer::clearItems() {
  clearPointerContainer(m_items);
  m_selectedItemIndex = -1;
  m_playing           = false;
  m_loading           = false;
  if (m_timerActive) {
    stopTimer();
    m_timerActive = false;
  }
}

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

void ThumbnailViewer::loadDirectory(const TFilePath &dirPath,
                                    const vector<string> &fileTypes) {
  clearItems();
  TFilePathSet fps;

  try {
    fps = TSystem::readDirectory(dirPath);
  } catch (...) {
    TMessage::error(toString(dirPath.getWideString()) +
                    ": can't read directory");
    updateContentSize();
    return;
  }

  fps = TSystem::packLevelNames(fps);
  fps.sort();

  for (TFilePathSet::iterator it = fps.begin(); it != fps.end(); it++) {
    try {
      TFilePath fp = *it;
      if (fp.getType() == "bmp" && fp.getName().find("_icon") != string::npos)
        continue;
      if (std::find(fileTypes.begin(), fileTypes.end(), fp.getType()) ==
          fileTypes.end())
        continue;
      Thumbnail *item = FileThumbnail::create(m_iconBox.getSize(), fp);
      if (item) m_items.push_back(item);
    } catch (...) {
      TMessage::error(toString(it->getWideString()) + ": can't read thumbnail");
    }
  }
  updateContentSize();
}

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

void ThumbnailViewer::updateContentSize() {
  int itemCount   = m_items.size();
  int columnCount = getColumnCount();
  int rowCount    = (itemCount + columnCount - 1) / columnCount;

  TRect rect;
  int yrange = rowCount * (m_itemSize.ly + m_itemSpace.ly) - m_itemSpace.ly +
               m_margins.ly * 2;
  setContentSize(TDimension(getLx(), yrange));
}

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

void ThumbnailViewer::scrollPage(int y) { doScrollTo_y(tmax(y, 0)); }