From d055c1fde2128514167b315f4d104b177e04a3de Mon Sep 17 00:00:00 2001 From: "reed@android.com" Date: Mon, 1 Mar 2010 14:54:05 +0000 Subject: [PATCH] convert matrix <--> matrix44 add glyphrec to bounder git-svn-id: http://skia.googlecode.com/svn/trunk@516 2bbb7eff-a529-9590-31e7-b0007b416f81 --- experimental/SkMatrix44.cpp | 38 +++++++++++++++++++++++++++++++++++++ experimental/SkMatrix44.h | 12 +++++++++++- include/core/SkBounder.h | 18 ++++++++++++++++-- include/core/SkMatrix.h | 6 ++++++ src/core/SkDraw.cpp | 23 ++++++++++++++++++---- 5 files changed, 90 insertions(+), 7 deletions(-) diff --git a/experimental/SkMatrix44.cpp b/experimental/SkMatrix44.cpp index b2b8c30465..841eec27d1 100644 --- a/experimental/SkMatrix44.cpp +++ b/experimental/SkMatrix44.cpp @@ -269,4 +269,42 @@ void SkMatrix44::dump() const { #endif } +/////////////////////////////////////////////////////////////////////////////// + +static void initFromMatrix(SkMScalar dst[4][4], const SkMatrix& src) { + sk_bzero(dst, 16 * sizeof(SkMScalar)); + dst[0][0] = src[SkMatrix::kMScaleX]; + dst[1][0] = src[SkMatrix::kMSkewX]; + dst[3][0] = src[SkMatrix::kMTransX]; + dst[0][1] = src[SkMatrix::kMSkewY]; + dst[1][1] = src[SkMatrix::kMScaleY]; + dst[3][1] = src[SkMatrix::kMTransY]; + dst[2][2] = dst[3][3] = 1; +} + +SkMatrix44::SkMatrix44(const SkMatrix& src) { + initFromMatrix(fMat, src); +} + +SkMatrix44& SkMatrix44::operator=(const SkMatrix& src) { + initFromMatrix(fMat, src); + return *this; +} + +SkMatrix44::operator SkMatrix() const { + SkMatrix dst; + dst.reset(); // setup our perspective correctly for identity + + dst[SkMatrix::kMScaleX] = SkMScalarToFloat(fMat[0][0]); + dst[SkMatrix::kMSkewX] = SkMScalarToFloat(fMat[1][0]); + dst[SkMatrix::kMTransX] = SkMScalarToFloat(fMat[3][0]); + + dst[SkMatrix::kMSkewY] = SkMScalarToFloat(fMat[0][1]); + dst[SkMatrix::kMScaleY] = SkMScalarToFloat(fMat[1][1]); + dst[SkMatrix::kMTransY] = SkMScalarToFloat(fMat[3][1]); + + return dst; +} + + diff --git a/experimental/SkMatrix44.h b/experimental/SkMatrix44.h index f86559056a..b0e7148c41 100644 --- a/experimental/SkMatrix44.h +++ b/experimental/SkMatrix44.h @@ -1,6 +1,7 @@ #ifndef SkMatrix44_DEFINED #define SkMatrix44_DEFINED +#include "SkMatrix.h" #include "SkScalar.h" // uncomment this to use doubles for matrix44 @@ -89,13 +90,22 @@ public: SkMatrix44(const SkMatrix44&); SkMatrix44(const SkMatrix44& a, const SkMatrix44& b); + SkMatrix44& operator=(const SkMatrix44& src) { + memcpy(this, &src, sizeof(*this)); + return *this; + } + bool operator==(const SkMatrix44& other) const { return !memcmp(this, &other, sizeof(*this)); } bool operator!=(const SkMatrix44& other) const { return !!memcmp(this, &other, sizeof(*this)); } - + + SkMatrix44(const SkMatrix&); + SkMatrix44& operator=(const SkMatrix& src); + operator SkMatrix() const; + bool isIdentity() const; void setIdentity(); void reset() { this->setIdentity(); } diff --git a/include/core/SkBounder.h b/include/core/SkBounder.h index 00fbbc6b65..bda5c97a98 100644 --- a/include/core/SkBounder.h +++ b/include/core/SkBounder.h @@ -19,7 +19,9 @@ #include "SkTypes.h" #include "SkRefCnt.h" +#include "SkPoint.h" +struct SkGlyph; struct SkIRect; struct SkPoint; struct SkRect; @@ -38,7 +40,8 @@ public: Returns the result from onIRect. */ bool doIRect(const SkIRect&); - bool doIRect(const SkIRect& , uint16_t glyphID); + bool doIRectGlyph(const SkIRect& , int x, int y, const SkGlyph&); + protected: /** Override in your subclass. This is called with the device bounds of an object (text, geometry, image) just before it is drawn. If your method @@ -50,10 +53,21 @@ protected: return false; } + /** Passed to onIRectGlyph with the information about the current glyph. + LSB and RSB are fixed-point (16.16) coordinates of the start and end + of the glyph's advance + */ + struct GlyphRec { + SkIPoint fLSB; //!< fixed-point left-side-bearing of the glyph + SkIPoint fRSB; //!< fixed-point right-side-bearing of the glyph + uint16_t fGlyphID; + uint16_t fFlags; //!< currently set to 0 + }; + /** Optionally, override in your subclass to receive the glyph ID when text drawing supplies the device bounds of the object. */ - virtual bool onIRect(const SkIRect& r, uint16_t glyphID) { + virtual bool onIRectGlyph(const SkIRect& r, const GlyphRec&) { return onIRect(r); } diff --git a/include/core/SkMatrix.h b/include/core/SkMatrix.h index 741c349a43..7ef7d8cd61 100644 --- a/include/core/SkMatrix.h +++ b/include/core/SkMatrix.h @@ -102,6 +102,12 @@ public: SkScalar getPerspX() const { return fMat[kMPersp0]; } SkScalar getPerspY() const { return fMat[kMPersp1]; } + SkScalar& operator[](int index) { + SkASSERT((unsigned)index < 9); + this->setTypeMask(kUnknown_Mask); + return fMat[index]; + } + void set(int index, SkScalar value) { SkASSERT((unsigned)index < 9); fMat[index] = value; diff --git a/src/core/SkDraw.cpp b/src/core/SkDraw.cpp index 8bf4c084f6..f112d4b088 100644 --- a/src/core/SkDraw.cpp +++ b/src/core/SkDraw.cpp @@ -1370,8 +1370,12 @@ static void D1G_Bounder(const SkDraw1Glyph& state, return; } } - - if (state.fBounder->doIRect(cr, glyph.getGlyphID())) { + + // we need to pass the origin, which we approximate with our + // (unadjusted) left,top coordinates (the caller called fixedfloor) + if (state.fBounder->doIRectGlyph(cr, + left - glyph.fLeft, + top - glyph.fTop, glyph)) { mask.fRowBytes = glyph.rowBytes(); mask.fFormat = static_cast(glyph.fMaskFormat); mask.fImage = (uint8_t*)aa; @@ -2249,9 +2253,20 @@ bool SkBounder::doIRect(const SkIRect& r) { return rr.intersect(fClip->getBounds(), r) && this->onIRect(rr); } -bool SkBounder::doIRect(const SkIRect& r, uint16_t glyphID) { +// TODO: change the prototype to take fixed, and update the callers +bool SkBounder::doIRectGlyph(const SkIRect& r, int x, int y, + const SkGlyph& glyph) { SkIRect rr; - return rr.intersect(fClip->getBounds(), r) && this->onIRect(rr, glyphID); + if (!rr.intersect(fClip->getBounds(), r)) { + return false; + } + GlyphRec rec; + rec.fLSB.set(SkIntToFixed(x), SkIntToFixed(y)); + rec.fRSB.set(rec.fLSB.fX + glyph.fAdvanceX, + rec.fLSB.fY + glyph.fAdvanceY); + rec.fGlyphID = glyph.getGlyphID(); + rec.fFlags = 0; + return this->onIRectGlyph(rr, rec); } bool SkBounder::doHairline(const SkPoint& pt0, const SkPoint& pt1, -- 2.34.1