#define LOG_TAG "Layer" #define LOG_NDEBUG 1 #include "config.h" #include "Layer.h" #include "AndroidLog.h" #include "SkCanvas.h" //#define DEBUG_DRAW_LAYER_BOUNDS //#define DEBUG_TRACK_NEW_DELETE #ifdef DEBUG_TRACK_NEW_DELETE static int gLayerAllocCount; #endif /////////////////////////////////////////////////////////////////////////////// Layer::Layer() { fParent = NULL; m_opacity = SK_Scalar1; m_size.set(0, 0); m_position.set(0, 0); m_anchorPoint.set(SK_ScalarHalf, SK_ScalarHalf); m_matrix.reset(); m_childrenMatrix.reset(); m_shouldInheritFromRootTransform = false; m_hasOverflowChildren = false; m_state = 0; #ifdef DEBUG_TRACK_NEW_DELETE gLayerAllocCount += 1; SkDebugf("Layer new: %d\n", gLayerAllocCount); #endif } Layer::Layer(const Layer& src) : INHERITED() { fParent = NULL; m_opacity = src.m_opacity; m_size = src.m_size; m_position = src.m_position; m_scrollOffset = src.m_scrollOffset; m_anchorPoint = src.m_anchorPoint; m_matrix = src.m_matrix; m_childrenMatrix = src.m_childrenMatrix; m_shouldInheritFromRootTransform = src.m_shouldInheritFromRootTransform; m_hasOverflowChildren = src.m_hasOverflowChildren; m_state = 0; #ifdef DEBUG_TRACK_NEW_DELETE gLayerAllocCount += 1; SkDebugf("Layer copy: %d\n", gLayerAllocCount); #endif } Layer::~Layer() { removeChildren(); #ifdef DEBUG_TRACK_NEW_DELETE gLayerAllocCount -= 1; SkDebugf("Layer delete: %d\n", gLayerAllocCount); #endif } /////////////////////////////////////////////////////////////////////////////// int Layer::countChildren() const { return m_children.count(); } Layer* Layer::getChild(int index) const { if ((unsigned)index < (unsigned)m_children.count()) { SkASSERT(m_children[index]->fParent == this); return m_children[index]; } return NULL; } Layer* Layer::addChild(Layer* child) { SkASSERT(this != child); child->ref(); child->detachFromParent(); SkASSERT(child->fParent == NULL); child->fParent = this; *m_children.append() = child; return child; } void Layer::detachFromParent() { if (fParent) { int index = fParent->m_children.find(this); SkASSERT(index >= 0); fParent->m_children.remove(index); fParent = NULL; unref(); // this call might delete us } } void Layer::removeChildren() { int count = m_children.count(); for (int i = 0; i < count; i++) { Layer* child = m_children[i]; SkASSERT(child->fParent == this); child->fParent = NULL; // in case it has more than one owner child->unref(); } m_children.reset(); } Layer* Layer::getRootLayer() const { const Layer* root = this; while (root->fParent != NULL) { root = root->fParent; } return const_cast(root); } /////////////////////////////////////////////////////////////////////////////// void Layer::getLocalTransform(SkMatrix* matrix) const { matrix->setTranslate(m_position.fX - m_scrollOffset.x(), m_position.fY - m_scrollOffset.y()); SkScalar tx = SkScalarMul(m_anchorPoint.fX, m_size.width()); SkScalar ty = SkScalarMul(m_anchorPoint.fY, m_size.height()); matrix->preTranslate(tx, ty); matrix->preConcat(getMatrix()); matrix->preTranslate(-tx, -ty); } void Layer::localToAncestor(const Layer* ancestor, SkMatrix* matrix) const { if (this == ancestor) { matrix->setIdentity(); return; } getLocalTransform(matrix); // Fixed position layers simply use the root layer's transform. if (shouldInheritFromRootTransform()) { ASSERT(!ancestor); matrix->postConcat(getRootLayer()->getMatrix()); return; } // Apply the local and child transforms for every layer between this layer // and ancestor. ASSERT(isAncestor(ancestor)); for (const Layer* layer = this->fParent; layer != ancestor; layer = layer->fParent) { SkMatrix tmp; layer->getLocalTransform(&tmp); tmp.preConcat(layer->getChildrenMatrix()); matrix->postConcat(tmp); } // If ancestor is not the root layer, apply its child transformation too. if (ancestor) matrix->postConcat(ancestor->getChildrenMatrix()); } /////////////////////////////////////////////////////////////////////////////// #include "SkString.h" void Layer::draw(SkCanvas* canvas, android::DrawExtra* extra, SkScalar opacity) { #if 0 SkString str1, str2; // getMatrix().toDumpString(&str1); // getChildrenMatrix().toDumpString(&str2); SkDebugf("--- drawlayer %p opacity %g size [%g %g] pos [%g %g] matrix %s children %s\n", this, opacity * getOpacity(), m_size.width(), m_size.height(), m_position.fX, m_position.fY, str1.c_str(), str2.c_str()); #endif opacity = SkScalarMul(opacity, getOpacity()); if (opacity <= 0) { // SkDebugf("---- abort drawing %p opacity %g\n", this, opacity); return; } SkAutoCanvasRestore acr(canvas, true); // apply our local transform { SkMatrix tmp; getLocalTransform(&tmp); if (shouldInheritFromRootTransform()) { // should we also apply the root's childrenMatrix? canvas->setMatrix(getRootLayer()->getMatrix()); } canvas->concat(tmp); } onDraw(canvas, opacity, extra, FlattenedLayers); #ifdef DEBUG_DRAW_LAYER_BOUNDS { SkRect r = SkRect::MakeSize(getSize()); SkPaint p; p.setAntiAlias(true); p.setStyle(SkPaint::kStroke_Style); p.setStrokeWidth(SkIntToScalar(2)); p.setColor(0xFFFF44DD); canvas->drawRect(r, p); canvas->drawLine(r.fLeft, r.fTop, r.fRight, r.fBottom, p); canvas->drawLine(r.fLeft, r.fBottom, r.fRight, r.fTop, p); } #endif int count = countChildren(); if (count > 0) { canvas->concat(getChildrenMatrix()); for (int i = 0; i < count; i++) { getChild(i)->draw(canvas, extra, opacity); } } } bool Layer::isAncestor(const Layer* possibleAncestor) const { if (!possibleAncestor) return true; for (const Layer* layer = getParent(); layer; layer = layer->getParent()) { if (layer == possibleAncestor) return true; } return false; } void Layer::setState(WebCore::GLWebViewState* state) { m_state = state; int count = countChildren(); for (int i = 0; i < count; i++) m_children[i]->setState(state); }