// vim: set ts=4 sw=4 tw=99 noet: // // AMX Mod X, based on AMX Mod by Aleksander Naszko ("OLO"). // Copyright (C) The AMX Mod X Development Team. // // This software is licensed under the GNU General Public License, version 3 or higher. // Additional exceptions apply. For full license details, see LICENSE.txt or visit: // https://alliedmods.net/amxmodx-license #include "amxmodx.h" #include "CTask.h" /*********************** CTask ***********************/ void CTaskMngr::CTask::set(CPluginMngr::CPlugin *pPlugin, int iFunc, int iFlags, cell iId, float fBase, int iParamsLen, const cell *pParams, int iRepeat, float fCurrentTime) { clear(); m_bFree = false; m_pPlugin = pPlugin; m_iFunc = iFunc; m_iId = iId; m_fBase = fBase; m_bInExecute = false; if (iFlags & 2) { m_bLoop = true; m_iRepeat = -1; } else if (iFlags & 1) { m_bLoop = true; m_iRepeat = iRepeat; } m_bAfterStart = (iFlags & 4) ? true : false; m_bBeforeEnd = (iFlags & 8) ? true : false; m_fNextExecTime = fCurrentTime + m_fBase; if (iParamsLen) { m_iParamLen = iParamsLen + 1; m_pParams = new cell[m_iParamLen]; memcpy(m_pParams, pParams, sizeof(cell)*iParamsLen); m_pParams[iParamsLen] = 0; } else { m_iParamLen = 0; m_pParams = NULL; } } void CTaskMngr::CTask::clear() { m_bFree = true; if (m_iFunc >= 0) { unregisterSPForward(m_iFunc); m_iFunc = -1; } if (m_pParams) { delete [] m_pParams; m_pParams = NULL; } m_pPlugin = NULL; m_iId = 0; m_fBase = 0.0f; m_iRepeat = 0; m_bLoop = false; m_bAfterStart = false; m_bBeforeEnd = false; m_fNextExecTime = 0.0f; } bool CTaskMngr::CTask::isFree() const { return m_bFree; } void CTaskMngr::CTask::changeBase(float fNewBase) { m_fBase = fNewBase; } void CTaskMngr::CTask::resetNextExecTime(float fCurrentTime) { // If we're here while we're executing we would add m_fBase twice if (!m_bInExecute) m_fNextExecTime = fCurrentTime + m_fBase; } void CTaskMngr::CTask::executeIfRequired(float fCurrentTime, float fTimeLimit, float fTimeLeft) { bool execute = false; bool done = false; if (m_bAfterStart) { if (fCurrentTime - fTimeLeft + 1.0f >= m_fBase) execute = true; } else if (m_bBeforeEnd) { if (fTimeLimit != 0.0f && (fTimeLeft + fTimeLimit * 60.0f) - fCurrentTime - 1.0f <= m_fBase) execute = true; } else if (m_fNextExecTime <= fCurrentTime) { execute = true; } if (execute) { //only bother calling if we have something to call if (!(m_bLoop && !m_iRepeat)) { m_bInExecute = true; if (m_iParamLen) // call with parameters { cell arr = prepareCellArray(m_pParams, m_iParamLen); executeForwards(m_iFunc, arr, m_iId); } else { executeForwards(m_iFunc, m_iId); } m_bInExecute = false; } if (isFree()) return; // set new exec time OR remove the task if needed if (m_bLoop) { if (m_iRepeat != -1 && --m_iRepeat <= 0) done = true; } else { done = true; } if (done) { clear(); } else { m_fNextExecTime += m_fBase; } } } CTaskMngr::CTask::CTask() { m_bFree = true; m_pPlugin = NULL; m_iFunc = -1; m_iId = 0; m_fBase = 0.0f; m_iRepeat = 0; m_bLoop = false; m_bAfterStart = false; m_bBeforeEnd = false; m_bInExecute = false; m_fNextExecTime = 0.0f; m_iParamLen = 0; m_pParams = NULL; } CTaskMngr::CTask::~CTask() { clear(); } /*********************** CTaskMngr ***********************/ CTaskMngr::CTaskMngr() { m_pTmr_CurrentTime = NULL; m_pTmr_TimeLimit = NULL; m_pTmr_TimeLeft = NULL; } CTaskMngr::~CTaskMngr() { clear(); } void CTaskMngr::registerTimers(float *pCurrentTime, float *pTimeLimit, float *pTimeLeft) { m_pTmr_CurrentTime = pCurrentTime; m_pTmr_TimeLimit = pTimeLimit; m_pTmr_TimeLeft = pTimeLeft; } void CTaskMngr::registerTask(CPluginMngr::CPlugin *pPlugin, int iFunc, int iFlags, cell iId, float fBase, int iParamsLen, const cell *pParams, int iRepeat) { // first, search for free tasks for (auto &task : m_Tasks) { if (task->isFree() && !task->inExecute()) { // found: reuse it task->set(pPlugin, iFunc, iFlags, iId, fBase, iParamsLen, pParams, iRepeat, *m_pTmr_CurrentTime); return; } } // not found: make a new one auto task = ke::AutoPtr(new CTask); if (!task) return; task->set(pPlugin, iFunc, iFlags, iId, fBase, iParamsLen, pParams, iRepeat, *m_pTmr_CurrentTime); m_Tasks.append(ke::Move(task)); } int CTaskMngr::removeTasks(int iId, AMX *pAmx) { int i = 0; for (auto &task : m_Tasks) { if (task->match(iId, pAmx)) { task->clear(); ++i; } } return i; } int CTaskMngr::changeTasks(int iId, AMX *pAmx, float fNewBase) { int i = 0; for (auto &task : m_Tasks) { if (task->match(iId, pAmx)) { task->changeBase(fNewBase); task->resetNextExecTime(*m_pTmr_CurrentTime); ++i; } } return i; } bool CTaskMngr::taskExists(int iId, AMX *pAmx) { for (auto &task : m_Tasks) { if (task->match(iId, pAmx)) { return true; } } return false; } void CTaskMngr::startFrame() { auto lastSize = m_Tasks.length(); for(auto i = 0u; i < lastSize; i++) { auto &task = m_Tasks[i]; if (task->isFree()) continue; task->executeIfRequired(*m_pTmr_CurrentTime, *m_pTmr_TimeLimit, *m_pTmr_TimeLeft); } } void CTaskMngr::clear() { m_Tasks.clear(); }