tools/drawMapTest/qaspectratiolayout.cpp
author nemo
Sun, 11 Mar 2012 20:12:31 -0400
changeset 6770 7d2c6cdb816a
parent 4477 63a21fac8bf7
permissions -rw-r--r--
Start on adding drowning bonus to bat/firepunch/whip. AI still is not smart enough to change direction when firepunching to face the water, or change the angle of the bat. Also try to weight cost of switches based on how many switches were available.

/*
 * Copyright (c) 2009 Nokia Corporation.
 */

#include "qaspectratiolayout.h"

QAspectRatioLayout::QAspectRatioLayout(QWidget* parent, int spacing) : QLayout(parent) {
        init(spacing);
}

QAspectRatioLayout::QAspectRatioLayout(int spacing) {
        init(spacing);
}

QAspectRatioLayout::~QAspectRatioLayout() {
        delete item;
        delete lastReceivedRect;
        delete _geometry;
}

void QAspectRatioLayout::init(int spacing) {
        item = 0;
        lastReceivedRect = new QRect(0, 0, 0, 0);
        _geometry = new QRect(0, 0, 0, 0);
        setSpacing(spacing);
}


/* Adds item if place isn't already taken. */
void QAspectRatioLayout::add(QLayoutItem* item) {
        if(!hasItem()) {
                replaceItem(item);
        }
}

/* Adds item if place isn't already taken. */
void QAspectRatioLayout::addItem(QLayoutItem* item) {
        if(!hasItem()) {
                replaceItem(item);
        }
}

/* Adds widget if place isn't already taken. */
void QAspectRatioLayout::addWidget(QWidget* widget) {
        if(!hasItem()) {
                replaceItem(new QWidgetItem(widget));
        }
}

/* Returns the item pointer and dereferences it here. */
QLayoutItem* QAspectRatioLayout::take() {
        QLayoutItem* item = 0;
        if(this->hasItem()) {
                item = this->item;
                this->item = 0;
        }
        return item;
}

/* Returns the item pointer and dereferences it here. */
QLayoutItem* QAspectRatioLayout::takeAt(int index) {
        if(index != 0) {
                return 0;
        }
        return this->take();
}

/* Returns the item pointer. */
QLayoutItem* QAspectRatioLayout::itemAt(int index) const {
        if(index != 0) {
                return 0;
        }
        if(hasItem()) {
                return this->item;
        }
        return 0;
}

/* Checks if we have an item. */
bool QAspectRatioLayout::hasItem() const {
        return this->item != 0;
}

/* Returns the count of items which can be either 0 or 1. */
int QAspectRatioLayout::count() const {
        int returnValue = 0;
        if(hasItem()) {
                returnValue = 1;
        }
        return returnValue;
}

/* Replaces the item with the new and returns the old. */
QLayoutItem* QAspectRatioLayout::replaceItem(QLayoutItem* item) {
        QLayoutItem* old = 0;
        if(this->hasItem()) {
                old = this->item;
        }
        this->item = item;
        setGeometry(*this->_geometry);
        return old;
}

/* Tells which way layout expands. */
Qt::Orientations QAspectRatioLayout::expandingDirections() const {
        return Qt::Horizontal | Qt::Vertical;
}

/* Tells which size is preferred. */
QSize QAspectRatioLayout::sizeHint() const {
        return this->item->minimumSize();
}

/* Tells minimum size. */
QSize QAspectRatioLayout::minimumSize() const {
        return this->item->minimumSize();
}

/*
 * Tells if heightForWidth calculations is handled.
 * It isn't since width isn't enough to calculate
 * proper size.
 */
bool QAspectRatioLayout::hasHeightForWidth() const {
        return false;
}

/* Replaces lastReceivedRect. */
void QAspectRatioLayout::setLastReceivedRect(const QRect& rect) {
        QRect* oldRect = this->lastReceivedRect;
        this->lastReceivedRect = new QRect(rect.topLeft(), rect.size());
        delete oldRect;
}

/* Returns geometry */
QRect QAspectRatioLayout::geometry() {
        return QRect(*this->_geometry);
}

/* Sets geometry to given size. */
void QAspectRatioLayout::setGeometry(const QRect& rect) {
        /*
         * We check if the item is set and
         * if size is the same previously received.
         * If either is false nothing is done.
         */
        if(!this->hasItem() ||
           areRectsEqual(*this->lastReceivedRect, rect)) {
                return;
        }
        /* Replace the last received rectangle. */
        setLastReceivedRect(rect);
        /* Calculate proper size for the item relative to the received size. */
        QSize properSize = calculateProperSize(rect.size());
        /* Calculate center location in the rect and with item size. */
        QPoint properLocation = calculateCenterLocation(rect.size(), properSize);
        /* Set items geometry */
        this->item->setGeometry(QRect(properLocation, properSize));
        QRect* oldRect = this->_geometry;
        /* Cache the calculated geometry. */
        this->_geometry = new QRect(properLocation, properSize);
        delete oldRect;
        /* Super classes setGeometry */
        QLayout::setGeometry(*this->_geometry);
}

/* Takes the shortest side and creates QSize
 * with the shortest side as width and height. */
QSize QAspectRatioLayout::calculateProperSize(QSize from) const {
        QSize properSize;
        if(from.height() * 2 < from.width()) {
                properSize.setHeight(from.height() - this->margin());
                properSize.setWidth(from.height() * 2 - this->margin());
        }
        else {
                properSize.setWidth(from.width() - this->margin());
                properSize.setHeight(from.width() / 2 - this->margin());
        }
        return properSize;
}

/* Calculates center location from the given height and width for item size. */
QPoint QAspectRatioLayout::calculateCenterLocation(QSize from,
                                                   QSize itemSize) const {
        QPoint centerLocation;
        if((from.width() - itemSize.width()) > 0) {
                centerLocation.setX((from.width() - itemSize.width())/2);
        }
        if((from.height() - itemSize.height()) > 0) {
                centerLocation.setY((from.height() - itemSize.height())/2);
        }
        return centerLocation;
}

/* Compares if two QRects are equal. */
bool QAspectRatioLayout::areRectsEqual(const QRect& a,
                                       const QRect& b) const {
        bool result = false;
        if(a.x() == b.x() &&
           a.y() == b.y() &&
           a.height() == b.height() &&
           a.width() == b.width()) {
                result = true;
        }
        return result;
}