/***************************************************************************** * Eliot * Copyright (C) 2012 Olivier Teulière * Authors: Olivier Teulière <ipkiss @@ gmail.com> * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA *****************************************************************************/ #include <QtGui/QHBoxLayout> #include <QtGui/QDragEnterEvent> #include <QtGui/QDragLeaveEvent> #include <QtGui/QDragMoveEvent> #include <QtGui/QDropEvent> #include "rack_widget.h" #include "tile_widget.h" #include "tile_layout.h" #include "play_model.h" #include "qtcommon.h" #include "public_game.h" #include "pldrack.h" #include "debug.h" using namespace std; INIT_LOGGER(qt, RackWidget); #define MIME_TYPE "text/x-tile" RackWidget::RackWidget(QWidget *parent) : QFrame(parent), m_game(NULL), m_playModel(NULL), m_showOnlyLastTurn(false) { setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding); TileLayout *layout = new TileLayout(1); layout->setSpacing(5); layout->setAlignment(Qt::AlignCenter); setLayout(layout); setAcceptDrops(true); } void RackWidget::setPlayModel(PlayModel *iPlayModel) { if (m_playModel != NULL) m_playModel->disconnect(this, SLOT(refresh())); if (iPlayModel != NULL) { QObject::connect(iPlayModel, SIGNAL(moveChanged(const Move &, const Move&)), this, SLOT(refresh())); } m_playModel = iPlayModel; } void RackWidget::setGame(const PublicGame *iGame) { m_game = iGame; if (m_game == NULL) { // Delete the widgets TileLayout *layout = (TileLayout*) this->layout(); layout->clear(); m_tilesVect.clear(); } refresh(); } void RackWidget::refresh() { if (m_game == NULL) return; if (m_showOnlyLastTurn && !m_game->isLastTurn()) return; // Get the tiles vector<Tile> tiles; m_game->getCurrentRack().getAllTiles(tiles); // Update the rack setRack(tiles); } void RackWidget::setRack(const vector<Tile> &iTiles) { const vector<Tile> &remainingTiles = filterRack(iTiles); unsigned tilesCount = remainingTiles.size(); // Make sure we have as many widgets as there are letters in the rack while (m_tilesVect.size() > tilesCount) { QtCommon::DestroyObject(m_tilesVect.back()); m_tilesVect.pop_back(); } while (m_tilesVect.size() < tilesCount) { TileWidget *tileWidget = new TileWidget(0, TileWidget::NONE, 0, m_tilesVect.size()); QObject::connect(tileWidget, SIGNAL(mousePressed(int, int, QMouseEvent*)), this, SLOT(tilePressed(int, int, QMouseEvent*))); tileWidget->setBorder(2); layout()->addWidget(tileWidget); m_tilesVect.push_back(tileWidget); } ASSERT(m_tilesVect.size() == tilesCount, "Invalid number of tiles"); // Update the widgets for (unsigned int i = 0; i < tilesCount; ++i) { TileWidget *tileWidget = m_tilesVect[i]; tileWidget->tileChanged(TileWidget::NORMAL, remainingTiles[i]); } } vector<Tile> RackWidget::filterRack(const vector<Tile> &iTiles) const { if (m_playModel == NULL || !m_playModel->getMove().isValid()) return iTiles; vector<Tile> result = iTiles; const Round &round = m_playModel->getMove().getRound(); for (unsigned i = 0; i < round.getWordLen(); ++i) { if (round.isPlayedFromRack(i)) { const Tile &t = round.getTile(i); vector<Tile>::iterator it; for (it = result.begin(); it != result.end(); ++it) { if (*it == t) { result.erase(it); break; } } } } return result; } // Drag & drop handling void RackWidget::tilePressed(int row, int col, QMouseEvent *event) { ASSERT(row == 0, "Multi-line racks are not supported"); ASSERT(col >= 0 && (unsigned)col < m_tilesVect.size(), "Invalid tile index: " << col); LOG_DEBUG("Starting drag for tile " << col); TileWidget *tileWidget = m_tilesVect[col]; // Save the initial column of the moved tile QByteArray itemData; QDataStream dataStream(&itemData, QIODevice::WriteOnly); dataStream << col; QMimeData *mimeData = new QMimeData; mimeData->setData(MIME_TYPE, itemData); // Create an image of the tile QPixmap pixmap(tileWidget->size()); tileWidget->render(&pixmap); // Initiate the drag QDrag *drag = new QDrag(this); drag->setMimeData(mimeData); drag->setHotSpot(event->pos()); drag->setPixmap(pixmap); if (!(drag->exec(Qt::MoveAction) == Qt::MoveAction)) { // TODO } } void RackWidget::dragEnterEvent(QDragEnterEvent *event) { if (event->mimeData()->hasFormat(MIME_TYPE)) event->accept(); else event->ignore(); } void RackWidget::dragLeaveEvent(QDragLeaveEvent *event) { event->accept(); } void RackWidget::dragMoveEvent(QDragMoveEvent *event) { if (event->mimeData()->hasFormat(MIME_TYPE)) { event->setDropAction(Qt::MoveAction); event->accept(); } else { event->ignore(); } } void RackWidget::dropEvent(QDropEvent *event) { int closestCol = findClosestTile(event->pos()); if (event->mimeData()->hasFormat(MIME_TYPE)) { // Retrieve the initial column of the moved tile QByteArray data = event->mimeData()->data(MIME_TYPE); QDataStream dataStream(&data, QIODevice::ReadOnly); int initialCol; dataStream >> initialCol; if (initialCol == closestCol) { event->ignore(); return; } LOG_DEBUG("Dropping tile " << initialCol << " closest to tile " << closestCol); // Get the initial tiles vector<Tile> tiles; Q_FOREACH(const TileWidget *tileWidget, m_tilesVect) { tiles.push_back(tileWidget->getTile()); } // Change the order Tile moved = tiles[initialCol]; tiles.erase(tiles.begin() + initialCol); tiles.insert(tiles.begin() + closestCol, moved); // Update the rack setRack(tiles); event->setDropAction(Qt::MoveAction); event->accept(); } else { event->ignore(); } } int RackWidget::findTile(const QPoint &iPos) const { for (unsigned i = 0; i < m_tilesVect.size(); ++i) { if (m_tilesVect[i]->geometry().contains(iPos)) return i; } return -1; } int RackWidget::findClosestTile(const QPoint &iPos) const { int minDist = geometry().bottomRight().manhattanLength(); int minIdx = -1; for (unsigned i = 0; i < m_tilesVect.size(); ++i) { QPoint distPoint = iPos - m_tilesVect[i]->geometry().center(); int dist = distPoint.manhattanLength(); if (dist < minDist) { minDist = dist; minIdx = i; } } return minIdx; }