| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305 | /* * CTradeBase.cpp, part of VCMI engine * * Authors: listed in file AUTHORS in main folder * * License: GNU General Public License v2.0 or later * Full text of license available in license.txt file, in main folder * */#include "StdInc.h"#include "CTradeBase.h"#include "../gui/CGuiHandler.h"#include "../render/Canvas.h"#include "../widgets/TextControls.h"#include "../windows/InfoWindows.h"#include "../CGameInfo.h"#include "../../lib/CGeneralTextHandler.h"#include "../../lib/mapObjects/CGHeroInstance.h"CTradeableItem::CTradeableItem(Point pos, EType Type, int ID, bool Left, int Serial)	: CIntObject(LCLICK | HOVER | SHOW_POPUP, pos)	, type(EType(-1)) // set to invalid, will be corrected in setType	, id(ID)	, serial(Serial)	, left(Left){	OBJECT_CONSTRUCTION_CAPTURING(255 - DISPOSE);	downSelection = false;	hlp = nullptr;	setType(Type);	if(image)	{		this->pos.w = image->pos.w;		this->pos.h = image->pos.h;	}}void CTradeableItem::setType(EType newType){	if(type != newType)	{		OBJECT_CONSTRUCTION_CUSTOM_CAPTURING(255 - DISPOSE);		type = newType;		if(getIndex() < 0)		{			image = std::make_shared<CAnimImage>(getFilename(), 0);			image->disable();		}		else		{			image = std::make_shared<CAnimImage>(getFilename(), getIndex());		}	}}void CTradeableItem::setID(int newID){	if(id != newID)	{		id = newID;		if(image)		{			int index = getIndex();			if(index < 0)				image->disable();			else			{				image->enable();				image->setFrame(index);			}		}	}}AnimationPath CTradeableItem::getFilename(){	switch(type)	{	case RESOURCE:		return AnimationPath::builtin("RESOURCE");	case PLAYER:		return AnimationPath::builtin("CREST58");	case ARTIFACT_TYPE:	case ARTIFACT_PLACEHOLDER:	case ARTIFACT_INSTANCE:		return AnimationPath::builtin("artifact");	case CREATURE:		return AnimationPath::builtin("TWCRPORT");	default:		return {};	}}int CTradeableItem::getIndex(){	if(id < 0)		return -1;	switch(type)	{	case RESOURCE:	case PLAYER:		return id;	case ARTIFACT_TYPE:	case ARTIFACT_INSTANCE:	case ARTIFACT_PLACEHOLDER:		return CGI->artifacts()->getByIndex(id)->getIconIndex();	case CREATURE:		return CGI->creatures()->getByIndex(id)->getIconIndex();	default:		return -1;	}}void CTradeableItem::showAll(Canvas & to){	Point posToBitmap;	Point posToSubCenter;	switch (type)	{	case RESOURCE:		posToBitmap = Point(19, 9);		posToSubCenter = Point(36, 59);		break;	case CREATURE_PLACEHOLDER:	case CREATURE:		posToSubCenter = Point(29, 77);		break;	case PLAYER:		posToSubCenter = Point(31, 76);		break;	case ARTIFACT_PLACEHOLDER:	case ARTIFACT_INSTANCE:		posToSubCenter = Point(19, 54);		if (downSelection)			posToSubCenter.y += 8;		break;	case ARTIFACT_TYPE:		posToSubCenter = Point(19, 58);		break;	}	if(image)	{		image->moveTo(pos.topLeft() + posToBitmap);		CIntObject::showAll(to);	}	to.drawText(pos.topLeft() + posToSubCenter, FONT_SMALL, Colors::WHITE, ETextAlignment::CENTER, subtitle);}void CTradeableItem::clickPressed(const Point & cursorPosition){	if(clickPressedCallback)		clickPressedCallback(shared_from_this());}void CTradeableItem::showAllAt(const Point & dstPos, const std::string & customSub, Canvas & to){	Rect oldPos = pos;	std::string oldSub = subtitle;	downSelection = true;	moveTo(dstPos);	subtitle = customSub;	showAll(to);	downSelection = false;	moveTo(oldPos.topLeft());	subtitle = oldSub;}void CTradeableItem::hover(bool on){	if(!on)	{		GH.statusbar()->clear();		return;	}	switch(type)	{	case CREATURE:	case CREATURE_PLACEHOLDER:		GH.statusbar()->write(boost::str(boost::format(CGI->generaltexth->allTexts[481]) % CGI->creh->objects[id]->getNamePluralTranslated()));		break;	case ARTIFACT_PLACEHOLDER:		if(id < 0)			GH.statusbar()->write(CGI->generaltexth->zelp[582].first);		else			GH.statusbar()->write(CGI->artifacts()->getByIndex(id)->getNameTranslated());		break;	}}void CTradeableItem::showPopupWindow(const Point & cursorPosition){	switch(type)	{	case CREATURE:	case CREATURE_PLACEHOLDER:		break;	case ARTIFACT_TYPE:	case ARTIFACT_PLACEHOLDER:		//TODO: it's would be better for market to contain actual CArtifactInstance and not just ids of certain artifact type so we can use getEffectiveDescription.		if (id >= 0)			CRClickPopup::createAndPush(CGI->artifacts()->getByIndex(id)->getDescriptionTranslated());		break;	}}std::string CTradeableItem::getName(int number) const{	switch(type)	{	case PLAYER:		return CGI->generaltexth->capColors[id];	case RESOURCE:		return CGI->generaltexth->restypes[id];	case CREATURE:		if (number == 1)			return CGI->creh->objects[id]->getNameSingularTranslated();		else			return CGI->creh->objects[id]->getNamePluralTranslated();	case ARTIFACT_TYPE:	case ARTIFACT_INSTANCE:		return CGI->artifacts()->getByIndex(id)->getNameTranslated();	}	logGlobal->error("Invalid trade item type: %d", (int)type);	return "";}const CArtifactInstance * CTradeableItem::getArtInstance() const{	switch(type)	{	case ARTIFACT_PLACEHOLDER:	case ARTIFACT_INSTANCE:		return hlp;	default:		return nullptr;	}}void CTradeableItem::setArtInstance(const CArtifactInstance * art){	assert(type == ARTIFACT_PLACEHOLDER || type == ARTIFACT_INSTANCE);	hlp = art;	if(art)		setID(art->artType->getId());	else		setID(-1);}SResourcesPanel::SResourcesPanel(CTradeableItem::ClickPressedFunctor clickPressedCallback, updatePanelFunctor updateSubtitles)	: updateSubtitles(updateSubtitles){	assert(resourcesForTrade.size() == slotsPos.size());	OBJECT_CONSTRUCTION_CUSTOM_CAPTURING(255 - DISPOSE);	for(const auto & res : resourcesForTrade)	{		slots.emplace_back(std::make_shared<CTradeableItem>(slotsPos[res.num], EType::RESOURCE, res.num, true, res.num));		slots.back()->clickPressedCallback = clickPressedCallback;		slots.back()->pos.w = 69; slots.back()->pos.h = 66;	}}void SResourcesPanel::updateSlots(){	if(updateSubtitles)		updateSubtitles();}CTradeBase::CTradeBase(const IMarket * market, const CGHeroInstance * hero)	: market(market)	, hero(hero){}void CTradeBase::removeItems(const std::set<std::shared_ptr<CTradeableItem>> & toRemove){	for(auto item : toRemove)		removeItem(item);}void CTradeBase::removeItem(std::shared_ptr<CTradeableItem> item){	items[item->left] -= item;	if(hRight == item)		hRight.reset();}void CTradeBase::getEmptySlots(std::set<std::shared_ptr<CTradeableItem>> & toRemove){	for(auto item : items[1])		if(!hero->getStackCount(SlotID(item->serial)))			toRemove.insert(item);}
 |