/* * Copyright (C) 2010 Apple Inc. All rights reserved. * Copyright (C) 2010 Nokia Corporation and/or its subsidiary(-ies) * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS'' * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF * THE POSSIBILITY OF SUCH DAMAGE. */ #include "config.h" #include "RunLoop.h" #include "WorkItem.h" #include <QApplication> #include <QAbstractEventDispatcher> #include <QObject> #include <QMetaMethod> #include <QMetaObject> #include <QTimerEvent> class RunLoop::TimerObject : public QObject { Q_OBJECT public: TimerObject(RunLoop* runLoop) : m_runLoop(runLoop) { int methodIndex = metaObject()->indexOfMethod("performWork()"); m_method = metaObject()->method(methodIndex); } Q_SLOT void performWork() { m_runLoop->performWork(); } inline void wakeUp() { m_method.invoke(this, Qt::QueuedConnection); } protected: virtual void timerEvent(QTimerEvent* event) { RunLoop::TimerBase::timerFired(m_runLoop, event->timerId()); } private: RunLoop* m_runLoop; QMetaMethod m_method; }; void RunLoop::run() { QCoreApplication::exec(); } void RunLoop::stop() { QCoreApplication::exit(); } RunLoop::RunLoop() : m_timerObject(new TimerObject(this)) { } RunLoop::~RunLoop() { delete m_timerObject; } void RunLoop::wakeUp() { m_timerObject->wakeUp(); } // RunLoop::Timer void RunLoop::TimerBase::timerFired(RunLoop* runLoop, int ID) { TimerMap::iterator it = runLoop->m_activeTimers.find(ID); ASSERT(it != runLoop->m_activeTimers.end()); TimerBase* timer = it->second; if (!timer->m_isRepeating) { // Stop the timer (calling stop would need another hash table lookup). runLoop->m_activeTimers.remove(it); runLoop->m_timerObject->killTimer(timer->m_ID); timer->m_ID = 0; } timer->fired(); } RunLoop::TimerBase::TimerBase(RunLoop* runLoop) : m_runLoop(runLoop) , m_ID(0) , m_isRepeating(false) { } RunLoop::TimerBase::~TimerBase() { stop(); } void RunLoop::TimerBase::start(double nextFireInterval, bool repeat) { stop(); int millis = static_cast<int>(nextFireInterval * 1000); m_isRepeating = repeat; m_ID = m_runLoop->m_timerObject->startTimer(millis); ASSERT(m_ID); m_runLoop->m_activeTimers.set(m_ID, this); } void RunLoop::TimerBase::stop() { if (!m_ID) return; TimerMap::iterator it = m_runLoop->m_activeTimers.find(m_ID); if (it == m_runLoop->m_activeTimers.end()) return; m_runLoop->m_activeTimers.remove(it); m_runLoop->m_timerObject->killTimer(m_ID); m_ID = 0; } bool RunLoop::TimerBase::isActive() const { return m_ID; } #include "RunLoopQt.moc"