2011-04-11 23:18:43 -07:00
|
|
|
/* vim: set shiftwidth=2 tabstop=8 autoindent cindent expandtab: */
|
2012-05-21 04:12:37 -07:00
|
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
2011-04-11 23:18:43 -07:00
|
|
|
|
2012-08-25 18:27:28 -07:00
|
|
|
#include "gfxPlatform.h"
|
2011-04-11 23:18:43 -07:00
|
|
|
#include "AnimationCommon.h"
|
|
|
|
#include "nsRuleData.h"
|
2012-12-11 13:12:43 -08:00
|
|
|
#include "nsCSSFrameConstructor.h"
|
2011-04-11 23:18:43 -07:00
|
|
|
#include "nsCSSValue.h"
|
|
|
|
#include "nsStyleContext.h"
|
2012-07-31 10:28:21 -07:00
|
|
|
#include "nsIFrame.h"
|
2012-07-31 10:28:22 -07:00
|
|
|
#include "nsLayoutUtils.h"
|
2012-12-11 13:12:43 -08:00
|
|
|
#include "mozilla/LookAndFeel.h"
|
|
|
|
#include "Layers.h"
|
|
|
|
#include "FrameLayerBuilder.h"
|
|
|
|
#include "nsDisplayList.h"
|
|
|
|
#include "mozilla/Preferences.h"
|
|
|
|
|
|
|
|
using namespace mozilla::layers;
|
2011-04-11 23:18:43 -07:00
|
|
|
|
|
|
|
namespace mozilla {
|
|
|
|
namespace css {
|
|
|
|
|
2012-08-21 18:48:47 -07:00
|
|
|
/* static */ bool
|
|
|
|
IsGeometricProperty(nsCSSProperty aProperty)
|
|
|
|
{
|
|
|
|
switch (aProperty) {
|
|
|
|
case eCSSProperty_bottom:
|
|
|
|
case eCSSProperty_height:
|
|
|
|
case eCSSProperty_left:
|
|
|
|
case eCSSProperty_right:
|
|
|
|
case eCSSProperty_top:
|
|
|
|
case eCSSProperty_width:
|
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-04-11 23:18:43 -07:00
|
|
|
CommonAnimationManager::CommonAnimationManager(nsPresContext *aPresContext)
|
|
|
|
: mPresContext(aPresContext)
|
|
|
|
{
|
|
|
|
PR_INIT_CLIST(&mElementData);
|
|
|
|
}
|
|
|
|
|
|
|
|
CommonAnimationManager::~CommonAnimationManager()
|
|
|
|
{
|
|
|
|
NS_ABORT_IF_FALSE(!mPresContext, "Disconnect should have been called");
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CommonAnimationManager::Disconnect()
|
|
|
|
{
|
|
|
|
// Content nodes might outlive the transition or animation manager.
|
|
|
|
RemoveAllElementData();
|
|
|
|
|
2012-07-30 07:20:58 -07:00
|
|
|
mPresContext = nullptr;
|
2011-04-11 23:18:43 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CommonAnimationManager::AddElementData(CommonElementAnimationData* aData)
|
|
|
|
{
|
|
|
|
if (PR_CLIST_IS_EMPTY(&mElementData)) {
|
|
|
|
// We need to observe the refresh driver.
|
|
|
|
nsRefreshDriver *rd = mPresContext->RefreshDriver();
|
|
|
|
rd->AddRefreshObserver(this, Flush_Style);
|
|
|
|
}
|
2012-07-31 10:28:22 -07:00
|
|
|
|
2011-04-11 23:18:43 -07:00
|
|
|
PR_INSERT_BEFORE(aData, &mElementData);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CommonAnimationManager::ElementDataRemoved()
|
|
|
|
{
|
|
|
|
// If we have no transitions or animations left, remove ourselves from
|
|
|
|
// the refresh driver.
|
|
|
|
if (PR_CLIST_IS_EMPTY(&mElementData)) {
|
|
|
|
mPresContext->RefreshDriver()->RemoveRefreshObserver(this, Flush_Style);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CommonAnimationManager::RemoveAllElementData()
|
|
|
|
{
|
|
|
|
while (!PR_CLIST_IS_EMPTY(&mElementData)) {
|
|
|
|
CommonElementAnimationData *head =
|
|
|
|
static_cast<CommonElementAnimationData*>(PR_LIST_HEAD(&mElementData));
|
|
|
|
head->Destroy();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* nsISupports implementation
|
|
|
|
*/
|
|
|
|
|
|
|
|
NS_IMPL_ISUPPORTS1(CommonAnimationManager, nsIStyleRuleProcessor)
|
|
|
|
|
|
|
|
nsRestyleHint
|
|
|
|
CommonAnimationManager::HasStateDependentStyle(StateRuleProcessorData* aData)
|
|
|
|
{
|
|
|
|
return nsRestyleHint(0);
|
|
|
|
}
|
|
|
|
|
2011-09-28 23:19:26 -07:00
|
|
|
bool
|
2011-04-11 23:18:43 -07:00
|
|
|
CommonAnimationManager::HasDocumentStateDependentStyle(StateRuleProcessorData* aData)
|
|
|
|
{
|
2011-10-17 07:59:28 -07:00
|
|
|
return false;
|
2011-04-11 23:18:43 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
nsRestyleHint
|
|
|
|
CommonAnimationManager::HasAttributeDependentStyle(AttributeRuleProcessorData* aData)
|
|
|
|
{
|
|
|
|
return nsRestyleHint(0);
|
|
|
|
}
|
|
|
|
|
2011-09-28 23:19:26 -07:00
|
|
|
/* virtual */ bool
|
2011-04-11 23:18:43 -07:00
|
|
|
CommonAnimationManager::MediumFeaturesChanged(nsPresContext* aPresContext)
|
|
|
|
{
|
2011-10-17 07:59:28 -07:00
|
|
|
return false;
|
2011-04-11 23:18:43 -07:00
|
|
|
}
|
|
|
|
|
2011-12-08 21:01:52 -08:00
|
|
|
/* virtual */ size_t
|
|
|
|
CommonAnimationManager::SizeOfExcludingThis(nsMallocSizeOfFun aMallocSizeOf) const
|
2011-08-01 11:25:20 -07:00
|
|
|
{
|
2012-01-02 18:19:14 -08:00
|
|
|
// Measurement of the following members may be added later if DMD finds it is
|
|
|
|
// worthwhile:
|
|
|
|
// - mElementData
|
|
|
|
//
|
|
|
|
// The following members are not measured
|
|
|
|
// - mPresContext, because it's non-owning
|
|
|
|
|
2011-12-08 21:01:52 -08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* virtual */ size_t
|
|
|
|
CommonAnimationManager::SizeOfIncludingThis(nsMallocSizeOfFun aMallocSizeOf) const
|
|
|
|
{
|
2012-01-25 00:52:51 -08:00
|
|
|
return aMallocSizeOf(this) + SizeOfExcludingThis(aMallocSizeOf);
|
2011-08-01 11:25:20 -07:00
|
|
|
}
|
|
|
|
|
2011-09-28 23:19:26 -07:00
|
|
|
/* static */ bool
|
2011-04-11 23:18:43 -07:00
|
|
|
CommonAnimationManager::ExtractComputedValueForTransition(
|
|
|
|
nsCSSProperty aProperty,
|
|
|
|
nsStyleContext* aStyleContext,
|
|
|
|
nsStyleAnimation::Value& aComputedValue)
|
|
|
|
{
|
2011-09-28 23:19:26 -07:00
|
|
|
bool result =
|
2011-04-11 23:18:43 -07:00
|
|
|
nsStyleAnimation::ExtractComputedValue(aProperty, aStyleContext,
|
|
|
|
aComputedValue);
|
|
|
|
if (aProperty == eCSSProperty_visibility) {
|
|
|
|
NS_ABORT_IF_FALSE(aComputedValue.GetUnit() ==
|
|
|
|
nsStyleAnimation::eUnit_Enumerated,
|
|
|
|
"unexpected unit");
|
|
|
|
aComputedValue.SetIntValue(aComputedValue.GetIntValue(),
|
|
|
|
nsStyleAnimation::eUnit_Visibility);
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMPL_ISUPPORTS1(AnimValuesStyleRule, nsIStyleRule)
|
|
|
|
|
|
|
|
/* virtual */ void
|
|
|
|
AnimValuesStyleRule::MapRuleInfoInto(nsRuleData* aRuleData)
|
|
|
|
{
|
|
|
|
nsStyleContext *contextParent = aRuleData->mStyleContext->GetParent();
|
|
|
|
if (contextParent && contextParent->HasPseudoElementData()) {
|
|
|
|
// Don't apply transitions or animations to things inside of
|
|
|
|
// pseudo-elements.
|
|
|
|
// FIXME (Bug 522599): Add tests for this.
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-08-22 08:56:38 -07:00
|
|
|
for (uint32_t i = 0, i_end = mPropertyValuePairs.Length(); i < i_end; ++i) {
|
2011-04-11 23:18:43 -07:00
|
|
|
PropertyValuePair &cv = mPropertyValuePairs[i];
|
|
|
|
if (aRuleData->mSIDs & nsCachedStyleData::GetBitForSID(
|
|
|
|
nsCSSProps::kSIDTable[cv.mProperty]))
|
|
|
|
{
|
|
|
|
nsCSSValue *prop = aRuleData->ValueFor(cv.mProperty);
|
|
|
|
if (prop->GetUnit() == eCSSUnit_Null) {
|
|
|
|
#ifdef DEBUG
|
2011-09-28 23:19:26 -07:00
|
|
|
bool ok =
|
2011-04-11 23:18:43 -07:00
|
|
|
#endif
|
2012-11-20 11:55:14 -08:00
|
|
|
nsStyleAnimation::UncomputeValue(cv.mProperty, cv.mValue, *prop);
|
2011-04-11 23:18:43 -07:00
|
|
|
NS_ABORT_IF_FALSE(ok, "could not store computed value");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
/* virtual */ void
|
2012-08-22 08:56:38 -07:00
|
|
|
AnimValuesStyleRule::List(FILE* out, int32_t aIndent) const
|
2011-04-11 23:18:43 -07:00
|
|
|
{
|
2012-11-20 11:55:14 -08:00
|
|
|
for (int32_t index = aIndent; --index >= 0; ) fputs(" ", out);
|
2012-11-20 13:22:35 -08:00
|
|
|
fputs("[anim values] { ", out);
|
2012-11-20 11:55:14 -08:00
|
|
|
for (uint32_t i = 0, i_end = mPropertyValuePairs.Length(); i < i_end; ++i) {
|
|
|
|
const PropertyValuePair &pair = mPropertyValuePairs[i];
|
|
|
|
nsAutoString value;
|
|
|
|
nsStyleAnimation::UncomputeValue(pair.mProperty, pair.mValue, value);
|
2012-11-20 13:22:35 -08:00
|
|
|
fprintf(out, "%s: %s; ", nsCSSProps::GetStringValue(pair.mProperty).get(),
|
|
|
|
NS_ConvertUTF16toUTF8(value).get());
|
2012-11-20 11:55:14 -08:00
|
|
|
}
|
2012-11-20 13:22:35 -08:00
|
|
|
fputs("}\n", out);
|
2011-04-11 23:18:43 -07:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
void
|
|
|
|
ComputedTimingFunction::Init(const nsTimingFunction &aFunction)
|
|
|
|
{
|
|
|
|
mType = aFunction.mType;
|
|
|
|
if (mType == nsTimingFunction::Function) {
|
|
|
|
mTimingFunction.Init(aFunction.mFunc.mX1, aFunction.mFunc.mY1,
|
|
|
|
aFunction.mFunc.mX2, aFunction.mFunc.mY2);
|
|
|
|
} else {
|
|
|
|
mSteps = aFunction.mSteps;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline double
|
2012-08-22 08:56:38 -07:00
|
|
|
StepEnd(uint32_t aSteps, double aPortion)
|
2011-04-11 23:18:43 -07:00
|
|
|
{
|
|
|
|
NS_ABORT_IF_FALSE(0.0 <= aPortion && aPortion <= 1.0, "out of range");
|
2012-08-22 08:56:38 -07:00
|
|
|
uint32_t step = uint32_t(aPortion * aSteps); // floor
|
2011-04-11 23:18:43 -07:00
|
|
|
return double(step) / double(aSteps);
|
|
|
|
}
|
|
|
|
|
|
|
|
double
|
|
|
|
ComputedTimingFunction::GetValue(double aPortion) const
|
|
|
|
{
|
|
|
|
switch (mType) {
|
|
|
|
case nsTimingFunction::Function:
|
|
|
|
return mTimingFunction.GetSplineValue(aPortion);
|
|
|
|
case nsTimingFunction::StepStart:
|
|
|
|
// There are diagrams in the spec that seem to suggest this check
|
|
|
|
// and the bounds point should not be symmetric with StepEnd, but
|
|
|
|
// should actually step up at rather than immediately after the
|
|
|
|
// fraction points. However, we rely on rounding negative values
|
|
|
|
// up to zero, so we can't do that. And it's not clear the spec
|
|
|
|
// really meant it.
|
|
|
|
return 1.0 - StepEnd(mSteps, 1.0 - aPortion);
|
|
|
|
default:
|
2011-10-17 07:59:28 -07:00
|
|
|
NS_ABORT_IF_FALSE(false, "bad type");
|
2011-04-11 23:18:43 -07:00
|
|
|
// fall through
|
|
|
|
case nsTimingFunction::StepEnd:
|
|
|
|
return StepEnd(mSteps, aPortion);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-07-31 10:28:21 -07:00
|
|
|
bool
|
|
|
|
CommonElementAnimationData::CanAnimatePropertyOnCompositor(const dom::Element *aElement,
|
2012-08-21 18:48:47 -07:00
|
|
|
nsCSSProperty aProperty,
|
2012-12-11 13:12:43 -08:00
|
|
|
CanAnimateFlags aFlags)
|
2012-07-31 10:28:21 -07:00
|
|
|
{
|
2012-08-06 13:33:23 -07:00
|
|
|
bool shouldLog = nsLayoutUtils::IsAnimationLoggingEnabled();
|
2012-08-25 18:27:28 -07:00
|
|
|
if (shouldLog && !gfxPlatform::OffMainThreadCompositingEnabled()) {
|
2012-08-25 18:27:28 -07:00
|
|
|
nsCString message;
|
|
|
|
message.AppendLiteral("Performance warning: Compositor disabled");
|
|
|
|
LogAsyncAnimationFailure(message);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-07-31 10:28:21 -07:00
|
|
|
nsIFrame* frame = aElement->GetPrimaryFrame();
|
2012-08-21 18:48:47 -07:00
|
|
|
if (IsGeometricProperty(aProperty)) {
|
|
|
|
if (shouldLog) {
|
2012-08-25 18:27:28 -07:00
|
|
|
nsCString message;
|
|
|
|
message.AppendLiteral("Performance warning: Async animation of geometric property '");
|
2012-12-11 13:12:47 -08:00
|
|
|
message.Append(nsCSSProps::GetStringValue(aProperty));
|
|
|
|
message.AppendLiteral("' is disabled");
|
2012-08-25 18:27:28 -07:00
|
|
|
LogAsyncAnimationFailure(message, aElement);
|
2012-08-21 18:48:47 -07:00
|
|
|
}
|
|
|
|
return false;
|
2012-08-05 10:03:43 -07:00
|
|
|
}
|
2012-08-02 23:32:13 -07:00
|
|
|
if (aProperty == eCSSProperty_transform) {
|
2012-08-21 18:48:47 -07:00
|
|
|
if (frame->Preserves3D() &&
|
2012-08-02 23:32:13 -07:00
|
|
|
frame->Preserves3DChildren()) {
|
2012-08-06 13:33:23 -07:00
|
|
|
if (shouldLog) {
|
2012-08-25 18:27:28 -07:00
|
|
|
nsCString message;
|
|
|
|
message.AppendLiteral("Gecko bug: Async animation of 'preserve-3d' transforms is not supported. See bug 779598");
|
|
|
|
LogAsyncAnimationFailure(message, aElement);
|
2012-08-02 23:32:13 -07:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2012-08-21 18:48:47 -07:00
|
|
|
if (frame->IsSVGTransformed()) {
|
2012-08-06 13:33:23 -07:00
|
|
|
if (shouldLog) {
|
2012-08-25 18:27:28 -07:00
|
|
|
nsCString message;
|
|
|
|
message.AppendLiteral("Gecko bug: Async 'transform' animations of frames with SVG transforms is not supported. See bug 779599");
|
|
|
|
LogAsyncAnimationFailure(message, aElement);
|
2012-08-02 23:32:13 -07:00
|
|
|
}
|
2012-07-31 10:28:21 -07:00
|
|
|
return false;
|
|
|
|
}
|
2012-12-11 13:12:43 -08:00
|
|
|
if (aFlags & CanAnimate_HasGeometricProperty) {
|
2012-08-21 18:48:47 -07:00
|
|
|
if (shouldLog) {
|
2012-08-25 18:27:28 -07:00
|
|
|
nsCString message;
|
|
|
|
message.AppendLiteral("Performance warning: Async animation of 'transform' not possible due to presence of geometric properties");
|
|
|
|
LogAsyncAnimationFailure(message, aElement);
|
2012-08-21 18:48:47 -07:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2012-08-02 23:32:13 -07:00
|
|
|
}
|
2013-05-23 00:09:26 -07:00
|
|
|
bool enabled = nsLayoutUtils::AreAsyncAnimationsEnabled();
|
|
|
|
if (!enabled && shouldLog) {
|
|
|
|
nsCString message;
|
|
|
|
message.AppendLiteral("Performance warning: Async animations are disabled");
|
|
|
|
LogAsyncAnimationFailure(message);
|
|
|
|
}
|
|
|
|
return enabled && (aFlags & CanAnimate_AllowPartial);
|
2012-07-31 10:28:21 -07:00
|
|
|
}
|
|
|
|
|
2012-08-25 18:27:28 -07:00
|
|
|
/* static */ void
|
|
|
|
CommonElementAnimationData::LogAsyncAnimationFailure(nsCString& aMessage,
|
|
|
|
const nsIContent* aContent)
|
|
|
|
{
|
|
|
|
if (aContent) {
|
|
|
|
aMessage.AppendLiteral(" [");
|
|
|
|
aMessage.Append(nsAtomCString(aContent->Tag()));
|
|
|
|
|
|
|
|
nsIAtom* id = aContent->GetID();
|
|
|
|
if (id) {
|
|
|
|
aMessage.AppendLiteral(" with id '");
|
|
|
|
aMessage.Append(nsAtomCString(aContent->GetID()));
|
|
|
|
aMessage.AppendLiteral("'");
|
|
|
|
}
|
|
|
|
aMessage.AppendLiteral("]");
|
|
|
|
}
|
2012-12-11 13:12:47 -08:00
|
|
|
aMessage.AppendLiteral("\n");
|
2012-08-25 18:27:28 -07:00
|
|
|
printf_stderr(aMessage.get());
|
|
|
|
}
|
2012-07-31 10:28:21 -07:00
|
|
|
|
2012-12-11 13:12:43 -08:00
|
|
|
bool
|
|
|
|
CommonElementAnimationData::CanThrottleTransformChanges(TimeStamp aTime)
|
|
|
|
{
|
2013-05-22 03:31:03 -07:00
|
|
|
if (!nsLayoutUtils::AreAsyncAnimationsEnabled()) {
|
2012-12-11 13:12:43 -08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we know that the animation cannot cause overflow,
|
|
|
|
// we can just disable flushes for this animation.
|
|
|
|
|
|
|
|
// If we don't show scrollbars, we don't care about overflow.
|
|
|
|
if (LookAndFeel::GetInt(LookAndFeel::eIntID_ShowHideScrollbars) == 0) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If this animation can cause overflow, we can throttle some of the ticks.
|
|
|
|
if ((aTime - mStyleRuleRefreshTime) < TimeDuration::FromMilliseconds(200)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the nearest scrollable ancestor has overflow:hidden,
|
|
|
|
// we don't care about overflow.
|
|
|
|
nsIScrollableFrame* scrollable =
|
|
|
|
nsLayoutUtils::GetNearestScrollableFrame(mElement->GetPrimaryFrame());
|
|
|
|
if (!scrollable) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsPresContext::ScrollbarStyles ss = scrollable->GetScrollbarStyles();
|
|
|
|
if (ss.mVertical == NS_STYLE_OVERFLOW_HIDDEN &&
|
|
|
|
ss.mHorizontal == NS_STYLE_OVERFLOW_HIDDEN &&
|
|
|
|
scrollable->GetLogicalScrollPosition() == nsPoint(0, 0)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
CommonElementAnimationData::CanThrottleAnimation(TimeStamp aTime)
|
|
|
|
{
|
|
|
|
nsIFrame* frame = mElement->GetPrimaryFrame();
|
|
|
|
if (!frame) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool hasTransform = HasAnimationOfProperty(eCSSProperty_transform);
|
|
|
|
bool hasOpacity = HasAnimationOfProperty(eCSSProperty_opacity);
|
|
|
|
if (hasOpacity) {
|
|
|
|
Layer* layer = FrameLayerBuilder::GetDedicatedLayer(
|
|
|
|
frame, nsDisplayItem::TYPE_OPACITY);
|
|
|
|
if (!layer || mAnimationGeneration > layer->GetAnimationGeneration()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!hasTransform) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
Layer* layer = FrameLayerBuilder::GetDedicatedLayer(
|
|
|
|
frame, nsDisplayItem::TYPE_TRANSFORM);
|
|
|
|
if (!layer || mAnimationGeneration > layer->GetAnimationGeneration()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return CanThrottleTransformChanges(aTime);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CommonElementAnimationData::UpdateAnimationGeneration(nsPresContext* aPresContext)
|
|
|
|
{
|
|
|
|
mAnimationGeneration =
|
|
|
|
aPresContext->PresShell()->FrameConstructor()->GetAnimationGeneration();
|
|
|
|
}
|
|
|
|
|
2011-04-11 23:18:43 -07:00
|
|
|
}
|
|
|
|
}
|