Blob Blame Raw


#include "textlist.h"
#include "tw/colors.h"
#include "tw/event.h"
#include "tw/scrollbar.h"
#include "tw/keycodes.h"

#include <vector>
#include <map>
#include <string>

using namespace std;
using namespace TwConsts;

namespace {
const int rowHeight = 20;  // 15
};

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

class TTextList::Data {
public:
  Data(TWidget *w)
      : m_w(w)
      , m_selAction(0)
      , m_dblClickAction(0)
      , m_scrollbar(new TScrollbar(w))
      , m_yoffset(0) {}

  ~Data() {
    if (m_selAction) delete m_selAction;

    if (m_dblClickAction) delete m_dblClickAction;

    std::map<string, TTextListItem *>::iterator it = m_items.begin();
    for (; it != m_items.end(); ++it) delete it->second;
  }

  int posToItem(const TPoint &p);
  void updateScrollBarStatus() {
    if (!m_scrollbar) return;
    unsigned int ly = m_w->getLy();

    if ((m_items.size() * rowHeight) > ly) {
      m_scrollbar->setValue(m_yoffset, 0, (m_items.size() + 1) * rowHeight - ly,
                            ly);
      m_scrollbar->show();  // m_scrollbar->setValue(m_yoffset,0, yrange-ly,
                            // ly);
    } else {
      m_yoffset = 0;
      m_scrollbar->hide();  // m_scrollbar->setValue(0,0, 0, 0);
    }
    m_scrollbar->invalidate();
  }

  TWidget *m_w;
  std::map<string, TTextListItem *> m_items;
  vector<string> m_selectedItems;

  TGenericTextListAction *m_selAction;
  TGenericTextListAction *m_dblClickAction;
  TScrollbar *m_scrollbar;
  int m_yoffset;
};

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

int TTextList::Data::posToItem(const TPoint &p) {
  /*  TDimension d = getSize();
int i = getItemCount()-1;

10 +
for(int y=10; y<d.ly && i>=0; y+=rowHeight, --i)
*/

  int y    = m_w->getSize().ly - p.y + m_yoffset;
  int item = y / rowHeight;
  return item;
}

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

TTextListItem::TTextListItem(const string &id, const string &caption)
    : m_id(id), m_caption(caption) {}

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

TTextList::TTextList(TWidget *parent, string name)
    : TWidget(parent, name), m_data(0) {
  m_data = new Data(this);
  m_data->m_scrollbar->setAction(
      new TScrollbarAction<TTextList>(this, &TTextList::scrollTo));
}

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

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

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

void TTextList::addItem(TTextListItem *item) {
  std::map<string, TTextListItem *>::iterator it =
      m_data->m_items.find(item->getId());

  if (it == m_data->m_items.end()) {
    m_data->m_items.insert(make_pair(item->getId(), item));
    m_data->updateScrollBarStatus();
  }
}

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

void TTextList::removeItem(const string &itemId) {
  std::map<string, TTextListItem *>::iterator it = m_data->m_items.find(itemId);
  if (it != m_data->m_items.end()) {
    m_data->m_items.erase(it);
    m_data->updateScrollBarStatus();
  }
}

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

void TTextList::clearAll() {
  m_data->m_items.clear();
  m_data->m_selectedItems.clear();
  invalidate();
  m_data->updateScrollBarStatus();
}

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

int TTextList::getItemCount() const { return m_data->m_items.size(); }

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

TTextListItem *TTextList::getItem(int i) const {
  if (i >= 0 && i < (int)m_data->m_items.size()) {
    std::map<string, TTextListItem *>::iterator it = m_data->m_items.begin();
    advance(it, i);
    return it->second;
  }
  return 0;
}

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

int TTextList::itemToIndex(const string &itemId) {
  std::map<string, TTextListItem *>::iterator it = m_data->m_items.find(itemId);

  if (it == m_data->m_items.end())
    return -1;
  else
    return distance(m_data->m_items.begin(), it);
}

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

int TTextList::getSelectedItemCount() const {
  return m_data->m_selectedItems.size();
}

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

TTextListItem *TTextList::getSelectedItem(int i) const {
  if (i < 0 || (int)m_data->m_selectedItems.size() <= i) return 0;

  string itemId = m_data->m_selectedItems[i];
  std::map<string, TTextListItem *>::iterator it = m_data->m_items.find(itemId);
  if (it != m_data->m_items.end())
    return it->second;
  else
    return 0;
}

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

string TTextList::getSelectedItemId(int i) const {
  if ((int)m_data->m_selectedItems.size() <= i) return "";
  assert(i >= 0 && i < (int)m_data->m_selectedItems.size());
  return m_data->m_selectedItems[i];
}
//------------------------------------------------------------------------------

void TTextList::select(int i, bool on) {
  assert(i >= 0 && i < (int)m_data->m_items.size());

  std::map<string, TTextListItem *>::iterator it = m_data->m_items.begin();
  advance(it, i);

  string id = it->first;
  vector<string>::iterator it2 =
      find(m_data->m_selectedItems.begin(), m_data->m_selectedItems.end(), id);

  if (on) {
    if (it2 == m_data->m_selectedItems.end())
      m_data->m_selectedItems.push_back(id);
  } else {
    if (it2 != m_data->m_selectedItems.end())
      m_data->m_selectedItems.erase(it2);
  }

  if (m_data->m_selAction) m_data->m_selAction->sendCommand(i);

  invalidate();
}

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

void TTextList::select(const string &itemId, bool on) {
  std::map<string, TTextListItem *>::iterator it = m_data->m_items.find(itemId);

  if (it != m_data->m_items.end()) {
    int i = distance(m_data->m_items.begin(), it);
    select(i, on);
  }
}

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

void TTextList::unselectAll() {
  m_data->m_selectedItems.clear();
  invalidate();
}

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

bool TTextList::isSelected(int i) const {
  assert(i >= 0 && i < (int)m_data->m_items.size());

  std::map<string, TTextListItem *>::iterator it = m_data->m_items.begin();
  advance(it, i);

  string id = it->first;

  vector<string>::iterator it2 =
      find(m_data->m_selectedItems.begin(), m_data->m_selectedItems.end(), id);

  return it2 != m_data->m_selectedItems.end();
}

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

bool TTextList::isSelected(const string &item) const {
  assert(false);
  return false;
}

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

void TTextList::scrollTo(int y) {
  y = (y / rowHeight) * rowHeight;
  if (m_data->m_yoffset == y) return;
  m_data->m_yoffset = y;
  m_data->updateScrollBarStatus();
  invalidate();
}

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

void TTextList::draw() {
  drawRect(TRect(TPoint(0, 0), getSize()));
  TDimension d = getSize();

  int y = d.ly - 1 - rowHeight;
  for (int i = m_data->m_yoffset / rowHeight; i < getItemCount() && y >= 0;
       ++i, y -= rowHeight) {
    if (isSelected(i)) {
      setColor(Blue, 2);
      fillRect(2, y - 2, getSize().lx - 1 - 2, y + rowHeight - 1 - 2);
    }

    setColor(Black);
    drawText(TPoint(10, y), getItem(i)->getCaption());
  }
}

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

void TTextList::setSelAction(TGenericTextListAction *action) {
  if (m_data->m_selAction) delete m_data->m_selAction;

  m_data->m_selAction = action;
}

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

void TTextList::setDblClickAction(TGenericTextListAction *action) {
  if (m_data->m_dblClickAction) delete m_data->m_dblClickAction;

  m_data->m_dblClickAction = action;
}

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

void TTextList::configureNotify(const TDimension &d) {
  m_data->m_scrollbar->setGeometry(d.lx - 20, 1, d.lx - 2, d.ly - 2);
  m_data->updateScrollBarStatus();
}

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

void TTextList::leftButtonDown(const TMouseEvent &e) {
  int i = m_data->posToItem(e.m_pos);
  if (i >= 0 && i < (int)m_data->m_items.size()) {
    if (!e.isShiftPressed()) unselectAll();
    select(i, !isSelected(i));
  }
}

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

void TTextList::leftButtonDoubleClick(const TMouseEvent &e) {
  int i = m_data->posToItem(e.m_pos);
  if (i >= 0 && i < (int)m_data->m_items.size()) {
    if (m_data->m_dblClickAction) m_data->m_dblClickAction->sendCommand(i);
  }
}

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

void TTextList::keyDown(int key, unsigned long mod, const TPoint &pos) {
  if ((key == TK_UpArrow) || (key == TK_DownArrow)) {
    int lastSelected = -1;
    for (int i                        = 0; i < getItemCount(); ++i)
      if (isSelected(i)) lastSelected = i;
    if (lastSelected == -1) return;
    int newSelected = (key == TK_UpArrow) ? lastSelected - 1 : lastSelected + 1;

    if (newSelected >= 0 && newSelected < (int)m_data->m_items.size()) {
      //   if (!e.isShiftPressed())
      unselectAll();
      select(newSelected, !isSelected(newSelected));
    }
  } else
    TWidget::keyDown(key, mod, pos);
}