346 lines
9.8 KiB
C++
346 lines
9.8 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* ***** BEGIN LICENSE BLOCK *****
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
*
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
|
* the License. You may obtain a copy of the License at
|
|
* http://www.mozilla.org/MPL/
|
|
*
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
* for the specific language governing rights and limitations under the
|
|
* License.
|
|
*
|
|
* The Original Code is the Mozilla SVG project.
|
|
*
|
|
* The Initial Developer of the Original Code is
|
|
* Crocodile Clips Ltd..
|
|
* Portions created by the Initial Developer are Copyright (C) 2001
|
|
* the Initial Developer. All Rights Reserved.
|
|
*
|
|
* Contributor(s):
|
|
* Alex Fritze <alex.fritze@crocodile-clips.com> (original author)
|
|
*
|
|
* Alternatively, the contents of this file may be used under the terms of
|
|
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
|
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
|
* of those above. If you wish to allow use of your version of this file only
|
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
|
* use your version of this file under the terms of the MPL, indicate your
|
|
* decision by deleting the provisions above and replace them with the notice
|
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
|
* the provisions above, a recipient may use your version of this file under
|
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
|
*
|
|
* ***** END LICENSE BLOCK ***** */
|
|
|
|
#include "nsIDOMSVGRect.h"
|
|
#include "nsIDOMSVGGElement.h"
|
|
#include "nsPresContext.h"
|
|
#include "nsISVGOuterSVGFrame.h"
|
|
#include "nsISVGRendererCanvas.h"
|
|
#include "nsISVGValue.h"
|
|
#include "nsIDOMSVGTransformable.h"
|
|
#include "nsIDOMSVGAnimTransformList.h"
|
|
#include "nsIDOMSVGTransformList.h"
|
|
#include "nsSVGDefsFrame.h"
|
|
#include "nsSVGUtils.h"
|
|
#include "nsINameSpaceManager.h"
|
|
#include "nsGkAtoms.h"
|
|
|
|
//----------------------------------------------------------------------
|
|
// Implementation
|
|
|
|
nsIFrame*
|
|
NS_NewSVGDefsFrame(nsIPresShell* aPresShell, nsIContent* aContent, nsStyleContext* aContext)
|
|
{
|
|
return new (aPresShell) nsSVGDefsFrame(aContext);
|
|
}
|
|
|
|
// Stub method specialized by subclasses. Not called by said
|
|
// specializations.
|
|
NS_IMETHODIMP
|
|
nsSVGDefsFrame::InitSVG()
|
|
{
|
|
return NS_OK;
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
// nsISupports methods
|
|
|
|
NS_INTERFACE_MAP_BEGIN(nsSVGDefsFrame)
|
|
NS_INTERFACE_MAP_ENTRY(nsISVGChildFrame)
|
|
NS_INTERFACE_MAP_ENTRY(nsISVGContainerFrame)
|
|
NS_INTERFACE_MAP_END_INHERITING(nsSVGDefsFrameBase)
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
// nsIFrame methods
|
|
NS_IMETHODIMP
|
|
nsSVGDefsFrame::Init(
|
|
nsIContent* aContent,
|
|
nsIFrame* aParent,
|
|
nsIFrame* aPrevInFlow)
|
|
{
|
|
nsresult rv;
|
|
rv = nsSVGDefsFrameBase::Init(aContent, aParent, aPrevInFlow);
|
|
|
|
InitSVG();
|
|
|
|
return rv;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsSVGDefsFrame::AppendFrames(nsIAtom* aListName,
|
|
nsIFrame* aFrameList)
|
|
{
|
|
// append == insert at end:
|
|
return InsertFrames(aListName, mFrames.LastChild(), aFrameList);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsSVGDefsFrame::InsertFrames(nsIAtom* aListName,
|
|
nsIFrame* aPrevFrame,
|
|
nsIFrame* aFrameList)
|
|
{
|
|
// memorize last new frame
|
|
nsIFrame* lastNewFrame = nsnull;
|
|
{
|
|
nsFrameList tmpList(aFrameList);
|
|
lastNewFrame = tmpList.LastChild();
|
|
}
|
|
|
|
// Insert the new frames
|
|
mFrames.InsertFrames(this, aPrevFrame, aFrameList);
|
|
|
|
// call InitialUpdate() on all new frames:
|
|
nsIFrame* end = nsnull;
|
|
if (lastNewFrame)
|
|
end = lastNewFrame->GetNextSibling();
|
|
|
|
for (nsIFrame* kid = aFrameList; kid != end;
|
|
kid = kid->GetNextSibling()) {
|
|
nsISVGChildFrame* SVGFrame=nsnull;
|
|
kid->QueryInterface(NS_GET_IID(nsISVGChildFrame),(void**)&SVGFrame);
|
|
if (SVGFrame) {
|
|
SVGFrame->InitialUpdate();
|
|
}
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsSVGDefsFrame::RemoveFrame(nsIAtom* aListName,
|
|
nsIFrame* aOldFrame)
|
|
{
|
|
nsCOMPtr<nsISVGRendererRegion> dirty_region;
|
|
|
|
nsISVGChildFrame* SVGFrame=nsnull;
|
|
aOldFrame->QueryInterface(NS_GET_IID(nsISVGChildFrame),(void**)&SVGFrame);
|
|
|
|
if (SVGFrame)
|
|
dirty_region = SVGFrame->GetCoveredRegion();
|
|
|
|
PRBool result = mFrames.DestroyFrame(aOldFrame);
|
|
|
|
nsISVGOuterSVGFrame* outerSVGFrame = nsSVGUtils::GetOuterSVGFrame(this);
|
|
NS_ASSERTION(outerSVGFrame, "no outer svg frame");
|
|
if (dirty_region && outerSVGFrame)
|
|
outerSVGFrame->InvalidateRegion(dirty_region, PR_TRUE);
|
|
|
|
NS_ASSERTION(result, "didn't find frame to delete");
|
|
return result ? NS_OK : NS_ERROR_FAILURE;
|
|
}
|
|
|
|
nsIAtom *
|
|
nsSVGDefsFrame::GetType() const
|
|
{
|
|
return nsLayoutAtoms::svgDefsFrame;
|
|
}
|
|
|
|
PRBool
|
|
nsSVGDefsFrame::IsFrameOfType(PRUint32 aFlags) const
|
|
{
|
|
return !(aFlags & ~nsIFrame::eSVG);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsSVGDefsFrame::AttributeChanged(PRInt32 aNameSpaceID,
|
|
nsIAtom* aAttribute,
|
|
PRInt32 aModType)
|
|
{
|
|
if (aNameSpaceID == kNameSpaceID_None &&
|
|
aAttribute == nsGkAtoms::transform) {
|
|
// make sure our cached transform matrix gets (lazily) updated
|
|
mCanvasTM = nsnull;
|
|
|
|
for (nsIFrame* kid = mFrames.FirstChild(); kid;
|
|
kid = kid->GetNextSibling()) {
|
|
nsISVGChildFrame* SVGFrame=nsnull;
|
|
kid->QueryInterface(NS_GET_IID(nsISVGChildFrame),(void**)&SVGFrame);
|
|
if (SVGFrame)
|
|
SVGFrame->NotifyCanvasTMChanged(PR_FALSE);
|
|
}
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
// nsISVGChildFrame methods
|
|
|
|
NS_IMETHODIMP
|
|
nsSVGDefsFrame::PaintSVG(nsISVGRendererCanvas* canvas)
|
|
{
|
|
// defs don't paint
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsSVGDefsFrame::GetFrameForPointSVG(float x, float y, nsIFrame** hit)
|
|
{
|
|
*hit = nsnull;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP_(already_AddRefed<nsISVGRendererRegion>)
|
|
nsSVGDefsFrame::GetCoveredRegion()
|
|
{
|
|
nsISVGRendererRegion *accu_region=nsnull;
|
|
|
|
return accu_region;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsSVGDefsFrame::InitialUpdate()
|
|
{
|
|
nsIFrame* kid = mFrames.FirstChild();
|
|
while (kid) {
|
|
nsISVGChildFrame* SVGFrame=0;
|
|
kid->QueryInterface(NS_GET_IID(nsISVGChildFrame),(void**)&SVGFrame);
|
|
if (SVGFrame) {
|
|
SVGFrame->InitialUpdate();
|
|
}
|
|
kid = kid->GetNextSibling();
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsSVGDefsFrame::NotifyCanvasTMChanged(PRBool suppressInvalidation)
|
|
{
|
|
// make sure our cached transform matrix gets (lazily) updated
|
|
mCanvasTM = nsnull;
|
|
|
|
for (nsIFrame* kid = mFrames.FirstChild(); kid;
|
|
kid = kid->GetNextSibling()) {
|
|
nsISVGChildFrame* SVGFrame=nsnull;
|
|
kid->QueryInterface(NS_GET_IID(nsISVGChildFrame),(void**)&SVGFrame);
|
|
if (SVGFrame) {
|
|
SVGFrame->NotifyCanvasTMChanged(suppressInvalidation);
|
|
}
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsSVGDefsFrame::NotifyRedrawSuspended()
|
|
{
|
|
for (nsIFrame* kid = mFrames.FirstChild(); kid;
|
|
kid = kid->GetNextSibling()) {
|
|
nsISVGChildFrame* SVGFrame=nsnull;
|
|
kid->QueryInterface(NS_GET_IID(nsISVGChildFrame),(void**)&SVGFrame);
|
|
if (SVGFrame) {
|
|
SVGFrame->NotifyRedrawSuspended();
|
|
}
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsSVGDefsFrame::NotifyRedrawUnsuspended()
|
|
{
|
|
for (nsIFrame* kid = mFrames.FirstChild(); kid;
|
|
kid = kid->GetNextSibling()) {
|
|
nsISVGChildFrame* SVGFrame=nsnull;
|
|
kid->QueryInterface(NS_GET_IID(nsISVGChildFrame),(void**)&SVGFrame);
|
|
if (SVGFrame) {
|
|
SVGFrame->NotifyRedrawUnsuspended();
|
|
}
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsSVGDefsFrame::GetBBox(nsIDOMSVGRect **_retval)
|
|
{
|
|
*_retval = nsnull;
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
//----------------------------------------------------------------------
|
|
// nsISVGContainerFrame methods:
|
|
|
|
already_AddRefed<nsIDOMSVGMatrix>
|
|
nsSVGDefsFrame::GetCanvasTM()
|
|
{
|
|
if (!mCanvasTM) {
|
|
// get our parent's tm and append local transforms (if any):
|
|
NS_ASSERTION(mParent, "null parent");
|
|
nsISVGContainerFrame *containerFrame;
|
|
mParent->QueryInterface(NS_GET_IID(nsISVGContainerFrame), (void**)&containerFrame);
|
|
if (!containerFrame) {
|
|
NS_ERROR("invalid parent");
|
|
return nsnull;
|
|
}
|
|
nsCOMPtr<nsIDOMSVGMatrix> parentTM = containerFrame->GetCanvasTM();
|
|
NS_ASSERTION(parentTM, "null TM");
|
|
|
|
// got the parent tm, now check for local tm:
|
|
nsCOMPtr<nsIDOMSVGMatrix> localTM;
|
|
{
|
|
nsCOMPtr<nsIDOMSVGTransformable> transformable = do_QueryInterface(mContent);
|
|
NS_ASSERTION(transformable, "wrong content element");
|
|
nsCOMPtr<nsIDOMSVGAnimatedTransformList> atl;
|
|
transformable->GetTransform(getter_AddRefs(atl));
|
|
NS_ASSERTION(atl, "null animated transform list");
|
|
nsCOMPtr<nsIDOMSVGTransformList> transforms;
|
|
atl->GetAnimVal(getter_AddRefs(transforms));
|
|
NS_ASSERTION(transforms, "null transform list");
|
|
PRUint32 numberOfItems;
|
|
transforms->GetNumberOfItems(&numberOfItems);
|
|
if (numberOfItems>0)
|
|
transforms->GetConsolidationMatrix(getter_AddRefs(localTM));
|
|
}
|
|
|
|
if (localTM)
|
|
parentTM->Multiply(localTM, getter_AddRefs(mCanvasTM));
|
|
else
|
|
mCanvasTM = parentTM;
|
|
}
|
|
|
|
nsIDOMSVGMatrix* retval = mCanvasTM.get();
|
|
NS_IF_ADDREF(retval);
|
|
return retval;
|
|
}
|
|
|
|
already_AddRefed<nsSVGCoordCtxProvider>
|
|
nsSVGDefsFrame::GetCoordContextProvider()
|
|
{
|
|
NS_ASSERTION(mParent, "null parent");
|
|
|
|
nsISVGContainerFrame *containerFrame;
|
|
mParent->QueryInterface(NS_GET_IID(nsISVGContainerFrame), (void**)&containerFrame);
|
|
if (!containerFrame) {
|
|
NS_ERROR("invalid container");
|
|
return nsnull;
|
|
}
|
|
|
|
return containerFrame->GetCoordContextProvider();
|
|
}
|