1 /**************************************************************
2  *
3  * Licensed to the Apache Software Foundation (ASF) under one
4  * or more contributor license agreements.  See the NOTICE file
5  * distributed with this work for additional information
6  * regarding copyright ownership.  The ASF licenses this file
7  * to you under the Apache License, Version 2.0 (the
8  * "License"); you may not use this file except in compliance
9  * with the License.  You may obtain a copy of the License at
10  *
11  *   http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing,
14  * software distributed under the License is distributed on an
15  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16  * KIND, either express or implied.  See the License for the
17  * specific language governing permissions and limitations
18  * under the License.
19  *
20  *************************************************************/
21 
22 
23 
24 // MARKER(update_precomp.py): autogen include statement, do not remove
25 #include "precompiled_framework.hxx"
26 #include <dispatch/helpagentdispatcher.hxx>
27 #include <threadhelp/readguard.hxx>
28 #include <threadhelp/writeguard.hxx>
29 #include <com/sun/star/awt/XWindow2.hpp>
30 #include <com/sun/star/awt/PosSize.hpp>
31 #include <com/sun/star/awt/Size.hpp>
32 #include <com/sun/star/awt/Rectangle.hpp>
33 #include <toolkit/helper/vclunohelper.hxx>
34 #include <svtools/helpopt.hxx>
35 #include <vcl/svapp.hxx>
36 #include <vcl/help.hxx>
37 
38 namespace css = ::com::sun::star;
39 
40 //........................................................................
41 namespace framework
42 {
43 
44 //-----------------------------------------------
45 DEFINE_XINTERFACE_4(HelpAgentDispatcher                         ,
46                     OWeakObject                                 ,
47                     DIRECT_INTERFACE (css::lang::XTypeProvider ),
48                     DIRECT_INTERFACE (css::frame::XDispatch    ),
49                     DIRECT_INTERFACE (css::awt::XWindowListener),
50                     DIRECT_INTERFACE (css::lang::XEventListener))
51 
52 //-----------------------------------------------
53 DEFINE_XTYPEPROVIDER_2(HelpAgentDispatcher     ,
54                        css::lang::XTypeProvider,
55                        css::frame::XDispatch   )
56 
57 //--------------------------------------------------------------------
58 HelpAgentDispatcher::HelpAgentDispatcher( const css::uno::Reference< css::frame::XFrame >& xParentFrame)
59 	: ThreadHelpBase	(&Application::GetSolarMutex())
60 	, m_sCurrentURL		(							  )
61 	, m_xContainerWindow(							  )
62 	, m_xAgentWindow	(							  )
63 	, m_aTimer			(							  )
64 	, m_xSelfHold		(							  )
65 {
66 	// It's required that this class has to be contructed with a valid frame.
67 	// And "valid" means: the frame must already bound to a valid container window.
68 	m_xContainerWindow = xParentFrame->getContainerWindow();
69 }
70 
71 //--------------------------------------------------------------------
72 HelpAgentDispatcher::~HelpAgentDispatcher()
73 {
74 	implts_stopTimer();
75 	implts_ignoreCurrentURL();
76 
77 	// Needed ... because it was create as "new VCLWindow()" ! Such windows must be disposed explicitly.
78 	css::uno::Reference< css::lang::XComponent > xAgentWindow(m_xAgentWindow, css::uno::UNO_QUERY);
79 	if (xAgentWindow.is())
80 		xAgentWindow->dispose();
81 }
82 
83 //--------------------------------------------------------------------
84 void SAL_CALL HelpAgentDispatcher::dispatch(const css::util::URL& 								   aURL ,
85 											const css::uno::Sequence< css::beans::PropertyValue >&)
86 	throw(css::uno::RuntimeException)
87 {
88 	// silently drop the request if the new URL was marked to be ignored next time.
89 	sal_Int32 nAllowedToIgnore = SvtHelpOptions().getAgentIgnoreURLCounter(aURL.Complete);
90 	if (nAllowedToIgnore < 1)
91 		return;
92 
93 	// stop the expiration timer for the old URL
94 	// The timer will add the old URL to the list of ignorable URLs.
95 	// So m_sCurrentURL must be set AFTER the timer was stopped !!!
96 	implts_stopTimer();
97 
98 	// SAFE ->
99 	WriteGuard aWriteLock(m_aLock);
100 	m_sCurrentURL = aURL.Complete;
101 	aWriteLock.unlock();
102 	// <- SAFE
103 
104 	// start the expiration timer for the new URL
105 	implts_startTimer();
106 
107 	// make sure the agent window is shown
108 	implts_showAgentWindow();
109 }
110 
111 //--------------------------------------------------------------------
112 void SAL_CALL HelpAgentDispatcher::addStatusListener(const css::uno::Reference< css::frame::XStatusListener >&,
113 													 const css::util::URL&)
114 	throw(css::uno::RuntimeException)
115 {
116 	// no status available
117 }
118 
119 //--------------------------------------------------------------------
120 void SAL_CALL HelpAgentDispatcher::removeStatusListener(const css::uno::Reference< css::frame::XStatusListener >&,
121 														const css::util::URL&)
122 	throw(css::uno::RuntimeException)
123 {
124 	// no status available
125 }
126 
127 //--------------------------------------------------------------------
128 void SAL_CALL HelpAgentDispatcher::windowResized(const css::awt::WindowEvent&)
129 	throw(css::uno::RuntimeException)
130 {
131 	implts_positionAgentWindow();
132 }
133 
134 //--------------------------------------------------------------------
135 void SAL_CALL HelpAgentDispatcher::windowMoved(const css::awt::WindowEvent&)
136 	throw(css::uno::RuntimeException)
137 {
138 	implts_positionAgentWindow();
139 }
140 
141 //--------------------------------------------------------------------
142 void SAL_CALL HelpAgentDispatcher::windowShown(const css::lang::EventObject&)
143 	throw(css::uno::RuntimeException)
144 {
145 	implts_showAgentWindow();
146 }
147 
148 //--------------------------------------------------------------------
149 void SAL_CALL HelpAgentDispatcher::windowHidden(const css::lang::EventObject&)
150 	throw(css::uno::RuntimeException)
151 {
152 	implts_hideAgentWindow();
153 }
154 
155 //--------------------------------------------------------------------
156 void SAL_CALL HelpAgentDispatcher::disposing(const css::lang::EventObject& aEvent)
157 	throw(css::uno::RuntimeException)
158 {
159 	// SAFE ->
160 	WriteGuard aWriteLock(m_aLock);
161 
162 	// Already disposed ?!
163 	if (! m_xContainerWindow.is())
164 		return;
165 	// Wrong broadcaster ?!
166 	if (aEvent.Source != m_xContainerWindow)
167 		return;
168 
169     css::uno::Reference< css::uno::XInterface > xSelfHoldUntilMethodEnds(static_cast< css::frame::XDispatch* >(this), css::uno::UNO_QUERY_THROW);
170 	m_xSelfHold.clear();
171 
172 	aWriteLock.unlock();
173 	// <- SAFE
174 
175 	implts_stopTimer();
176 	implts_hideAgentWindow();
177 	implts_ignoreCurrentURL();
178 
179 	// SAFE ->
180 	aWriteLock.lock();
181 	m_xContainerWindow.clear();
182 	css::uno::Reference< css::lang::XComponent > xAgentWindow(m_xAgentWindow, css::uno::UNO_QUERY);
183 	m_xAgentWindow.clear();
184 	aWriteLock.unlock();
185 	// <- SAFE
186 
187 	// Needed ... because it was create as "new VCLWindow()" ! Such windows must be disposed explicitly.
188 	if (xAgentWindow.is())
189 		xAgentWindow->dispose();
190 }
191 
192 //--------------------------------------------------------------------
193 void HelpAgentDispatcher::helpRequested()
194 {
195 	implts_stopTimer();
196 	implts_hideAgentWindow();
197 	implts_acceptCurrentURL();
198 }
199 
200 //-----------------------------------------------
201 void HelpAgentDispatcher::closeAgent()
202 {
203 	implts_stopTimer();
204 	implts_hideAgentWindow();
205 	implts_ignoreCurrentURL();
206 }
207 
208 //--------------------------------------------------------------------
209 void HelpAgentDispatcher::implts_acceptCurrentURL()
210 {
211 	// SAFE ->
212 	WriteGuard aWriteLock(m_aLock);
213 
214 	::rtl::OUString sAcceptedURL  = m_sCurrentURL;
215 					m_sCurrentURL = ::rtl::OUString();
216 
217 	aWriteLock.unlock();
218 	// <- SAFE
219 
220 	// We must make sure that this URL isn't marked as ignored by the user.
221 	// Otherwhise the user wont see the corresponding help content in the future.
222 	SvtHelpOptions().resetAgentIgnoreURLCounter(sAcceptedURL);
223 
224 	// show the right help content
225 	// SOLAR SAFE ->
226 	{
227 		::vos::OGuard aSolarLock(Application::GetSolarMutex());
228 		Help* pHelp = Application::GetHelp();
229 		if (pHelp)
230 			pHelp->Start(sAcceptedURL, NULL);
231 	}
232 	// <- SOLAR SAFE
233 }
234 
235 //--------------------------------------------------------------------
236 void HelpAgentDispatcher::implts_ignoreCurrentURL()
237 {
238 	// SAFE ->
239 	WriteGuard aWriteLock(m_aLock);
240 
241 	::rtl::OUString sIgnoredURL   = m_sCurrentURL;
242 					m_sCurrentURL = ::rtl::OUString();
243 
244 	aWriteLock.unlock();
245 	// <- SAFE
246 
247 	if (sIgnoredURL.getLength())
248 		SvtHelpOptions().decAgentIgnoreURLCounter(sIgnoredURL);
249 }
250 
251 //--------------------------------------------------------------------
252 void HelpAgentDispatcher::implts_stopTimer()
253 {
254 	// SAFE ->
255 	WriteGuard aWriteLock(m_aLock);
256 	m_xSelfHold.clear();
257 	aWriteLock.unlock();
258 	// <- SAFE
259 
260 	// SOLAR SAFE ->
261 	// Timer access needs no "own lock" ! It lives if we live ...
262 	// But it requires locking of the solar mutex ... because it's a vcl based timer.
263 	{
264 		::vos::OGuard aSolarLock(Application::GetSolarMutex());
265 		if (! m_aTimer.IsActive())
266 			return;
267 		m_aTimer.Stop();
268 	}
269 	// <- SOLAR SAFE
270 }
271 
272 //--------------------------------------------------------------------
273 void HelpAgentDispatcher::implts_startTimer()
274 {
275 	// SOLAR SAFE ->
276 	// Timer access needs no "own lock" ! It lives if we live ...
277 	// But it requires locking of the solar mutex ... because it's a vcl based timer.
278 	{
279 		::vos::OGuard aSolarLock(Application::GetSolarMutex());
280 		if (m_aTimer.IsActive())
281 			return;
282 	}
283 	// <- SOLAR SAFE
284 
285 	// SAFE ->
286 	// Timer uses pointer to this help agent dispatcher ...
287 	// But normaly we are ref counted. So we must make sure that this
288 	// dispatcher isn't killed during the timer runs .-)
289 	WriteGuard aWriteLock(m_aLock);
290 	m_xSelfHold = css::uno::Reference< css::uno::XInterface >(static_cast< css::frame::XDispatch* >(this), css::uno::UNO_QUERY_THROW);
291 	aWriteLock.unlock();
292 	// <- SAFE
293 
294 	sal_Int32 nTime = SvtHelpOptions().GetHelpAgentTimeoutPeriod();
295 
296 	// SOLAR SAFE ->
297 	// Timer access needs no "own lock" ! It lives if we live ...
298 	// But it requires locking of the solar mutex ... because it's a vcl based timer.
299 	{
300 		::vos::OGuard aSolarLock(Application::GetSolarMutex());
301 		m_aTimer.SetTimeout(nTime*1000); // sec => ms !
302 		m_aTimer.Start();
303 	}
304 }
305 
306 //-----------------------------------------------
307 IMPL_LINK(HelpAgentDispatcher, implts_timerExpired, void*,)
308 {
309     // This method is called by using a pointer to us.
310     // But we must be aware that we can be destroyed hardly
311     // if our uno reference will be gone!
312     // => Hold this object alive till this method finish its work.
313 	// SAFE ->
314 	WriteGuard aWriteLock(m_aLock);
315     css::uno::Reference< css::uno::XInterface > xSelfHoldUntilMethodEnds(static_cast< css::frame::XDispatch* >(this), css::uno::UNO_QUERY_THROW);
316 	m_xSelfHold.clear();
317 	aWriteLock.unlock();
318 	// <- SAFE
319 
320 	implts_hideAgentWindow();
321 	implts_ignoreCurrentURL();
322 
323 	return 0;
324 }
325 
326 //--------------------------------------------------------------------
327 void HelpAgentDispatcher::implts_showAgentWindow()
328 {
329 	// SAFE ->
330 	ReadGuard aReadLock(m_aLock);
331 	css::uno::Reference< css::awt::XWindow2 > xContainerWindow(m_xContainerWindow, css::uno::UNO_QUERY_THROW);
332 	aReadLock.unlock();
333 	// <- SAFE
334 
335 	css::uno::Reference< css::awt::XWindow > xAgentWindow = implts_ensureAgentWindow();
336 
337 	if (
338 		(xContainerWindow.is() 		  ) &&
339 		(xAgentWindow.is() 		  	  ) &&
340 		(xContainerWindow->isVisible())
341 	   )
342 	{
343 		// make sure that agent window resists at the right place .-)
344 		implts_positionAgentWindow();
345 		xAgentWindow->setVisible(sal_True);
346 	}
347 }
348 
349 //--------------------------------------------------------------------
350 void HelpAgentDispatcher::implts_hideAgentWindow()
351 {
352 	css::uno::Reference< css::awt::XWindow > xAgentWindow = implts_ensureAgentWindow();
353 	if (xAgentWindow.is())
354 		xAgentWindow->setVisible(sal_False);
355 }
356 
357 //--------------------------------------------------------------------
358 void HelpAgentDispatcher::implts_positionAgentWindow()
359 {
360 	// SAFE ->
361 	ReadGuard aReadLock(m_aLock);
362 	css::uno::Reference< css::awt::XWindow > xContainerWindow = m_xContainerWindow;
363 	aReadLock.unlock();
364 	// <- SAFE
365 
366 	css::uno::Reference< css::awt::XWindow > xAgentWindow = implts_ensureAgentWindow();
367 	if (
368 		(! xContainerWindow.is())  ||
369 		(! xAgentWindow.is()    )
370 	   )
371 		return;
372 
373 		  ::svt::HelpAgentWindow* pAgentWindow   = (::svt::HelpAgentWindow*)VCLUnoHelper::GetWindow(xAgentWindow);
374 	const css::awt::Rectangle 	  aContainerSize = xContainerWindow->getPosSize();
375 	const Size                	  aAgentSize     = pAgentWindow->getPreferredSizePixel();
376 
377     sal_Int32 nW = aAgentSize.Width() ;
378     sal_Int32 nH = aAgentSize.Height();
379 
380     if (nW < 1)
381         nW = 100;
382     if (nH < 1)
383         nH = 100;
384 
385 	sal_Int32 nX = aContainerSize.Width  - nW;
386 	sal_Int32 nY = aContainerSize.Height - nH;
387 
388 	// TODO: use a surrogate if the container window is too small to contain the full-sized agent window
389 	xAgentWindow->setPosSize(nX, nY, nW, nH, css::awt::PosSize::POSSIZE);
390 }
391 
392 //--------------------------------------------------------------------
393 css::uno::Reference< css::awt::XWindow > HelpAgentDispatcher::implts_ensureAgentWindow()
394 {
395 	// SAFE ->
396 	ReadGuard aReadLock(m_aLock);
397 	if (m_xAgentWindow.is())
398 		return m_xAgentWindow;
399 	css::uno::Reference< css::awt::XWindow > xContainerWindow = m_xContainerWindow;
400 	aReadLock.unlock();
401 	// <- SAFE
402 
403 	if (!xContainerWindow.is())
404 		return css::uno::Reference< css::awt::XWindow >();
405 
406 	::svt::HelpAgentWindow* pAgentWindow = 0;
407 	// SOLAR SAFE ->
408 	{
409 		::vos::OGuard aSolarLock(Application::GetSolarMutex());
410 		// create the agent window
411 		Window*	pContainerWindow = VCLUnoHelper::GetWindow(xContainerWindow);
412 				pAgentWindow 	 = new ::svt::HelpAgentWindow(pContainerWindow);
413 		pAgentWindow->setCallback(this);
414 	}
415 	// <- SOLAR SAFE
416 
417 	// SAFE ->
418 	WriteGuard aWriteLock(m_aLock);
419 	m_xAgentWindow = VCLUnoHelper::GetInterface(pAgentWindow);
420 	css::uno::Reference< css::awt::XWindow > xAgentWindow = m_xAgentWindow;
421 	aWriteLock.unlock();
422 	// <- SAFE
423 
424 	// add as window listener to the container window so we can maintain the property position of the agent window
425 	xContainerWindow->addWindowListener(this);
426 
427 	// SOLAR SAFE ->
428 	{
429 		::vos::OGuard aSolarLock(Application::GetSolarMutex());
430 		// establish callback for our internal used timer.
431 		// Note: Its only active, if the timer will be started ...
432 		m_aTimer.SetTimeoutHdl(LINK(this, HelpAgentDispatcher, implts_timerExpired));
433 	}
434 	// <- SOLAR SAFE
435 
436 	return xAgentWindow;
437 }
438 
439 } // namespace framework
440 
441