1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
3 * This file is part of the LibreOffice project.
5 * This Source Code Form is subject to the terms of the Mozilla Public
6 * License, v. 2.0. If a copy of the MPL was not distributed with this
7 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
9 * This file incorporates work covered by the following license notice:
11 * Licensed to the Apache Software Foundation (ASF) under one or more
12 * contributor license agreements. See the NOTICE file distributed
13 * with this work for additional information regarding copyright
14 * ownership. The ASF licenses this file to you under the Apache
15 * License, Version 2.0 (the "License"); you may not use this file
16 * except in compliance with the License. You may obtain a copy of
17 * the License at http://www.apache.org/licenses/LICENSE-2.0 .
21 #include <sal/config.h>
22 #include <sal/log.hxx>
24 #include <cppuhelper/basemutex.hxx>
25 #include <cppuhelper/compbase.hxx>
27 #include <tools/diagnose_ex.h>
28 #include <vcl/svapp.hxx>
30 #include <factory.hxx>
32 #include <salinst.hxx>
33 #include <salsession.hxx>
35 #include <com/sun/star/frame/XSessionManagerClient.hpp>
36 #include <com/sun/star/lang/XMultiServiceFactory.hpp>
37 #include <com/sun/star/frame/XSessionManagerListener2.hpp>
41 using namespace com::sun::star::uno
;
42 using namespace com::sun::star::lang
;
43 using namespace com::sun::star::frame
;
45 SalSession::~SalSession()
50 private cppu::BaseMutex
,
51 public cppu::WeakComponentImplHelper
< XSessionManagerClient
>
55 css::uno::Reference
< XSessionManagerListener
> m_xListener
;
56 bool m_bInteractionRequested
;
57 bool m_bInteractionDone
;
60 explicit Listener( const css::uno::Reference
< XSessionManagerListener
>& xListener
)
61 : m_xListener( xListener
),
62 m_bInteractionRequested( false ),
63 m_bInteractionDone( false ),
68 std::vector
< Listener
> m_aListeners
;
69 std::unique_ptr
< SalSession
> m_xSession
;
70 bool m_bInteractionRequested
;
71 bool m_bInteractionGranted
;
72 bool m_bInteractionDone
;
75 static void SalSessionEventProc( void* pData
, SalSessionEvent
* pEvent
);
77 virtual ~VCLSession() override
{}
79 virtual void SAL_CALL
addSessionManagerListener( const css::uno::Reference
< XSessionManagerListener
>& xListener
) override
;
80 virtual void SAL_CALL
removeSessionManagerListener( const css::uno::Reference
< XSessionManagerListener
>& xListener
) override
;
81 virtual void SAL_CALL
queryInteraction( const css::uno::Reference
< XSessionManagerListener
>& xListener
) override
;
82 virtual void SAL_CALL
interactionDone( const css::uno::Reference
< XSessionManagerListener
>& xListener
) override
;
83 virtual void SAL_CALL
saveDone( const css::uno::Reference
< XSessionManagerListener
>& xListener
) override
;
84 virtual sal_Bool SAL_CALL
cancelShutdown() override
;
86 void SAL_CALL
disposing() override
;
88 void callSaveRequested( bool bShutdown
);
89 void callShutdownCancelled();
90 void callInteractionGranted( bool bGranted
);
97 VCLSession::VCLSession()
98 : cppu::WeakComponentImplHelper
< XSessionManagerClient
>( m_aMutex
),
99 m_xSession( ImplGetSVData()->mpDefInst
->CreateSalSession() ),
100 m_bInteractionRequested( false ),
101 m_bInteractionGranted( false ),
102 m_bInteractionDone( false ),
105 SAL_INFO("vcl.se", "VCLSession::VCLSession" );
108 m_xSession
->SetCallback( SalSessionEventProc
, this );
111 void VCLSession::callSaveRequested( bool bShutdown
)
113 SAL_INFO("vcl.se", "VCLSession::callSaveRequested" );
115 std::vector
< Listener
> aListeners
;
117 osl::MutexGuard
aGuard( m_aMutex
);
118 // reset listener states
119 for (auto & listener
: m_aListeners
) {
120 listener
.m_bSaveDone
= listener
.m_bInteractionRequested
= listener
.m_bInteractionDone
= false;
123 // copy listener vector since calling a listener may remove it.
124 aListeners
= m_aListeners
;
125 // set back interaction state
127 m_bInteractionDone
= false;
128 // without session we assume UI is always possible,
129 // so it was requested and granted
130 m_bInteractionRequested
= m_bInteractionGranted
= !m_xSession
;
132 // answer the session manager even if no listeners available anymore
133 SAL_WARN_IF( aListeners
.empty(), "vcl.se", "saveRequested but no listeners !" );
135 SAL_INFO("vcl.se.debug", " aListeners.empty() = " << (aListeners
.empty() ? "true" : "false") <<
136 ", bShutdown = " << (bShutdown
? "true" : "false"));
137 if( aListeners
.empty() )
140 m_xSession
->saveDone();
145 SolarMutexReleaser aReleaser
;
146 for (auto const & listener
: aListeners
)
147 listener
.m_xListener
->doSave( bShutdown
, false/*bCancelable*/ );
150 void VCLSession::callInteractionGranted( bool bInteractionGranted
)
152 SAL_INFO("vcl.se", "VCLSession::callInteractionGranted" );
154 std::vector
< Listener
> aListeners
;
156 osl::MutexGuard
aGuard( m_aMutex
);
157 // copy listener vector since calling a listener may remove it.
158 for (auto const & listener
: m_aListeners
)
159 if( listener
.m_bInteractionRequested
)
160 aListeners
.push_back( listener
);
162 m_bInteractionGranted
= bInteractionGranted
;
164 // answer the session manager even if no listeners available anymore
165 SAL_WARN_IF( aListeners
.empty(), "vcl.se", "interactionGranted but no listeners !" );
167 SAL_INFO("vcl.se.debug", " aListeners.empty() = " << (aListeners
.empty() ? "true" : "false") <<
168 ", bInteractionGranted = " << (bInteractionGranted
? "true" : "false"));
169 if( aListeners
.empty() )
172 m_xSession
->interactionDone();
177 SolarMutexReleaser aReleaser
;
178 for (auto const & listener
: aListeners
)
179 listener
.m_xListener
->approveInteraction( bInteractionGranted
);
182 void VCLSession::callShutdownCancelled()
184 SAL_INFO("vcl.se", "VCLSession::callShutdownCancelled");
186 std::vector
< Listener
> aListeners
;
188 osl::MutexGuard
aGuard( m_aMutex
);
189 // copy listener vector since calling a listener may remove it.
190 aListeners
= m_aListeners
;
191 // set back interaction state
192 m_bInteractionRequested
= m_bInteractionDone
= m_bInteractionGranted
= false;
195 SolarMutexReleaser aReleaser
;
196 for (auto const & listener
: aListeners
)
197 listener
.m_xListener
->shutdownCanceled();
200 void VCLSession::callQuit()
202 SAL_INFO("vcl.se", "VCLSession::callQuit");
204 std::vector
< Listener
> aListeners
;
206 osl::MutexGuard
aGuard( m_aMutex
);
207 // copy listener vector since calling a listener may remove it.
208 aListeners
= m_aListeners
;
209 // set back interaction state
210 m_bInteractionRequested
= m_bInteractionDone
= m_bInteractionGranted
= false;
213 SolarMutexReleaser aReleaser
;
214 for (auto const & listener
: aListeners
)
216 css::uno::Reference
< XSessionManagerListener2
> xListener2( listener
.m_xListener
, UNO_QUERY
);
217 if( xListener2
.is() )
218 xListener2
->doQuit();
222 void VCLSession::SalSessionEventProc( void* pData
, SalSessionEvent
* pEvent
)
224 SAL_INFO("vcl.se", "VCLSession::SalSessionEventProc");
226 VCLSession
* pThis
= static_cast< VCLSession
* >( pData
);
227 switch( pEvent
->m_eType
)
231 SAL_INFO("vcl.se.debug", " EventProcType = Interaction");
232 SalSessionInteractionEvent
* pIEv
= static_cast<SalSessionInteractionEvent
*>(pEvent
);
233 pThis
->callInteractionGranted( pIEv
->m_bInteractionGranted
);
238 SAL_INFO("vcl.se.debug", " EventProcType = SaveRequest");
239 SalSessionSaveRequestEvent
* pSEv
= static_cast<SalSessionSaveRequestEvent
*>(pEvent
);
240 pThis
->callSaveRequested( pSEv
->m_bShutdown
);
244 SAL_INFO("vcl.se.debug", " EventProcType = ShutdownCancel");
245 pThis
->callShutdownCancelled();
248 SAL_INFO("vcl.se.debug", " EventProcType = Quit");
254 void SAL_CALL
VCLSession::addSessionManagerListener( const css::uno::Reference
<XSessionManagerListener
>& xListener
)
256 SAL_INFO("vcl.se", "VCLSession::addSessionManagerListener" );
258 osl::MutexGuard
aGuard( m_aMutex
);
260 SAL_INFO("vcl.se.debug", " m_aListeners.size() = " << m_aListeners
.size() );
261 m_aListeners
.emplace_back( xListener
);
264 void SAL_CALL
VCLSession::removeSessionManagerListener( const css::uno::Reference
<XSessionManagerListener
>& xListener
)
266 SAL_INFO("vcl.se", "VCLSession::removeSessionManagerListener" );
268 osl::MutexGuard
aGuard( m_aMutex
);
270 SAL_INFO("vcl.se.debug", " m_aListeners.size() = " << m_aListeners
.size() );
272 m_aListeners
.erase(std::remove_if(m_aListeners
.begin(), m_aListeners
.end(), [&](Listener
& listener
) {return xListener
== listener
.m_xListener
;}), m_aListeners
.end());
275 void SAL_CALL
VCLSession::queryInteraction( const css::uno::Reference
<XSessionManagerListener
>& xListener
)
277 SAL_INFO("vcl.se", "VCLSession::queryInteraction");
279 SAL_INFO("vcl.se.debug", " m_bInteractionGranted = " << (m_bInteractionGranted
? "true" : "false") <<
280 ", m_bInteractionRequested = "<< (m_bInteractionRequested
? "true" : "false"));
281 if( m_bInteractionGranted
)
283 if( m_bInteractionDone
)
284 xListener
->approveInteraction( false );
286 xListener
->approveInteraction( true );
290 osl::MutexGuard
aGuard( m_aMutex
);
291 if( ! m_bInteractionRequested
)
293 m_xSession
->queryInteraction();
294 m_bInteractionRequested
= true;
296 for (auto & listener
: m_aListeners
)
298 if( listener
.m_xListener
== xListener
)
300 SAL_INFO("vcl.se.debug", " listener.m_xListener == xListener");
301 listener
.m_bInteractionRequested
= true;
302 listener
.m_bInteractionDone
= false;
307 void SAL_CALL
VCLSession::interactionDone( const css::uno::Reference
< XSessionManagerListener
>& xListener
)
309 SAL_INFO("vcl.se", "VCLSession::interactionDone");
311 osl::MutexGuard
aGuard( m_aMutex
);
312 int nRequested
= 0, nDone
= 0;
313 for (auto & listener
: m_aListeners
)
315 if( listener
.m_bInteractionRequested
)
318 if( xListener
== listener
.m_xListener
)
319 listener
.m_bInteractionDone
= true;
321 if( listener
.m_bInteractionDone
)
325 SAL_INFO("vcl.se.debug", " nDone = " << nDone
<<
326 ", nRequested =" << nRequested
);
327 if( nDone
== nRequested
&& nDone
> 0 )
329 m_bInteractionDone
= true;
331 m_xSession
->interactionDone();
335 void SAL_CALL
VCLSession::saveDone( const css::uno::Reference
< XSessionManagerListener
>& xListener
)
337 SAL_INFO("vcl.se", "VCLSession::saveDone");
339 osl::MutexGuard
aGuard( m_aMutex
);
341 bool bSaveDone
= true;
342 for (auto & listener
: m_aListeners
)
344 if( listener
.m_xListener
== xListener
)
345 listener
.m_bSaveDone
= true;
346 if( ! listener
.m_bSaveDone
)
350 SAL_INFO("vcl.se.debug", " bSaveDone = " << (bSaveDone
? "true" : "false"));
352 if( bSaveDone
&& !m_bSaveDone
)
356 m_xSession
->saveDone();
360 sal_Bool SAL_CALL
VCLSession::cancelShutdown()
362 SAL_INFO("vcl.se", "VCLSession::cancelShutdown");
364 return m_xSession
&& m_xSession
->cancelShutdown();
367 void VCLSession::disposing() {
368 SAL_INFO("vcl.se", "VCLSession::disposing");
370 std::vector
<Listener
> vector
;
372 osl::MutexGuard
g(m_aMutex
);
373 vector
.swap(m_aListeners
);
375 css::lang::EventObject
src(static_cast<OWeakObject
*>(this));
376 for (auto const & listener
: vector
) {
378 listener
.m_xListener
->disposing(src
);
379 SAL_INFO("vcl.se.debug", " call Listener disposing");
380 } catch (css::uno::RuntimeException
&) {
381 TOOLS_WARN_EXCEPTION("vcl.se", "ignoring");
386 // service implementation
388 OUString
vcl_session_getImplementationName()
390 SAL_INFO("vcl.se", "vcl_session_getImplementationName");
392 return "com.sun.star.frame.VCLSessionManagerClient";
395 Sequence
< OUString
> vcl_session_getSupportedServiceNames()
397 SAL_INFO("vcl.se", "vcl_session_getSupportedServiceNames");
399 Sequence
< OUString
> aRet
{ "com.sun.star.frame.SessionManagerClient" };
403 css::uno::Reference
< XInterface
> vcl_session_createInstance( SAL_UNUSED_PARAMETER
const css::uno::Reference
< XMultiServiceFactory
> & )
405 SAL_INFO("vcl.se", "vcl_session_createInstance");
407 return static_cast< cppu::OWeakObject
* >(new VCLSession
);
410 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */