|  |  | 
|  | /* | 
|  | * Copyright 2010 The Android Open Source Project | 
|  | * | 
|  | * Use of this source code is governed by a BSD-style license that can be | 
|  | * found in the LICENSE file. | 
|  | */ | 
|  |  | 
|  |  | 
|  | #include "SkPDFFormXObject.h" | 
|  |  | 
|  | #include "SkMatrix.h" | 
|  | #include "SkPDFCatalog.h" | 
|  | #include "SkPDFDevice.h" | 
|  | #include "SkPDFResourceDict.h" | 
|  | #include "SkPDFUtils.h" | 
|  | #include "SkStream.h" | 
|  | #include "SkTypes.h" | 
|  |  | 
|  | SkPDFFormXObject::SkPDFFormXObject(SkPDFDevice* device) { | 
|  | // We don't want to keep around device because we'd have two copies | 
|  | // of content, so reference or copy everything we need (content and | 
|  | // resources). | 
|  | SkTSet<SkPDFObject*> emptySet; | 
|  | SkPDFResourceDict* resourceDict = device->getResourceDict(); | 
|  | resourceDict->getReferencedResources(emptySet, &fResources, false); | 
|  |  | 
|  | SkAutoTUnref<SkStream> content(device->content()); | 
|  | setData(content.get()); | 
|  |  | 
|  | SkAutoTUnref<SkPDFArray> bboxArray(device->copyMediaBox()); | 
|  | init(NULL, resourceDict, bboxArray); | 
|  |  | 
|  | // We invert the initial transform and apply that to the xobject so that | 
|  | // it doesn't get applied twice. We can't just undo it because it's | 
|  | // embedded in things like shaders and images. | 
|  | if (!device->initialTransform().isIdentity()) { | 
|  | SkMatrix inverse; | 
|  | if (!device->initialTransform().invert(&inverse)) { | 
|  | // The initial transform should be invertible. | 
|  | SkASSERT(false); | 
|  | inverse.reset(); | 
|  | } | 
|  | insert("Matrix", SkPDFUtils::MatrixToArray(inverse))->unref(); | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Creates a FormXObject from a content stream and associated resources. | 
|  | */ | 
|  | SkPDFFormXObject::SkPDFFormXObject(SkStream* content, SkRect bbox, | 
|  | SkPDFResourceDict* resourceDict) { | 
|  | SkTSet<SkPDFObject*> emptySet; | 
|  | resourceDict->getReferencedResources(emptySet, &fResources, false); | 
|  |  | 
|  | setData(content); | 
|  |  | 
|  | SkAutoTUnref<SkPDFArray> bboxArray(SkPDFUtils::RectToArray(bbox)); | 
|  | init("DeviceRGB", resourceDict, bboxArray); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Common initialization code. | 
|  | * Note that bbox is unreferenced here, so calling code does not need worry. | 
|  | */ | 
|  | void SkPDFFormXObject::init(const char* colorSpace, | 
|  | SkPDFDict* resourceDict, SkPDFArray* bbox) { | 
|  | insertName("Type", "XObject"); | 
|  | insertName("Subtype", "Form"); | 
|  | insert("Resources", resourceDict); | 
|  | insert("BBox", bbox); | 
|  |  | 
|  | // Right now SkPDFFormXObject is only used for saveLayer, which implies | 
|  | // isolated blending.  Do this conditionally if that changes. | 
|  | SkAutoTUnref<SkPDFDict> group(new SkPDFDict("Group")); | 
|  | group->insertName("S", "Transparency"); | 
|  |  | 
|  | if (colorSpace != NULL) { | 
|  | group->insertName("CS", colorSpace); | 
|  | } | 
|  | group->insert("I", new SkPDFBool(true))->unref();  // Isolated. | 
|  | insert("Group", group.get()); | 
|  | } | 
|  |  | 
|  | SkPDFFormXObject::~SkPDFFormXObject() { | 
|  | fResources.unrefAll(); | 
|  | } | 
|  |  | 
|  | void SkPDFFormXObject::getResources( | 
|  | const SkTSet<SkPDFObject*>& knownResourceObjects, | 
|  | SkTSet<SkPDFObject*>* newResourceObjects) { | 
|  | GetResourcesHelper(&fResources.toArray(), | 
|  | knownResourceObjects, | 
|  | newResourceObjects); | 
|  | } |