2011-08-28 15:17:58 +02:00
|
|
|
/*
|
2012-01-02 17:58:37 +01:00
|
|
|
* Copyright (c) 2010-2012 OTClient <https://github.com/edubart/otclient>
|
2011-08-28 15:17:58 +02:00
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
|
|
* in the Software without restriction, including without limitation the rights
|
|
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in
|
|
|
|
* all copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
* THE SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
2011-08-15 16:11:24 +02:00
|
|
|
#include "tile.h"
|
|
|
|
#include "item.h"
|
2012-06-21 19:54:20 +02:00
|
|
|
#include "thingtypemanager.h"
|
2011-08-17 06:45:55 +02:00
|
|
|
#include "map.h"
|
2011-08-20 04:08:27 +02:00
|
|
|
#include "game.h"
|
2011-08-17 06:45:55 +02:00
|
|
|
#include "localplayer.h"
|
2011-08-31 22:22:57 +02:00
|
|
|
#include "effect.h"
|
2012-07-14 03:10:24 +02:00
|
|
|
#include "protocolgame.h"
|
2011-08-20 04:08:27 +02:00
|
|
|
#include <framework/graphics/fontmanager.h>
|
2011-08-15 16:11:24 +02:00
|
|
|
|
2012-06-23 23:30:54 +02:00
|
|
|
Tile::Tile(const Position& position) :
|
|
|
|
m_position(position),
|
|
|
|
m_drawElevation(0),
|
|
|
|
m_flags(0),
|
|
|
|
m_minimapColorByte(0)
|
2011-08-15 16:11:24 +02:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2012-01-30 19:18:10 +01:00
|
|
|
void Tile::draw(const Point& dest, float scaleFactor, int drawFlags)
|
2011-08-17 06:45:55 +02:00
|
|
|
{
|
2012-02-02 17:37:52 +01:00
|
|
|
bool animate = drawFlags & Otc::DrawAnimations;
|
2012-01-31 18:06:55 +01:00
|
|
|
|
2011-08-31 22:22:57 +02:00
|
|
|
// first bottom items
|
2012-02-02 21:54:49 +01:00
|
|
|
if(drawFlags & (Otc::DrawGround | Otc::DrawGroundBorders | Otc::DrawOnBottom)) {
|
2012-02-08 00:54:33 +01:00
|
|
|
m_drawElevation = 0;
|
2012-01-30 19:18:10 +01:00
|
|
|
for(const ThingPtr& thing : m_things) {
|
|
|
|
if(!thing->isGround() && !thing->isGroundBorder() && !thing->isOnBottom())
|
|
|
|
break;
|
|
|
|
|
2012-02-02 17:37:52 +01:00
|
|
|
if((thing->isGround() && drawFlags & Otc::DrawGround) ||
|
|
|
|
(thing->isGroundBorder() && drawFlags & Otc::DrawGroundBorders) ||
|
2012-02-02 21:54:49 +01:00
|
|
|
(thing->isOnBottom() && drawFlags & Otc::DrawOnBottom))
|
2012-02-02 22:20:34 +01:00
|
|
|
thing->draw(dest - m_drawElevation*scaleFactor, scaleFactor, animate);
|
2012-01-30 19:18:10 +01:00
|
|
|
|
2012-02-02 22:20:34 +01:00
|
|
|
m_drawElevation += thing->getElevation();
|
|
|
|
if(m_drawElevation > Otc::MAX_ELEVATION)
|
|
|
|
m_drawElevation = Otc::MAX_ELEVATION;
|
2012-01-30 19:18:10 +01:00
|
|
|
}
|
2011-08-31 23:41:04 +02:00
|
|
|
}
|
|
|
|
|
2012-02-02 21:54:49 +01:00
|
|
|
int redrawPreviousTopW = 0;
|
|
|
|
int redrawPreviousTopH = 0;
|
2012-02-02 21:10:14 +01:00
|
|
|
|
2012-02-02 21:54:49 +01:00
|
|
|
if(drawFlags & Otc::DrawItems) {
|
2012-02-02 21:10:14 +01:00
|
|
|
// now common items in reverse order
|
2012-01-30 19:18:10 +01:00
|
|
|
for(auto it = m_things.rbegin(); it != m_things.rend(); ++it) {
|
|
|
|
const ThingPtr& thing = *it;
|
2012-06-05 17:36:27 +02:00
|
|
|
if(thing->isOnTop() || thing->isOnBottom() || thing->isGroundBorder() || thing->isGround() || thing->isCreature())
|
2012-01-30 19:18:10 +01:00
|
|
|
break;
|
2012-02-02 22:20:34 +01:00
|
|
|
thing->draw(dest - m_drawElevation*scaleFactor, scaleFactor, animate);
|
2012-01-30 19:18:10 +01:00
|
|
|
|
2012-02-02 21:10:14 +01:00
|
|
|
if(thing->isLyingCorpse()) {
|
2012-06-21 19:54:20 +02:00
|
|
|
redrawPreviousTopW = std::max(thing->getWidth(), redrawPreviousTopW);
|
|
|
|
redrawPreviousTopH = std::max(thing->getHeight(), redrawPreviousTopH);
|
2012-02-02 21:10:14 +01:00
|
|
|
}
|
|
|
|
|
2012-02-02 22:20:34 +01:00
|
|
|
m_drawElevation += thing->getElevation();
|
|
|
|
if(m_drawElevation > Otc::MAX_ELEVATION)
|
|
|
|
m_drawElevation = Otc::MAX_ELEVATION;
|
2012-01-30 19:18:10 +01:00
|
|
|
}
|
2011-08-31 02:28:06 +02:00
|
|
|
}
|
2011-08-20 04:08:27 +02:00
|
|
|
|
2012-02-02 21:54:49 +01:00
|
|
|
// after we render 2x2 lying corpses, we must redraw previous creatures/ontop above them
|
|
|
|
if(redrawPreviousTopH > 0 || redrawPreviousTopW > 0) {
|
|
|
|
int topRedrawFlags = drawFlags & (Otc::DrawCreatures | Otc::DrawEffects | Otc::DrawOnTop | Otc::DrawAnimations);
|
|
|
|
if(topRedrawFlags) {
|
2012-02-03 14:21:13 +01:00
|
|
|
for(int x=-redrawPreviousTopW;x<=0;++x) {
|
|
|
|
for(int y=-redrawPreviousTopH;y<=0;++y) {
|
2012-02-02 21:10:14 +01:00
|
|
|
if(x == 0 && y == 0)
|
|
|
|
continue;
|
|
|
|
const TilePtr& tile = g_map.getTile(m_position.translated(x,y));
|
2012-02-02 21:54:49 +01:00
|
|
|
if(tile)
|
|
|
|
tile->draw(dest + Point(x*Otc::TILE_PIXELS, y*Otc::TILE_PIXELS)*scaleFactor, scaleFactor, topRedrawFlags);
|
2012-02-02 21:10:14 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-30 22:28:08 +01:00
|
|
|
// creatures
|
2012-01-30 19:18:10 +01:00
|
|
|
if(drawFlags & Otc::DrawCreatures) {
|
2012-02-02 17:37:52 +01:00
|
|
|
if(animate) {
|
|
|
|
for(const CreaturePtr& creature : m_walkingCreatures) {
|
2012-03-19 18:54:47 +01:00
|
|
|
if(creature->isRemoved())
|
|
|
|
continue;
|
2012-02-02 22:20:34 +01:00
|
|
|
creature->draw(Point(dest.x + ((creature->getPosition().x - m_position.x)*Otc::TILE_PIXELS - m_drawElevation)*scaleFactor,
|
|
|
|
dest.y + ((creature->getPosition().y - m_position.y)*Otc::TILE_PIXELS - m_drawElevation)*scaleFactor), scaleFactor, animate);
|
2012-01-30 22:28:08 +01:00
|
|
|
|
2012-02-02 17:37:52 +01:00
|
|
|
}
|
2012-01-30 22:28:08 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
for(auto it = m_things.rbegin(); it != m_things.rend(); ++it) {
|
2012-06-05 17:36:27 +02:00
|
|
|
const ThingPtr& thing = *it;
|
|
|
|
if(!thing->isCreature())
|
|
|
|
continue;
|
|
|
|
CreaturePtr creature = thing->asCreature();
|
2012-02-02 17:37:52 +01:00
|
|
|
if(creature && (!creature->isWalking() || !animate))
|
2012-02-03 00:01:57 +01:00
|
|
|
creature->draw(dest - m_drawElevation*scaleFactor, scaleFactor, animate);
|
2011-09-01 00:42:52 +02:00
|
|
|
}
|
2011-08-31 02:28:06 +02:00
|
|
|
}
|
2011-08-20 04:08:27 +02:00
|
|
|
|
2011-08-31 22:22:57 +02:00
|
|
|
// effects
|
2012-01-30 19:18:10 +01:00
|
|
|
if(drawFlags & Otc::DrawEffects) {
|
|
|
|
for(const EffectPtr& effect : m_effects)
|
2012-02-03 00:01:57 +01:00
|
|
|
effect->draw(dest - m_drawElevation*scaleFactor, scaleFactor, animate);
|
2012-01-30 19:18:10 +01:00
|
|
|
}
|
2011-08-20 04:08:27 +02:00
|
|
|
|
2011-08-31 22:22:57 +02:00
|
|
|
// top items
|
2012-02-02 21:54:49 +01:00
|
|
|
if(drawFlags & Otc::DrawOnTop) {
|
2012-01-30 19:18:10 +01:00
|
|
|
for(const ThingPtr& thing : m_things) {
|
|
|
|
if(thing->isOnTop())
|
2012-02-03 00:01:57 +01:00
|
|
|
thing->draw(dest, scaleFactor, animate);
|
2012-01-30 19:18:10 +01:00
|
|
|
}
|
2011-08-31 02:28:06 +02:00
|
|
|
}
|
2011-08-31 22:22:57 +02:00
|
|
|
}
|
2011-08-31 02:28:06 +02:00
|
|
|
|
2011-08-31 22:22:57 +02:00
|
|
|
void Tile::clean()
|
|
|
|
{
|
2012-02-02 21:10:14 +01:00
|
|
|
while(!m_things.empty())
|
|
|
|
removeThing(m_things.front());
|
2011-08-31 22:22:57 +02:00
|
|
|
}
|
2011-08-31 01:39:14 +02:00
|
|
|
|
2012-01-30 22:28:08 +01:00
|
|
|
void Tile::addWalkingCreature(const CreaturePtr& creature)
|
|
|
|
{
|
|
|
|
m_walkingCreatures.push_back(creature);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Tile::removeWalkingCreature(const CreaturePtr& creature)
|
|
|
|
{
|
|
|
|
auto it = std::find(m_walkingCreatures.begin(), m_walkingCreatures.end(), creature);
|
|
|
|
if(it != m_walkingCreatures.end())
|
|
|
|
m_walkingCreatures.erase(it);
|
|
|
|
}
|
|
|
|
|
2011-08-31 22:22:57 +02:00
|
|
|
ThingPtr Tile::addThing(const ThingPtr& thing, int stackPos)
|
2011-08-15 16:11:24 +02:00
|
|
|
{
|
|
|
|
if(!thing)
|
2011-08-31 22:22:57 +02:00
|
|
|
return nullptr;
|
|
|
|
|
2011-12-30 15:15:23 +01:00
|
|
|
if(EffectPtr effect = thing->asEffect()) {
|
|
|
|
m_effects.push_back(effect);
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2012-07-15 07:46:56 +02:00
|
|
|
// the items stackpos follows this order:
|
|
|
|
// 0 - ground
|
|
|
|
// 1 - ground borders
|
|
|
|
// 2 - bottom (walls)
|
|
|
|
// 3 - on top (doors)
|
|
|
|
// 4 - creatures, from top to bottom
|
|
|
|
// 5 - items, from top to bottom
|
2011-08-31 22:22:57 +02:00
|
|
|
if(stackPos < 0) {
|
|
|
|
int priority = thing->getStackPriority();
|
2012-07-15 08:16:40 +02:00
|
|
|
bool prepend = (stackPos == -2 || priority <= 3);
|
2011-08-31 22:22:57 +02:00
|
|
|
for(stackPos = 0; stackPos < (int)m_things.size(); ++stackPos) {
|
|
|
|
int otherPriority = m_things[stackPos]->getStackPriority();
|
2012-07-15 07:46:56 +02:00
|
|
|
if((prepend && otherPriority > priority) || (!prepend && otherPriority >= priority))
|
2011-08-31 22:22:57 +02:00
|
|
|
break;
|
2011-08-15 16:11:24 +02:00
|
|
|
}
|
2011-08-31 22:22:57 +02:00
|
|
|
} else if(stackPos > (int)m_things.size())
|
|
|
|
stackPos = m_things.size();
|
|
|
|
|
|
|
|
ThingPtr oldObject;
|
|
|
|
if(stackPos < (int)m_things.size())
|
|
|
|
oldObject = m_things[stackPos];
|
|
|
|
m_things.insert(m_things.begin() + stackPos, thing);
|
2012-01-30 01:00:12 +01:00
|
|
|
|
2012-05-10 14:09:44 +02:00
|
|
|
if(m_things.size() > MAX_THINGS)
|
|
|
|
removeThing(m_things[MAX_THINGS]);
|
|
|
|
|
2012-06-22 07:26:22 +02:00
|
|
|
update();
|
2011-08-31 22:22:57 +02:00
|
|
|
return oldObject;
|
2011-08-16 07:47:35 +02:00
|
|
|
}
|
|
|
|
|
2012-02-02 21:10:14 +01:00
|
|
|
bool Tile::removeThing(ThingPtr thing)
|
2012-01-30 01:00:12 +01:00
|
|
|
{
|
|
|
|
if(!thing)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
bool removed = false;
|
|
|
|
|
|
|
|
if(EffectPtr effect = thing->asEffect()) {
|
|
|
|
auto it = std::find(m_effects.begin(), m_effects.end(), effect);
|
|
|
|
if(it != m_effects.end()) {
|
|
|
|
m_effects.erase(it);
|
|
|
|
removed = true;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
auto it = std::find(m_things.begin(), m_things.end(), thing);
|
|
|
|
if(it != m_things.end()) {
|
|
|
|
m_things.erase(it);
|
|
|
|
removed = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// reset values managed by this tile
|
2012-06-22 07:26:22 +02:00
|
|
|
if(removed)
|
|
|
|
update();
|
2012-01-30 01:00:12 +01:00
|
|
|
|
|
|
|
return removed;
|
|
|
|
}
|
|
|
|
|
2011-08-31 22:22:57 +02:00
|
|
|
ThingPtr Tile::getThing(int stackPos)
|
2011-08-16 07:47:35 +02:00
|
|
|
{
|
2011-08-31 22:22:57 +02:00
|
|
|
if(stackPos >= 0 && stackPos < (int)m_things.size())
|
|
|
|
return m_things[stackPos];
|
|
|
|
return nullptr;
|
2011-08-15 16:11:24 +02:00
|
|
|
}
|
|
|
|
|
2012-07-15 07:46:56 +02:00
|
|
|
EffectPtr Tile::getEffect(uint16 id)
|
|
|
|
{
|
|
|
|
for(const EffectPtr& effect : m_effects)
|
|
|
|
if(effect->getId() == id)
|
|
|
|
return effect;
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2012-05-10 00:19:05 +02:00
|
|
|
bool Tile::hasThing(const ThingPtr& thing)
|
|
|
|
{
|
|
|
|
return std::find(m_things.begin(), m_things.end(), thing) != m_things.end();
|
|
|
|
}
|
|
|
|
|
2012-01-03 23:27:31 +01:00
|
|
|
int Tile::getThingStackpos(const ThingPtr& thing)
|
|
|
|
{
|
|
|
|
for(uint stackpos = 0; stackpos < m_things.size(); ++stackpos)
|
|
|
|
if(thing == m_things[stackpos])
|
|
|
|
return stackpos;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2012-07-15 07:46:56 +02:00
|
|
|
ThingPtr Tile:: getTopThing()
|
2012-01-03 21:41:00 +01:00
|
|
|
{
|
2012-07-15 08:16:40 +02:00
|
|
|
if(isEmpty())
|
|
|
|
return nullptr;
|
2012-07-15 07:46:56 +02:00
|
|
|
for(const ThingPtr& thing : m_things) {
|
|
|
|
if(!thing->isGround() && !thing->isGroundBorder() && !thing->isOnBottom() && !thing->isOnTop() && !thing->isCreature())
|
|
|
|
return thing;
|
|
|
|
}
|
2012-01-03 21:41:00 +01:00
|
|
|
return m_things[m_things.size() - 1];
|
|
|
|
}
|
|
|
|
|
2012-06-23 16:26:18 +02:00
|
|
|
std::vector<ItemPtr> Tile::getItems()
|
|
|
|
{
|
|
|
|
std::vector<ItemPtr> items;
|
|
|
|
for(const ThingPtr& thing : m_things) {
|
|
|
|
if(ItemPtr item = thing->asItem())
|
|
|
|
items.push_back(item);
|
|
|
|
}
|
|
|
|
return items;
|
|
|
|
}
|
|
|
|
|
2011-08-31 22:22:57 +02:00
|
|
|
std::vector<CreaturePtr> Tile::getCreatures()
|
2011-08-17 06:45:55 +02:00
|
|
|
{
|
2011-08-31 22:22:57 +02:00
|
|
|
std::vector<CreaturePtr> creatures;
|
|
|
|
for(const ThingPtr& thing : m_things) {
|
|
|
|
if(CreaturePtr creature = thing->asCreature())
|
|
|
|
creatures.push_back(creature);
|
|
|
|
}
|
|
|
|
return creatures;
|
2011-08-15 16:11:24 +02:00
|
|
|
}
|
|
|
|
|
2011-08-31 22:22:57 +02:00
|
|
|
ItemPtr Tile::getGround()
|
2011-08-15 16:11:24 +02:00
|
|
|
{
|
2011-08-31 22:22:57 +02:00
|
|
|
ThingPtr firstObject = getThing(0);
|
|
|
|
if(!firstObject)
|
|
|
|
return nullptr;
|
2012-01-30 01:00:12 +01:00
|
|
|
if(firstObject->isGround())
|
2011-08-31 22:22:57 +02:00
|
|
|
return firstObject->asItem();
|
|
|
|
return nullptr;
|
2011-08-15 16:11:24 +02:00
|
|
|
}
|
2011-08-31 01:39:14 +02:00
|
|
|
|
2012-01-15 22:19:52 +01:00
|
|
|
int Tile::getGroundSpeed()
|
|
|
|
{
|
|
|
|
int groundSpeed = 100;
|
|
|
|
if(ItemPtr ground = getGround())
|
|
|
|
groundSpeed = ground->getGroundSpeed();
|
|
|
|
return groundSpeed;
|
|
|
|
}
|
|
|
|
|
2012-01-04 14:02:35 +01:00
|
|
|
ThingPtr Tile::getTopLookThing()
|
2011-12-30 19:14:50 +01:00
|
|
|
{
|
2012-01-04 14:39:08 +01:00
|
|
|
if(isEmpty())
|
|
|
|
return nullptr;
|
|
|
|
|
2012-01-06 17:16:47 +01:00
|
|
|
for(uint i = 0; i < m_things.size(); ++i) {
|
2012-01-04 14:02:35 +01:00
|
|
|
ThingPtr thing = m_things[i];
|
2012-01-30 01:00:12 +01:00
|
|
|
if(!thing->isIgnoreLook() && (!thing->isGround() && !thing->isGroundBorder() && !thing->isOnBottom() && !thing->isOnTop()))
|
2012-01-04 14:02:35 +01:00
|
|
|
return thing;
|
2012-01-02 20:01:48 +01:00
|
|
|
}
|
2012-01-04 14:02:35 +01:00
|
|
|
|
2012-01-04 14:39:08 +01:00
|
|
|
return m_things[0];
|
2012-01-04 14:02:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
ThingPtr Tile::getTopUseThing()
|
|
|
|
{
|
|
|
|
if(isEmpty())
|
|
|
|
return nullptr;
|
|
|
|
|
2012-01-06 17:16:47 +01:00
|
|
|
for(uint i = 0; i < m_things.size(); ++i) {
|
2012-01-04 14:02:35 +01:00
|
|
|
ThingPtr thing = m_things[i];
|
2012-06-05 17:36:27 +02:00
|
|
|
if(thing->isForceUse() || (!thing->isGround() && !thing->isGroundBorder() && !thing->isOnBottom() && !thing->isOnTop() && !thing->isCreature()))
|
2012-01-04 14:02:35 +01:00
|
|
|
return thing;
|
|
|
|
}
|
|
|
|
|
|
|
|
return m_things[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
CreaturePtr Tile::getTopCreature()
|
|
|
|
{
|
|
|
|
CreaturePtr creature;
|
2012-01-06 17:16:47 +01:00
|
|
|
for(uint i = 0; i < m_things.size(); ++i) {
|
2012-01-04 14:02:35 +01:00
|
|
|
ThingPtr thing = m_things[i];
|
2012-01-30 01:00:12 +01:00
|
|
|
if(thing->asLocalPlayer()) // return local player if there is no other creature
|
2012-01-04 14:02:35 +01:00
|
|
|
creature = thing->asCreature();
|
2012-06-05 17:36:27 +02:00
|
|
|
else if(thing->isCreature() && !thing->isLocalPlayer())
|
2012-01-04 14:02:35 +01:00
|
|
|
return thing->asCreature();
|
|
|
|
}
|
2012-06-03 02:00:09 +02:00
|
|
|
if(!creature && !m_walkingCreatures.empty())
|
|
|
|
creature = m_walkingCreatures.back();
|
2012-01-04 14:02:35 +01:00
|
|
|
return creature;
|
2011-12-30 19:14:50 +01:00
|
|
|
}
|
|
|
|
|
2012-01-20 02:12:26 +01:00
|
|
|
ThingPtr Tile::getTopMoveThing()
|
|
|
|
{
|
|
|
|
if(isEmpty())
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
for(uint i = 0; i < m_things.size(); ++i) {
|
|
|
|
ThingPtr thing = m_things[i];
|
2012-06-05 17:36:27 +02:00
|
|
|
if(!thing->isGround() && !thing->isGroundBorder() && !thing->isOnBottom() && !thing->isOnTop() && !thing->isCreature()) {
|
2012-01-20 02:12:26 +01:00
|
|
|
if(i > 0 && thing->isNotMoveable())
|
|
|
|
return m_things[i-1];
|
|
|
|
return thing;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return m_things[0];
|
|
|
|
}
|
|
|
|
|
2012-01-09 21:54:37 +01:00
|
|
|
ThingPtr Tile::getTopMultiUseThing()
|
|
|
|
{
|
|
|
|
// this is related to classic controls, getting top item, forceuse or creature
|
|
|
|
if(isEmpty())
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
for(uint i = 0; i < m_things.size(); ++i) {
|
|
|
|
ThingPtr thing = m_things[i];
|
|
|
|
if(thing->isForceUse() || (!thing->isGround() && !thing->isGroundBorder() && !thing->isOnBottom() && !thing->isOnTop())) {
|
2012-07-18 01:49:21 +02:00
|
|
|
if(i > 0 && thing->isSplash())
|
2012-01-09 21:54:37 +01:00
|
|
|
return m_things[i-1];
|
|
|
|
return thing;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-04-05 05:20:40 +02:00
|
|
|
for(uint i = 0; i < m_things.size(); ++i) {
|
|
|
|
ThingPtr thing = m_things[i];
|
|
|
|
if(!thing->isGround() && !thing->isGroundBorder() && !thing->isOnTop())
|
|
|
|
return thing;
|
|
|
|
}
|
|
|
|
|
2012-01-09 21:54:37 +01:00
|
|
|
return m_things[0];
|
|
|
|
}
|
|
|
|
|
2011-11-05 21:34:49 +01:00
|
|
|
bool Tile::isWalkable()
|
|
|
|
{
|
|
|
|
if(!getGround())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
for(const ThingPtr& thing : m_things) {
|
2012-01-30 01:00:12 +01:00
|
|
|
if(thing->isNotWalkable())
|
2011-11-05 21:34:49 +01:00
|
|
|
return false;
|
2011-12-30 15:15:23 +01:00
|
|
|
|
|
|
|
if(CreaturePtr creature = thing->asCreature()) {
|
|
|
|
if(!creature->getPassable())
|
|
|
|
return false;
|
|
|
|
}
|
2011-11-05 21:34:49 +01:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2012-03-24 16:22:40 +01:00
|
|
|
bool Tile::isPathable()
|
|
|
|
{
|
|
|
|
if(!isWalkable())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
for(const ThingPtr& thing : m_things) {
|
|
|
|
if(thing->isNotPathable())
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2011-09-02 00:00:46 +02:00
|
|
|
bool Tile::isFullGround()
|
|
|
|
{
|
2012-01-30 01:00:12 +01:00
|
|
|
ItemPtr ground = getGround();
|
|
|
|
if(ground && ground->isFullGround())
|
2011-09-02 00:00:46 +02:00
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2011-08-31 22:22:57 +02:00
|
|
|
bool Tile::isFullyOpaque()
|
2011-08-31 01:39:14 +02:00
|
|
|
{
|
2011-08-31 22:22:57 +02:00
|
|
|
ThingPtr firstObject = getThing(0);
|
2012-01-30 01:00:12 +01:00
|
|
|
return firstObject && firstObject->isFullGround();
|
2011-08-31 01:39:14 +02:00
|
|
|
}
|
2011-09-02 00:00:46 +02:00
|
|
|
|
|
|
|
bool Tile::isLookPossible()
|
|
|
|
{
|
|
|
|
for(const ThingPtr& thing : m_things) {
|
2012-06-21 19:54:20 +02:00
|
|
|
if(thing->blockProjectile())
|
2011-09-02 00:00:46 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
2011-11-10 07:53:16 +01:00
|
|
|
|
2012-01-04 15:30:28 +01:00
|
|
|
bool Tile::isClickable()
|
|
|
|
{
|
2012-01-30 01:00:12 +01:00
|
|
|
bool hasGround = false;
|
|
|
|
bool hasOnBottom = false;
|
|
|
|
bool hasIgnoreLook = false;
|
2012-01-04 15:30:28 +01:00
|
|
|
for(const ThingPtr& thing : m_things) {
|
2012-01-30 01:00:12 +01:00
|
|
|
if(thing->isGround())
|
2012-01-04 15:30:28 +01:00
|
|
|
hasGround = true;
|
2012-01-30 01:00:12 +01:00
|
|
|
if(thing->isOnBottom())
|
2012-01-04 15:30:28 +01:00
|
|
|
hasOnBottom = true;
|
2012-01-30 01:00:12 +01:00
|
|
|
if(thing->isIgnoreLook())
|
2012-01-04 15:30:28 +01:00
|
|
|
hasIgnoreLook = true;
|
|
|
|
|
|
|
|
if((hasGround || hasOnBottom) && !hasIgnoreLook)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-01-30 01:00:12 +01:00
|
|
|
bool Tile::isEmpty()
|
|
|
|
{
|
|
|
|
return m_things.size() == 0;
|
|
|
|
}
|
|
|
|
|
2012-02-02 17:55:42 +01:00
|
|
|
bool Tile::mustHookEast()
|
|
|
|
{
|
|
|
|
for(const ThingPtr& thing : m_things)
|
|
|
|
if(thing->isHookEast())
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Tile::mustHookSouth()
|
|
|
|
{
|
|
|
|
for(const ThingPtr& thing : m_things)
|
|
|
|
if(thing->isHookSouth())
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2011-12-30 15:15:23 +01:00
|
|
|
bool Tile::hasCreature()
|
|
|
|
{
|
|
|
|
for(const ThingPtr& thing : m_things)
|
|
|
|
if(thing->asCreature())
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-01-30 01:00:12 +01:00
|
|
|
bool Tile::limitsFloorsView()
|
2012-01-02 20:01:48 +01:00
|
|
|
{
|
2012-01-30 01:00:12 +01:00
|
|
|
// ground and walls limits the view
|
|
|
|
ThingPtr firstThing = getThing(0);
|
|
|
|
if(firstThing && !firstThing->isDontHide() && (firstThing->isGround() || firstThing->isOnBottom()))
|
|
|
|
return true;
|
|
|
|
return false;
|
2012-01-02 20:01:48 +01:00
|
|
|
}
|
2012-02-02 21:10:14 +01:00
|
|
|
|
|
|
|
bool Tile::canErase()
|
|
|
|
{
|
|
|
|
return m_walkingCreatures.empty() && m_effects.empty() && m_things.empty();
|
|
|
|
}
|
|
|
|
|
2012-06-22 07:26:22 +02:00
|
|
|
void Tile::update()
|
|
|
|
{
|
|
|
|
m_minimapColorByte = 0;
|
|
|
|
for(const ThingPtr& thing : m_things) {
|
|
|
|
if(!thing->isGround() && !thing->isGroundBorder() && !thing->isOnBottom() && !thing->isOnTop())
|
|
|
|
break;
|
|
|
|
uint8 c = thing->getMinimapColor();
|
|
|
|
if(c != 0)
|
|
|
|
m_minimapColorByte = c;
|
|
|
|
}
|
|
|
|
}
|