#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);
}