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 "SkDrawGroup.h"
     11 #include "SkAnimateMaker.h"
     12 #include "SkAnimatorScript.h"
     13 #include "SkCanvas.h"
     14 #include "SkDisplayApply.h"
     15 #include "SkPaint.h"
     16 #ifdef SK_DEBUG
     17 #include "SkDisplayList.h"
     18 #endif
     19 
     20 #if SK_USE_CONDENSED_INFO == 0
     21 
     22 const SkMemberInfo SkGroup::fInfo[] = {
     23     SK_MEMBER(condition, String),
     24     SK_MEMBER(enableCondition, String)
     25 };
     26 
     27 #endif
     28 
     29 DEFINE_GET_MEMBER(SkGroup);
     30 
     31 SkGroup::SkGroup() : fParentList(nullptr), fOriginal(nullptr) {
     32 }
     33 
     34 SkGroup::~SkGroup() {
     35     if (fOriginal)  // has been copied
     36         return;
     37     int index = 0;
     38     int max = fCopies.count() << 5;
     39     for (SkADrawable** ptr = fChildren.begin(); ptr < fChildren.end(); ptr++) {
     40         if (index >= max || markedForDelete(index))
     41             delete *ptr;
     42 //      else {
     43 //          SkApply* apply = (SkApply*) *ptr;
     44 //          SkASSERT(apply->isApply());
     45 //          SkASSERT(apply->getScope());
     46 //          delete apply->getScope();
     47 //      }
     48         index++;
     49     }
     50 }
     51 
     52 bool SkGroup::addChild(SkAnimateMaker& , SkDisplayable* child) {
     53     SkASSERT(child);
     54 //  SkASSERT(child->isDrawable());
     55     *fChildren.append() = (SkADrawable*) child;
     56     if (child->isGroup()) {
     57         SkGroup* groupie = (SkGroup*) child;
     58         SkASSERT(groupie->fParentList == nullptr);
     59         groupie->fParentList = &fChildren;
     60     }
     61     return true;
     62 }
     63 
     64 bool SkGroup::contains(SkDisplayable* match) {
     65     for (SkADrawable** ptr = fChildren.begin(); ptr < fChildren.end(); ptr++) {
     66         SkADrawable* drawable = *ptr;
     67         if (drawable == match || drawable->contains(match))
     68             return true;
     69     }
     70     return false;
     71 }
     72 
     73 SkGroup* SkGroup::copy() {
     74     SkGroup* result = new SkGroup();
     75     result->fOriginal = this;
     76     result->fChildren = fChildren;
     77     return result;
     78 }
     79 
     80 SkBool SkGroup::copySet(int index) {
     81     return (fCopies[index >> 5] & 1 << (index & 0x1f)) != 0;
     82 }
     83 
     84 SkDisplayable* SkGroup::deepCopy(SkAnimateMaker* maker) {
     85     SkDisplayable* copy = INHERITED::deepCopy(maker);
     86     for (SkADrawable** ptr = fChildren.begin(); ptr < fChildren.end(); ptr++) {
     87         SkDisplayable* displayable = (SkDisplayable*)*ptr;
     88         SkDisplayable* deeperCopy = displayable->deepCopy(maker);
     89         ((SkGroup*)copy)->addChild(*maker, deeperCopy);
     90     }
     91     return copy;
     92 }
     93 
     94 bool SkGroup::doEvent(SkDisplayEvent::Kind kind, SkEventState* state) {
     95     bool handled = false;
     96     for (SkADrawable** ptr = fChildren.begin(); ptr < fChildren.end(); ptr++) {
     97         SkADrawable* drawable = *ptr;
     98         if (drawable->isDrawable() == false)
     99             continue;
    100         handled |= drawable->doEvent(kind, state);
    101     }
    102     return handled;
    103 }
    104 
    105 bool SkGroup::draw(SkAnimateMaker& maker) {
    106     bool conditionTrue = ifCondition(maker, this, condition);
    107     bool result = false;
    108     for (SkADrawable** ptr = fChildren.begin(); ptr < fChildren.end(); ptr++) {
    109         SkADrawable* drawable = *ptr;
    110         if (drawable->isDrawable() == false)
    111             continue;
    112         if (conditionTrue == false) {
    113             if (drawable->isApply())
    114                 ((SkApply*) drawable)->disable();
    115             continue;
    116         }
    117         maker.validate();
    118         result |= drawable->draw(maker);
    119         maker.validate();
    120     }
    121     return result;
    122 }
    123 
    124 #ifdef SK_DUMP_ENABLED
    125 void SkGroup::dump(SkAnimateMaker* maker) {
    126     dumpBase(maker);
    127     if (condition.size() > 0)
    128         SkDebugf("condition=\"%s\" ", condition.c_str());
    129     if (enableCondition.size() > 0)
    130         SkDebugf("enableCondition=\"%s\" ", enableCondition.c_str());
    131     dumpDrawables(maker);
    132 }
    133 
    134 void SkGroup::dumpDrawables(SkAnimateMaker* maker) {
    135     SkDisplayList::fIndent += 4;
    136     int save = SkDisplayList::fDumpIndex;
    137     SkDisplayList::fDumpIndex = 0;
    138     bool closedYet = false;
    139     for (SkADrawable** ptr = fChildren.begin(); ptr < fChildren.end(); ptr++) {
    140         if (closedYet == false) {
    141             closedYet = true;
    142             SkDebugf(">\n");
    143         }
    144         SkADrawable* drawable = *ptr;
    145         drawable->dump(maker);
    146         SkDisplayList::fDumpIndex++;
    147     }
    148     SkDisplayList::fIndent -= 4;
    149     SkDisplayList::fDumpIndex = save;
    150     if (closedYet) //we had children, now it's time to close the group
    151         dumpEnd(maker);
    152     else    //no children
    153         SkDebugf("/>\n");
    154 }
    155 
    156 void SkGroup::dumpEvents() {
    157     for (SkADrawable** ptr = fChildren.begin(); ptr < fChildren.end(); ptr++) {
    158         SkADrawable* drawable = *ptr;
    159         drawable->dumpEvents();
    160     }
    161 }
    162 #endif
    163 
    164 bool SkGroup::enable(SkAnimateMaker& maker ) {
    165     reset();
    166     for (SkADrawable** ptr = fChildren.begin(); ptr < fChildren.end(); ptr++) {
    167         SkADrawable* drawable = *ptr;
    168         if (ifCondition(maker, drawable, enableCondition) == false)
    169             continue;
    170         drawable->enable(maker);
    171     }
    172     return true;    // skip add; already added so that scope is findable by children
    173 }
    174 
    175 int SkGroup::findGroup(SkADrawable* match,  SkTDDrawableArray** list,
    176                  SkGroup** parent, SkGroup** found, SkTDDrawableArray** grandList) {
    177     *list = &fChildren;
    178     for (SkADrawable** ptr = fChildren.begin(); ptr < fChildren.end(); ptr++) {
    179         SkADrawable* drawable = *ptr;
    180         if (drawable->isGroup()) {
    181             SkGroup* childGroup = (SkGroup*) drawable;
    182             if (childGroup->fOriginal == match)
    183                 goto foundMatch;
    184         }
    185         if (drawable == match) {
    186 foundMatch:
    187             *parent = this;
    188             return (int) (ptr - fChildren.begin());
    189         }
    190     }
    191     *grandList = &fChildren;
    192     return SkDisplayList::SearchForMatch(match, list, parent, found, grandList);
    193 }
    194 
    195 bool SkGroup::hasEnable() const {
    196     return true;
    197 }
    198 
    199 bool SkGroup::ifCondition(SkAnimateMaker& maker, SkADrawable*,
    200         SkString& conditionString) {
    201     if (conditionString.size() == 0)
    202         return true;
    203     int32_t result;
    204     bool success = SkAnimatorScript::EvaluateInt(maker, this, conditionString.c_str(), &result);
    205 #ifdef SK_DUMP_ENABLED
    206     if (maker.fDumpGConditions) {
    207         SkDebugf("group: ");
    208         dumpBase(&maker);
    209         SkDebugf("condition=%s ", conditionString.c_str());
    210         if (success == false)
    211             SkDebugf("(script failed)\n");
    212         else
    213             SkDebugf("success=%s\n", result != 0 ? "true" : "false");
    214     }
    215 #endif
    216     return success && result != 0;
    217 }
    218 
    219 void SkGroup::initialize() {
    220     for (SkADrawable** ptr = fChildren.begin(); ptr < fChildren.end(); ptr++) {
    221         SkADrawable* drawable = *ptr;
    222         if (drawable->isDrawable() == false)
    223             continue;
    224         drawable->initialize();
    225     }
    226 }
    227 
    228 void SkGroup::markCopyClear(int index) {
    229     if (index < 0)
    230         index = fChildren.count();
    231     fCopies[index >> 5] &= ~(1 << (index & 0x1f));
    232 }
    233 
    234 void SkGroup::markCopySet(int index) {
    235     if (index < 0)
    236         index = fChildren.count();
    237     fCopies[index >> 5] |= 1 << (index & 0x1f);
    238 }
    239 
    240 void SkGroup::markCopySize(int index) {
    241     if (index < 0)
    242         index = fChildren.count() + 1;
    243     int oldLongs = fCopies.count();
    244     int newLongs = (index >> 5) + 1;
    245     if (oldLongs < newLongs) {
    246         fCopies.setCount(newLongs);
    247         memset(&fCopies[oldLongs], 0, (newLongs - oldLongs) << 2);
    248     }
    249 }
    250 
    251 void SkGroup::reset() {
    252     if (fOriginal)  // has been copied
    253         return;
    254     int index = 0;
    255     int max = fCopies.count() << 5;
    256     for (SkADrawable** ptr = fChildren.begin(); ptr < fChildren.end(); ptr++) {
    257         if (index >= max || copySet(index) == false)
    258             continue;
    259         SkApply* apply = (SkApply*) *ptr;
    260         SkASSERT(apply->isApply());
    261         SkASSERT(apply->getScope());
    262         *ptr = apply->getScope();
    263         markCopyClear(index);
    264         index++;
    265     }
    266 }
    267 
    268 bool SkGroup::resolveIDs(SkAnimateMaker& maker, SkDisplayable* orig, SkApply* apply) {
    269     SkGroup* original = (SkGroup*) orig;
    270     SkTDDrawableArray& originalChildren = original->fChildren;
    271     SkADrawable** originalPtr = originalChildren.begin();
    272     SkADrawable** ptr = fChildren.begin();
    273     SkADrawable** end = fChildren.end();
    274     SkADrawable** origChild = ((SkGroup*) orig)->fChildren.begin();
    275     while (ptr < end) {
    276         SkADrawable* drawable = *ptr++;
    277         maker.resolveID(drawable, *origChild++);
    278         if (drawable->resolveIDs(maker, *originalPtr++, apply) == true)
    279             return true; // failed
    280     }
    281     return false;
    282 }
    283 
    284 void SkGroup::setSteps(int steps) {
    285     for (SkADrawable** ptr = fChildren.begin(); ptr < fChildren.end(); ptr++) {
    286         SkADrawable* drawable = *ptr;
    287         if (drawable->isDrawable() == false)
    288             continue;
    289         drawable->setSteps(steps);
    290     }
    291 }
    292 
    293 #ifdef SK_DEBUG
    294 void SkGroup::validate() {
    295     for (SkADrawable** ptr = fChildren.begin(); ptr < fChildren.end(); ptr++) {
    296         SkADrawable* drawable = *ptr;
    297         drawable->validate();
    298     }
    299 }
    300 #endif
    301 
    302 #if SK_USE_CONDENSED_INFO == 0
    303 
    304 const SkMemberInfo SkSave::fInfo[] = {
    305     SK_MEMBER_INHERITED
    306 };
    307 
    308 #endif
    309 
    310 DEFINE_GET_MEMBER(SkSave);
    311 
    312 bool SkSave::draw(SkAnimateMaker& maker) {
    313     maker.fCanvas->save();
    314     SkPaint* save = maker.fPaint;
    315     SkPaint local = SkPaint(*maker.fPaint);
    316     maker.fPaint = &local;
    317     bool result = INHERITED::draw(maker);
    318     maker.fPaint = save;
    319     maker.fCanvas->restore();
    320     return result;
    321 }
    322