LUMIERA.clone/src/proc/asset/meta/time-grid.cpp
2012-01-06 03:16:22 +01:00

179 lines
5.6 KiB
C++

/*
TimeGrid - reference scale for quantised time
Copyright (C) Lumiera.org
2010, Hermann Vosseler <Ichthyostega@web.de>
This program is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License as
published by the Free Software Foundation; either version 2 of
the License, or (at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
* *****************************************************/
#include "proc/asset/meta/time-grid.hpp"
#include "proc/asset/entry-id.hpp"
#include "proc/assetmanager.hpp"
#include "lib/time/quantiser.hpp"
#include "lib/time/timevalue.hpp"
#include "lib/time/display.hpp"
#include "lib/advice.hpp"
#include "lib/util.hpp"
//#include "include/logging.h"
#include <boost/format.hpp>
#include <string>
using util::cStr;
using util::isnil;
using boost::format;
using boost::str;
using std::string;
namespace proc {
namespace asset {
namespace meta {
namespace error = lumiera::error;
/** */
TimeGrid::TimeGrid (EntryID<TimeGrid> const& nameID)
: Meta (nameID.getIdent())
{ }
using lib::time::Time;
using lib::time::TimeValue;
using lib::time::TimeSpan;
using lib::time::Duration;
using lib::time::Offset;
using lib::time::FSecs;
using lib::time::PQuant;
using lib::time::Quantiser;
using lib::time::FixedFrameQuantiser;
using std::tr1::dynamic_pointer_cast;
namespace advice = lib::advice;
namespace {
/** @internal helper to retrieve the smart-ptr
* from the AssetManager, then attach a further
* smart-ptr-to-Quantiser to that, which then can be
* published via the \link advice.hpp "advice system"\endlink
* @note this allows to use a time grid just "by name",
* without explicit dependance to the Session / Assets
*/
inline PGrid
publishWrapped (TimeGrid& newGrid)
{
PGrid gridImplementation = AssetManager::instance().wrap (newGrid);
PQuant quantiser (dynamic_pointer_cast<const Quantiser>(gridImplementation));
Literal bindingID (cStr(newGrid.ident.name));
advice::Provision<PQuant>(bindingID).setAdvice(quantiser);
return gridImplementation;
}
}
/**
* TimeGrid implementation: a trivial time grid,
* starting at a given point in time and using a
* constant grid spacing.
*
* @note The actual implementation is mixed in,
* together with the Quantiser API; the intended use
* of this implementation is to publish it via the advice
* framework, when building and registering the meta asset.
*/
class SimpleTimeGrid
: public TimeGrid
, public FixedFrameQuantiser
{
public:
SimpleTimeGrid (Time start, Duration frameDuration, EntryID<TimeGrid> const& name)
: TimeGrid (name)
, FixedFrameQuantiser(frameDuration,start)
{ }
SimpleTimeGrid (Time start, FrameRate frames_per_second, EntryID<TimeGrid> const& name)
: TimeGrid (name)
, FixedFrameQuantiser(frames_per_second,start)
{ }
};
/** Setup of a TimeGrid: validate the settings configured into this builder instance,
* then decide on the implementation strategy for the time grid. Convert the given
* frames per second into an appropriate gridSpacing time and build a suitable
* name-ID to denote the TimeGrid-meta-Asset to be built.
* @return shared_ptr holding onto the new asset::Meta, which has already been
* registered with the AssetManager.
* @throw error::Config in case of invalid frames-per-second. The AssetManager
* might raise further exception when asset registration fails.
* @note the newly created grid is automatically published through the Advice System.
* This allows client code to pick up that grid definition just by using the
* Grid ID, without requiring an explicit link to the session or Asset subsystem.
* @todo currently (12/2010) the AssetManager is unable to detect duplicate assets.
* Later on the intention is that in such cases, instead of creating a new grid
* we'll silently return the already registered existing and equivalent grid.
*/
P<TimeGrid>
Builder<TimeGrid>::commit()
{
if (predecessor_)
throw error::Invalid("compound and variable time grids are a planned feature"
, error::LUMIERA_ERROR_UNIMPLEMENTED);
ENSURE (fps_, "infinite grid should have been detected by FrameRate ctor");
if (isnil (id_))
{
format gridIdFormat("grid(%f_%d)");
id_ = str(gridIdFormat % fps_ % _raw(origin_));
}
EntryID<TimeGrid> nameID (id_);
return publishWrapped (*new SimpleTimeGrid(origin_, fps_, nameID));
}
/* === TimeGrid shortcut builder functions === */
PGrid
TimeGrid::build (Symbol gridID, FrameRate frames_per_second)
{
return build (gridID,frames_per_second, Time(0,0));
}
PGrid
TimeGrid::build (Symbol gridID, FrameRate frames_per_second, Time origin)
{
string name(gridID);
Builder<TimeGrid> spec(name);
spec.fps_ = frames_per_second;
spec.origin_ = origin;
return spec.commit();
}
}}} // namespace asset::meta