show only unlocked Questsgroups and Quests and summarize XPs recursive regarding choosed path by character

This commit is contained in:
coderkun 2014-03-30 04:07:23 +02:00
commit 40c1ecca50
10 changed files with 317 additions and 176 deletions

View file

@ -256,11 +256,34 @@
$questgroup = $this->getQuestgroupById($questgroupId);
// Chack all Quests
$quests = $this->Quests->getQuestsForQuestgroup($questgroup['id']);
foreach($quests as &$quest) {
if(!$this->Quests->hasCharacterSolvedQuest($quest['id'], $characterId)) {
$currentQuest = $this->Quests->getFirstQuestOfQuestgroup($questgroup['id']);
if(!is_null($currentQuest))
{
if(!$this->Quests->hasCharacterSolvedQuest($currentQuest['id'], $characterId)) {
return false;
}
// Get next Quests
while(!is_null($currentQuest) && !empty($nextQuests = $this->Quests->getNextQuests($currentQuest['id'])))
{
// Get choosed Quest
$currentQuest = null;
foreach($nextQuests as &$nextQuest) {
if($this->Quests->hasCharacterEnteredQuest($nextQuest['id'], $characterId)) {
$currentQuest = $nextQuest;
}
}
// Check Quest
if(is_null($currentQuest)) {
return false;
}
// Check status
if(!$this->Quests->hasCharacterSolvedQuest($currentQuest['id'], $characterId)) {
return false;
}
}
}
// Check all child Questgroups
@ -328,7 +351,7 @@
* sub-Questgroups.
*
* @param int $questgroupId ID of Questgroup
* @return Sum of XPs
* @return int Sum of XPs
*/
public function getAchievableXPsForQuestgroup($questgroupId)
{
@ -340,16 +363,9 @@
$questgroup['hierarchy'] = $this->Questgroupshierarchy->getHierarchyForQuestgroup($questgroup['id']);
// Quests of current Questgroup
$quests = $this->Quests->getQuestsForQuestgroup($questgroup['id']);
foreach($quests as &$quest)
{
$xps += $quest['xps'];
// Related Questgroups
$relatedQuestgroups = $this->getRelatedQuestsgroupsOfQuest($quest['id']);
foreach($relatedQuestgroups as $group) {
$xps += $this->getAchievableXPsForQuestgroup($group['id']);
}
$quest = $this->Quests->getFirstQuestOfQuestgroup($questgroup['id']);
if(!is_null($quest)) {
$xps += $this->getAchievableXPsForQuest($quest);
}
// XPs of child Questgroups
@ -372,13 +388,45 @@
}
/**
* Summarize XPs of the given Quest, its following Quests and
* its related Questgroups.
*
* @param array $quest Quest to summarize XPs for
* @return int Sum of XPs
*/
public function getAchievableXPsForQuest($quest)
{
// XPs for the given Quest
$xps = $quest['xps'];
// Related Questgroups
$relatedQuestgroups = $this->getRelatedQuestsgroupsOfQuest($quest['id']);
foreach($relatedQuestgroups as &$relatedQuestgroup) {
$xps += $this->getAchievableXPsForQuestgroup($relatedQuestgroup['id']);
}
// Next Quests
$nextQuests = $this->Quests->getNextQuests($quest['id']);
$nextXPs = array(0);
foreach($nextQuests as &$nextQuest)
{
$nextXPs[] = $this->getAchievableXPsForQuest($nextQuest);
}
$xps += max($nextXPs);
return $xps;
}
/**
* Summarize XPs of all Quests for a Questgroup and its
* sub-Questgroups solved by a Character.
*
* @param int $questgroupId ID of Questgroup
* @param int $characterId ID of Character
* @return Sum of XPs
* @return int Sum of XPs
*/
public function getAchievedXPsForQuestgroup($questgroupId, $characterId)
{
@ -389,18 +437,9 @@
$questgroup = $this->getQuestgroupById($questgroupId);
// Quests of current Questgroup
$quests = $this->Quests->getQuestsForQuestgroup($questgroup['id']);
foreach($quests as &$quest)
{
if($this->Quests->hasCharacterSolvedQuest($quest['id'], $characterId)) {
$xps += $quest['xps'];
}
// Related Questgroups
$relatedQuestgroups = $this->getRelatedQuestsgroupsOfQuest($quest['id']);
foreach($relatedQuestgroups as $group) {
$xps += $this->getAchievedXPsForQuestgroup($group['id'], $characterId);
}
$quest = $this->Quests->getFirstQuestOfQuestgroup($questgroup['id']);
if(!is_null($quest)) {
$xps += $this->getAchievedXPsForQuest($quest, $characterId);
}
// XPs of child Questgroups
@ -423,6 +462,47 @@
}
/**
* Summarize XPs of the given Quest, its following Quests and
* its related Questgroups solved by a Character.
*
* @param int $quest Quest to summarize XPs for
* @param int $characterId ID of Character
* @return int Sum of XPs
*/
public function getAchievedXPsForQuest($quest, $characterId)
{
$xps = 0;
// XPs for the given Quest
if($this->Quests->hasCharacterSolvedQuest($quest['id'], $characterId))
{
$xps += $quest['xps'];
// Next Quests
$nextQuests = $this->Quests->getNextQuests($quest['id']);
foreach($nextQuests as &$nextQuest)
{
if($this->Quests->hasCharacterEnteredQuest($nextQuest['id'], $characterId))
{
$xps += $this->getAchievedXPsForQuest($nextQuest, $characterId);
break;
}
}
}
// Related Questgroups
$relatedQuestgroups = $this->getRelatedQuestsgroupsOfQuest($quest['id']);
foreach($relatedQuestgroups as &$relatedQuestgroup) {
$xps += $this->getAchievedXPsForQuestgroup($relatedQuestgroup['id'], $characterId);
}
// Return summarized XPs
return $xps;
}
/**

View file

@ -19,6 +19,24 @@
*/
class QuestsModel extends \hhu\z\Model
{
/**
* Quest-status: Entered
*
* @var int;
*/
const QUEST_STATUS_ENTERED = 1;
/**
* Quest-status: Solved
*
* @var int;
*/
const QUEST_STATUS_SOLVED = 0;
/**
* Quest-status: Unsolved
*
* @var int;
*/
const QUEST_STATUS_UNSOLVED = -1;
@ -34,45 +52,6 @@
/**
* Get all Quests for the given Questgroup.
*
* @param int $questgroupId ID of a Questgroup
* @return array Quests of the given Questgroup
*/
public function getQuestsForQuestgroup($questgroupId)
{
$quests = array();
// Get first Quest
$quest = $this->db->query(
'SELECT id, questtype_id, title, url, xps, task '.
'FROM quests '.
'LEFT JOIN quests_previousquests ON quests_previousquests.quest_id = quests.id '.
'WHERE questgroup_id = ? AND quests_previousquests.previous_quest_id IS NULL',
'i',
$questgroupId
);
if(empty($quest)) {
return $quests;
}
$quest = $quest[0];
array_push($quests, $quest);
// Get next Quests
// TODO Multiple next Quests
while(($nextQuests = $this->getNextQuests($quest['id'])) != null)
{
$quest = $nextQuests[0];
array_push($quests, $quest);
}
// Return Quests
return $quests;
}
/**
* Get a Quest and its data by its URL.
*
@ -127,6 +106,31 @@
}
/**
* Get the first Quest of a Qusetgroup.
*
* @param int $questId ID of Questgroup
* @return array Data of first Quest
*/
public function getFirstQuestOfQuestgroup($questgroupId)
{
$data = $this->db->query(
'SELECT id, questtype_id, title, url, xps, task '.
'FROM quests '.
'LEFT JOIN quests_previousquests ON quests_previousquests.quest_id = quests.id '.
'WHERE questgroup_id = ? AND quests_previousquests.previous_quest_id IS NULL',
'i',
$questgroupId
);
if(!empty($data)) {
return $data[0];
}
return null;
}
/**
* Get Quests that follow-up a Quest.
*
@ -167,6 +171,18 @@
}
/**
* Mark a Quest as entered for a Character.
*
* @param int $questId ID of Quest to mark as entered
* @param int $characterId ID of Character that entered the Quest
*/
public function setQuestEntered($questId, $characterId)
{
$this->setQuestStatus($questId, $characterId, static::QUEST_STATUS_ENTERED);
}
/**
* Mark a Quest as solved for a Character.
*
@ -175,16 +191,7 @@
*/
public function setQuestSolved($questId, $characterId)
{
$this->db->query(
'INSERT INTO quests_characters '.
'(quest_id, character_id, status) '.
'VALUES '.
'(?, ?, ?)',
'iii',
$questId,
$characterId,
0
);
$this->setQuestStatus($questId, $characterId, static::QUEST_STATUS_SOLVED);
}
@ -196,16 +203,31 @@
*/
public function setQuestUnsolved($questId, $characterId)
{
$this->db->query(
'INSERT INTO quests_characters '.
'(quest_id, character_id, status) '.
'VALUES '.
'(?, ?, ?)',
'iii',
$this->setQuestStatus($questId, $characterId, static::QUEST_STATUS_UNSOLVED);
}
/**
* Determine if the given Character has entered a Quest.
*
* @param int $questId ID of Quest to check
* @param int $characterId ID of Character to check
* @result boolean Whether Character has entered the Quest or not
*/
public function hasCharacterEnteredQuest($questId, $characterId)
{
$count = $this->db->query(
'SELECT count(id) AS c '.
'FROM quests_characters '.
'WHERE quest_id = ? AND character_id = ? AND status IN (?,?,?)',
'iiiii',
$questId,
$characterId,
-1
static::QUEST_STATUS_ENTERED, static::QUEST_STATUS_SOLVED, static::QUEST_STATUS_UNSOLVED
);
return (!empty($count) && intval($count[0]['c']) > 0);
}
@ -221,10 +243,11 @@
$count = $this->db->query(
'SELECT count(id) AS c '.
'FROM quests_characters '.
'WHERE quest_id = ? AND character_id = ? AND status IN (-1,0)',
'ii',
'WHERE quest_id = ? AND character_id = ? AND status IN (?,?)',
'iiii',
$questId,
$characterId
$characterId,
static::QUEST_STATUS_SOLVED, static::QUEST_STATUS_UNSOLVED
);
@ -244,10 +267,11 @@
$count = $this->db->query(
'SELECT count(id) AS c '.
'FROM quests_characters '.
'WHERE quest_id = ? AND character_id = ? AND status = 0',
'ii',
'WHERE quest_id = ? AND character_id = ? AND status = ?',
'iii',
$questId,
$characterId
$characterId,
static::QUEST_STATUS_SOLVED
);
@ -267,10 +291,11 @@
'SELECT character_id, created, text '.
'FROM questtypes_submit_characters '.
'WHERE quest_id = ? AND EXISTS ('.
'SELECT character_id FROM quests_characters WHERE quest_id = questtypes_submit_characters.quest_id AND status = 0'.
'SELECT character_id FROM quests_characters WHERE quest_id = questtypes_submit_characters.quest_id AND status = ?'.
')',
'i',
$questId
'ii',
$questId,
static::QUEST_STATUS_SOLVED
);
}
@ -287,10 +312,35 @@
'SELECT character_id, created, text '.
'FROM questtypes_submit_characters '.
'WHERE quest_id = ? AND NOT EXISTS ('.
'SELECT character_id FROM quests_characters WHERE quest_id = questtypes_submit_characters.quest_id AND status = 0'.
'SELECT character_id FROM quests_characters WHERE quest_id = questtypes_submit_characters.quest_id AND status = ?'.
')',
'i',
$questId
'ii',
$questId,
static::QUEST_STATUS_SOLVED
);
}
/**
* Mark a Quest for a Character.
*
* @param int $questId ID of Quest to mark
* @param int $characterId ID of Character to mark the Quest for
* @param int $status Quest status to mark
*/
private function setQuestStatus($questId, $characterId, $status)
{
$this->db->query(
'INSERT INTO quests_characters '.
'(quest_id, character_id, status) '.
'VALUES '.
'(?, ?, ?) ',
'iii',
$questId,
$characterId,
$status
);
}