/* ***** 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 mozilla.org code. * * The Initial Developer of the Original Code is * Netscape Communications Corporation. * Portions created by the Initial Developer are Copyright (C) 2001 * the Initial Developer. All Rights Reserved. * * Contributor(s): * Joe Hewitt (original author) * Christopher A. Aillon * * 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 "inFlasher.h" #include "inLayoutUtils.h" #include "nsIServiceManager.h" #include "nsIViewManager.h" #include "nsIDeviceContext.h" #include "nsIWidget.h" #include "nsIPresShell.h" #include "nsIFrame.h" #include "nsReadableUtils.h" #include "prprf.h" static NS_DEFINE_CID(kInspectorCSSUtilsCID, NS_INSPECTORCSSUTILS_CID); /////////////////////////////////////////////////////////////////////////////// inFlasher::inFlasher() : mColor(NS_RGB(0,0,0)), mThickness(0), mInvert(PR_FALSE) { mCSSUtils = do_GetService(kInspectorCSSUtilsCID); } inFlasher::~inFlasher() { } NS_IMPL_ISUPPORTS1(inFlasher, inIFlasher) /////////////////////////////////////////////////////////////////////////////// // inIFlasher NS_IMETHODIMP inFlasher::GetColor(nsAString& aColor) { // Copied from nsGenericHTMLElement::ColorToString() char buf[10]; PR_snprintf(buf, sizeof(buf), "#%02x%02x%02x", NS_GET_R(mColor), NS_GET_G(mColor), NS_GET_B(mColor)); CopyASCIItoUTF16(buf, aColor); return NS_OK; } NS_IMETHODIMP inFlasher::SetColor(const nsAString& aColor) { NS_ENSURE_FALSE(aColor.IsEmpty(), NS_ERROR_ILLEGAL_VALUE); nsAutoString colorStr; colorStr.Assign(aColor); if (colorStr.CharAt(0) != '#') { if (NS_ColorNameToRGB(colorStr, &mColor)) { return NS_OK; } } else { colorStr.Cut(0, 1); if (NS_HexToRGB(colorStr, &mColor)) { return NS_OK; } } return NS_ERROR_ILLEGAL_VALUE; } NS_IMETHODIMP inFlasher::GetThickness(PRUint16 *aThickness) { NS_PRECONDITION(aThickness, "Null pointer"); *aThickness = mThickness; return NS_OK; } NS_IMETHODIMP inFlasher::SetThickness(PRUint16 aThickness) { mThickness = aThickness; return NS_OK; } NS_IMETHODIMP inFlasher::GetInvert(PRBool *aInvert) { NS_PRECONDITION(aInvert, "Null pointer"); *aInvert = mInvert; return NS_OK; } NS_IMETHODIMP inFlasher::SetInvert(PRBool aInvert) { mInvert = aInvert; return NS_OK; } NS_IMETHODIMP inFlasher::RepaintElement(nsIDOMElement* aElement) { nsCOMPtr window = inLayoutUtils::GetWindowFor(aElement); if (!window) return NS_OK; nsCOMPtr presShell = inLayoutUtils::GetPresShellFor(window); if (!presShell) return NS_OK; nsIFrame* frame = inLayoutUtils::GetFrameFor(aElement, presShell); if (!frame) return NS_OK; nsIFrame* parentWithView = frame->GetAncestorWithViewExternal(); if (parentWithView) { nsIView* view = parentWithView->GetViewExternal(); if (view) { nsIViewManager* viewManager = view->GetViewManager(); if (viewManager) { nsRect rect = parentWithView->GetRect(); viewManager->UpdateView(view, rect, NS_VMREFRESH_NO_SYNC); } } } return NS_OK; } NS_IMETHODIMP inFlasher::DrawElementOutline(nsIDOMElement* aElement) { nsCOMPtr window = inLayoutUtils::GetWindowFor(aElement); if (!window) return NS_OK; nsCOMPtr presShell = inLayoutUtils::GetPresShellFor(window); nsPresContext *presContext = presShell->GetPresContext(); float p2t = presContext->PixelsToTwips(); PRBool isFirstFrame = PR_TRUE; nsCOMPtr rcontext; nsIFrame* frame = inLayoutUtils::GetFrameFor(aElement, presShell); while (frame) { if (!rcontext) { presShell->CreateRenderingContext(frame, getter_AddRefs(rcontext)); } // get view bounds in case this frame is being scrolled nsRect rect = frame->GetRect(); nsPoint origin = inLayoutUtils::GetClientOrigin(presContext, frame); rect.MoveTo(origin); mCSSUtils->AdjustRectForMargins(frame, rect); if (mInvert) { rcontext->InvertRect(rect); } frame = frame->GetNextInFlow(); PRBool isLastFrame = (frame == nsnull); DrawOutline(rect.x, rect.y, rect.width, rect.height, p2t, rcontext, isFirstFrame, isLastFrame); isFirstFrame = PR_FALSE; } return NS_OK; } NS_IMETHODIMP inFlasher::ScrollElementIntoView(nsIDOMElement *aElement) { NS_PRECONDITION(aElement, "Dude, where's my arg?!"); nsCOMPtr window = inLayoutUtils::GetWindowFor(aElement); if (!window) { return NS_OK; } nsCOMPtr presShell = inLayoutUtils::GetPresShellFor(window); NS_ASSERTION(presShell, "Dude, where's my pres shell?!"); nsIFrame* frame = inLayoutUtils::GetFrameFor(aElement, presShell); if (!frame) { return NS_OK; } presShell->ScrollFrameIntoView(frame, NS_PRESSHELL_SCROLL_ANYWHERE /* VPercent */, NS_PRESSHELL_SCROLL_ANYWHERE /* HPercent */); return NS_OK; } /////////////////////////////////////////////////////////////////////////////// // inFlasher void inFlasher::DrawOutline(nscoord aX, nscoord aY, nscoord aWidth, nscoord aHeight, float aP2T, nsIRenderingContext* aRenderContext, PRBool aDrawBegin, PRBool aDrawEnd) { aRenderContext->SetColor(mColor); DrawLine(aX, aY, aWidth, DIR_HORIZONTAL, BOUND_OUTER, aP2T, aRenderContext); if (aDrawBegin) { DrawLine(aX, aY, aHeight, DIR_VERTICAL, BOUND_OUTER, aP2T, aRenderContext); } DrawLine(aX, aY+aHeight, aWidth, DIR_HORIZONTAL, BOUND_INNER, aP2T, aRenderContext); if (aDrawEnd) { DrawLine(aX+aWidth, aY, aHeight, DIR_VERTICAL, BOUND_INNER, aP2T, aRenderContext); } } void inFlasher::DrawLine(nscoord aX, nscoord aY, nscoord aLength, PRBool aDir, PRBool aBounds, float aP2T, nsIRenderingContext* aRenderContext) { nscoord thickTwips = NSIntPixelsToTwips(mThickness, aP2T); if (aDir) { // horizontal aRenderContext->FillRect(aX, aY+(aBounds?0:-thickTwips), aLength, thickTwips); } else { // vertical aRenderContext->FillRect(aX+(aBounds?0:-thickTwips), aY, thickTwips, aLength); } }