| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011 | #include "StdInc.h"#include "CIntObjectClasses.h"#include "../CBitmapHandler.h"#include "SDL_Pixels.h"#include "SDL_Extensions.h"#include "../Graphics.h"#include "../CAnimation.h"#include "CCursorHandler.h"#include "../CGameInfo.h"#include "../../CCallback.h"#include "../../lib/CConfigHandler.h"#include "../battle/CBattleInterface.h"#include "../battle/CBattleInterfaceClasses.h"#include "../CPlayerInterface.h"#include "../CMessage.h"#include "../CMusicHandler.h"#include "../GUIClasses.h"#include "CGuiHandler.h"#include "../CAdvmapInterface.h"#include "../../lib/CGeneralTextHandler.h" //for Unicode related stuffCPicture::CPicture( SDL_Surface *BG, int x, int y, bool Free ){	init();	bg = BG;	freeSurf = Free;	pos.x += x;	pos.y += y;	pos.w = BG->w;	pos.h = BG->h;}CPicture::CPicture( const std::string &bmpname, int x, int y ){	init();	bg = BitmapHandler::loadBitmap(bmpname);	freeSurf = true;;	pos.x += x;	pos.y += y;	if(bg)	{		pos.w = bg->w;		pos.h = bg->h;	}	else	{		pos.w = pos.h = 0;	}}CPicture::CPicture(const Rect &r, const SDL_Color &color, bool screenFormat /*= false*/){	init();	createSimpleRect(r, screenFormat, SDL_MapRGB(bg->format, color.r, color.g,color.b));}CPicture::CPicture(const Rect &r, ui32 color, bool screenFormat /*= false*/){	init();	createSimpleRect(r, screenFormat, color);}CPicture::CPicture(SDL_Surface *BG, const Rect &SrcRect, int x /*= 0*/, int y /*= 0*/, bool free /*= false*/){	needRefresh = false;	srcRect = new Rect(SrcRect);	pos.x += x;	pos.y += y;	pos.w = srcRect->w;	pos.h = srcRect->h;	bg = BG;	freeSurf = free;}void CPicture::setSurface(SDL_Surface *to){	bg = to;	if (srcRect)	{		pos.w = srcRect->w;		pos.h = srcRect->h;	}	else	{		pos.w = bg->w;		pos.h = bg->h;	}}CPicture::~CPicture(){	if(freeSurf)		SDL_FreeSurface(bg);	delete srcRect;}void CPicture::init(){	needRefresh = false;	srcRect = nullptr;}void CPicture::show(SDL_Surface * to){	if (needRefresh)		showAll(to);}void CPicture::showAll(SDL_Surface * to){	if(bg)	{		if(srcRect)		{			SDL_Rect srcRectCpy = *srcRect;			SDL_Rect dstRect = srcRectCpy;			dstRect.x = pos.x;			dstRect.y = pos.y;			CSDL_Ext::blitSurface(bg, &srcRectCpy, to, &dstRect);		}		else			blitAt(bg, pos, to);	}}void CPicture::convertToScreenBPP(){	SDL_Surface *hlp = bg;	bg = SDL_ConvertSurface(hlp,screen->format,0);	CSDL_Ext::setDefaultColorKey(bg);		SDL_FreeSurface(hlp);}void CPicture::setAlpha(int value){		#ifdef VCMI_SDL1	SDL_SetAlpha(bg, SDL_SRCALPHA, value);		#else	SDL_SetSurfaceAlphaMod(bg,value);	#endif // 0}void CPicture::scaleTo(Point size){	SDL_Surface * scaled = CSDL_Ext::scaleSurface(bg, size.x, size.y);	if(freeSurf)		SDL_FreeSurface(bg);	setSurface(scaled);	freeSurf = false;}void CPicture::createSimpleRect(const Rect &r, bool screenFormat, ui32 color){	pos += r;	pos.w = r.w;	pos.h = r.h;	if(screenFormat)		bg = CSDL_Ext::newSurface(r.w, r.h);	else		bg = SDL_CreateRGBSurface(SDL_SWSURFACE, r.w, r.h, 8, 0, 0, 0, 0);	SDL_FillRect(bg, nullptr, color);	freeSurf = true;}void CPicture::colorizeAndConvert(PlayerColor player){	assert(bg);	colorize(player);	convertToScreenBPP();}void CPicture::colorize(PlayerColor player){	assert(bg);	graphics->blueToPlayersAdv(bg, player);}CFilledTexture::CFilledTexture(std::string imageName, Rect position):    CIntObject(0, position.topLeft()),    texture(BitmapHandler::loadBitmap(imageName)){	pos.w = position.w;	pos.h = position.h;}CFilledTexture::~CFilledTexture(){	SDL_FreeSurface(texture);}void CFilledTexture::showAll(SDL_Surface *to){	CSDL_Ext::CClipRectGuard guard(to, pos);	CSDL_Ext::fillTexture(to, texture);}CButtonBase::CButtonBase(){	swappedImages = keepFrame = false;	bitmapOffset = 0;	state=NORMAL;	image = nullptr;	text = nullptr;}CButtonBase::~CButtonBase(){}void CButtonBase::update(){	if (text)	{		if (state == PRESSED)			text->moveTo(Point(pos.x+pos.w/2+1, pos.y+pos.h/2+1));		else			text->moveTo(Point(pos.x+pos.w/2, pos.y+pos.h/2));	}	int newPos = (int)state + bitmapOffset;	if (newPos < 0)		newPos = 0;	if (state == HIGHLIGHTED && image->size() < 4)		newPos = image->size()-1;	if (swappedImages)	{		if (newPos == 0) newPos = 1;		else if (newPos == 1) newPos = 0;	}	if (!keepFrame)		image->setFrame(newPos);	if (active)		redraw();}void CButtonBase::addTextOverlay( const std::string &Text, EFonts font, SDL_Color color){	OBJ_CONSTRUCTION_CAPTURING_ALL;	delete text;	text = new CLabel(pos.w/2, pos.h/2, font, CENTER, color, Text);	update();}void CButtonBase::setOffset(int newOffset){	if (bitmapOffset == newOffset)		return;	bitmapOffset = newOffset;	update();}void CButtonBase::setState(ButtonState newState){	if (state == newState)		return;	state = newState;	update();}CButtonBase::ButtonState CButtonBase::getState(){	return state;}bool CButtonBase::isBlocked(){	return state == BLOCKED;}bool CButtonBase::isHighlighted(){	return state == HIGHLIGHTED;}void CButtonBase::block(bool on){	setState(on?BLOCKED:NORMAL);}CAdventureMapButton::CAdventureMapButton (){	hoverable = actOnDown = borderEnabled = soundDisabled = false;	CSDL_Ext::colorSetAlpha(borderColor,1);// represents a transparent color, used for HighlightableButton	addUsedEvents(LCLICK | RCLICK | HOVER | KEYBOARD);}CAdventureMapButton::CAdventureMapButton( const std::string &Name, const std::string &HelpBox, const CFunctionList<void()> &Callback, int x, int y,  const std::string &defName,int key, std::vector<std::string> * add, bool playerColoredButton ){	std::map<int,std::string> pom;	pom[0] = Name;	init(Callback, pom, HelpBox, playerColoredButton, defName, add, x, y, key);}CAdventureMapButton::CAdventureMapButton( const std::string &Name, const std::string &HelpBox, const CFunctionList<void()> &Callback, config::ButtonInfo *info, int key/*=0*/ ){	std::map<int,std::string> pom;	pom[0] = Name;	init(Callback, pom, HelpBox, info->playerColoured, info->defName, &info->additionalDefs, info->x, info->y, key);}CAdventureMapButton::CAdventureMapButton( const std::pair<std::string, std::string> &help, const CFunctionList<void()> &Callback, int x, int y, const std::string &defName, int key/*=0*/, std::vector<std::string> * add /*= nullptr*/, bool playerColoredButton /*= false */ ){	std::map<int,std::string> pom;	pom[0] = help.first;	init(Callback, pom, help.second, playerColoredButton, defName, add, x, y, key);}void CAdventureMapButton::onButtonClicked(){	// debug logging to figure out pressed button (and as result - player actions) in case of crash	logAnim->traceStream() << "Button clicked at " << pos.x << "x" << pos.y;	CIntObject * parent = this->parent;	std::string prefix = "Parent is";	while (parent)	{		logAnim->traceStream() << prefix << typeid(*parent).name() << " at " << parent->pos.x << "x" << parent->pos.y;		parent = parent->parent;		prefix = '\t' + prefix;	}	callback();}void CAdventureMapButton::clickLeft(tribool down, bool previousState){	if(isBlocked())		return;	if (down)	{		if (!soundDisabled)			CCS->soundh->playSound(soundBase::button);		setState(PRESSED);	}	else if(hoverable && hovered)		setState(HIGHLIGHTED);	else		setState(NORMAL);	if (actOnDown && down)	{		onButtonClicked();	}	else if (!actOnDown && previousState && (down==false))	{		onButtonClicked();	}}void CAdventureMapButton::clickRight(tribool down, bool previousState){	if(down && helpBox.size()) //there is no point to show window with nothing inside...		CRClickPopup::createAndPush(helpBox);}void CAdventureMapButton::hover (bool on){	if(hoverable)	{		if(on)			setState(HIGHLIGHTED);		else			setState(NORMAL);	}	if(pressedL && on)		setState(PRESSED);	std::string *name = (vstd::contains(hoverTexts,getState()))		? (&hoverTexts[getState()])		: (vstd::contains(hoverTexts,0) ? (&hoverTexts[0]) : nullptr);	if(name && name->size() && !isBlocked()) //if there is no name, there is nohing to display also	{		if (LOCPLINT && LOCPLINT->battleInt) //for battle buttons		{			if(on && LOCPLINT->battleInt->console->alterTxt == "")			{				LOCPLINT->battleInt->console->alterTxt = *name;				LOCPLINT->battleInt->console->whoSetAlter = 1;			}			else if (LOCPLINT->battleInt->console->alterTxt == *name)			{				LOCPLINT->battleInt->console->alterTxt = "";				LOCPLINT->battleInt->console->whoSetAlter = 0;			}		}		else if(GH.statusbar) //for other buttons		{			if (on)				GH.statusbar->setText(*name);			else if ( GH.statusbar->getText()==(*name) )				GH.statusbar->clear();		}	}}void CAdventureMapButton::init(const CFunctionList<void()> &Callback, const std::map<int,std::string> &Name, const std::string &HelpBox, bool playerColoredButton, const std::string &defName, std::vector<std::string> * add, int x, int y, int key){	currentImage = -1;	addUsedEvents(LCLICK | RCLICK | HOVER | KEYBOARD);	callback = Callback;	hoverable = actOnDown = borderEnabled = soundDisabled = false;	CSDL_Ext::colorSetAlpha(borderColor,1);// represents a transparent color, used for HighlightableButton	hoverTexts = Name;	helpBox=HelpBox;	if (key != SDLK_UNKNOWN)		assignedKeys.insert(key);	pos.x += x;	pos.y += y;	if (!defName.empty())		imageNames.push_back(defName);	if (add)		for (auto & elem : *add)			imageNames.push_back(elem);	setIndex(0, playerColoredButton);}void CAdventureMapButton::setIndex(size_t index, bool playerColoredButton){	if (index == currentImage || index>=imageNames.size())		return;	currentImage = index;	setImage(new CAnimation(imageNames[index]), playerColoredButton);}void CAdventureMapButton::setImage(CAnimation* anim, bool playerColoredButton, int animFlags){	OBJ_CONSTRUCTION_CAPTURING_ALL;	delete image;	image = new CAnimImage(anim, getState(), 0, 0, 0, animFlags);	if (playerColoredButton)		image->playerColored(LOCPLINT->playerID);	pos.w = image->pos.w;	pos.h = image->pos.h;}void CAdventureMapButton::setPlayerColor(PlayerColor player){	if (image)		image->playerColored(player);}void CAdventureMapButton::showAll(SDL_Surface * to){	CIntObject::showAll(to);		#ifdef VCMI_SDL1	if (borderEnabled && borderColor.unused == 0)		CSDL_Ext::drawBorder(to, pos.x-1, pos.y-1, pos.w+2, pos.h+2, int3(borderColor.r, borderColor.g, borderColor.b));		#else	if (borderEnabled && borderColor.a == 0)		CSDL_Ext::drawBorder(to, pos.x-1, pos.y-1, pos.w+2, pos.h+2, int3(borderColor.r, borderColor.g, borderColor.b));		#endif // 0}void CHighlightableButton::select(bool on){	selected = on;	if (on)	{		borderEnabled = true;		setState(HIGHLIGHTED);		callback();	}	else	{		borderEnabled = false;		setState(NORMAL);		callback2();	}	if(hoverTexts.size()>1)	{		hover(false);		hover(true);	}}void CHighlightableButton::clickLeft(tribool down, bool previousState){	if(isBlocked())		return;	if (down && !(onlyOn && isHighlighted()))	{		CCS->soundh->playSound(soundBase::button);		setState(PRESSED);	}	if(previousState)//mouse up	{		if(down == false && getState() == PRESSED)			select(!selected);		else			setState(selected?HIGHLIGHTED:NORMAL);	}}CHighlightableButton::CHighlightableButton( const CFunctionList<void()> &onSelect, const CFunctionList<void()> &onDeselect, const std::map<int,std::string> &Name, const std::string &HelpBox, bool playerColoredButton, const std::string &defName, std::vector<std::string> * add, int x, int y, int key): onlyOn(false), selected(false), callback2(onDeselect){	init(onSelect,Name,HelpBox,playerColoredButton,defName,add,x,y,key);}CHighlightableButton::CHighlightableButton( const std::pair<std::string, std::string> &help, const CFunctionList<void()> &onSelect, int x, int y, const std::string &defName, int myid, int key/*=0*/, std::vector<std::string> * add /*= nullptr*/, bool playerColoredButton /*= false */ ): onlyOn(false), selected(false) // TODO: callback2(???){	ID = myid;	std::map<int,std::string> pom;	pom[0] = help.first;	init(onSelect, pom, help.second, playerColoredButton, defName, add, x, y, key);}CHighlightableButton::CHighlightableButton( const std::string &Name, const std::string &HelpBox, const CFunctionList<void()> &onSelect, int x, int y, const std::string &defName, int myid, int key/*=0*/, std::vector<std::string> * add /*= nullptr*/, bool playerColoredButton /*= false */ ): onlyOn(false), selected(false) // TODO: callback2(???){	ID = myid;	std::map<int,std::string> pom;	pom[0] = Name;	init(onSelect, pom,HelpBox, playerColoredButton, defName, add, x, y, key);}void CHighlightableButtonsGroup::addButton(CHighlightableButton* bt){	if (bt->parent)		bt->parent->removeChild(bt);	addChild(bt);	bt->recActions = defActions;//FIXME: not needed?    bt->callback += std::bind(&CHighlightableButtonsGroup::selectionChanged,this,bt->ID);	bt->onlyOn = true;	buttons.push_back(bt);}void CHighlightableButtonsGroup::addButton(const std::map<int,std::string> &tooltip, const std::string &HelpBox, const std::string &defName, int x, int y, int uid, const CFunctionList<void()> &OnSelect, int key){	OBJ_CONSTRUCTION_CAPTURING_ALL;	CHighlightableButton *bt = new CHighlightableButton(OnSelect, 0, tooltip, HelpBox, false, defName, nullptr, x, y, key);	if(musicLike)	{		bt->setOffset(buttons.size()-3);	}	bt->ID = uid;    bt->callback += std::bind(&CHighlightableButtonsGroup::selectionChanged,this,bt->ID);	bt->onlyOn = true;	buttons.push_back(bt);}CHighlightableButtonsGroup::CHighlightableButtonsGroup(const CFunctionList<void(int)> &OnChange, bool musicLikeButtons): onChange(OnChange), musicLike(musicLikeButtons){}CHighlightableButtonsGroup::~CHighlightableButtonsGroup(){}void CHighlightableButtonsGroup::select(int id, bool mode){	assert(!buttons.empty());	CHighlightableButton *bt = buttons.front();	if(mode)	{		for(auto btn : buttons)			if (btn->ID == id)				bt = btn;	}	else	{		bt = buttons[id];	}	bt->select(true);	selectionChanged(bt->ID);}void CHighlightableButtonsGroup::selectionChanged(int to){	for(auto & elem : buttons)		if(elem->ID!=to && elem->isHighlighted())			elem->select(false);	onChange(to);	if (parent)		parent->redraw();}void CHighlightableButtonsGroup::show(SDL_Surface * to){	if (musicLike)	{		for(auto & elem : buttons)			if(elem->isHighlighted())				elem->show(to);	}	else		CIntObject::show(to);}void CHighlightableButtonsGroup::showAll(SDL_Surface * to){	if (musicLike)	{		for(auto & elem : buttons)			if(elem->isHighlighted())				elem->showAll(to);	}	else		CIntObject::showAll(to);}void CHighlightableButtonsGroup::block( ui8 on ){	for(auto & elem : buttons)	{		elem->block(on);	}}void CSlider::sliderClicked(){	if(!(active & MOVE))		addUsedEvents(MOVE);}void CSlider::mouseMoved (const SDL_MouseMotionEvent & sEvent){	double v = 0;	if(horizontal)	{		if(	std::abs(sEvent.y-(pos.y+pos.h/2)) > pos.h/2+40  ||  std::abs(sEvent.x-(pos.x+pos.w/2)) > pos.w/2  )			return;		v = sEvent.x - pos.x - 24;		v *= positions;		v /= (pos.w - 48);	}	else	{		if(std::abs(sEvent.x-(pos.x+pos.w/2)) > pos.w/2+40  ||  std::abs(sEvent.y-(pos.y+pos.h/2)) > pos.h/2  )			return;		v = sEvent.y - pos.y - 24;		v *= positions;		v /= (pos.h - 48);	}	v += 0.5;	if(v!=value)	{		moveTo(v);		redrawSlider();	}}void CSlider::redrawSlider(){	//slider->show(screenBuf);}void CSlider::moveLeft(){	moveTo(value-1);}void CSlider::moveRight(){	moveTo(value+1);}void CSlider::moveTo(int to){	vstd::amax(to, 0);	vstd::amin(to, positions);	//same, old position?	if(value == to)		return;	value = to;	if(horizontal)	{		if(positions)		{			double part = static_cast<double>(to) / positions;			part*=(pos.w-48);			int newPos = part + pos.x + 16 - slider->pos.x;			slider->moveBy(Point(newPos, 0));		}		else			slider->moveTo(Point(pos.x+16, pos.y));	}	else	{		if(positions)		{			double part = static_cast<double>(to) / positions;			part*=(pos.h-48);			int newPos = part + pos.y + 16 - slider->pos.y;			slider->moveBy(Point(0, newPos));		}		else			slider->moveTo(Point(pos.x, pos.y+16));	}	if(moved)		moved(to);}void CSlider::clickLeft(tribool down, bool previousState){	if(down && !slider->isBlocked())	{		double pw = 0;		double rw = 0;		if(horizontal)		{			pw = GH.current->motion.x-pos.x-25;			rw = pw / static_cast<double>(pos.w - 48);		}		else		{			pw = GH.current->motion.y-pos.y-24;			rw = pw / (pos.h-48);		}		if(pw < -8  ||  pw > (horizontal ? pos.w : pos.h) - 40)			return;		// 		if (rw>1) return;		// 		if (rw<0) return;		slider->clickLeft(true, slider->pressedL);		moveTo(rw * positions  +  0.5);		return;	}	if(active & MOVE)		removeUsedEvents(MOVE);}CSlider::~CSlider(){}CSlider::CSlider(int x, int y, int totalw, std::function<void(int)> Moved, int Capacity, int Amount, int Value, bool Horizontal, int style):    capacity(Capacity),    amount(Amount),    scrollStep(1),    horizontal(Horizontal),    moved(Moved){	OBJ_CONSTRUCTION_CAPTURING_ALL;	setAmount(amount);	addUsedEvents(LCLICK | KEYBOARD | WHEEL);	strongInterest = true;	left = new CAdventureMapButton();	right = new CAdventureMapButton();	slider = new CAdventureMapButton();	pos.x += x;	pos.y += y;	if(horizontal)	{		left->pos.y = slider->pos.y = right->pos.y = pos.y;		left->pos.x = pos.x;		right->pos.x = pos.x + totalw - 16;	}	else	{		left->pos.x = slider->pos.x = right->pos.x = pos.x;		left->pos.y = pos.y;		right->pos.y = pos.y + totalw - 16;	}    left->callback = std::bind(&CSlider::moveLeft,this);    right->callback = std::bind(&CSlider::moveRight,this);    slider->callback = std::bind(&CSlider::sliderClicked,this);	left->pos.w = left->pos.h = right->pos.w = right->pos.h = slider->pos.w = slider->pos.h = 16;	if(horizontal)	{		pos.h = 16;		pos.w = totalw;	}	else	{		pos.w = 16;		pos.h = totalw;	}	if(style == 0)	{		std::string name = horizontal?"IGPCRDIV.DEF":"OVBUTN2.DEF";		//NOTE: this images do not have "blocked" frames. They should be implemented somehow (e.g. palette transform or something...)		//use source def to create custom animations. Format "name.def:123" will load this frame from def file		auto animLeft = new CAnimation();		animLeft->setCustom(name + ":0", 0);		animLeft->setCustom(name + ":1", 1);		left->setImage(animLeft);		auto animRight = new CAnimation();		animRight->setCustom(name + ":2", 0);		animRight->setCustom(name + ":3", 1);		right->setImage(animRight);		auto animSlider = new CAnimation();		animSlider->setCustom(name + ":4", 0);		slider->setImage(animSlider);	}	else	{		left->setImage(new CAnimation(horizontal ? "SCNRBLF.DEF" : "SCNRBUP.DEF"));		right->setImage(new CAnimation(horizontal ? "SCNRBRT.DEF" : "SCNRBDN.DEF"));		slider->setImage(new CAnimation("SCNRBSL.DEF"));	}	slider->actOnDown = true;	slider->soundDisabled = true;	left->soundDisabled = true;	right->soundDisabled = true;	value = -1;	moveTo(Value);}void CSlider::block( bool on ){	left->block(on);	right->block(on);	slider->block(on);}void CSlider::setAmount( int to ){	amount = to;	positions = to - capacity;	vstd::amax(positions, 0);}void CSlider::showAll(SDL_Surface * to){	CSDL_Ext::fillRectBlack(to, &pos);	CIntObject::showAll(to);}void CSlider::wheelScrolled(bool down, bool in){	moveTo(value + 3 * (down ? +scrollStep : -scrollStep));}void CSlider::keyPressed(const SDL_KeyboardEvent & key){	if(key.state != SDL_PRESSED) return;	int moveDest = 0;	switch(key.keysym.sym)	{	case SDLK_UP:	case SDLK_LEFT:		moveDest = value - scrollStep;		break;	case SDLK_DOWN:	case SDLK_RIGHT:		moveDest = value + scrollStep;		break;	case SDLK_PAGEUP:		moveDest = value - capacity + scrollStep;		break;	case SDLK_PAGEDOWN:		moveDest = value + capacity - scrollStep;		break;	case SDLK_HOME:		moveDest = 0;		break;	case SDLK_END:		moveDest = amount - capacity;		break;	default:		return;	}	moveTo(moveDest);}void CSlider::moveToMax(){	moveTo(amount);}static void intDeleter(CIntObject* object){	delete object;}CObjectList::CObjectList(CreateFunc create, DestroyFunc destroy):createObject(create),destroyObject(destroy){	if (!destroyObject)		destroyObject = intDeleter;}void CObjectList::deleteItem(CIntObject* item){	if (!item)		return;	removeChild(item);	destroyObject(item);}CIntObject* CObjectList::createItem(size_t index){	OBJ_CONSTRUCTION_CAPTURING_ALL;	CIntObject * item = createObject(index);	if (item == nullptr)		item = new CIntObject();	item->recActions = defActions;	addChild(item);	return item;}CTabbedInt::CTabbedInt(CreateFunc create, DestroyFunc destroy, Point position, size_t ActiveID):CObjectList(create, destroy),activeTab(nullptr),activeID(ActiveID){	pos += position;	reset();}void CTabbedInt::setActive(size_t which){	if (which != activeID)	{		activeID = which;		reset();	}}void CTabbedInt::reset(){	deleteItem(activeTab);	activeTab = createItem(activeID);	activeTab->moveTo(pos.topLeft());	if (active)		redraw();}CIntObject * CTabbedInt::getItem(){	return activeTab;}CListBox::CListBox(CreateFunc create, DestroyFunc destroy, Point Pos, Point ItemOffset, size_t VisibleSize,				   size_t TotalSize, size_t InitialPos, int Slider, Rect SliderPos):	CObjectList(create, destroy),	first(InitialPos),	totalSize(TotalSize),	itemOffset(ItemOffset),    slider(nullptr){	pos += Pos;	items.resize(VisibleSize, nullptr);	if (Slider & 1)	{		OBJ_CONSTRUCTION_CAPTURING_ALL;        slider = new CSlider(SliderPos.x, SliderPos.y, SliderPos.w, std::bind(&CListBox::moveToPos, this, _1),			VisibleSize, TotalSize, InitialPos, Slider & 2, Slider & 4);	}	reset();}// Used to move active items after changing list positionvoid CListBox::updatePositions(){	Point itemPos = pos.topLeft();	for (auto & elem : items)	{		(elem)->moveTo(itemPos);		itemPos += itemOffset;	}	if (active)	{		redraw();		if (slider)			slider->moveTo(first);	}}void CListBox::reset(){	size_t current = first;	for (auto & elem : items)	{		deleteItem(elem);		elem = createItem(current++);	}	updatePositions();}void CListBox::resize(size_t newSize){	totalSize = newSize;	if (slider)		slider->setAmount(totalSize);	reset();}size_t CListBox::size(){	return totalSize;}CIntObject * CListBox::getItem(size_t which){	if (which < first || which > first + items.size() || which > totalSize)		return nullptr;	size_t i=first;	for (auto iter = items.begin(); iter != items.end(); iter++, i++)		if( i == which)			return *iter;	return nullptr;}size_t CListBox::getIndexOf(CIntObject *item){	size_t i=first;	for (auto iter = items.begin(); iter != items.end(); iter++, i++)		if(*iter == item)			return i;	return size_t(-1);}void CListBox::scrollTo(size_t which){	//scroll up	if (first > which)		moveToPos(which);	//scroll down	else if (first + items.size() <= which && which < totalSize)		moveToPos(which - items.size() + 1);}void CListBox::moveToPos(size_t which){	//Calculate new position	size_t maxPossible;	if (totalSize > items.size())		maxPossible = totalSize - items.size();	else		maxPossible = 0;	size_t newPos = std::min(which, maxPossible);	//If move distance is 1 (most of calls from Slider) - use faster shifts instead of resetting all items	if (first - newPos == 1)		moveToPrev();	else if (newPos - first == 1)		moveToNext();	else if (newPos != first)	{		first = newPos;		reset();	}}void CListBox::moveToNext(){	//Remove front item and insert new one to end	if (first + items.size() < totalSize)	{		first++;		deleteItem(items.front());		items.pop_front();		items.push_back(createItem(first+items.size()));		updatePositions();	}}void CListBox::moveToPrev(){	//Remove last item and insert new one at start	if (first)	{		first--;		deleteItem(items.back());		items.pop_back();		items.push_front(createItem(first));		updatePositions();	}}size_t CListBox::getPos(){	return first;}const std::list<CIntObject *> &CListBox::getItems(){	return items;}void CSimpleWindow::show(SDL_Surface * to){	if(bitmap)		blitAt(bitmap,pos.x,pos.y,to);}CSimpleWindow::~CSimpleWindow(){	if (bitmap)	{		SDL_FreeSurface(bitmap);		bitmap=nullptr;	}}void CHoverableArea::hover (bool on){	if (on)		GH.statusbar->setText(hoverText);	else if (GH.statusbar->getText()==hoverText)		GH.statusbar->clear();}CHoverableArea::CHoverableArea(){	addUsedEvents(HOVER);}CHoverableArea::~CHoverableArea(){}void LRClickableAreaWText::clickLeft(tribool down, bool previousState){	if(!down && previousState)	{		LOCPLINT->showInfoDialog(text);	}}void LRClickableAreaWText::clickRight(tribool down, bool previousState){	adventureInt->handleRightClick(text, down);}LRClickableAreaWText::LRClickableAreaWText(){	init();}LRClickableAreaWText::LRClickableAreaWText(const Rect &Pos, const std::string &HoverText /*= ""*/, const std::string &ClickText /*= ""*/){	init();	pos = Pos + pos;	hoverText = HoverText;	text = ClickText;}LRClickableAreaWText::~LRClickableAreaWText(){}void LRClickableAreaWText::init(){	addUsedEvents(LCLICK | RCLICK | HOVER);}std::string CLabel::visibleText(){	return text;}void CLabel::showAll(SDL_Surface * to){	CIntObject::showAll(to);	if(!visibleText().empty())		blitLine(to, pos, visibleText());}CLabel::CLabel(int x, int y, EFonts Font /*= FONT_SMALL*/, EAlignment Align, const SDL_Color &Color /*= Colors::WHITE*/, const std::string &Text /*= ""*/):CTextContainer(Align, Font, Color), text(Text){	type |= REDRAW_PARENT;	autoRedraw = true;	pos.x += x;	pos.y += y;	pos.w = pos.h = 0;	bg = nullptr;	if (alignment == TOPLEFT) // causes issues for MIDDLE	{		pos.w = graphics->fonts[font]->getStringWidth(visibleText().c_str());		pos.h = graphics->fonts[font]->getLineHeight();	}}Point CLabel::getBorderSize(){	return Point(0, 0);}std::string CLabel::getText(){	return text;}void CLabel::setText(const std::string &Txt){	text = Txt;	if(autoRedraw)	{		if(bg || !parent)			redraw();		else			parent->redraw();	}}CMultiLineLabel::CMultiLineLabel(Rect position, EFonts Font, EAlignment Align, const SDL_Color &Color, const std::string &Text):    CLabel(position.x, position.y, Font, Align, Color, Text),    visibleSize(0, 0, position.w, position.h){	pos.w = position.w;	pos.h = position.h;	splitText(Text);}void CMultiLineLabel::setVisibleSize(Rect visibleSize){	this->visibleSize = visibleSize;	redraw();}void CMultiLineLabel::scrollTextBy(int distance){	scrollTextTo(visibleSize.y + distance);}void CMultiLineLabel::scrollTextTo(int distance){	Rect size = visibleSize;	size.y = distance;	setVisibleSize(size);}void CMultiLineLabel::setText(const std::string &Txt){	splitText(Txt);	CLabel::setText(Txt);}void CTextContainer::blitLine(SDL_Surface *to, Rect destRect, std::string what){	const IFont * f = graphics->fonts[font];	Point where = destRect.topLeft();	// input is rect in which given text should be placed	// calculate proper position for top-left corner of the text	if (alignment == TOPLEFT)	{		where.x += getBorderSize().x;		where.y += getBorderSize().y;	}	if (alignment == CENTER)	{		where.x += (int(destRect.w) - int(f->getStringWidth(what))) / 2;		where.y += (int(destRect.h) - int(f->getLineHeight())) / 2;	}	if (alignment == BOTTOMRIGHT)	{		where.x += getBorderSize().x + destRect.w - f->getStringWidth(what);		where.y += getBorderSize().y + destRect.h - f->getLineHeight();	}	size_t begin = 0;	std::string delimeters = "{}";	size_t currDelimeter = 0;	do	{		size_t end = what.find_first_of(delimeters[currDelimeter % 2], begin);		if (begin != end)		{			std::string toPrint = what.substr(begin, end - begin);			if (currDelimeter % 2) // Enclosed in {} text - set to yellow				f->renderTextLeft(to, toPrint, Colors::YELLOW, where);			else // Non-enclosed text, use default color				f->renderTextLeft(to, toPrint, color, where);			begin = end;			where.x += f->getStringWidth(toPrint);		}		currDelimeter++;	}	while (begin++ != std::string::npos);}CTextContainer::CTextContainer(EAlignment alignment, EFonts font, SDL_Color color):	alignment(alignment),	font(font),	color(color){}void CMultiLineLabel::showAll(SDL_Surface * to){	CIntObject::showAll(to);	const IFont * f = graphics->fonts[font];	// calculate which lines should be visible	int totalLines = lines.size();	int beginLine  = visibleSize.y;	int endLine    = getTextLocation().h + visibleSize.y;	if (beginLine < 0)		beginLine = 0;	else		beginLine /= f->getLineHeight();	if (endLine < 0)		endLine = 0;	else		endLine /= f->getLineHeight();	endLine++;	// and where they should be displayed	Point lineStart = getTextLocation().topLeft() - visibleSize + Point(0, beginLine * f->getLineHeight());	Point lineSize  = Point(getTextLocation().w, f->getLineHeight());	CSDL_Ext::CClipRectGuard guard(to, getTextLocation()); // to properly trim text that is too big to fit	for (int i = beginLine; i < std::min(totalLines, endLine); i++)	{		if (!lines[i].empty()) //non-empty line			blitLine(to, Rect(lineStart, lineSize), lines[i]);		lineStart.y += f->getLineHeight();	}}void CMultiLineLabel::splitText(const std::string &Txt){	lines.clear();	const IFont * f = graphics->fonts[font];	int lineHeight =  f->getLineHeight();	lines = CMessage::breakText(Txt, pos.w, font);	 textSize.y = lineHeight * lines.size();	 textSize.x = 0;	for(const std::string &line : lines)		vstd::amax( textSize.x, f->getStringWidth(line.c_str()));	redraw();}Rect CMultiLineLabel::getTextLocation(){	// this method is needed for vertical alignment alignment of text	// when height of available text is smaller than height of widget	// in this case - we should add proper offset to display text at required position	if (pos.h <= textSize.y)		return pos;	Point textSize(pos.w, graphics->fonts[font]->getLineHeight() * lines.size());	Point textOffset(pos.w - textSize.x, pos.h - textSize.y);	switch(alignment)	{	case TOPLEFT:     return Rect(pos.topLeft(), textSize);	case CENTER:      return Rect(pos.topLeft() + textOffset / 2, textSize);	case BOTTOMRIGHT: return Rect(pos.topLeft() + textOffset, textSize);	}	assert(0);	return Rect();}CLabelGroup::CLabelGroup(EFonts Font, EAlignment Align, const SDL_Color &Color):	font(Font), align(Align), color(Color){}void CLabelGroup::add(int x, int y, const std::string &text){	OBJ_CONSTRUCTION_CAPTURING_ALL;	new CLabel(x, y, font, align, color, text);}CTextBox::CTextBox(std::string Text, const Rect &rect, int SliderStyle, EFonts Font /*= FONT_SMALL*/, EAlignment Align /*= TOPLEFT*/, const SDL_Color &Color /*= Colors::WHITE*/):    sliderStyle(SliderStyle),    slider(nullptr){	OBJ_CONSTRUCTION_CAPTURING_ALL;	label = new CMultiLineLabel(rect, Font, Align, Color);	type |= REDRAW_PARENT;	pos.x += rect.x;	pos.y += rect.y;	pos.h = rect.h;	pos.w = rect.w;	assert(pos.w >= 40); //we need some space	setText(Text);}void CTextBox::sliderMoved(int to){	label->scrollTextTo(to);}void CTextBox::resize(Point newSize){	pos.w = newSize.x;	pos.h = newSize.y;	label->pos.w = pos.w;	label->pos.h = pos.h;	if (slider)		vstd::clear_pointer(slider); // will be recreated if needed later	setText(label->getText()); // force refresh}void CTextBox::setText(const std::string &text){	label->setText(text);	if (label->textSize.y <= label->pos.h && slider)	{		// slider is no longer needed		vstd::clear_pointer(slider);		label->pos.w = pos.w;		label->setText(text);	}	else if (label->textSize.y > label->pos.h && !slider)	{		// create slider and update widget		label->pos.w = pos.w - 32;		label->setText(text);		OBJ_CONSTRUCTION_CAPTURING_ALL;        slider = new CSlider(pos.w - 32, 0, pos.h, std::bind(&CTextBox::sliderMoved, this, _1),		                     label->pos.h, label->textSize.y, 0, false, sliderStyle);		slider->scrollStep = graphics->fonts[label->font]->getLineHeight();	}}void CGStatusBar::setText(const std::string & Text){	if(!textLock)		CLabel::setText(Text);}void CGStatusBar::clear(){	setText("");}CGStatusBar::CGStatusBar(CPicture *BG, EFonts Font /*= FONT_SMALL*/, EAlignment Align /*= CENTER*/, const SDL_Color &Color /*= Colors::WHITE*/): CLabel(BG->pos.x, BG->pos.y, Font, Align, Color, ""){	init();	bg = BG;	addChild(bg);	pos = bg->pos;	getBorderSize();    textLock = false;}CGStatusBar::CGStatusBar(int x, int y, std::string name/*="ADROLLVR.bmp"*/, int maxw/*=-1*/): CLabel(x, y, FONT_SMALL, CENTER){	OBJ_CONSTRUCTION_CAPTURING_ALL;	init();	bg = new CPicture(name);	pos = bg->pos;	if((unsigned int)maxw < pos.w)	{		vstd::amin(pos.w, maxw);		bg->srcRect = new Rect(0, 0, maxw, pos.h);	}    textLock = false;}CGStatusBar::~CGStatusBar(){	GH.statusbar = oldStatusBar;}void CGStatusBar::show(SDL_Surface * to){    showAll(to);}void CGStatusBar::init(){	oldStatusBar = GH.statusbar;	GH.statusbar = this;}Point CGStatusBar::getBorderSize(){	//Width of borders where text should not be printed	static const Point borderSize(5,1);	switch(alignment)	{	case TOPLEFT:     return Point(borderSize.x, borderSize.y);	case CENTER:      return Point(pos.w/2, pos.h/2);	case BOTTOMRIGHT: return Point(pos.w - borderSize.x, pos.h - borderSize.y);	}	assert(0);	return Point();}void CGStatusBar::lock(bool shouldLock){    textLock = shouldLock;}CTextInput::CTextInput(const Rect &Pos, EFonts font, const CFunctionList<void(const std::string &)> &CB):    CLabel(Pos.x, Pos.y, font, CENTER),    cb(CB){	type |= REDRAW_PARENT;	focus = false;	pos.h = Pos.h;	pos.w = Pos.w;	captureAllKeys = true;	bg = nullptr;	addUsedEvents(LCLICK | KEYBOARD | TEXTINPUT);	giveFocus();}CTextInput::CTextInput( const Rect &Pos, const Point &bgOffset, const std::string &bgName, const CFunctionList<void(const std::string &)> &CB ):cb(CB){	focus = false;	pos += Pos;	captureAllKeys = true;	OBJ_CONSTRUCTION;	bg = new CPicture(bgName, bgOffset.x, bgOffset.y);	addUsedEvents(LCLICK | KEYBOARD | TEXTINPUT);	giveFocus();}CTextInput::CTextInput(const Rect &Pos, SDL_Surface *srf){	focus = false;	pos += Pos;	captureAllKeys = true;	OBJ_CONSTRUCTION;	bg = new CPicture(Pos, 0, true);	Rect hlp = Pos;	if(srf)		CSDL_Ext::blitSurface(srf, &hlp, *bg, nullptr);	else		SDL_FillRect(*bg, nullptr, 0);	pos.w = bg->pos.w;	pos.h = bg->pos.h;	bg->pos = pos;	addUsedEvents(LCLICK | KEYBOARD | TEXTINPUT);	giveFocus();}void CTextInput::focusGot(){	CSDL_Ext::startTextInput(&pos);	}void CTextInput::focusLost(){	CSDL_Ext::stopTextInput();}std::string CTextInput::visibleText(){	return focus ? text + newText + "_" : text;}void CTextInput::clickLeft( tribool down, bool previousState ){	if(down && !focus)		giveFocus();}void CTextInput::keyPressed( const SDL_KeyboardEvent & key ){	if(!focus || key.state != SDL_PRESSED)		return;	if(key.keysym.sym == SDLK_TAB)	{		moveFocus();		GH.breakEventHandling();		return;	}	bool redrawNeeded = false;	#ifdef VCMI_SDL1	std::string oldText = text;	#endif // 0		switch(key.keysym.sym)	{	case SDLK_DELETE: // have index > ' ' so it won't be filtered out by default section		return;	case SDLK_BACKSPACE:		if(!newText.empty())		{			Unicode::trimRight(newText);			redrawNeeded = true;		}		else if(!text.empty())		{			Unicode::trimRight(text);			redrawNeeded = true;		}					break;	default:		#ifdef VCMI_SDL1		if (key.keysym.unicode < ' ')			return;		else		{			text += key.keysym.unicode; //TODO 16-/>8			redrawNeeded = true;		}					#endif // 0		break;	}	#ifdef VCMI_SDL1	filters(text, oldText);	#endif // 0	if (redrawNeeded)	{		redraw();		cb(text);	}	}void CTextInput::setText( const std::string &nText, bool callCb ){	CLabel::setText(nText);	if(callCb)		cb(text);}bool CTextInput::captureThisEvent(const SDL_KeyboardEvent & key){	if(key.keysym.sym == SDLK_RETURN || key.keysym.sym == SDLK_KP_ENTER)		return false;		#ifdef VCMI_SDL1	//this should allow all non-printable keys to go through (for example arrows)	if (key.keysym.unicode < ' ')		return false;	return true;	#else	return false;	#endif}#ifndef VCMI_SDL1void CTextInput::textInputed(const SDL_TextInputEvent & event){	if(!focus)		return;	std::string oldText = text;		text += event.text;			filters(text,oldText);	if (text != oldText)	{		redraw();		cb(text);	}		newText = "";}void CTextInput::textEdited(const SDL_TextEditingEvent & event){	if(!focus)		return;			newText = event.text;	redraw();	cb(text+newText);	}#endifvoid CTextInput::filenameFilter(std::string & text, const std::string &){	static const std::string forbiddenChars = "<>:\"/\\|?*\r\n"; //if we are entering a filename, some special characters won't be allowed	size_t pos;	while ((pos = text.find_first_of(forbiddenChars)) != std::string::npos)		text.erase(pos, 1);}void CTextInput::numberFilter(std::string & text, const std::string & oldText, int minValue, int maxValue){	assert(minValue < maxValue);	if (text.empty())		text = "0";	size_t pos = 0;	if (text[0] == '-') //allow '-' sign as first symbol only		pos++;	while (pos < text.size())	{		if (text[pos] < '0' || text[pos] > '9')		{			text = oldText;			return; //new text is not number.		}		pos++;	}	try	{		int value = boost::lexical_cast<int>(text);		if (value < minValue)			text = boost::lexical_cast<std::string>(minValue);		else if (value > maxValue)			text = boost::lexical_cast<std::string>(maxValue);	}	catch(boost::bad_lexical_cast &)	{		//Should never happen. Unless I missed some cases        logGlobal->warnStream() << "Warning: failed to convert "<< text << " to number!";		text = oldText;	}}CFocusable::CFocusable(){	focusables.push_back(this);}CFocusable::~CFocusable(){	if(inputWithFocus == this)	{		focusLost();		inputWithFocus = nullptr;	}		focusables -= this;}void CFocusable::giveFocus(){	if(inputWithFocus)	{		inputWithFocus->focus = false;		inputWithFocus->focusLost();		inputWithFocus->redraw();	}	focus = true;	inputWithFocus = this;	focusGot();	redraw();	}void CFocusable::moveFocus(){	auto i = vstd::find(focusables, this),		ourIt = i;	for(i++; i != ourIt; i++)	{		if(i == focusables.end())			i = focusables.begin();		if((*i)->active)		{			(*i)->giveFocus();			break;;		}	}}CWindowObject::CWindowObject(int options_, std::string imageName, Point centerAt):    CIntObject(getUsedEvents(options_), Point()),    shadow(nullptr),    options(options_),    background(createBg(imageName, options & PLAYER_COLORED)){	assert(parent == nullptr); //Safe to remove, but windows should not have parent	if (options & RCLICK_POPUP)		CCS->curh->hide();	if (background)		pos = background->center(centerAt);	else		center(centerAt);	if (!(options & SHADOW_DISABLED))		setShadow(true);}CWindowObject::CWindowObject(int options_, std::string imageName):    CIntObject(getUsedEvents(options_), Point()),    shadow(nullptr),    options(options_),    background(createBg(imageName, options & PLAYER_COLORED)){	assert(parent == nullptr); //Safe to remove, but windows should not have parent	if (options & RCLICK_POPUP)		CCS->curh->hide();	if (background)		pos = background->center();	else		center(Point(screen->w/2, screen->h/2));	if (!(options & SHADOW_DISABLED))		setShadow(true);}CWindowObject::~CWindowObject(){	setShadow(false);}CPicture * CWindowObject::createBg(std::string imageName, bool playerColored){	OBJ_CONSTRUCTION_CAPTURING_ALL;	if (imageName.empty())		return nullptr;	auto  image = new CPicture(imageName);	if (playerColored)		image->colorize(LOCPLINT->playerID);	return image;}void CWindowObject::setBackground(std::string filename){	OBJ_CONSTRUCTION_CAPTURING_ALL;	delete background;	background = createBg(filename, options & PLAYER_COLORED);	if (background)		pos = background->center(Point(pos.w/2 + pos.x, pos.h/2 + pos.y));	updateShadow();}int CWindowObject::getUsedEvents(int options){	if (options & RCLICK_POPUP)		return RCLICK;	return 0;}void CWindowObject::updateShadow(){	setShadow(false);	if (!(options & SHADOW_DISABLED))		setShadow(true);}void CWindowObject::setShadow(bool on){	//size of shadow	static const int size = 8;	if (on == bool(shadow))		return;	vstd::clear_pointer(shadow);	//object too small to cast shadow	if (pos.h <= size || pos.w <= size)		return;	if (on)	{		//helper to set last row		auto blitAlphaRow = [](SDL_Surface *surf, size_t row)		{			Uint8 * ptr = (Uint8*)surf->pixels + surf->pitch * (row);			for (size_t i=0; i< surf->w; i++)			{				Channels::px<4>::a.set(ptr, 128);				ptr+=4;			}		};		// helper to set last column		auto blitAlphaCol = [](SDL_Surface *surf, size_t col)		{			Uint8 * ptr = (Uint8*)surf->pixels + 4 * (col);			for (size_t i=0; i< surf->h; i++)			{				Channels::px<4>::a.set(ptr, 128);				ptr+= surf->pitch;			}		};		static SDL_Surface * shadowCornerTempl = nullptr;		static SDL_Surface * shadowBottomTempl = nullptr;		static SDL_Surface * shadowRightTempl = nullptr;		//one-time initialization		if (!shadowCornerTempl)		{			//create "template" surfaces			shadowCornerTempl = CSDL_Ext::createSurfaceWithBpp<4>(size, size);			shadowBottomTempl = CSDL_Ext::createSurfaceWithBpp<4>(1, size);			shadowRightTempl  = CSDL_Ext::createSurfaceWithBpp<4>(size, 1);			Uint32 shadowColor = SDL_MapRGBA(shadowCornerTempl->format, 0, 0, 0, 192);			//fill with shadow body color			SDL_FillRect(shadowCornerTempl, nullptr, shadowColor);			SDL_FillRect(shadowBottomTempl, nullptr, shadowColor);			SDL_FillRect(shadowRightTempl,  nullptr, shadowColor);			//fill last row and column with more transparent color			blitAlphaCol(shadowRightTempl , size-1);			blitAlphaCol(shadowCornerTempl, size-1);			blitAlphaRow(shadowBottomTempl, size-1);			blitAlphaRow(shadowCornerTempl, size-1);		}		OBJ_CONSTRUCTION_CAPTURING_ALL;		//FIXME: do something with this points		Point shadowStart;		if (options & BORDERED)			shadowStart = Point(size - 14, size - 14);		else			shadowStart = Point(size, size);		Point shadowPos;		if (options & BORDERED)			shadowPos = Point(pos.w + 14, pos.h + 14);		else			shadowPos = Point(pos.w, pos.h);		Point fullsize;		if (options & BORDERED)			fullsize = Point(pos.w + 28, pos.h + 29);		else			fullsize = Point(pos.w, pos.h);		//create base 8x8 piece of shadow		SDL_Surface * shadowCorner = CSDL_Ext::copySurface(shadowCornerTempl);		SDL_Surface * shadowBottom = CSDL_Ext::scaleSurfaceFast(shadowBottomTempl, fullsize.x - size, size);		SDL_Surface * shadowRight  = CSDL_Ext::scaleSurfaceFast(shadowRightTempl,  size, fullsize.y - size);		blitAlphaCol(shadowBottom, 0);		blitAlphaRow(shadowRight, 0);		//generate "shadow" object with these 3 pieces in it		shadow = new CIntObject;		shadow->addChild(new CPicture(shadowCorner, shadowPos.x, shadowPos.y));		shadow->addChild(new CPicture(shadowRight,  shadowPos.x, shadowStart.y));		shadow->addChild(new CPicture(shadowBottom, shadowStart.x, shadowPos.y));	}}void CWindowObject::showAll(SDL_Surface *to){	CIntObject::showAll(to);	if ((options & BORDERED) && (pos.h != to->h || pos.w != to->w))		CMessage::drawBorder(LOCPLINT ? LOCPLINT->playerID : PlayerColor(1), to, pos.w+28, pos.h+29, pos.x-14, pos.y-15);}void CWindowObject::close(){	GH.popIntTotally(this);}void CWindowObject::clickRight(tribool down, bool previousState){	close();	CCS->curh->show();}
 |