|  | @@ -2267,7 +2267,7 @@ bool CGameHandler::recruitCreatures(ObjectInstanceID objid, ObjectInstanceID dst
 | 
	
		
			
				|  |  |  		COMPLAIN_RET_FALSE_IF(artId == ArtifactID::CATAPULT, "Catapult cannot be recruited!");
 | 
	
		
			
				|  |  |  		COMPLAIN_RET_FALSE_IF(nullptr == art, "Invalid war machine artifact");
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -		return giveHeroNewArtifact(hero, art);
 | 
	
		
			
				|  |  | +		return giveHeroNewArtifact(hero, artId, ArtifactPosition::FIRST_AVAILABLE);
 | 
	
		
			
				|  |  |  	}
 | 
	
		
			
				|  |  |  	else
 | 
	
		
			
				|  |  |  	{
 | 
	
	
		
			
				|  | @@ -2502,7 +2502,7 @@ bool CGameHandler::moveArtifact(const PlayerColor & player, const ArtifactLocati
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  	auto hero = getHero(dst.artHolder);
 | 
	
		
			
				|  |  |  	if(ArtifactUtils::checkSpellbookIsNeeded(hero, srcArtifact->artType->getId(), dstSlot))
 | 
	
		
			
				|  |  | -		giveHeroNewArtifact(hero, ArtifactID(ArtifactID::SPELLBOOK).toArtifact(), ArtifactPosition::SPELLBOOK);
 | 
	
		
			
				|  |  | +		giveHeroNewArtifact(hero, ArtifactID::SPELLBOOK, ArtifactPosition::SPELLBOOK);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  	ma.artsPack0.push_back(BulkMoveArtifacts::LinkedSlots(src.slot, dstSlot));
 | 
	
		
			
				|  |  |  	if(src.artHolder != dst.artHolder)
 | 
	
	
		
			
				|  | @@ -2543,7 +2543,7 @@ bool CGameHandler::bulkMoveArtifacts(const PlayerColor & player, ObjectInstanceI
 | 
	
		
			
				|  |  |  			if(auto dstHero = getHero(dstId))
 | 
	
		
			
				|  |  |  			{
 | 
	
		
			
				|  |  |  				if(ArtifactUtils::checkSpellbookIsNeeded(dstHero, artifact->getTypeId(), dstSlot))
 | 
	
		
			
				|  |  | -					giveHeroNewArtifact(dstHero, ArtifactID(ArtifactID::SPELLBOOK).toArtifact(), ArtifactPosition::SPELLBOOK);
 | 
	
		
			
				|  |  | +					giveHeroNewArtifact(dstHero, ArtifactID::SPELLBOOK, ArtifactPosition::SPELLBOOK);
 | 
	
		
			
				|  |  |  			}
 | 
	
		
			
				|  |  |  		}
 | 
	
		
			
				|  |  |  	};
 | 
	
	
		
			
				|  | @@ -2736,7 +2736,7 @@ bool CGameHandler::assembleArtifacts(ObjectInstanceID heroID, ArtifactPosition a
 | 
	
		
			
				|  |  |  		}
 | 
	
		
			
				|  |  |  		
 | 
	
		
			
				|  |  |  		if(ArtifactUtils::checkSpellbookIsNeeded(hero, assembleTo, artifactSlot))
 | 
	
		
			
				|  |  | -			giveHeroNewArtifact(hero, ArtifactID(ArtifactID::SPELLBOOK).toArtifact(), ArtifactPosition::SPELLBOOK);
 | 
	
		
			
				|  |  | +			giveHeroNewArtifact(hero, ArtifactID::SPELLBOOK, ArtifactPosition::SPELLBOOK);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  		AssembledArtifact aa;
 | 
	
		
			
				|  |  |  		aa.al = dstLoc;
 | 
	
	
		
			
				|  | @@ -2793,7 +2793,7 @@ bool CGameHandler::buyArtifact(ObjectInstanceID hid, ArtifactID aid)
 | 
	
		
			
				|  |  |  			return false;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  		giveResource(hero->getOwner(),EGameResID::GOLD,-GameConstants::SPELLBOOK_GOLD_COST);
 | 
	
		
			
				|  |  | -		giveHeroNewArtifact(hero, ArtifactID(ArtifactID::SPELLBOOK).toArtifact(), ArtifactPosition::SPELLBOOK);
 | 
	
		
			
				|  |  | +		giveHeroNewArtifact(hero, ArtifactID::SPELLBOOK, ArtifactPosition::SPELLBOOK);
 | 
	
		
			
				|  |  |  		assert(hero->getArt(ArtifactPosition::SPELLBOOK));
 | 
	
		
			
				|  |  |  		giveSpells(town,hero);
 | 
	
		
			
				|  |  |  		return true;
 | 
	
	
		
			
				|  | @@ -2821,7 +2821,7 @@ bool CGameHandler::buyArtifact(ObjectInstanceID hid, ArtifactID aid)
 | 
	
		
			
				|  |  |  			}
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  			giveResource(hero->getOwner(),EGameResID::GOLD,-price);
 | 
	
		
			
				|  |  | -			return giveHeroNewArtifact(hero, art);
 | 
	
		
			
				|  |  | +			return giveHeroNewArtifact(hero, aid, ArtifactPosition::FIRST_AVAILABLE);
 | 
	
		
			
				|  |  |  		}
 | 
	
		
			
				|  |  |  		else
 | 
	
		
			
				|  |  |  			COMPLAIN_RET("This machine is unavailable here!");
 | 
	
	
		
			
				|  | @@ -2874,7 +2874,7 @@ bool CGameHandler::buyArtifact(const IMarket *m, const CGHeroInstance *h, GameRe
 | 
	
		
			
				|  |  |  		COMPLAIN_RET("Cannot find selected artifact on the list");
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  	sendAndApply(&saa);
 | 
	
		
			
				|  |  | -	giveHeroNewArtifact(h, aid.toArtifact(), ArtifactPosition::FIRST_AVAILABLE);
 | 
	
		
			
				|  |  | +	giveHeroNewArtifact(h, aid, ArtifactPosition::FIRST_AVAILABLE);
 | 
	
		
			
				|  |  |  	return true;
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
	
		
			
				|  | @@ -3416,7 +3416,7 @@ bool CGameHandler::dig(const CGHeroInstance *h)
 | 
	
		
			
				|  |  |  		iw.text.appendLocalString(EMetaText::GENERAL_TXT, 58); //"Congratulations! After spending many hours digging here, your hero has uncovered the " ...
 | 
	
		
			
				|  |  |  		iw.text.appendName(grail); // ... " The Grail"
 | 
	
		
			
				|  |  |  		iw.soundID = soundBase::ULTIMATEARTIFACT;
 | 
	
		
			
				|  |  | -		giveHeroNewArtifact(h, grail.toArtifact(), ArtifactPosition::FIRST_AVAILABLE); //give grail
 | 
	
		
			
				|  |  | +		giveHeroNewArtifact(h, grail, ArtifactPosition::FIRST_AVAILABLE); //give grail
 | 
	
		
			
				|  |  |  		sendAndApply(&iw);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  		iw.soundID = soundBase::invalid;
 | 
	
	
		
			
				|  | @@ -3756,13 +3756,21 @@ bool CGameHandler::putArtifact(const ArtifactLocation & al, const CArtifactInsta
 | 
	
		
			
				|  |  |  	}
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -bool CGameHandler::giveHeroNewArtifact(const CGHeroInstance * h, const CArtifact * artType, ArtifactPosition pos)
 | 
	
		
			
				|  |  | +bool CGameHandler::giveHeroNewArtifact(
 | 
	
		
			
				|  |  | +	const CGHeroInstance * h, const CArtifact * artType, const SpellID & spellId, const ArtifactPosition & pos)
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  |  	assert(artType);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +	NewArtifact na;
 | 
	
		
			
				|  |  | +	na.artHolder = h->id;
 | 
	
		
			
				|  |  | +	na.artId = artType->getId();
 | 
	
		
			
				|  |  | +	na.spellId = spellId;
 | 
	
		
			
				|  |  | +	na.pos = pos;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |  	if(pos == ArtifactPosition::FIRST_AVAILABLE)
 | 
	
		
			
				|  |  |  	{
 | 
	
		
			
				|  |  | -		if(!artType->canBePutAt(h, ArtifactUtils::getArtAnyPosition(h, artType->getId())))
 | 
	
		
			
				|  |  | +		na.pos = ArtifactUtils::getArtAnyPosition(h, artType->getId());
 | 
	
		
			
				|  |  | +		if(!artType->canBePutAt(h, na.pos))
 | 
	
		
			
				|  |  |  			COMPLAIN_RET("Cannot put artifact in that slot!");
 | 
	
		
			
				|  |  |  	}
 | 
	
		
			
				|  |  |  	else if(ArtifactUtils::isSlotBackpack(pos))
 | 
	
	
		
			
				|  | @@ -3774,18 +3782,18 @@ bool CGameHandler::giveHeroNewArtifact(const CGHeroInstance * h, const CArtifact
 | 
	
		
			
				|  |  |  	{
 | 
	
		
			
				|  |  |  		COMPLAIN_RET_FALSE_IF(!artType->canBePutAt(h, pos, false), "Cannot put artifact in that slot!");
 | 
	
		
			
				|  |  |  	}
 | 
	
		
			
				|  |  | +	sendAndApply(&na);
 | 
	
		
			
				|  |  | +	return true;
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -	auto * newArtInst = new CArtifactInstance();
 | 
	
		
			
				|  |  | -	newArtInst->artType = artType; // *NOT* via settype -> all bonus-related stuff must be done by NewArtifact apply
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -	NewArtifact na;
 | 
	
		
			
				|  |  | -	na.art = newArtInst;
 | 
	
		
			
				|  |  | -	sendAndApply(&na); // -> updates newArtInst!!!
 | 
	
		
			
				|  |  | +bool CGameHandler::giveHeroNewArtifact(const CGHeroInstance * h, const ArtifactID & artId, const ArtifactPosition & pos)
 | 
	
		
			
				|  |  | +{
 | 
	
		
			
				|  |  | +	return giveHeroNewArtifact(h, artId.toArtifact(), SpellID::NONE, pos);
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -	if(putArtifact(ArtifactLocation(h->id, pos), newArtInst, false))
 | 
	
		
			
				|  |  | -		return true;
 | 
	
		
			
				|  |  | -	else
 | 
	
		
			
				|  |  | -		return false;
 | 
	
		
			
				|  |  | +bool CGameHandler::giveHeroNewScroll(const CGHeroInstance * h, const SpellID & spellId, const ArtifactPosition & pos)
 | 
	
		
			
				|  |  | +{
 | 
	
		
			
				|  |  | +	return giveHeroNewArtifact(h, ArtifactID(ArtifactID::SPELL_SCROLL).toArtifact(), spellId, pos);
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  void CGameHandler::spawnWanderingMonsters(CreatureID creatureID)
 |