Home | History | Annotate | Download | only in animator
      1 
      2 /*
      3  * Copyright 2006 The Android Open Source Project
      4  *
      5  * Use of this source code is governed by a BSD-style license that can be
      6  * found in the LICENSE file.
      7  */
      8 
      9 
     10 #include "SkAnimator.h"
     11 #include "SkAnimateMaker.h"
     12 #include "SkCanvas.h"
     13 #include "SkDisplayApply.h"
     14 #include "SkDisplayMovie.h"
     15 #include "SkDisplayTypes.h"
     16 #include "SkDisplayXMLParser.h"
     17 #include "SkStream.h"
     18 #include "SkScript.h"
     19 #include "SkScript2.h" //   compiled script experiment
     20 #include "SkSystemEventTypes.h"
     21 #include "SkTypedArray.h"
     22 #ifdef SK_BUILD_FOR_ANDROID
     23 #include "SkDrawExtraPathEffect.h"
     24 #endif
     25 #ifdef SK_DEBUG
     26 #include "SkTime.h"
     27 #endif
     28 
     29 #if defined SK_BUILD_FOR_WIN32 && defined SK_DEBUG
     30     #define _static
     31     extern const char gMathPrimerText[];
     32     extern const char gMathPrimerBinary[];
     33 #else
     34     #define _static static
     35 #endif
     36 
     37 _static const char gMathPrimerText[] =
     38 "<screenplay>"
     39     "<Math id=\"Math\"/>"
     40     "<Number id=\"Number\"/>"
     41 "</screenplay>";
     42 
     43 #define gMathPrimer gMathPrimerText
     44 
     45 SkAnimator::SkAnimator() : fMaker(nullptr) {
     46     initialize();
     47 }
     48 
     49 SkAnimator::~SkAnimator() { delete fMaker; }
     50 
     51 void SkAnimator::addExtras(SkExtras* extras) {
     52     *fMaker->fExtras.append() = extras;
     53 }
     54 
     55 bool SkAnimator::appendStream(SkStream* stream) {
     56     return decodeStream(stream);
     57 }
     58 
     59 bool SkAnimator::decodeMemory(const void* buffer, size_t size)
     60 {
     61     fMaker->fFileName.reset();
     62     SkDisplayXMLParser parser(*fMaker);
     63     return parser.parse((const char*)buffer, size);
     64 }
     65 
     66 bool SkAnimator::decodeStream(SkStream* stream)
     67 {
     68     SkDisplayXMLParser parser(*fMaker);
     69     bool result = parser.parse(*stream);
     70     fMaker->setErrorString();
     71     return result;
     72 }
     73 
     74 bool SkAnimator::decodeDOM(const SkDOM& dom, const SkDOMNode* node)
     75 {
     76     fMaker->fFileName.reset();
     77     SkDisplayXMLParser parser(*fMaker);
     78     return parser.parse(dom, node);
     79 }
     80 
     81 bool SkAnimator::decodeURI(const char uri[]) {
     82 //  SkDebugf("animator decode %s\n", uri);
     83 
     84 //    SkStream* stream = SkStream::GetURIStream(fMaker->fPrefix.c_str(), uri);
     85     SkAutoTDelete<SkStream> stream(SkStream::NewFromFile(uri));
     86     if (stream.get()) {
     87         this->setURIBase(uri);
     88         return decodeStream(stream);
     89     } else {
     90         return false;
     91     }
     92 }
     93 
     94 bool SkAnimator::doCharEvent(SkUnichar code) {
     95     if (code == 0)
     96         return false;
     97     struct SkEventState state;
     98     state.fCode = code;
     99     fMaker->fEnableTime = fMaker->getAppTime();
    100     bool result = fMaker->fEvents.doEvent(*fMaker, SkDisplayEvent::kKeyChar, &state);
    101     fMaker->notifyInval();
    102     return result;
    103 }
    104 
    105 bool SkAnimator::doClickEvent(int clickState, SkScalar x, SkScalar y) {
    106     SkASSERT(clickState >= 0 && clickState <= 2);
    107     struct SkEventState state;
    108     state.fX = x;
    109     state.fY = y;
    110     fMaker->fEnableTime = fMaker->getAppTime();
    111     bool result = fMaker->fEvents.doEvent(*fMaker,
    112         clickState == 0 ? SkDisplayEvent::kMouseDown :
    113         clickState == 1 ? SkDisplayEvent::kMouseDrag :
    114         SkDisplayEvent::kMouseUp, &state);
    115     fMaker->notifyInval();
    116     return result;
    117 }
    118 
    119 bool SkAnimator::doKeyEvent(SkKey code) {
    120     if (code == 0)
    121         return false;
    122     struct SkEventState state;
    123     state.fCode = code;
    124     fMaker->fEnableTime = fMaker->getAppTime();
    125     bool result = fMaker->fEvents.doEvent(*fMaker, SkDisplayEvent::kKeyPress, &state);
    126     fMaker->notifyInval();
    127     return result;
    128 }
    129 
    130 bool SkAnimator::doKeyUpEvent(SkKey code) {
    131     if (code == 0)
    132         return false;
    133     struct SkEventState state;
    134     state.fCode = code;
    135     fMaker->fEnableTime = fMaker->getAppTime();
    136     bool result = fMaker->fEvents.doEvent(*fMaker, SkDisplayEvent::kKeyPressUp, &state);
    137     fMaker->notifyInval();
    138     return result;
    139 }
    140 
    141 bool SkAnimator::doUserEvent(const SkEvent& evt) {
    142     fMaker->fEnableTime = fMaker->getAppTime();
    143     return onEvent(evt);
    144 }
    145 
    146 SkAnimator::DifferenceType SkAnimator::draw(SkCanvas* canvas, SkPaint* paint, SkMSec time) {
    147     if (paint == nullptr)
    148         return draw(canvas, time);
    149     fMaker->fScreenplay.time = time;
    150     fMaker->fCanvas = canvas;
    151     fMaker->fPaint = paint;
    152     fMaker->fDisplayList.fHasUnion = false;
    153     int result = fMaker->fDisplayList.draw(*fMaker, time);
    154     if (result)
    155         result += fMaker->fDisplayList.fHasUnion;
    156     return (DifferenceType) result;
    157 }
    158 
    159 SkAnimator::DifferenceType SkAnimator::draw(SkCanvas* canvas, SkMSec time) {
    160     SkPaint paint;
    161     return draw(canvas, &paint, time);
    162 }
    163 
    164 #ifdef SK_DEBUG
    165 void SkAnimator::eventDone(const SkEvent& ) {
    166 }
    167 #endif
    168 
    169 bool SkAnimator::findClickEvent(SkScalar x, SkScalar y) {
    170     struct SkEventState state;
    171     state.fDisable = true;
    172     state.fX = x;
    173     state.fY = y;
    174     fMaker->fEnableTime = fMaker->getAppTime();
    175     bool result = fMaker->fEvents.doEvent(*fMaker, SkDisplayEvent::kMouseDown, &state);
    176     fMaker->notifyInval();
    177     return result;
    178 }
    179 
    180 const SkAnimator* SkAnimator::getAnimator(const SkDisplayable* displayable) const {
    181     if (displayable->getType() != SkType_Movie)
    182         return nullptr;
    183     const SkDisplayMovie* movie = (const SkDisplayMovie*) displayable;
    184     return movie->getAnimator();
    185 }
    186 
    187 const SkDisplayable* SkAnimator::getElement(const char* id) {
    188     SkDisplayable* element;
    189     if (fMaker->find(id, &element) == false)
    190         return nullptr;
    191     return (const SkDisplayable*) element;
    192 }
    193 
    194 SkElementType SkAnimator::getElementType(const SkDisplayable* ae) {
    195     SkDisplayable* element = (SkDisplayable*) ae;
    196     const SkMemberInfo* info = SkDisplayType::GetMembers(fMaker, element->getType(), nullptr);
    197     return (SkElementType) SkDisplayType::Find(fMaker, info);
    198 }
    199 
    200 SkElementType SkAnimator::getElementType(const char* id) {
    201     const SkDisplayable* element = getElement(id);
    202     return getElementType(element);
    203 }
    204 
    205 const SkMemberInfo* SkAnimator::getField(const SkDisplayable* ae, const char* field) {
    206     SkDisplayable* element = (SkDisplayable*) ae;
    207     const SkMemberInfo* info = element->getMember(field);
    208     return (const SkMemberInfo*) info;
    209 }
    210 
    211 const SkMemberInfo* SkAnimator::getField(const char* elementID, const char* field) {
    212     const SkDisplayable* element = getElement(elementID);
    213     return getField(element, field);
    214 }
    215 
    216 SkFieldType SkAnimator::getFieldType(const SkMemberInfo* ai) {
    217     const SkMemberInfo* info = (const SkMemberInfo*) ai;
    218     return (SkFieldType) info->getType();
    219 }
    220 
    221 SkFieldType SkAnimator::getFieldType(const char* id, const char* fieldID) {
    222     const SkMemberInfo* field = getField(id, fieldID);
    223     return getFieldType(field);
    224 }
    225 
    226 static bool getArrayCommon(const SkDisplayable* ae, const SkMemberInfo* ai,
    227                            int index, SkOperand* operand) {
    228     const SkDisplayable* element = (const SkDisplayable*) ae;
    229     const SkMemberInfo* info = (const SkMemberInfo*) ai;
    230     SkASSERT(info->fType == SkType_Array);
    231     return info->getArrayValue(element, index, operand);
    232 }
    233 
    234 int32_t SkAnimator::getArrayInt(const SkDisplayable* ae,
    235         const SkMemberInfo* ai, int index) {
    236     SkOperand operand;
    237     bool result = getArrayCommon(ae, ai, index, &operand);
    238     return result ? operand.fS32 : SK_NaN32;
    239 }
    240 
    241 int32_t SkAnimator::getArrayInt(const char* id, const char* fieldID, int index) {
    242     const SkDisplayable* element = getElement(id);
    243     if (element == nullptr)
    244         return SK_NaN32;
    245     const SkMemberInfo* field = getField(element, fieldID);
    246     if (field == nullptr)
    247         return SK_NaN32;
    248     return getArrayInt(element, field, index);
    249 }
    250 
    251 SkScalar SkAnimator::getArrayScalar(const SkDisplayable* ae,
    252         const SkMemberInfo* ai, int index) {
    253     SkOperand operand;
    254     bool result = getArrayCommon(ae, ai, index, &operand);
    255     return result ? operand.fScalar : SK_ScalarNaN;
    256 }
    257 
    258 SkScalar SkAnimator::getArrayScalar(const char* id, const char* fieldID, int index) {
    259     const SkDisplayable* element = getElement(id);
    260     if (element == nullptr)
    261         return SK_ScalarNaN;
    262     const SkMemberInfo* field = getField(element, fieldID);
    263     if (field == nullptr)
    264         return SK_ScalarNaN;
    265     return getArrayScalar(element, field, index);
    266 }
    267 
    268 const char* SkAnimator::getArrayString(const SkDisplayable* ae,
    269         const SkMemberInfo* ai, int index) {
    270     SkOperand operand;
    271     bool result = getArrayCommon(ae, ai, index, &operand);
    272     return result ? operand.fString->c_str() : nullptr;
    273 }
    274 
    275 const char* SkAnimator::getArrayString(const char* id, const char* fieldID, int index) {
    276     const SkDisplayable* element = getElement(id);
    277     if (element == nullptr)
    278         return nullptr;
    279     const SkMemberInfo* field = getField(element, fieldID);
    280     if (field == nullptr)
    281         return nullptr;
    282     return getArrayString(element, field, index);
    283 }
    284 
    285 SkMSec SkAnimator::getInterval() {
    286     return fMaker->fMinimumInterval == (SkMSec) -1 ? 0 : fMaker->fMinimumInterval;
    287 }
    288 
    289 void SkAnimator::getInvalBounds(SkRect* inval) {
    290     if (fMaker->fDisplayList.fHasUnion) {
    291         inval->fLeft = SkIntToScalar(fMaker->fDisplayList.fInvalBounds.fLeft);
    292         inval->fTop = SkIntToScalar(fMaker->fDisplayList.fInvalBounds.fTop);
    293         inval->fRight = SkIntToScalar(fMaker->fDisplayList.fInvalBounds.fRight);
    294         inval->fBottom = SkIntToScalar(fMaker->fDisplayList.fInvalBounds.fBottom);
    295     } else {
    296         inval->fLeft = inval->fTop = -SK_ScalarMax;
    297         inval->fRight = inval->fBottom = SK_ScalarMax;
    298     }
    299 }
    300 
    301 const SkXMLParserError* SkAnimator::getParserError() {
    302     return &fMaker->fError;
    303 }
    304 
    305 const char* SkAnimator::getParserErrorString() {
    306     if (fMaker->fErrorString.size() == 0 && fMaker->fError.hasError())
    307         fMaker->setErrorString();
    308     return fMaker->fErrorString.c_str();
    309 }
    310 
    311 int32_t SkAnimator::getInt(const SkDisplayable* element, const SkMemberInfo* info) {
    312     if (info->fType != SkType_MemberProperty) {
    313         SkOperand operand;
    314         if (info->getType() == SkType_Int) {
    315             info->getValue(element, &operand, 1);
    316             return operand.fS32;
    317         }
    318         return SK_NaN32;
    319     }
    320     SkScriptValue scriptValue;
    321     bool success = element->getProperty(info->propertyIndex(), &scriptValue);
    322     if (success && scriptValue.fType == SkType_Int)
    323         return scriptValue.fOperand.fS32;
    324     return SK_NaN32;
    325 }
    326 
    327 int32_t SkAnimator::getInt(const char* id, const char* fieldID) {
    328     const SkDisplayable* element = getElement(id);
    329     if (element == nullptr)
    330         return SK_NaN32;
    331     const SkMemberInfo* field = getField(element, fieldID);
    332     if (field == nullptr)
    333         return SK_NaN32;
    334     return getInt(element, field);
    335 }
    336 
    337 SkScalar SkAnimator::getScalar(const SkDisplayable* element, const SkMemberInfo* info) {
    338     if (info->fType != SkType_MemberProperty) {
    339         SkOperand operand;
    340         if (info->getType() == SkType_Float) {
    341             info->getValue(element, &operand, 1);
    342             return operand.fScalar;
    343         }
    344         return SK_ScalarNaN;
    345     }
    346     SkScriptValue scriptValue;
    347     bool success = element->getProperty(info->propertyIndex(), &scriptValue);
    348     if (success && scriptValue.fType == SkType_Float)
    349         return scriptValue.fOperand.fScalar;
    350     return SK_ScalarNaN;
    351 }
    352 
    353 SkScalar SkAnimator::getScalar(const char* id, const char* fieldID) {
    354     const SkDisplayable* element = getElement(id);
    355     if (element == nullptr)
    356         return SK_ScalarNaN;
    357     const SkMemberInfo* field = getField(element, fieldID);
    358     if (field == nullptr)
    359         return SK_ScalarNaN;
    360     return getScalar(element, field);
    361 }
    362 
    363 const char* SkAnimator::getString(const SkDisplayable* ae,
    364         const SkMemberInfo* ai) {
    365     const SkDisplayable* element = (const SkDisplayable*) ae;
    366     const SkMemberInfo* info = (const SkMemberInfo*) ai;
    367     SkString* temp;
    368     info->getString(element, &temp);
    369     return temp->c_str();
    370 }
    371 
    372 const char* SkAnimator::getString(const char* id, const char* fieldID) {
    373     const SkDisplayable* element = getElement(id);
    374     if (element == nullptr)
    375         return nullptr;
    376     const SkMemberInfo* field = getField(element, fieldID);
    377     if (field == nullptr)
    378         return nullptr;
    379     return getString(element, field);
    380 }
    381 
    382 const char* SkAnimator::getURIBase() {
    383     return fMaker->fPrefix.c_str();
    384 }
    385 
    386 void SkAnimator::initialize() {
    387     delete fMaker;
    388     fMaker = new SkAnimateMaker(this, nullptr, nullptr);
    389     decodeMemory(gMathPrimer, sizeof(gMathPrimer)-1);
    390 #ifdef SK_BUILD_FOR_ANDROID
    391     InitializeSkExtraPathEffects(this);
    392 #endif
    393 }
    394 
    395 
    396 #ifdef SK_DEBUG
    397 bool SkAnimator::isTrackingEvents() {
    398     return false;
    399 }
    400 #endif
    401 
    402 bool SkAnimator::onEvent(const SkEvent& evt) {
    403 #ifdef SK_DEBUG
    404     SkAnimator* root = fMaker->getRoot();
    405     if (root == nullptr)
    406         root = this;
    407     if (root->isTrackingEvents())
    408         root->eventDone(evt);
    409 #endif
    410     if (evt.isType(SK_EventType_OnEnd)) {
    411         SkEventState eventState;
    412         SkDEBUGCODE(bool success =) evt.findPtr("anim", (void**) &eventState.fDisplayable);
    413         SkASSERT(success);
    414         SkDEBUGCODE(success =) evt.findS32("time", (int32_t*) &fMaker->fEnableTime);
    415         SkASSERT(success);
    416         fMaker->fAdjustedStart = fMaker->getAppTime() - fMaker->fEnableTime;
    417         fMaker->fEvents.doEvent(*fMaker, SkDisplayEvent::kOnEnd, &eventState);
    418         fMaker->fAdjustedStart = 0;
    419         goto inval;
    420     }
    421     if (evt.isType(SK_EventType_Delay)) {
    422         fMaker->doDelayedEvent();
    423         goto inval;
    424     }
    425     {
    426         const char* id = evt.findString("id");
    427         if (id == nullptr)
    428             return false;
    429         SkDisplayable** firstMovie = fMaker->fMovies.begin();
    430         SkDisplayable** endMovie = fMaker->fMovies.end();
    431         for (SkDisplayable** ptr = firstMovie; ptr < endMovie; ptr++) {
    432             SkDisplayMovie* movie = (SkDisplayMovie*) *ptr;
    433             movie->doEvent(evt);
    434         }
    435         {
    436             SkDisplayable* event;
    437             if (fMaker->find(id, &event) == false)
    438                 return false;
    439     #if defined SK_DEBUG && defined SK_DEBUG_ANIMATION_TIMING
    440             SkString debugOut;
    441             SkMSec realTime = fMaker->getAppTime();
    442             debugOut.appendS32(realTime - fMaker->fDebugTimeBase);
    443             debugOut.append(" onEvent id=");
    444             debugOut.append(id);
    445     #endif
    446             SkMSec time = evt.getFast32();
    447             if (time != 0) {
    448                 SkMSec app  = fMaker->getAppTime();
    449                 fMaker->setEnableTime(app, time);
    450     #if defined SK_DEBUG && defined SK_DEBUG_ANIMATION_TIMING
    451                 debugOut.append(" time=");
    452                 debugOut.appendS32(time - fMaker->fDebugTimeBase);
    453                 debugOut.append(" adjust=");
    454                 debugOut.appendS32(fMaker->fAdjustedStart);
    455     #endif
    456             }
    457     #if defined SK_DEBUG && defined SK_DEBUG_ANIMATION_TIMING
    458             SkDebugf("%s\n", debugOut.c_str());
    459     #endif
    460             SkASSERT(event->isEvent());
    461             SkDisplayEvent* displayEvent = (SkDisplayEvent*) event;
    462             displayEvent->populateInput(*fMaker, evt);
    463             displayEvent->enableEvent(*fMaker);
    464         }
    465     }
    466 inval:
    467     fMaker->notifyInval();
    468     return true;
    469 }
    470 
    471 void SkAnimator::onEventPost(SkEvent* evt, SkEventSinkID sinkID)
    472 {
    473 #ifdef SK_DEBUG
    474     SkAnimator* root = fMaker->getRoot();
    475     if (root) {
    476         root->onEventPost(evt, sinkID);
    477         return;
    478     }
    479 #else
    480     SkASSERT(sinkID == this->getSinkID() || this->getHostEventSinkID() == sinkID);
    481 #endif
    482     evt->setTargetID(sinkID)->post();
    483 }
    484 
    485 void SkAnimator::onEventPostTime(SkEvent* evt, SkEventSinkID sinkID, SkMSec time)
    486 {
    487 #ifdef SK_DEBUG
    488     SkAnimator* root = fMaker->getRoot();
    489     if (root) {
    490         root->onEventPostTime(evt, sinkID, time);
    491         return;
    492     }
    493 #else
    494     SkASSERT(sinkID == this->getSinkID() || this->getHostEventSinkID() == sinkID);
    495 #endif
    496     evt->setTargetID(sinkID)->postTime(time);
    497 }
    498 
    499 void SkAnimator::reset() {
    500     fMaker->fDisplayList.reset();
    501 }
    502 
    503 SkEventSinkID SkAnimator::getHostEventSinkID() const {
    504     return fMaker->fHostEventSinkID;
    505 }
    506 
    507 void SkAnimator::setHostEventSinkID(SkEventSinkID target) {
    508     fMaker->fHostEventSinkID = target;
    509 }
    510 
    511 void SkAnimator::onSetHostHandler(Handler ) {
    512 }
    513 
    514 void SkAnimator::setJavaOwner(Handler ) {
    515 }
    516 
    517 bool SkAnimator::setArrayString(const char* id, const char* fieldID, const char** array, int num)
    518 {
    519     SkTypedArray tArray(SkType_String);
    520     tArray.setCount(num);
    521     for (int i = 0; i < num; i++) {
    522         SkOperand op;
    523         op.fString = new SkString(array[i]);
    524         tArray[i] = op;
    525     }
    526     return setArray(id, fieldID, tArray);
    527 }
    528 bool SkAnimator::setArrayInt(const char* id, const char* fieldID, const int* array, int num)
    529 {
    530     SkTypedArray tArray(SkType_Int);
    531     tArray.setCount(num);
    532     for (int i = 0; i < num; i++) {
    533         SkOperand op;
    534         op.fS32 = array[i];
    535         tArray[i] = op;
    536     }
    537     return setArray(id, fieldID, tArray);
    538 }
    539 
    540 bool SkAnimator::setArray(SkDisplayable* element, const SkMemberInfo* info, SkTypedArray array) {
    541     if (info->fType != SkType_Array)
    542         return false;   //the field is not an array
    543     //i think we can handle the case where the displayable itself is an array differently from the
    544     //case where it has an array - for one thing, if it is an array, i think we can change its type
    545     //if it's not, we cannot
    546     SkDisplayTypes type = element->getType();
    547     if (type == SkType_Array) {
    548         SkDisplayArray* dispArray = (SkDisplayArray*) element;
    549         dispArray->values = array;
    550         return true;
    551     }
    552     else
    553         return false;   //currently i don't care about this case
    554 }
    555 
    556 bool SkAnimator::setArray(const char* id, const char* fieldID, SkTypedArray array) {
    557     SkDisplayable* element = (SkDisplayable*) getElement(id);
    558     //should I go ahead and change all 'nullptr's to 'nullptr'?
    559     if (element == nullptr)
    560         return false;
    561     const SkMemberInfo* field = getField(element, fieldID);
    562     if (field == nullptr)
    563         return false;
    564     return setArray(element, field, array);
    565 }
    566 
    567 bool SkAnimator::setInt(SkDisplayable* element, const SkMemberInfo* info, int32_t s32) {
    568     if (info->fType != SkType_MemberProperty) {
    569         SkOperand operand;
    570         operand.fS32 = s32;
    571         SkASSERT(info->getType() == SkType_Int);
    572         info->setValue(element, &operand, 1);
    573     } else {
    574         SkScriptValue scriptValue;
    575         scriptValue.fType = SkType_Int;
    576         scriptValue.fOperand.fS32 = s32;
    577         element->setProperty(info->propertyIndex(), scriptValue);
    578     }
    579     return true;
    580 }
    581 
    582 bool SkAnimator::setInt(const char* id, const char* fieldID, int32_t s32) {
    583     SkDisplayable* element = (SkDisplayable*) getElement(id);
    584     if (element == nullptr)
    585         return false;
    586     const SkMemberInfo* field = getField(element, fieldID);
    587     if (field == nullptr)
    588         return false;
    589     return setInt(element, field, s32);
    590 }
    591 
    592 bool SkAnimator::setScalar(SkDisplayable* element, const SkMemberInfo* info, SkScalar scalar) {
    593     if (info->fType != SkType_MemberProperty) {
    594         SkOperand operand;
    595         operand.fScalar = scalar;
    596         SkASSERT(info->getType() == SkType_Float);
    597         info->setValue(element, &operand, 1);
    598     } else {
    599         SkScriptValue scriptValue;
    600         scriptValue.fType = SkType_Float;
    601         scriptValue.fOperand.fScalar = scalar;
    602         element->setProperty(info->propertyIndex(), scriptValue);
    603     }
    604     return true;
    605 }
    606 
    607 bool SkAnimator::setScalar(const char* id, const char* fieldID, SkScalar scalar) {
    608     SkDisplayable* element = (SkDisplayable*) getElement(id);
    609     if (element == nullptr)
    610         return false;
    611     const SkMemberInfo* field = getField(element, fieldID);
    612     if (field == nullptr)
    613         return false;
    614     return setScalar(element, field, scalar);
    615 }
    616 
    617 bool SkAnimator::setString(SkDisplayable* element,
    618         const SkMemberInfo* info, const char* str) {
    619     // !!! until this is fixed, can't call script with global references from here
    620     info->setValue(*fMaker, nullptr, 0, info->fCount, element, info->getType(), str, strlen(str));
    621     return true;
    622 }
    623 
    624 bool SkAnimator::setString(const char* id, const char* fieldID, const char* str) {
    625     SkDisplayable* element = (SkDisplayable*) getElement(id);
    626     if (element == nullptr)
    627         return false;
    628     const SkMemberInfo* field = getField(element, fieldID);
    629     if (field == nullptr)
    630         return false;
    631     return setString(element, field, str);
    632 }
    633 
    634 void SkAnimator::setTimeline(const Timeline& timeline) {
    635     fMaker->fTimeline = &timeline;
    636 }
    637 
    638 void SkAnimator::setURIBase(const char* uri) {
    639     if (uri)
    640     {
    641         const char* tail = strrchr(uri, '/');
    642         if (tail) {
    643             SkString prefix(uri, tail - uri + 1);
    644             if (uri[0] != '.' /*SkStream::IsAbsoluteURI(uri)*/)
    645                 fMaker->fPrefix.reset();
    646             fMaker->fPrefix.append(prefix);
    647             fMaker->fFileName.set(tail + 1);
    648         } else
    649             fMaker->fFileName.set(uri);
    650     }
    651 }
    652 
    653 #ifdef SK_DEBUG
    654 bool SkAnimator::NoLeaks() {
    655 #ifdef SK_BUILD_FOR_MAC
    656     if (SkDisplayable::fAllocations.count() == 0)
    657         return true;
    658 //  return SkDisplayable::fAllocationCount == 0;
    659     SkDebugf("!!! leaked %d displayables:\n", SkDisplayable::fAllocations.count());
    660     for (SkDisplayable** leak = SkDisplayable::fAllocations.begin(); leak < SkDisplayable::fAllocations.end(); leak++)
    661         SkDebugf("%08x %s\n", *leak, (*leak)->id);
    662 #endif
    663     return false;
    664 }
    665 #endif
    666 
    667 #ifdef SK_SUPPORT_UNITTEST
    668 #include "SkAnimatorScript.h"
    669 #include "SkBase64.h"
    670 #include "SkParse.h"
    671 #include "SkMemberInfo.h"
    672 
    673 #define unittestline(type)  { #type , type::UnitTest }
    674 #endif
    675 
    676 
    677 #ifdef SK_SUPPORT_UNITTEST
    678 void SkAnimator::Init(bool runUnitTests) {
    679     if (runUnitTests == false)
    680         return;
    681     static const struct {
    682         const char* fTypeName;
    683         void (*fUnitTest)( );
    684     } gUnitTests[] = {
    685         unittestline(SkBase64),
    686         unittestline(SkDisplayType),
    687         unittestline(SkParse),
    688         unittestline(SkScriptEngine),
    689 //      unittestline(SkScriptEngine2),  // compiled script experiment
    690         unittestline(SkAnimatorScript)
    691     };
    692     for (int i = 0; i < (int)SK_ARRAY_COUNT(gUnitTests); i++)
    693     {
    694         SkDebugf("SkAnimator: Running UnitTest for %s\n", gUnitTests[i].fTypeName);
    695         gUnitTests[i].fUnitTest();
    696         SkDebugf("SkAnimator: End UnitTest for %s\n", gUnitTests[i].fTypeName);
    697     }
    698 }
    699 #else
    700 void SkAnimator::Init(bool) {}
    701 #endif
    702 
    703 void SkAnimator::Term() {
    704 }
    705