Bug 470455 - test_database_sync_embed_visits.js leaks, r=sdwilsh
[wine-gecko.git] / accessible / src / msaa / nsTextAccessibleWrap.cpp
blob00c9c5ef24c538bd3a4251e2a3efb191ec6479f0
1 /* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
2 /* ***** BEGIN LICENSE BLOCK *****
3 * Version: MPL 1.1/GPL 2.0/LGPL 2.1
5 * The contents of this file are subject to the Mozilla Public License Version
6 * 1.1 (the "License"); you may not use this file except in compliance with
7 * the License. You may obtain a copy of the License at
8 * http://www.mozilla.org/MPL/
10 * Software distributed under the License is distributed on an "AS IS" basis,
11 * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
12 * for the specific language governing rights and limitations under the
13 * License.
15 * The Original Code is mozilla.org code.
17 * The Initial Developer of the Original Code is
18 * Netscape Corp.
19 * Portions created by the Initial Developer are Copyright (C) 2003
20 * the Initial Developer. All Rights Reserved.
22 * Contributor(s):
23 * Aaron Leventhal
25 * Alternatively, the contents of this file may be used under the terms of
26 * either the GNU General Public License Version 2 or later (the "GPL"), or
27 * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
28 * in which case the provisions of the GPL or the LGPL are applicable instead
29 * of those above. If you wish to allow use of your version of this file only
30 * under the terms of either the GPL or the LGPL, and not to allow others to
31 * use your version of this file under the terms of the MPL, indicate your
32 * decision by deleting the provisions above and replace them with the notice
33 * and other provisions required by the GPL or the LGPL. If you do not delete
34 * the provisions above, a recipient may use your version of this file under
35 * the terms of any one of the MPL, the GPL or the LGPL.
37 * ***** END LICENSE BLOCK ***** */
39 // NOTE: alphabetically ordered
40 #include "nsTextAccessibleWrap.h"
41 #include "ISimpleDOMText_i.c"
42 #include "nsIAccessibleDocument.h"
43 #include "nsIFontMetrics.h"
44 #include "nsIFrame.h"
45 #include "nsPresContext.h"
46 #include "nsIPresShell.h"
47 #include "nsIRenderingContext.h"
48 #include "nsIWidget.h"
49 #include "nsIComponentManager.h"
51 // --------------------------------------------------------
52 // nsTextAccessibleWrap Accessible
53 // --------------------------------------------------------
55 nsTextAccessibleWrap::nsTextAccessibleWrap(nsIDOMNode* aDOMNode, nsIWeakReference* aShell):
56 nsTextAccessible(aDOMNode, aShell)
60 STDMETHODIMP_(ULONG) nsTextAccessibleWrap::AddRef()
62 return nsAccessNode::AddRef();
65 STDMETHODIMP_(ULONG) nsTextAccessibleWrap::Release()
67 return nsAccessNode::Release();
70 STDMETHODIMP nsTextAccessibleWrap::QueryInterface(REFIID iid, void** ppv)
72 *ppv = nsnull;
74 if (IID_IUnknown == iid || IID_ISimpleDOMText == iid)
75 *ppv = static_cast<ISimpleDOMText*>(this);
77 if (nsnull == *ppv)
78 return nsAccessibleWrap::QueryInterface(iid, ppv);
80 (reinterpret_cast<IUnknown*>(*ppv))->AddRef();
81 return S_OK;
84 STDMETHODIMP nsTextAccessibleWrap::get_domText(
85 /* [retval][out] */ BSTR __RPC_FAR *aDomText)
87 __try {
88 *aDomText = NULL;
90 if (!mDOMNode) {
91 return E_FAIL; // Node already shut down
93 nsAutoString nodeValue;
95 mDOMNode->GetNodeValue(nodeValue);
96 if (nodeValue.IsEmpty())
97 return S_FALSE;
99 *aDomText = ::SysAllocStringLen(nodeValue.get(), nodeValue.Length());
100 if (!*aDomText)
101 return E_OUTOFMEMORY;
103 } __except(FilterA11yExceptions(::GetExceptionCode(), GetExceptionInformation())) { }
105 return S_OK;
108 STDMETHODIMP nsTextAccessibleWrap::get_clippedSubstringBounds(
109 /* [in] */ unsigned int aStartIndex,
110 /* [in] */ unsigned int aEndIndex,
111 /* [out] */ int __RPC_FAR *aX,
112 /* [out] */ int __RPC_FAR *aY,
113 /* [out] */ int __RPC_FAR *aWidth,
114 /* [out] */ int __RPC_FAR *aHeight)
116 __try {
117 *aX = *aY = *aWidth = *aHeight = 0;
118 nscoord x, y, width, height, docX, docY, docWidth, docHeight;
119 HRESULT rv = get_unclippedSubstringBounds(aStartIndex, aEndIndex, &x, &y, &width, &height);
120 if (FAILED(rv)) {
121 return rv;
124 nsCOMPtr<nsIAccessibleDocument> docAccessible(GetDocAccessible());
125 nsCOMPtr<nsIAccessible> accessible(do_QueryInterface(docAccessible));
126 NS_ASSERTION(accessible, "There must always be a doc accessible, but there isn't");
128 accessible->GetBounds(&docX, &docY, &docWidth, &docHeight);
130 nsRect unclippedRect(x, y, width, height);
131 nsRect docRect(docX, docY, docWidth, docHeight);
132 nsRect clippedRect;
134 clippedRect.IntersectRect(unclippedRect, docRect);
136 *aX = clippedRect.x;
137 *aY = clippedRect.y;
138 *aWidth = clippedRect.width;
139 *aHeight = clippedRect.height;
140 } __except(FilterA11yExceptions(::GetExceptionCode(), GetExceptionInformation())) { }
142 return S_OK;
145 STDMETHODIMP nsTextAccessibleWrap::get_unclippedSubstringBounds(
146 /* [in] */ unsigned int aStartIndex,
147 /* [in] */ unsigned int aEndIndex,
148 /* [out] */ int __RPC_FAR *aX,
149 /* [out] */ int __RPC_FAR *aY,
150 /* [out] */ int __RPC_FAR *aWidth,
151 /* [out] */ int __RPC_FAR *aHeight)
153 __try {
154 *aX = *aY = *aWidth = *aHeight = 0;
156 if (!mDOMNode) {
157 return E_FAIL; // Node already shut down
160 if (NS_FAILED(GetCharacterExtents(aStartIndex, aEndIndex,
161 aX, aY, aWidth, aHeight))) {
162 return NS_ERROR_FAILURE;
164 } __except(FilterA11yExceptions(::GetExceptionCode(), GetExceptionInformation())) { }
166 return S_OK;
170 STDMETHODIMP nsTextAccessibleWrap::scrollToSubstring(
171 /* [in] */ unsigned int aStartIndex,
172 /* [in] */ unsigned int aEndIndex)
174 __try {
175 nsresult rv =
176 nsCoreUtils::ScrollSubstringTo(GetFrame(), mDOMNode, aStartIndex,
177 mDOMNode, aEndIndex,
178 nsIAccessibleScrollType::SCROLL_TYPE_ANYWHERE);
179 if (NS_FAILED(rv))
180 return E_FAIL;
181 } __except(FilterA11yExceptions(::GetExceptionCode(), GetExceptionInformation())) { }
182 return S_OK;
185 nsIFrame* nsTextAccessibleWrap::GetPointFromOffset(nsIFrame *aContainingFrame,
186 PRInt32 aOffset,
187 PRBool aPreferNext,
188 nsPoint& aOutPoint)
190 nsIFrame *textFrame = nsnull;
191 PRInt32 outOffset;
192 aContainingFrame->GetChildFrameContainingOffset(aOffset, aPreferNext, &outOffset, &textFrame);
193 if (!textFrame) {
194 return nsnull;
197 textFrame->GetPointFromOffset(aOffset, &aOutPoint);
198 return textFrame;
202 * Given an offset, the x, y, width, and height values are filled appropriately.
204 nsresult nsTextAccessibleWrap::GetCharacterExtents(PRInt32 aStartOffset, PRInt32 aEndOffset,
205 PRInt32* aX, PRInt32* aY,
206 PRInt32* aWidth, PRInt32* aHeight)
208 *aX = *aY = *aWidth = *aHeight = 0;
209 nsPresContext *presContext = GetPresContext();
210 NS_ENSURE_TRUE(presContext, NS_ERROR_FAILURE);
212 nsIFrame *frame = GetFrame();
213 NS_ENSURE_TRUE(frame, NS_ERROR_FAILURE);
215 nsIWidget *widget = frame->GetWindow();
216 NS_ENSURE_TRUE(widget, NS_ERROR_FAILURE);
218 nsPoint startPoint, endPoint;
219 nsIFrame *startFrame = GetPointFromOffset(frame, aStartOffset, PR_TRUE, startPoint);
220 nsIFrame *endFrame = GetPointFromOffset(frame, aEndOffset, PR_FALSE, endPoint);
221 if (!startFrame || !endFrame) {
222 return E_FAIL;
225 nsRect sum(0, 0, 0, 0);
226 nsIFrame *iter = startFrame;
227 nsIFrame *stopLoopFrame = endFrame->GetNextContinuation();
228 for (; iter != stopLoopFrame; iter = iter->GetNextContinuation()) {
229 nsRect rect = iter->GetScreenRectExternal();
230 nscoord start = (iter == startFrame) ? presContext->AppUnitsToDevPixels(startPoint.x) : 0;
231 nscoord end = (iter == endFrame) ? presContext->AppUnitsToDevPixels(endPoint.x) :
232 rect.width;
233 rect.x += start;
234 rect.width = end - start;
235 sum.UnionRect(sum, rect);
238 *aX = sum.x;
239 *aY = sum.y;
240 *aWidth = sum.width;
241 *aHeight = sum.height;
243 return NS_OK;
246 STDMETHODIMP nsTextAccessibleWrap::get_fontFamily(
247 /* [retval][out] */ BSTR __RPC_FAR *aFontFamily)
249 __try {
250 *aFontFamily = NULL;
252 nsIFrame *frame = GetFrame();
253 nsCOMPtr<nsIPresShell> presShell = GetPresShell();
254 if (!frame || !presShell || !presShell->GetPresContext()) {
255 return E_FAIL;
258 nsCOMPtr<nsIRenderingContext> rc;
259 presShell->CreateRenderingContext(frame, getter_AddRefs(rc));
260 if (!rc) {
261 return E_FAIL;
264 const nsStyleFont *font = frame->GetStyleFont();
266 const nsStyleVisibility *visibility = frame->GetStyleVisibility();
268 if (NS_FAILED(rc->SetFont(font->mFont, visibility->mLangGroup,
269 presShell->GetPresContext()->GetUserFontSet()))) {
270 return E_FAIL;
273 nsCOMPtr<nsIDeviceContext> deviceContext;
274 rc->GetDeviceContext(*getter_AddRefs(deviceContext));
275 if (!deviceContext) {
276 return E_FAIL;
279 nsIFontMetrics *fm;
280 rc->GetFontMetrics(fm);
281 if (!fm) {
282 return E_FAIL;
285 nsAutoString fontFamily;
286 deviceContext->FirstExistingFont(fm->Font(), fontFamily);
287 if (fontFamily.IsEmpty())
288 return S_FALSE;
290 *aFontFamily = ::SysAllocStringLen(fontFamily.get(), fontFamily.Length());
291 if (!*aFontFamily)
292 return E_OUTOFMEMORY;
294 } __except(FilterA11yExceptions(::GetExceptionCode(), GetExceptionInformation())) { }
296 return S_OK;