Blob Blame Raw


#include "tgrid.h"
#include "tw/colors.h"
#include "tw/event.h"

using namespace TwConsts;

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

class TGridColumn::Imp
{
public:
	vector<TGridCell> m_cells;
	string m_name;
	TGridColumn::Alignment m_alignment;

	Imp(const string &name, Alignment alignment)
		: m_name(name), m_alignment(alignment) {}

	~Imp() {}

	Imp &operator=(const Imp &src)
	{
		m_cells = src.m_cells;
		return *this;
	}

private:
	//not implemented
	Imp(const Imp &);
};

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

TGridColumn::TGridColumn(const string &name, Alignment alignment)
	: m_imp(new Imp(name, alignment))
{
}

TGridColumn::~TGridColumn()
{
	delete m_imp;
}

TGridColumn *TGridColumn::createEmpty()
{
	return new TGridColumn();
}

int TGridColumn::getRowCount() const
{
	return m_imp->m_cells.size();
}

const TGridCell &TGridColumn::getCell(int row) const
{
	assert(row >= 0 && row < (int)m_imp->m_cells.size());
	return m_imp->m_cells[row];
}

void TGridColumn::setCell(int row, const TGridCell &cell)
{
	assert(m_imp);
	assert(row >= 0);

	//checkColumn();

	int firstRow = 0;
	if (m_imp->m_cells.empty()) //se la colonna e' vuota
	{
		//    if(!cell.isEmpty())
		{
			m_imp->m_cells.push_back(cell);
			firstRow = row;
		}
		return;
	}

	int oldCellCount = m_imp->m_cells.size();
	assert(oldCellCount > 0);
	int lastRow = firstRow + oldCellCount - 1;

	if (row < firstRow) //prima
	{
		//    if(cell.isEmpty()) return; //non faccio nulla
		int delta = firstRow - row;
		assert(delta > 0);
		m_imp->m_cells.insert(m_imp->m_cells.begin(), delta - 1, TGridCell()); //celle vuote
		m_imp->m_cells.insert(m_imp->m_cells.begin(), cell);				   //devo settare la prima comp. del vettore
		firstRow = row;														   //row 'e la nuova firstrow
																			   //    updateIcon();
																			   //    checkColumn();
		return;
	} else if (row > lastRow) //dopo
	{
		//    if(cell.isEmpty()) return; //non faccio nulla
		int count = row - lastRow - 1;
		//se necessario, inserisco celle vuote
		for (int i = 0; i < count; ++i)
			m_imp->m_cells.push_back(TGridCell());
		m_imp->m_cells.push_back(cell);
		//    checkColumn();
		return;
	}
	//"[r0,r1]"
	int index = row - firstRow;
	assert(0 <= index && index < (int)m_imp->m_cells.size());
	m_imp->m_cells[index] = cell;
	//  if(index == 0) updateIcon();

	/*
  if(cell.isEmpty())
  {
    if(row==lastRow)
    {
      // verifico la presenza di celle bianche alla fine     
      while(!m_imp->m_cells.empty() && m_imp->m_cells.back().isEmpty()) 
        m_imp->m_cells.pop_back();     
    }
    else
    if(row==firstRow)
    {
      // verifico la presenza di celle bianche all'inizio
      while(!m_imp->m_cells.empty() && m_imp->m_cells.front().isEmpty()) 
      {
       m_imp->m_cells.erase(m_imp->m_cells.begin());  
       firstRow++;
      }
    }
  
    if(m_imp->m_cells.empty())
      firstRow = 0;
  }
  checkColumn();
*/
}

void TGridColumn::getCells(int row, int rowCount, TGridCell cells[]);
void TGridColumn::setCells(int row, int rowCount, const TGridCell cells[]);

void TGridColumn::insertEmptyCells(int row, int rowCount)
{
	assert(m_imp);
	if (m_imp->m_cells.empty())
		return; //se la colonna e' vuota non devo inserire celle

	int firstRow = 0;
	if (row >= firstRow + (int)m_imp->m_cells.size())
		return;			 //dopo:non inserisco nulla
	if (row <= firstRow) //prima
	{
		firstRow += rowCount;
	} else //in mezzo
	{
		int delta = row - firstRow;
		std::vector<TGridCell>::iterator it = m_imp->m_cells.begin();
		std::advance(it, delta);
		m_imp->m_cells.insert(it, rowCount, TGridCell());
	}
}

void TGridColumn::removeCells(int row, int rowCount)
{
	if (rowCount <= 0)
		return;
	if (m_imp->m_cells.empty())
		return; //se la colonna e' vuota
	assert(m_imp);

	int firstRow = 0; //m_imp->m_first;
	int cellCount = m_imp->m_cells.size();

	if (row >= firstRow + cellCount)
		return; //sono "sotto" l'ultima cella
	if (row < firstRow) {
		if (row + rowCount <= firstRow) //"sono sopra la prima cella"
		{								// aggiorno solo m_first
			firstRow -= rowCount;
			return;
		}
		rowCount += row - firstRow; //rowCount = row+rowCount-firstRow;
		firstRow = row;
	}

	assert(row >= firstRow);
	// le celle sotto firstRow+cellCount sono gia' vuote
	if (rowCount > firstRow + cellCount - row)
		rowCount = firstRow + cellCount - row;
	if (rowCount <= 0)
		return;

	if (row == firstRow) {
		//cancello all'inizio
		assert(rowCount <= cellCount);
		std::vector<TGridCell>::iterator it = m_imp->m_cells.begin();
		std::vector<TGridCell>::iterator it2 = m_imp->m_cells.begin();
		std::advance(it2, rowCount);
		//m_imp->m_cells.erase(&m_imp->m_cells[0],&m_imp->m_cells[rowCount]);
		m_imp->m_cells.erase(it, it2);
		// verifico la presenza di celle bianche all'inizio
		while (!m_imp->m_cells.empty() /*&& m_imp->m_cells.front().isEmpty()*/) {
			m_imp->m_cells.erase(m_imp->m_cells.begin());
			firstRow++;
		}
	} else {
		// cancello dopo l'inizio
		int d = row - firstRow;
		std::vector<TGridCell>::iterator it = m_imp->m_cells.begin();
		std::vector<TGridCell>::iterator it2 = m_imp->m_cells.begin();
		std::advance(it, d);
		std::advance(it2, d + rowCount);
		//m_imp->m_cells.erase(&m_imp->m_cells[d],&m_imp->m_cells[d+rowCount]);
		m_imp->m_cells.erase(it, it2);
		if (row + rowCount == firstRow + cellCount) {
			// verifico la presenza di celle bianche alla fine
			while (!m_imp->m_cells.empty() /*&& m_imp->m_cells.back().isEmpty()*/) {
				m_imp->m_cells.pop_back();
			}
		}
	}

	if (m_imp->m_cells.empty()) {
		firstRow = 0;
	}
	//updateIcon();
}

void TGridColumn::clearCells(int row, int rowCount)
{
}

string TGridColumn::getName() const
{
	return m_imp->m_name;
}

TGridColumn::Alignment TGridColumn::getAlignment() const
{
	return m_imp->m_alignment;
}

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

class TGrid::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;
	}

	int posToRow(const TPoint &p);

	void updateScrollBarStatus()
	{
		if (!m_scrollbar)
			return;
		int ly = m_w->getLy();

		assert(m_columnSet.getColumnCount() > 0);
		TGridColumnP col = m_columnSet.getColumn(0);

		if ((col->getRowCount() * m_rowHeight) > ly)
			m_scrollbar->show(); //m_scrollbar->setValue(m_yoffset,0, yrange-ly, ly);
		else
			m_scrollbar->hide(); //m_scrollbar->setValue(0,0, 0, 0);
		m_scrollbar->invalidate();
	}

	TWidget *m_w;

	TColumnSetT<TGridColumn> m_columnSet;

	vector<int> m_selectedRows;

	TGenericGridAction *m_selAction;
	TGenericGridAction *m_dblClickAction;
	TScrollbar *m_scrollbar;
	int m_yoffset;

	bool m_colSeparatorDragged;
	int m_prevColSeparatorX;
	int m_colSeparatorX;
	int m_colIndex;

	static int m_rowHeight;
};

int TGrid::Data::m_rowHeight = 20;

int TGrid::Data::posToRow(const TPoint &p)
{
	int y = m_w->getSize().ly - p.y + m_yoffset;
	int item = y / m_rowHeight;
	return item;
}

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

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

	//setBackgroundColor(White);
}

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

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

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

void TGrid::addColumn(const string &name, int width, TGridColumn::Alignment align)
{
	TGridColumnP column = new TGridColumn(name, align);
	column->setWidth(width);
	m_data->m_columnSet.insertColumn(m_data->m_columnSet.getColumnCount(), column);
}

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

void TGrid::addRow()
{
	int colCount = m_data->m_columnSet.getColumnCount();
	for (int i = 0; i < colCount; ++i) {
		TGridColumnP col = m_data->m_columnSet.getColumn(i);
		col->setCell(col->getRowCount(), "");
	}
}

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

void TGrid::removeRow(int row)
{
	int colCount = m_data->m_columnSet.getColumnCount();
	for (int i = 0; i < colCount; ++i) {
		TGridColumnP col = m_data->m_columnSet.getColumn(i);
		col->removeCells(row);
	}
}

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

int TGrid::getColIndex(const string &colName)
{
	int colCount = m_data->m_columnSet.getColumnCount();
	for (int i = 0; i < colCount; ++i) {
		TGridColumnP col = m_data->m_columnSet.getColumn(i);
		if (colName == col->getName())
			return i;
	}

	return -1;
}

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

void TGrid::setCell(int row, int col, const string &text)
{
	assert(row >= 0 && col >= 0);
	TGridColumnP column = m_data->m_columnSet.touchColumn(col);
	column->setCell(row, TGridCell(text));
}

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

int TGrid::getColCount() const
{
	return m_data->m_columnSet.getColumnCount();
}

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

int TGrid::getRowCount() const
{
	assert(m_data->m_columnSet.getColumnCount() > 0);
	TGridColumnP col = m_data->m_columnSet.getColumn(0);
	return col->getRowCount();
}

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

int TGrid::getSelectedRowCount() const
{
	return m_data->m_selectedRows.size();
}

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

int TGrid::getSelectedRowIndex(int i) const
{
	assert(i >= 0 && i < (int)m_data->m_selectedRows.size());
	return m_data->m_selectedRows[i];
}

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

void TGrid::select(int row, bool on)
{
	assert(row >= 0 && row < (int)getRowCount());
	std::vector<int>::iterator it =
		std::find(m_data->m_selectedRows.begin(), m_data->m_selectedRows.end(), row);
	if (on) {
		if (it == m_data->m_selectedRows.end())
			m_data->m_selectedRows.push_back(row);
	} else {
		if (it != m_data->m_selectedRows.end())
			m_data->m_selectedRows.erase(it);
	}
	if (m_data->m_selAction)
		m_data->m_selAction->sendCommand(row);

	invalidate();
}

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

void TGrid::unselectAll()
{
	m_data->m_selectedRows.clear();
	invalidate();
}

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

bool TGrid::isSelected(int row) const
{
	std::vector<int>::iterator it =
		std::find(m_data->m_selectedRows.begin(), m_data->m_selectedRows.end(), row);

	return it != m_data->m_selectedRows.end();
}

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

void TGrid::scrollTo(int y)
{
	if (m_data->m_yoffset == y)
		return;
	m_data->m_yoffset = y;
	invalidate();
}

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

void TGrid::draw()
{
	TDimension size = getSize();

	//TRect gridHeaderPlacement(TPoint(0,size.ly-1), TDimension(size.lx-1, -m_data->m_rowHeight));

	TRect gridHeaderPlacement(0, size.ly - 1 - m_data->m_rowHeight, size.lx - 1, size.ly - 1);

	setColor(Gray180 /*White*/);
	fillRect(gridHeaderPlacement);

	int gridHeaderY0 = gridHeaderPlacement.getP00().y;

	for (int i = 0; i < m_data->m_columnSet.getColumnCount(); ++i) {
		TGridColumnP col = m_data->m_columnSet.getColumn(i);
		int x0, x1;
		col->getCoords(x0, x1);

		// draw column header
		setColor(Black);
		drawText(TRect(x0, gridHeaderY0, x1, gridHeaderY0 + m_data->m_rowHeight - 1), col->getName());

		int rowY = gridHeaderY0; // - m_data->m_rowHeight;

		int y = size.ly - 1 - m_data->m_rowHeight;
		for (int j = m_data->m_yoffset / m_data->m_rowHeight; j < col->getRowCount() && y; ++j, y -= m_data->m_rowHeight) {
			drawHLine(TPoint(0, rowY), size.lx);

			TRect cellRect(x0, rowY - m_data->m_rowHeight - 1, x1, rowY);

			if (isSelected(j)) {
				setColor(Blue, 1);
				fillRect(cellRect);
			}

			setColor(Black);

			TWidget::Alignment align = TWidget::CENTER;
			switch (col->getAlignment()) {
			case TGridColumn::CenterAlignment:
				align = TWidget::CENTER;
				break;
			case TGridColumn::RightAlignment:
				align = TWidget::END;
				break;

			case TGridColumn::LeftAlignment:
				align = TWidget::BEGIN;
				break;
			}

			drawText(cellRect, col->getCell(j).m_text, align);

			rowY -= m_data->m_rowHeight;
		}

		setColor(Gray210);
		drawVLine(TPoint(x0, 0), size.ly);
		drawVLine(TPoint(x1, 0), size.ly);
	}

	if (m_data->m_colSeparatorDragged) {
		drawVLine(TPoint(m_data->m_colSeparatorX, 0), size.ly);
	}
}

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

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

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

void TGrid::leftButtonDown(const TMouseEvent &e)
{
	m_data->m_colSeparatorDragged = false;
	int rowCount = getRowCount();

	int i = m_data->posToRow(e.m_pos);
	if (i == 0) {
		// la riga 0 e' l'header della tabella

		for (int j = 0; j < m_data->m_columnSet.getColumnCount(); ++j) {
			TGridColumnP col = m_data->m_columnSet.getColumn(j);
			int x0, x1;
			col->getCoords(x0, x1);
			if (x0 - 3 < e.m_pos.x && e.m_pos.x < x0 + 3) {
				m_data->m_colSeparatorDragged = true;
				m_data->m_colSeparatorX = x0;
				m_data->m_prevColSeparatorX = x0;
				m_data->m_colIndex = j;
			}
		}
	} else {
		i--;
		if (i >= 0 && i < (int)rowCount) {
			if (!e.isShiftPressed())
				unselectAll();
			select(i, !isSelected(i));
		}
	}
}

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

void TGrid::leftButtonDrag(const TMouseEvent &e)
{
	if (m_data->m_colSeparatorDragged) {
		m_data->m_colSeparatorX += (e.m_pos.x - m_data->m_prevColSeparatorX);
		m_data->m_prevColSeparatorX = e.m_pos.x;
		invalidate();
	}
}

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

void TGrid::leftButtonUp(const TMouseEvent &e)
{
	if (m_data->m_colSeparatorDragged) {
		TGridColumnP col0 = m_data->m_columnSet.getColumn(m_data->m_colIndex - 1);
		TGridColumnP col1 = m_data->m_columnSet.getColumn(m_data->m_colIndex);

		int col0X0, col0X1;
		col0->getCoords(col0X0, col0X1);

		int newWidth = m_data->m_colSeparatorX - col0X0;
		int totWidth = col0->getWidth() + col1->getWidth();

		col0->setWidth(newWidth);
		col1->setWidth(totWidth - newWidth);

		m_data->m_colSeparatorDragged = false;
		invalidate();
	}
}