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_desktop.hxx"
26
27 #include "sal/config.h"
28
29 #include <cstddef>
30 #include <limits>
31 #include <map>
32 #include <memory>
33 #include <utility>
34 #include <vector>
35
36
37 #include "boost/optional.hpp"
38 #include "com/sun/star/awt/Rectangle.hpp"
39 #include "com/sun/star/awt/WindowAttribute.hpp"
40 #include "com/sun/star/awt/WindowClass.hpp"
41 #include "com/sun/star/awt/WindowDescriptor.hpp"
42 #include "com/sun/star/awt/XToolkit.hpp"
43 #include "com/sun/star/awt/XWindow.hpp"
44 #include "com/sun/star/awt/XWindowPeer.hpp"
45 #include "com/sun/star/beans/NamedValue.hpp"
46 #include "com/sun/star/beans/Optional.hpp"
47 #include "com/sun/star/beans/PropertyValue.hpp"
48 #include "com/sun/star/beans/XPropertySet.hpp"
49 #include "com/sun/star/container/XNameAccess.hpp"
50 #include "com/sun/star/container/XNameContainer.hpp"
51 #include "com/sun/star/deployment/DeploymentException.hpp"
52 #include "com/sun/star/deployment/UpdateInformationProvider.hpp"
53 #include "com/sun/star/deployment/XPackage.hpp"
54 #include "com/sun/star/deployment/XExtensionManager.hpp"
55 #include "com/sun/star/deployment/ExtensionManager.hpp"
56 #include "com/sun/star/deployment/XUpdateInformationProvider.hpp"
57 #include "com/sun/star/frame/XDesktop.hpp"
58 #include "com/sun/star/frame/XDispatch.hpp"
59 #include "com/sun/star/frame/XDispatchProvider.hpp"
60 #include "com/sun/star/lang/IllegalArgumentException.hpp"
61 #include "com/sun/star/lang/XMultiComponentFactory.hpp"
62 #include "com/sun/star/lang/XSingleServiceFactory.hpp"
63 #include "com/sun/star/system/SystemShellExecuteFlags.hpp"
64 #include "com/sun/star/system/SystemShellExecute.hpp"
65 #include "com/sun/star/task/XAbortChannel.hpp"
66 #include "com/sun/star/task/XJob.hpp"
67 #include "com/sun/star/ucb/CommandAbortedException.hpp"
68 #include "com/sun/star/ucb/CommandFailedException.hpp"
69 #include "com/sun/star/ucb/XCommandEnvironment.hpp"
70 #include "com/sun/star/uno/Any.hxx"
71 #include "com/sun/star/uno/Exception.hpp"
72 #include "com/sun/star/uno/Reference.hxx"
73 #include "com/sun/star/uno/RuntimeException.hpp"
74 #include "com/sun/star/uno/Sequence.hxx"
75 #include "com/sun/star/uno/XInterface.hpp"
76 #include "com/sun/star/util/URL.hpp"
77 #include "com/sun/star/util/XChangesBatch.hpp"
78 #include "com/sun/star/util/XURLTransformer.hpp"
79 #include "com/sun/star/xml/dom/XElement.hpp"
80 #include "com/sun/star/xml/dom/XNode.hpp"
81 #include "osl/diagnose.h"
82 #include "rtl/bootstrap.hxx"
83 #include "rtl/ref.hxx"
84 #include "rtl/string.h"
85 #include "rtl/ustrbuf.hxx"
86 #include "rtl/ustring.h"
87 #include "rtl/ustring.hxx"
88 #include "sal/types.h"
89 #include "svtools/svlbitm.hxx"
90 #include "svtools/svlbox.hxx"
91 #include <svtools/controldims.hrc>
92 #include "svx/checklbx.hxx"
93 #include "tools/gen.hxx"
94 #include "tools/link.hxx"
95 #include "tools/resid.hxx"
96 #include "tools/resmgr.hxx"
97 #include "tools/solar.h"
98 #include "tools/string.hxx"
99 #include "vcl/button.hxx"
100 #include "vcl/dialog.hxx"
101 #include "vcl/fixed.hxx"
102 #include "vcl/image.hxx"
103 #include "vcl/msgbox.hxx"
104 #include "vcl/svapp.hxx"
105 #include "vos/mutex.hxx"
106
107 #include "comphelper/processfactory.hxx"
108
109 #include "dp_dependencies.hxx"
110 #include "dp_descriptioninfoset.hxx"
111 #include "dp_identifier.hxx"
112 #include "dp_version.hxx"
113 #include "dp_misc.h"
114 #include "dp_update.hxx"
115
116 #include "dp_gui.h"
117 #include "dp_gui.hrc"
118 #include "dp_gui_thread.hxx"
119 #include "dp_gui_updatedata.hxx"
120 #include "dp_gui_updatedialog.hxx"
121 #include "dp_gui_shared.hxx"
122 #include "dp_gui_system.hxx"
123
124 class KeyEvent;
125 class MouseEvent;
126 class Window;
127 namespace com { namespace sun { namespace star { namespace uno {
128 class XComponentContext;
129 } } } }
130
131 using namespace ::com::sun::star;
132 using dp_gui::UpdateDialog;
133
134 namespace {
135
136 static sal_Unicode const LF = 0x000A;
137 static sal_Unicode const CR = 0x000D;
138 static const sal_uInt16 CMD_ENABLE_UPDATE = 1;
139 static const sal_uInt16 CMD_IGNORE_UPDATE = 2;
140 static const sal_uInt16 CMD_IGNORE_ALL_UPDATES = 3;
141
142 #define OUSTR(x) ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM(x) )
143
144 #define IGNORED_UPDATES OUSTR("/org.openoffice.Office.ExtensionManager/ExtensionUpdateData/IgnoredUpdates")
145 #define PROPERTY_VERSION OUSTR("Version")
146
147 enum Kind { ENABLED_UPDATE, DISABLED_UPDATE, SPECIFIC_ERROR };
148
confineToParagraph(rtl::OUString const & text)149 rtl::OUString confineToParagraph(rtl::OUString const & text) {
150 // Confine arbitrary text to a single paragraph in a dp_gui::AutoScrollEdit.
151 // This assumes that U+000A and U+000D are the only paragraph separators in
152 // a dp_gui::AutoScrollEdit, and that replacing them with a single space
153 // each is acceptable:
154 return text.replace(LF, ' ').replace(CR, ' ');
155 }
156 }
157
158 struct UpdateDialog::DisabledUpdate {
159 rtl::OUString name;
160 uno::Sequence< rtl::OUString > unsatisfiedDependencies;
161 // We also want to show release notes and publisher for disabled updates
162 ::com::sun::star::uno::Reference< ::com::sun::star::xml::dom::XNode > aUpdateInfo;
163 sal_uInt16 m_nID;
164 };
165
166 struct UpdateDialog::SpecificError {
167 rtl::OUString name;
168 rtl::OUString message;
169 sal_uInt16 m_nID;
170 };
171
172 //------------------------------------------------------------------------------
173 struct UpdateDialog::IgnoredUpdate {
174 rtl::OUString sExtensionID;
175 rtl::OUString sVersion;
176 bool bRemoved;
177
178 IgnoredUpdate( const rtl::OUString &rExtensionID, const rtl::OUString &rVersion );
179 };
180
181 //------------------------------------------------------------------------------
IgnoredUpdate(const rtl::OUString & rExtensionID,const rtl::OUString & rVersion)182 UpdateDialog::IgnoredUpdate::IgnoredUpdate( const rtl::OUString &rExtensionID, const rtl::OUString &rVersion ):
183 sExtensionID( rExtensionID ),
184 sVersion( rVersion ),
185 bRemoved( false )
186 {}
187
188 //------------------------------------------------------------------------------
189 struct UpdateDialog::Index
190 {
191 Kind m_eKind;
192 bool m_bIgnored;
193 sal_uInt16 m_nID;
194 sal_uInt16 m_nIndex;
195 rtl::OUString m_aName;
196
197 Index( Kind theKind, sal_uInt16 nID, sal_uInt16 nIndex, const rtl::OUString &rName );
198 };
199
200 //------------------------------------------------------------------------------
Index(Kind theKind,sal_uInt16 nID,sal_uInt16 nIndex,const rtl::OUString & rName)201 UpdateDialog::Index::Index( Kind theKind, sal_uInt16 nID, sal_uInt16 nIndex, const rtl::OUString &rName ):
202 m_eKind( theKind ),
203 m_bIgnored( false ),
204 m_nID( nID ),
205 m_nIndex( nIndex ),
206 m_aName( rName )
207 {}
208
209 //------------------------------------------------------------------------------
210 //------------------------------------------------------------------------------
211 //------------------------------------------------------------------------------
212 class UpdateDialog::Thread: public dp_gui::Thread {
213 public:
214 Thread(
215 uno::Reference< uno::XComponentContext > const & context,
216 UpdateDialog & dialog,
217 const std::vector< uno::Reference< deployment::XPackage > > & vExtensionList);
218
219 void stop();
220
221 private:
222 Thread(UpdateDialog::Thread &); // not defined
223 void operator =(UpdateDialog::Thread &); // not defined
224
225 struct Entry {
226 explicit Entry(
227 uno::Reference< deployment::XPackage > const & thePackage,
228 rtl::OUString const & theVersion);
229
230 uno::Reference< deployment::XPackage > package;
231 rtl::OUString version;
232 //Indicates that the extension provides its own update URLs.
233 //If this is true, then we must not use the default update
234 //URL to find the update information.
235 bool bProvidesOwnUpdate;
236 uno::Reference< xml::dom::XNode > info;
237 UpdateDialog::DisabledUpdate disableUpdate;
238 dp_gui::UpdateData updateData;
239 };
240
241 // A multimap in case an extension is installed in "user", "shared" or "bundled"
242 typedef std::map< rtl::OUString, Entry > Map;
243
244 virtual ~Thread();
245
246 virtual void execute();
247 #if 0
248 void handleGeneralError(uno::Any const & exception) const;
249 #endif
250 void handleSpecificError(
251 uno::Reference< deployment::XPackage > const & package,
252 uno::Any const & exception) const;
253
254 uno::Sequence< uno::Reference< xml::dom::XElement > >
255 getUpdateInformation(
256 uno::Reference< deployment::XPackage > const & package,
257 uno::Sequence< rtl::OUString > const & urls,
258 rtl::OUString const & identifier) const;
259
260 void getOwnUpdateInformation(
261 uno::Reference< deployment::XPackage > const & package,
262 Map * map);
263
264 ::rtl::OUString getUpdateDisplayString(
265 dp_gui::UpdateData const & data, ::rtl::OUString const & version = ::rtl::OUString()) const;
266
267 void prepareUpdateData(
268 ::com::sun::star::uno::Reference< ::com::sun::star::xml::dom::XNode > const & updateInfo,
269 UpdateDialog::DisabledUpdate & out_du,
270 dp_gui::UpdateData & out_data) const;
271
272 bool update(
273 UpdateDialog::DisabledUpdate & du,
274 dp_gui::UpdateData & data) const;
275
276 uno::Reference< uno::XComponentContext > m_context;
277 UpdateDialog & m_dialog;
278 std::vector< uno::Reference< deployment::XPackage > > m_vExtensionList;
279 uno::Reference< deployment::XUpdateInformationProvider > m_updateInformation;
280 uno::Reference< task::XInteractionHandler > m_xInteractionHdl;
281
282 // guarded by Application::GetSolarMutex():
283 uno::Reference< task::XAbortChannel > m_abort;
284 bool m_stop;
285 };
286
Thread(uno::Reference<uno::XComponentContext> const & context,UpdateDialog & dialog,const std::vector<uno::Reference<deployment::XPackage>> & vExtensionList)287 UpdateDialog::Thread::Thread(
288 uno::Reference< uno::XComponentContext > const & context,
289 UpdateDialog & dialog,
290 const std::vector< uno::Reference< deployment::XPackage > > &vExtensionList):
291 m_context(context),
292 m_dialog(dialog),
293 m_vExtensionList(vExtensionList),
294 m_updateInformation(
295 deployment::UpdateInformationProvider::create(context)),
296 m_stop(false)
297 {
298 if( m_context.is() )
299 {
300 uno::Reference< lang::XMultiComponentFactory > xServiceManager( m_context->getServiceManager() );
301
302 if( xServiceManager.is() )
303 {
304 m_xInteractionHdl = uno::Reference< task::XInteractionHandler > (
305 xServiceManager->createInstanceWithContext( OUSTR( "com.sun.star.task.InteractionHandler" ), m_context),
306 uno::UNO_QUERY );
307 if ( m_xInteractionHdl.is() )
308 m_updateInformation->setInteractionHandler( m_xInteractionHdl );
309 }
310 }
311 }
312
stop()313 void UpdateDialog::Thread::stop() {
314 uno::Reference< task::XAbortChannel > abort;
315 {
316 vos::OGuard g(Application::GetSolarMutex());
317 abort = m_abort;
318 m_stop = true;
319 }
320 if (abort.is()) {
321 abort->sendAbort();
322 }
323 m_updateInformation->cancel();
324 }
325
Entry(uno::Reference<deployment::XPackage> const & thePackage,rtl::OUString const & theVersion)326 UpdateDialog::Thread::Entry::Entry(
327 uno::Reference< deployment::XPackage > const & thePackage,
328 rtl::OUString const & theVersion):
329
330 package(thePackage),
331 version(theVersion),
332 bProvidesOwnUpdate(false),
333 updateData(thePackage)
334 {
335 }
336
~Thread()337 UpdateDialog::Thread::~Thread()
338 {
339 if ( m_xInteractionHdl.is() )
340 m_updateInformation->setInteractionHandler( uno::Reference< task::XInteractionHandler > () );
341 }
342
execute()343 void UpdateDialog::Thread::execute()
344 {
345 {
346 vos::OGuard g( Application::GetSolarMutex() );
347 if ( m_stop ) {
348 return;
349 }
350 }
351 uno::Reference<deployment::XExtensionManager> extMgr =
352 deployment::ExtensionManager::get(m_context);
353
354 std::vector<std::pair<uno::Reference<deployment::XPackage>, uno::Any > > errors;
355
356 dp_misc::UpdateInfoMap updateInfoMap = dp_misc::getOnlineUpdateInfos(
357 m_context, extMgr, m_updateInformation, &m_vExtensionList, errors);
358
359 typedef std::vector<std::pair<uno::Reference<deployment::XPackage>,
360 uno::Any> >::const_iterator ITERROR;
361 for (ITERROR ite = errors.begin(); ite != errors.end(); ite ++)
362 handleSpecificError(ite->first, ite->second);
363
364 for (dp_misc::UpdateInfoMap::iterator i(updateInfoMap.begin()); i != updateInfoMap.end(); i++)
365 {
366 dp_misc::UpdateInfo const & info = i->second;
367 UpdateData updateData(info.extension);
368 DisabledUpdate disableUpdate;
369 //determine if online updates meet the requirements
370 prepareUpdateData(info.info, disableUpdate, updateData);
371
372 //determine if the update is installed in the user or shared repository
373 rtl::OUString sOnlineVersion;
374 if (info.info.is())
375 sOnlineVersion = info.version;
376 rtl::OUString sVersionUser;
377 rtl::OUString sVersionShared;
378 rtl::OUString sVersionBundled;
379 uno::Sequence< uno::Reference< deployment::XPackage> > extensions;
380 try {
381 extensions = extMgr->getExtensionsWithSameIdentifier(
382 dp_misc::getIdentifier(info.extension), info.extension->getName(),
383 uno::Reference<ucb::XCommandEnvironment>());
384 } catch (lang::IllegalArgumentException& ) {
385 OSL_ASSERT(0);
386 continue;
387 } catch (css::ucb::CommandFailedException& ) {
388 OSL_ASSERT(0);
389 continue;
390 }
391 OSL_ASSERT(extensions.getLength() == 3);
392 if (extensions[0].is() )
393 sVersionUser = extensions[0]->getVersion();
394 if (extensions[1].is() )
395 sVersionShared = extensions[1]->getVersion();
396 if (extensions[2].is() )
397 sVersionBundled = extensions[2]->getVersion();
398
399 bool bSharedReadOnly = extMgr->isReadOnlyRepository(OUSTR("shared"));
400
401 dp_misc::UPDATE_SOURCE sourceUser = dp_misc::isUpdateUserExtension(
402 bSharedReadOnly, sVersionUser, sVersionShared, sVersionBundled, sOnlineVersion);
403 dp_misc::UPDATE_SOURCE sourceShared = dp_misc::isUpdateSharedExtension(
404 bSharedReadOnly, sVersionShared, sVersionBundled, sOnlineVersion);
405
406 uno::Reference<deployment::XPackage> updateSource;
407 if (sourceUser != dp_misc::UPDATE_SOURCE_NONE)
408 {
409 if (sourceUser == dp_misc::UPDATE_SOURCE_SHARED)
410 {
411 updateData.aUpdateSource = extensions[1];
412 updateData.updateVersion = extensions[1]->getVersion();
413 }
414 else if (sourceUser == dp_misc::UPDATE_SOURCE_BUNDLED)
415 {
416 updateData.aUpdateSource = extensions[2];
417 updateData.updateVersion = extensions[2]->getVersion();
418 }
419 if (!update(disableUpdate, updateData))
420 return;
421 }
422
423 if (sourceShared != dp_misc::UPDATE_SOURCE_NONE)
424 {
425 if (sourceShared == dp_misc::UPDATE_SOURCE_BUNDLED)
426 {
427 updateData.aUpdateSource = extensions[2];
428 updateData.updateVersion = extensions[2]->getVersion();
429 }
430 updateData.bIsShared = true;
431 if (!update(disableUpdate, updateData))
432 return;
433 }
434 }
435
436
437 vos::OGuard g(Application::GetSolarMutex());
438 if (!m_stop) {
439 m_dialog.checkingDone();
440 }
441 }
442 #if 0
443 void UpdateDialog::Thread::handleGeneralError(uno::Any const & exception)
444 const
445 {
446 rtl::OUString message;
447 uno::Exception e;
448 if (exception >>= e) {
449 message = e.Message;
450 }
451 vos::OGuard g(Application::GetSolarMutex());
452 if (!m_stop) {
453 m_dialog.addGeneralError(message);
454 }
455 }
456 #endif
457 //Parameter package can be null
handleSpecificError(uno::Reference<deployment::XPackage> const & package,uno::Any const & exception) const458 void UpdateDialog::Thread::handleSpecificError(
459 uno::Reference< deployment::XPackage > const & package,
460 uno::Any const & exception) const
461 {
462 UpdateDialog::SpecificError data;
463 if (package.is())
464 data.name = package->getDisplayName();
465 uno::Exception e;
466 if (exception >>= e) {
467 data.message = e.Message;
468 }
469 vos::OGuard g(Application::GetSolarMutex());
470 if (!m_stop) {
471 m_dialog.addSpecificError(data);
472 }
473 }
474
getUpdateDisplayString(dp_gui::UpdateData const & data,::rtl::OUString const & version) const475 ::rtl::OUString UpdateDialog::Thread::getUpdateDisplayString(
476 dp_gui::UpdateData const & data, ::rtl::OUString const & version) const
477 {
478 OSL_ASSERT(data.aInstalledPackage.is());
479 rtl::OUStringBuffer b(data.aInstalledPackage->getDisplayName());
480 b.append(static_cast< sal_Unicode >(' '));
481 {
482 vos::OGuard g( Application::GetSolarMutex() );
483 if(!m_stop)
484 b.append(m_dialog.m_version);
485 }
486 b.append(static_cast< sal_Unicode >(' '));
487 if (version.getLength())
488 b.append(version);
489 else
490 b.append(data.updateVersion);
491
492 if (data.sWebsiteURL.getLength())
493 {
494 b.append(static_cast< sal_Unicode >(' '));
495 {
496 vos::OGuard g( Application::GetSolarMutex() );
497 if(!m_stop)
498 b.append(m_dialog.m_browserbased);
499 }
500 }
501 return b.makeStringAndClear();
502 }
503
504 /** out_data will only be filled if all dependencies are ok.
505 */
prepareUpdateData(uno::Reference<xml::dom::XNode> const & updateInfo,UpdateDialog::DisabledUpdate & out_du,dp_gui::UpdateData & out_data) const506 void UpdateDialog::Thread::prepareUpdateData(
507 uno::Reference< xml::dom::XNode > const & updateInfo,
508 UpdateDialog::DisabledUpdate & out_du,
509 dp_gui::UpdateData & out_data) const
510 {
511 if (!updateInfo.is())
512 return;
513 dp_misc::DescriptionInfoset infoset(m_context, updateInfo);
514 OSL_ASSERT(infoset.getVersion().getLength() != 0);
515 uno::Sequence< uno::Reference< xml::dom::XElement > > ds(
516 dp_misc::Dependencies::check(infoset));
517
518 out_du.aUpdateInfo = updateInfo;
519 out_du.unsatisfiedDependencies.realloc(ds.getLength());
520 for (sal_Int32 i = 0; i < ds.getLength(); ++i) {
521 out_du.unsatisfiedDependencies[i] = dp_misc::Dependencies::getErrorText(ds[i]);
522 }
523
524 const ::boost::optional< ::rtl::OUString> updateWebsiteURL(infoset.getLocalizedUpdateWebsiteURL());
525
526 out_du.name = getUpdateDisplayString(out_data, infoset.getVersion());
527
528 if (out_du.unsatisfiedDependencies.getLength() == 0)
529 {
530 out_data.aUpdateInfo = updateInfo;
531 out_data.updateVersion = infoset.getVersion();
532 if (updateWebsiteURL)
533 out_data.sWebsiteURL = *updateWebsiteURL;
534 }
535 }
536
update(UpdateDialog::DisabledUpdate & du,dp_gui::UpdateData & data) const537 bool UpdateDialog::Thread::update(
538 UpdateDialog::DisabledUpdate & du,
539 dp_gui::UpdateData & data) const
540 {
541 bool ret = false;
542 if (du.unsatisfiedDependencies.getLength() == 0)
543 {
544 vos::OGuard g(Application::GetSolarMutex());
545 if (!m_stop) {
546 m_dialog.addEnabledUpdate(getUpdateDisplayString(data), data);
547 }
548 ret = !m_stop;
549 } else {
550 vos::OGuard g(Application::GetSolarMutex());
551 if (!m_stop) {
552 m_dialog.addDisabledUpdate(du);
553 }
554 ret = !m_stop;
555 }
556 return ret;
557 }
558
559 // UpdateDialog ----------------------------------------------------------
UpdateDialog(uno::Reference<uno::XComponentContext> const & context,Window * parent,const std::vector<uno::Reference<deployment::XPackage>> & vExtensionList,std::vector<dp_gui::UpdateData> * updateData)560 UpdateDialog::UpdateDialog(
561 uno::Reference< uno::XComponentContext > const & context,
562 Window * parent,
563 const std::vector<uno::Reference< deployment::XPackage > > &vExtensionList,
564 std::vector< dp_gui::UpdateData > * updateData):
565 ModalDialog(parent,DpGuiResId(RID_DLG_UPDATE)),
566 m_context(context),
567 m_checking(this, DpGuiResId(RID_DLG_UPDATE_CHECKING)),
568 m_throbber(this, DpGuiResId(RID_DLG_UPDATE_THROBBER)),
569 m_update(this, DpGuiResId(RID_DLG_UPDATE_UPDATE)),
570 m_updates(
571 *this, DpGuiResId(RID_DLG_UPDATE_UPDATES),
572 Image(DpGuiResId(RID_DLG_UPDATE_NORMALALERT)),
573 Image(DpGuiResId(RID_DLG_UPDATE_HIGHCONTRASTALERT))),
574 m_all(this, DpGuiResId(RID_DLG_UPDATE_ALL)),
575 m_description(this, DpGuiResId(RID_DLG_UPDATE_DESCRIPTION)),
576 m_PublisherLabel(this, DpGuiResId(RID_DLG_UPDATE_PUBLISHER_LABEL)),
577 m_PublisherLink(this, DpGuiResId(RID_DLG_UPDATE_PUBLISHER_LINK)),
578 m_ReleaseNotesLabel(this, DpGuiResId(RID_DLG_UPDATE_RELEASENOTES_LABEL)),
579 m_ReleaseNotesLink(this, DpGuiResId(RID_DLG_UPDATE_RELEASENOTES_LINK)),
580 m_descriptions(this, DpGuiResId(RID_DLG_UPDATE_DESCRIPTIONS)),
581 m_line(this, DpGuiResId(RID_DLG_UPDATE_LINE)),
582 m_help(this, DpGuiResId(RID_DLG_UPDATE_HELP)),
583 m_ok(this, DpGuiResId(RID_DLG_UPDATE_OK)),
584 m_close(this, DpGuiResId(RID_DLG_UPDATE_CLOSE)),
585 m_error(String(DpGuiResId(RID_DLG_UPDATE_ERROR))),
586 m_none(String(DpGuiResId(RID_DLG_UPDATE_NONE))),
587 m_noInstallable(String(DpGuiResId(RID_DLG_UPDATE_NOINSTALLABLE))),
588 m_failure(String(DpGuiResId(RID_DLG_UPDATE_FAILURE))),
589 m_unknownError(String(DpGuiResId(RID_DLG_UPDATE_UNKNOWNERROR))),
590 m_noDescription(String(DpGuiResId(RID_DLG_UPDATE_NODESCRIPTION))),
591 m_noInstall(String(DpGuiResId(RID_DLG_UPDATE_NOINSTALL))),
592 m_noDependency(String(DpGuiResId(RID_DLG_UPDATE_NODEPENDENCY))),
593 m_noDependencyCurVer(String(DpGuiResId(RID_DLG_UPDATE_NODEPENDENCY_CUR_VER))),
594 m_browserbased(String(DpGuiResId(RID_DLG_UPDATE_BROWSERBASED))),
595 m_version(String(DpGuiResId(RID_DLG_UPDATE_VERSION))),
596 m_ignoredUpdate(String(DpGuiResId(RID_DLG_UPDATE_IGNORED_UPDATE))),
597 m_updateData(*updateData),
598 m_thread(
599 new UpdateDialog::Thread(
600 context, *this, vExtensionList)),
601 m_nFirstLineDelta(0),
602 m_nOneLineMissing(0),
603 m_nLastID(1),
604 m_bModified( false )
605 // TODO: check!
606 // ,
607 // m_extensionManagerDialog(extensionManagerDialog)
608 {
609 OSL_ASSERT(updateData != NULL);
610
611 m_xExtensionManager = deployment::ExtensionManager::get( context );
612
613 uno::Reference< awt::XToolkit > toolkit;
614 try {
615 toolkit = uno::Reference< awt::XToolkit >(
616 (uno::Reference< lang::XMultiComponentFactory >(
617 m_context->getServiceManager(),
618 uno::UNO_QUERY_THROW)->
619 createInstanceWithContext(
620 rtl::OUString(
621 RTL_CONSTASCII_USTRINGPARAM("com.sun.star.awt.Toolkit")),
622 m_context)),
623 uno::UNO_QUERY_THROW);
624 } catch (uno::RuntimeException &) {
625 throw;
626 } catch (uno::Exception & e) {
627 throw uno::RuntimeException(e.Message, e.Context);
628 }
629 m_updates.SetSelectHdl(LINK(this, UpdateDialog, selectionHandler));
630 m_all.SetToggleHdl(LINK(this, UpdateDialog, allHandler));
631 m_ok.SetClickHdl(LINK(this, UpdateDialog, okHandler));
632 m_close.SetClickHdl(LINK(this, UpdateDialog, closeHandler));
633 if ( ! dp_misc::office_is_running())
634 m_help.Disable();
635 FreeResource();
636
637 initDescription();
638 getIgnoredUpdates();
639 }
640
641 //------------------------------------------------------------------------------
~UpdateDialog()642 UpdateDialog::~UpdateDialog()
643 {
644 storeIgnoredUpdates();
645
646 for ( std::vector< UpdateDialog::Index* >::iterator i( m_ListboxEntries.begin() ); i != m_ListboxEntries.end(); ++i )
647 {
648 delete (*i);
649 }
650 for ( std::vector< UpdateDialog::IgnoredUpdate* >::iterator i( m_ignoredUpdates.begin() ); i != m_ignoredUpdates.end(); ++i )
651 {
652 delete (*i);
653 }
654 }
655
656 //------------------------------------------------------------------------------
Close()657 sal_Bool UpdateDialog::Close() {
658 m_thread->stop();
659 return ModalDialog::Close();
660 }
661
Execute()662 short UpdateDialog::Execute() {
663 m_throbber.start();
664 m_thread->launch();
665 return ModalDialog::Execute();
666 }
667
668 //------------------------------------------------------------------------------
669 //------------------------------------------------------------------------------
670 //------------------------------------------------------------------------------
CheckListBox(UpdateDialog & dialog,ResId const & resource,Image const & normalStaticImage,Image const & highContrastStaticImage)671 UpdateDialog::CheckListBox::CheckListBox( UpdateDialog & dialog, ResId const & resource,
672 Image const & normalStaticImage,
673 Image const & highContrastStaticImage ):
674 SvxCheckListBox( &dialog, resource, normalStaticImage, highContrastStaticImage ),
675 m_ignoreUpdate( String( DpGuiResId( RID_DLG_UPDATE_IGNORE ) ) ),
676 m_ignoreAllUpdates( String( DpGuiResId( RID_DLG_UPDATE_IGNORE_ALL ) ) ),
677 m_enableUpdate( String( DpGuiResId( RID_DLG_UPDATE_ENABLE ) ) ),
678 m_dialog(dialog)
679 {}
680
681 //------------------------------------------------------------------------------
~CheckListBox()682 UpdateDialog::CheckListBox::~CheckListBox() {}
683
684 //------------------------------------------------------------------------------
getItemCount() const685 sal_uInt16 UpdateDialog::CheckListBox::getItemCount() const {
686 sal_uLong i = GetEntryCount();
687 OSL_ASSERT(i <= std::numeric_limits< sal_uInt16 >::max());
688 return sal::static_int_cast< sal_uInt16 >(i);
689 }
690
691 //------------------------------------------------------------------------------
MouseButtonDown(MouseEvent const & event)692 void UpdateDialog::CheckListBox::MouseButtonDown( MouseEvent const & event )
693 {
694 // When clicking on a selected entry in an SvxCheckListBox, the entry's
695 // checkbox is toggled on mouse button down:
696 SvxCheckListBox::MouseButtonDown( event );
697
698 if ( event.IsRight() )
699 {
700 handlePopupMenu( event.GetPosPixel() );
701 }
702
703 m_dialog.enableOk();
704 }
705
706 //------------------------------------------------------------------------------
MouseButtonUp(MouseEvent const & event)707 void UpdateDialog::CheckListBox::MouseButtonUp(MouseEvent const & event) {
708 // When clicking on an entry's checkbox in an SvxCheckListBox, the entry's
709 // checkbox is toggled on mouse button up:
710 SvxCheckListBox::MouseButtonUp(event);
711 m_dialog.enableOk();
712 }
713
KeyInput(KeyEvent const & event)714 void UpdateDialog::CheckListBox::KeyInput(KeyEvent const & event) {
715 SvxCheckListBox::KeyInput(event);
716 m_dialog.enableOk();
717 }
718
719 //------------------------------------------------------------------------------
handlePopupMenu(const Point & rPos)720 void UpdateDialog::CheckListBox::handlePopupMenu( const Point &rPos )
721 {
722 SvListEntry *pData = GetEntry( rPos );
723
724 if ( pData )
725 {
726 sal_uInt16 nEntryPos = GetSelectEntryPos();
727 UpdateDialog::Index * p = static_cast< UpdateDialog::Index * >( GetEntryData( nEntryPos ) );
728
729 if ( ( p->m_eKind == ENABLED_UPDATE ) || ( p->m_eKind == DISABLED_UPDATE ) )
730 {
731 PopupMenu aPopup;
732
733 if ( p->m_bIgnored )
734 aPopup.InsertItem( CMD_ENABLE_UPDATE, m_enableUpdate );
735 else
736 {
737 aPopup.InsertItem( CMD_IGNORE_UPDATE, m_ignoreUpdate );
738 aPopup.InsertItem( CMD_IGNORE_ALL_UPDATES, m_ignoreAllUpdates );
739 }
740
741 sal_uInt16 aCmd = aPopup.Execute( this, rPos );
742 if ( ( aCmd == CMD_IGNORE_UPDATE ) || ( aCmd == CMD_IGNORE_ALL_UPDATES ) )
743 {
744 p->m_bIgnored = true;
745 if ( p->m_eKind == ENABLED_UPDATE )
746 {
747 RemoveEntry( nEntryPos );
748 m_dialog.addAdditional( p, SvLBoxButtonKind_disabledCheckbox );
749 }
750 if ( aCmd == CMD_IGNORE_UPDATE )
751 m_dialog.setIgnoredUpdate( p, true, false );
752 else
753 m_dialog.setIgnoredUpdate( p, true, true );
754 // TODO: reselect entry to display new description!
755 }
756 else if ( aCmd == CMD_ENABLE_UPDATE )
757 {
758 p->m_bIgnored = false;
759 if ( p->m_eKind == ENABLED_UPDATE )
760 {
761 RemoveEntry( nEntryPos );
762 m_dialog.insertItem( p, SvLBoxButtonKind_enabledCheckbox );
763 }
764 m_dialog.setIgnoredUpdate( p, false, false );
765 }
766 }
767 }
768 }
769
770 //------------------------------------------------------------------------------
771 //------------------------------------------------------------------------------
772 //------------------------------------------------------------------------------
insertItem(UpdateDialog::Index * pEntry,SvLBoxButtonKind kind)773 sal_uInt16 UpdateDialog::insertItem( UpdateDialog::Index *pEntry, SvLBoxButtonKind kind )
774 {
775 m_updates.InsertEntry( pEntry->m_aName, LISTBOX_APPEND, static_cast< void * >( pEntry ), kind );
776
777 for ( sal_uInt16 i = m_updates.getItemCount(); i != 0 ; )
778 {
779 i -= 1;
780 UpdateDialog::Index const * p = static_cast< UpdateDialog::Index const * >( m_updates.GetEntryData( i ) );
781 if ( p == pEntry )
782 return i;
783 }
784 OSL_ASSERT(0);
785 return 0;
786 }
787
788 //------------------------------------------------------------------------------
addAdditional(UpdateDialog::Index * index,SvLBoxButtonKind kind)789 void UpdateDialog::addAdditional( UpdateDialog::Index * index, SvLBoxButtonKind kind )
790 {
791 m_all.Enable();
792 if (m_all.IsChecked())
793 {
794 insertItem( index, kind );
795 m_update.Enable();
796 m_updates.Enable();
797 m_description.Enable();
798 m_descriptions.Enable();
799 }
800 }
801
802 //------------------------------------------------------------------------------
addEnabledUpdate(rtl::OUString const & name,dp_gui::UpdateData & data)803 void UpdateDialog::addEnabledUpdate( rtl::OUString const & name,
804 dp_gui::UpdateData & data )
805 {
806 sal_uInt16 nIndex = sal::static_int_cast< sal_uInt16 >( m_enabledUpdates.size() );
807 UpdateDialog::Index *pEntry = new UpdateDialog::Index( ENABLED_UPDATE, m_nLastID, nIndex, name );
808
809 data.m_nID = m_nLastID;
810 m_nLastID += 1;
811
812 m_enabledUpdates.push_back( data );
813 m_ListboxEntries.push_back( pEntry );
814
815 if ( ! isIgnoredUpdate( pEntry ) )
816 {
817 sal_uInt16 nPos = insertItem( pEntry, SvLBoxButtonKind_enabledCheckbox );
818 m_updates.CheckEntryPos( nPos );
819 }
820 else
821 addAdditional( pEntry, SvLBoxButtonKind_disabledCheckbox );
822
823 m_update.Enable();
824 m_updates.Enable();
825 m_description.Enable();
826 m_descriptions.Enable();
827 }
828
829 //------------------------------------------------------------------------------
addDisabledUpdate(UpdateDialog::DisabledUpdate & data)830 void UpdateDialog::addDisabledUpdate( UpdateDialog::DisabledUpdate & data )
831 {
832 sal_uInt16 nIndex = sal::static_int_cast< sal_uInt16 >( m_disabledUpdates.size() );
833 UpdateDialog::Index *pEntry = new UpdateDialog::Index( DISABLED_UPDATE, m_nLastID, nIndex, data.name );
834
835 data.m_nID = m_nLastID;
836 m_nLastID += 1;
837
838 m_disabledUpdates.push_back( data );
839 m_ListboxEntries.push_back( pEntry );
840
841 isIgnoredUpdate( pEntry );
842 addAdditional( pEntry, SvLBoxButtonKind_disabledCheckbox );
843 }
844
845 //------------------------------------------------------------------------------
addSpecificError(UpdateDialog::SpecificError & data)846 void UpdateDialog::addSpecificError( UpdateDialog::SpecificError & data )
847 {
848 sal_uInt16 nIndex = sal::static_int_cast< sal_uInt16 >( m_specificErrors.size() );
849 UpdateDialog::Index *pEntry = new UpdateDialog::Index( SPECIFIC_ERROR, m_nLastID, nIndex, data.name );
850
851 data.m_nID = m_nLastID;
852 m_nLastID += 1;
853
854 m_specificErrors.push_back( data );
855 m_ListboxEntries.push_back( pEntry );
856
857 addAdditional( pEntry, SvLBoxButtonKind_staticImage);
858 }
859
checkingDone()860 void UpdateDialog::checkingDone() {
861 m_checking.Hide();
862 m_throbber.stop();
863 m_throbber.Hide();
864 if (m_updates.getItemCount() == 0)
865 {
866 clearDescription();
867 m_description.Enable();
868 m_descriptions.Enable();
869
870 if ( m_disabledUpdates.empty() && m_specificErrors.empty() && m_ignoredUpdates.empty() )
871 showDescription( m_none, false );
872 else
873 showDescription( m_noInstallable, false );
874 }
875
876 enableOk();
877 }
878
enableOk()879 void UpdateDialog::enableOk() {
880 if (!m_checking.IsVisible()) {
881 m_ok.Enable(m_updates.GetCheckedEntryCount() != 0);
882 }
883 }
884
885 // *********************************************************************************
createNotifyJob(bool bPrepareOnly,uno::Sequence<uno::Sequence<rtl::OUString>> & rItemList)886 void UpdateDialog::createNotifyJob( bool bPrepareOnly,
887 uno::Sequence< uno::Sequence< rtl::OUString > > &rItemList )
888 {
889 if ( !dp_misc::office_is_running() )
890 return;
891
892 // notify update check job
893 try
894 {
895 uno::Reference< lang::XMultiServiceFactory > xFactory( ::comphelper::getProcessServiceFactory() );
896 uno::Reference< lang::XMultiServiceFactory > xConfigProvider(
897 xFactory->createInstance( OUSTR( "com.sun.star.configuration.ConfigurationProvider" )),
898 uno::UNO_QUERY_THROW);
899
900 beans::PropertyValue aProperty;
901 aProperty.Name = OUSTR( "nodepath" );
902 aProperty.Value = uno::makeAny( OUSTR("org.openoffice.Office.Addons/AddonUI/OfficeHelp/UpdateCheckJob") );
903
904 uno::Sequence< uno::Any > aArgumentList( 1 );
905 aArgumentList[0] = uno::makeAny( aProperty );
906
907 uno::Reference< container::XNameAccess > xNameAccess(
908 xConfigProvider->createInstanceWithArguments(
909 OUSTR("com.sun.star.configuration.ConfigurationAccess"), aArgumentList ),
910 uno::UNO_QUERY_THROW );
911
912 util::URL aURL;
913 xNameAccess->getByName(OUSTR("URL")) >>= aURL.Complete;
914
915 uno::Reference < util::XURLTransformer > xTransformer( xFactory->createInstance( OUSTR( "com.sun.star.util.URLTransformer" ) ),
916 uno::UNO_QUERY_THROW );
917
918 xTransformer->parseStrict(aURL);
919
920 uno::Reference < frame::XDesktop > xDesktop( xFactory->createInstance( OUSTR( "com.sun.star.frame.Desktop" ) ),
921 uno::UNO_QUERY_THROW );
922 uno::Reference< frame::XDispatchProvider > xDispatchProvider( xDesktop->getCurrentFrame(),
923 uno::UNO_QUERY_THROW );
924 uno::Reference< frame::XDispatch > xDispatch = xDispatchProvider->queryDispatch(aURL, rtl::OUString(), 0);
925
926 if( xDispatch.is() )
927 {
928 uno::Sequence< beans::PropertyValue > aPropList(2);
929 aProperty.Name = OUSTR( "updateList" );
930 aProperty.Value = uno::makeAny( rItemList );
931 aPropList[0] = aProperty;
932 aProperty.Name = OUSTR( "prepareOnly" );
933 aProperty.Value = uno::makeAny( bPrepareOnly );
934 aPropList[1] = aProperty;
935
936 xDispatch->dispatch(aURL, aPropList );
937 }
938 }
939 catch( const uno::Exception& e )
940 {
941 dp_misc::TRACE( OUSTR("Caught exception: ")
942 + e.Message + OUSTR("\n thread terminated.\n\n"));
943 }
944 }
945
946 // *********************************************************************************
notifyMenubar(bool bPrepareOnly,bool bRecheckOnly)947 void UpdateDialog::notifyMenubar( bool bPrepareOnly, bool bRecheckOnly )
948 {
949 if ( !dp_misc::office_is_running() )
950 return;
951
952 uno::Sequence< uno::Sequence< rtl::OUString > > aItemList;
953 sal_Int32 nCount = 0;
954
955 if ( ! bRecheckOnly )
956 {
957 for ( sal_Int16 i = 0; i < m_updates.getItemCount(); ++i )
958 {
959 uno::Sequence< rtl::OUString > aItem(2);
960
961 UpdateDialog::Index const * p = static_cast< UpdateDialog::Index const * >(m_updates.GetEntryData(i));
962
963 if ( p->m_eKind == ENABLED_UPDATE )
964 {
965 dp_gui::UpdateData aUpdData = m_enabledUpdates[ p->m_nIndex ];
966 aItem[0] = dp_misc::getIdentifier( aUpdData.aInstalledPackage );
967
968 dp_misc::DescriptionInfoset aInfoset( m_context, aUpdData.aUpdateInfo );
969 aItem[1] = aInfoset.getVersion();
970 }
971 else if ( p->m_eKind == DISABLED_UPDATE )
972 continue;
973 else
974 continue;
975
976 aItemList.realloc( nCount + 1 );
977 aItemList[ nCount ] = aItem;
978 nCount += 1;
979 }
980 }
981
982 storeIgnoredUpdates();
983 createNotifyJob( bPrepareOnly, aItemList );
984 }
985
986 // *********************************************************************************
987
initDescription()988 void UpdateDialog::initDescription()
989 {
990 m_PublisherLabel.Hide();
991 m_PublisherLink.Hide();
992 m_ReleaseNotesLabel.Hide();
993 m_ReleaseNotesLink.Hide();
994 m_descriptions.Hide();
995
996 Link aLink = LINK( this, UpdateDialog, hyperlink_clicked );
997 m_PublisherLink.SetClickHdl( aLink );
998 m_ReleaseNotesLink.SetClickHdl( aLink );
999
1000 long nTextWidth = m_PublisherLabel.GetCtrlTextWidth( m_PublisherLabel.GetText() );
1001 long nTemp = m_ReleaseNotesLabel.GetTextWidth( m_ReleaseNotesLabel.GetText() );
1002 if ( nTemp > nTextWidth )
1003 nTextWidth = nTemp;
1004 nTextWidth = nTextWidth * 110 / 100;
1005
1006 Size aNewSize = m_PublisherLabel.GetSizePixel();
1007 if ( nTextWidth > aNewSize.Width() )
1008 {
1009 long nDelta = nTextWidth - aNewSize.Width();
1010 aNewSize.Width() = nTextWidth;
1011 m_PublisherLabel.SetSizePixel( aNewSize );
1012 m_ReleaseNotesLabel.SetSizePixel( aNewSize );
1013
1014 aNewSize = m_PublisherLink.GetSizePixel();
1015 aNewSize.Width() = aNewSize.Width() - nDelta;
1016 Point aNewPos = m_PublisherLink.GetPosPixel();
1017 aNewPos.X() = aNewPos.X() + nDelta;
1018 m_PublisherLink.SetPosSizePixel( aNewPos, aNewSize );
1019 aNewPos.Y() = m_ReleaseNotesLink.GetPosPixel().Y();
1020 m_ReleaseNotesLink.SetPosSizePixel( aNewPos, aNewSize );
1021 }
1022
1023 m_aFirstLinePos = m_descriptions.GetPosPixel();
1024 m_aFirstLineSize = m_descriptions.GetSizePixel();
1025 Size aMarginSize = LogicToPixel( Size( RSC_SP_CTRL_GROUP_X, RSC_SP_CTRL_GROUP_Y ), MAP_APPFONT );
1026 Point aThirdLinePos = m_ReleaseNotesLabel.GetPosPixel();
1027 aThirdLinePos.Y() = aThirdLinePos.Y() + m_ReleaseNotesLabel.GetSizePixel().Height() + aMarginSize.Height();
1028 m_nFirstLineDelta = aThirdLinePos.Y() - m_aFirstLinePos.Y();
1029 m_nOneLineMissing = m_ReleaseNotesLabel.GetPosPixel().Y() - m_PublisherLabel.GetPosPixel().Y();
1030 }
1031
clearDescription()1032 void UpdateDialog::clearDescription()
1033 {
1034 String sEmpty;
1035 m_PublisherLabel.Hide();
1036 m_PublisherLink.Hide();
1037 m_PublisherLink.SetDescription( sEmpty );
1038 m_PublisherLink.SetURL( sEmpty );
1039 m_ReleaseNotesLabel.Hide();
1040 m_ReleaseNotesLink.Hide();
1041 m_ReleaseNotesLink.SetURL( sEmpty );
1042 if ( m_PublisherLabel.GetPosPixel().Y() == m_ReleaseNotesLabel.GetPosPixel().Y() )
1043 {
1044 Point aNewPos = m_ReleaseNotesLabel.GetPosPixel();
1045 aNewPos.Y() += m_nOneLineMissing;
1046 m_ReleaseNotesLabel.SetPosPixel( aNewPos );
1047 aNewPos = m_ReleaseNotesLink.GetPosPixel();
1048 aNewPos.Y() += m_nOneLineMissing;
1049 m_ReleaseNotesLink.SetPosPixel( aNewPos );
1050 }
1051 m_descriptions.Hide();
1052 m_descriptions.Clear();
1053 m_descriptions.SetPosSizePixel( m_aFirstLinePos, m_aFirstLineSize );
1054 }
1055
showDescription(uno::Reference<xml::dom::XNode> const & aUpdateInfo)1056 bool UpdateDialog::showDescription(uno::Reference< xml::dom::XNode > const & aUpdateInfo)
1057 {
1058 dp_misc::DescriptionInfoset infoset(m_context, aUpdateInfo);
1059 return showDescription(infoset.getLocalizedPublisherNameAndURL(),
1060 infoset.getLocalizedReleaseNotesURL());
1061 }
1062
showDescription(uno::Reference<deployment::XPackage> const & aExtension)1063 bool UpdateDialog::showDescription(uno::Reference< deployment::XPackage > const & aExtension)
1064 {
1065 OSL_ASSERT(aExtension.is());
1066 beans::StringPair pubInfo = aExtension->getPublisherInfo();
1067 return showDescription(std::make_pair(pubInfo.First, pubInfo.Second),
1068 OUSTR(""));
1069 }
1070
showDescription(std::pair<rtl::OUString,rtl::OUString> const & pairPublisher,rtl::OUString const & sReleaseNotes)1071 bool UpdateDialog::showDescription(std::pair< rtl::OUString, rtl::OUString > const & pairPublisher,
1072 rtl::OUString const & sReleaseNotes)
1073 {
1074 rtl::OUString sPub = pairPublisher.first;
1075 rtl::OUString sURL = pairPublisher.second;
1076
1077 if ( sPub.getLength() == 0 && sURL.getLength() == 0 && sReleaseNotes.getLength() == 0 )
1078 // nothing to show
1079 return false;
1080
1081 bool bPublisher = false;
1082 if ( sPub.getLength() > 0 )
1083 {
1084 m_PublisherLabel.Show();
1085 m_PublisherLink.Show();
1086 m_PublisherLink.SetDescription( sPub );
1087 m_PublisherLink.SetURL( sURL );
1088 bPublisher = true;
1089 }
1090
1091 if ( sReleaseNotes.getLength() > 0 )
1092 {
1093 if ( !bPublisher )
1094 {
1095 m_ReleaseNotesLabel.SetPosPixel( m_PublisherLabel.GetPosPixel() );
1096 m_ReleaseNotesLink.SetPosPixel( m_PublisherLink.GetPosPixel() );
1097 }
1098 m_ReleaseNotesLabel.Show();
1099 m_ReleaseNotesLink.Show();
1100 m_ReleaseNotesLink.SetURL( sReleaseNotes );
1101 }
1102 return true;
1103 }
1104
showDescription(const String & rDescription,bool bWithPublisher)1105 bool UpdateDialog::showDescription( const String& rDescription, bool bWithPublisher )
1106 {
1107 if ( rDescription.Len() == 0 )
1108 // nothing to show
1109 return false;
1110
1111 if ( bWithPublisher )
1112 {
1113 bool bOneLineMissing = !m_ReleaseNotesLabel.IsVisible() || !m_PublisherLabel.IsVisible();
1114 Point aNewPos = m_aFirstLinePos;
1115 aNewPos.Y() += m_nFirstLineDelta;
1116 if ( bOneLineMissing )
1117 aNewPos.Y() -= m_nOneLineMissing;
1118 Size aNewSize = m_aFirstLineSize;
1119 aNewSize.Height() -= m_nFirstLineDelta;
1120 if ( bOneLineMissing )
1121 aNewSize.Height() += m_nOneLineMissing;
1122 m_descriptions.SetPosSizePixel( aNewPos, aNewSize );
1123 }
1124 m_descriptions.Show();
1125 m_descriptions.SetDescription( rDescription );
1126 return true;
1127 }
1128
isReadOnly(const uno::Reference<deployment::XPackage> & xPackage) const1129 bool UpdateDialog::isReadOnly( const uno::Reference< deployment::XPackage > &xPackage ) const
1130 {
1131 if ( m_xExtensionManager.is() && xPackage.is() )
1132 {
1133 return m_xExtensionManager->isReadOnlyRepository( xPackage->getRepositoryName() );
1134 }
1135 else
1136 return true;
1137 }
1138
1139 //------------------------------------------------------------------------------
getIgnoredUpdates()1140 void UpdateDialog::getIgnoredUpdates()
1141 {
1142 uno::Reference< lang::XMultiServiceFactory > xConfig( m_context->getServiceManager()->createInstanceWithContext(
1143 OUSTR("com.sun.star.configuration.ConfigurationProvider"), m_context ), uno::UNO_QUERY_THROW);
1144 beans::NamedValue aValue( OUSTR("nodepath"), uno::Any( IGNORED_UPDATES ) );
1145 uno::Sequence< uno::Any > args(1);
1146 args[0] <<= aValue;
1147
1148 uno::Reference< container::XNameAccess > xNameAccess( xConfig->createInstanceWithArguments( OUSTR("com.sun.star.configuration.ConfigurationAccess"), args), uno::UNO_QUERY_THROW );
1149 uno::Sequence< rtl::OUString > aElementNames = xNameAccess->getElementNames();
1150
1151 for ( sal_Int32 i = 0; i < aElementNames.getLength(); i++ )
1152 {
1153 ::rtl::OUString aIdentifier = aElementNames[i];
1154 ::rtl::OUString aVersion;
1155
1156 uno::Any aPropValue( uno::Reference< beans::XPropertySet >( xNameAccess->getByName( aIdentifier ), uno::UNO_QUERY_THROW )->getPropertyValue( PROPERTY_VERSION ) );
1157 aPropValue >>= aVersion;
1158 IgnoredUpdate *pData = new IgnoredUpdate( aIdentifier, aVersion );
1159 m_ignoredUpdates.push_back( pData );
1160 }
1161 }
1162
1163 //------------------------------------------------------------------------------
storeIgnoredUpdates()1164 void UpdateDialog::storeIgnoredUpdates()
1165 {
1166 if ( m_bModified && ( m_ignoredUpdates.size() != 0 ) )
1167 {
1168 uno::Reference< lang::XMultiServiceFactory > xConfig( m_context->getServiceManager()->createInstanceWithContext(
1169 OUSTR("com.sun.star.configuration.ConfigurationProvider"), m_context ), uno::UNO_QUERY_THROW );
1170 beans::NamedValue aValue( OUSTR("nodepath"), uno::Any( IGNORED_UPDATES ) );
1171 uno::Sequence< uno::Any > args(1);
1172 args[0] <<= aValue;
1173
1174 uno::Reference< container::XNameContainer > xNameContainer( xConfig->createInstanceWithArguments(
1175 OUSTR("com.sun.star.configuration.ConfigurationUpdateAccess"), args ), uno::UNO_QUERY_THROW );
1176
1177 for ( std::vector< UpdateDialog::IgnoredUpdate* >::iterator i( m_ignoredUpdates.begin() ); i != m_ignoredUpdates.end(); ++i )
1178 {
1179 if ( xNameContainer->hasByName( (*i)->sExtensionID ) )
1180 {
1181 if ( (*i)->bRemoved )
1182 xNameContainer->removeByName( (*i)->sExtensionID );
1183 else
1184 uno::Reference< beans::XPropertySet >( xNameContainer->getByName( (*i)->sExtensionID ), uno::UNO_QUERY_THROW )->setPropertyValue( PROPERTY_VERSION, uno::Any( (*i)->sVersion ) );
1185 }
1186 else if ( ! (*i)->bRemoved )
1187 {
1188 uno::Reference< beans::XPropertySet > elem( uno::Reference< lang::XSingleServiceFactory >( xNameContainer, uno::UNO_QUERY_THROW )->createInstance(), uno::UNO_QUERY_THROW );
1189 elem->setPropertyValue( PROPERTY_VERSION, uno::Any( (*i)->sVersion ) );
1190 xNameContainer->insertByName( (*i)->sExtensionID, uno::Any( elem ) );
1191 }
1192 }
1193
1194 uno::Reference< util::XChangesBatch > xChangesBatch( xNameContainer, uno::UNO_QUERY );
1195 if ( xChangesBatch.is() && xChangesBatch->hasPendingChanges() )
1196 xChangesBatch->commitChanges();
1197 }
1198
1199 m_bModified = false;
1200 }
1201
1202 //------------------------------------------------------------------------------
isIgnoredUpdate(UpdateDialog::Index * index)1203 bool UpdateDialog::isIgnoredUpdate( UpdateDialog::Index * index )
1204 {
1205 bool bIsIgnored = false;
1206
1207 if ( m_ignoredUpdates.size() != 0 )
1208 {
1209 rtl::OUString aExtensionID;
1210 rtl::OUString aVersion;
1211
1212 if ( index->m_eKind == ENABLED_UPDATE )
1213 {
1214 dp_gui::UpdateData aUpdData = m_enabledUpdates[ index->m_nIndex ];
1215 aExtensionID = dp_misc::getIdentifier( aUpdData.aInstalledPackage );
1216 aVersion = aUpdData.updateVersion;
1217 }
1218 else if ( index->m_eKind == DISABLED_UPDATE )
1219 {
1220 DisabledUpdate &rData = m_disabledUpdates[ index->m_nIndex ];
1221 dp_misc::DescriptionInfoset aInfoset( m_context, rData.aUpdateInfo );
1222 ::boost::optional< ::rtl::OUString > aID( aInfoset.getIdentifier() );
1223 if ( aID )
1224 aExtensionID = *aID;
1225 aVersion = aInfoset.getVersion();
1226 }
1227
1228 for ( std::vector< UpdateDialog::IgnoredUpdate* >::iterator i( m_ignoredUpdates.begin() ); i != m_ignoredUpdates.end(); ++i )
1229 {
1230 if ( (*i)->sExtensionID == aExtensionID )
1231 {
1232 if ( ( (*i)->sVersion.getLength() == 0 ) || ( (*i)->sVersion == aVersion ) )
1233 {
1234 bIsIgnored = true;
1235 index->m_bIgnored = true;
1236 }
1237 else // when we find another update of an ignored version, we will remove the old one to keep the ignored list small
1238 (*i)->bRemoved = true;
1239 break;
1240 }
1241 }
1242 }
1243
1244 return bIsIgnored;
1245 }
1246
1247 //------------------------------------------------------------------------------
setIgnoredUpdate(UpdateDialog::Index * pIndex,bool bIgnore,bool bIgnoreAll)1248 void UpdateDialog::setIgnoredUpdate( UpdateDialog::Index *pIndex, bool bIgnore, bool bIgnoreAll )
1249 {
1250 rtl::OUString aExtensionID;
1251 rtl::OUString aVersion;
1252
1253 m_bModified = true;
1254
1255 if ( pIndex->m_eKind == ENABLED_UPDATE )
1256 {
1257 dp_gui::UpdateData aUpdData = m_enabledUpdates[ pIndex->m_nIndex ];
1258 aExtensionID = dp_misc::getIdentifier( aUpdData.aInstalledPackage );
1259 if ( !bIgnoreAll )
1260 aVersion = aUpdData.updateVersion;
1261 }
1262 else if ( pIndex->m_eKind == DISABLED_UPDATE )
1263 {
1264 DisabledUpdate &rData = m_disabledUpdates[ pIndex->m_nIndex ];
1265 dp_misc::DescriptionInfoset aInfoset( m_context, rData.aUpdateInfo );
1266 ::boost::optional< ::rtl::OUString > aID( aInfoset.getIdentifier() );
1267 if ( aID )
1268 aExtensionID = *aID;
1269 if ( !bIgnoreAll )
1270 aVersion = aInfoset.getVersion();
1271 }
1272
1273 if ( aExtensionID.getLength() )
1274 {
1275 bool bFound = false;
1276 for ( std::vector< UpdateDialog::IgnoredUpdate* >::iterator i( m_ignoredUpdates.begin() ); i != m_ignoredUpdates.end(); ++i )
1277 {
1278 if ( (*i)->sExtensionID == aExtensionID )
1279 {
1280 (*i)->sVersion = aVersion;
1281 (*i)->bRemoved = !bIgnore;
1282 bFound = true;
1283 break;
1284 }
1285 }
1286 if ( bIgnore && !bFound )
1287 {
1288 IgnoredUpdate *pData = new IgnoredUpdate( aExtensionID, aVersion );
1289 m_ignoredUpdates.push_back( pData );
1290 }
1291 }
1292 }
1293
1294 //------------------------------------------------------------------------------
IMPL_LINK(UpdateDialog,selectionHandler,void *,EMPTYARG)1295 IMPL_LINK(UpdateDialog, selectionHandler, void *, EMPTYARG)
1296 {
1297 rtl::OUStringBuffer b;
1298 bool bInserted = false;
1299 UpdateDialog::Index const * p = static_cast< UpdateDialog::Index const * >(
1300 m_updates.GetEntryData(m_updates.GetSelectEntryPos()));
1301 clearDescription();
1302
1303 if ( p != NULL )
1304 {
1305 sal_uInt16 pos = p->m_nIndex;
1306
1307 switch (p->m_eKind)
1308 {
1309 case ENABLED_UPDATE:
1310 {
1311 if ( m_enabledUpdates[ pos ].aUpdateSource.is() )
1312 bInserted = showDescription( m_enabledUpdates[ pos ].aUpdateSource );
1313 else
1314 bInserted = showDescription( m_enabledUpdates[ pos ].aUpdateInfo );
1315
1316 if ( p->m_bIgnored )
1317 b.append( m_ignoredUpdate );
1318
1319 break;
1320 }
1321 case DISABLED_UPDATE:
1322 {
1323 bInserted = showDescription( m_disabledUpdates[pos].aUpdateInfo );
1324
1325 if ( p->m_bIgnored )
1326 b.append( m_ignoredUpdate );
1327
1328 UpdateDialog::DisabledUpdate & data = m_disabledUpdates[ pos ];
1329 if (data.unsatisfiedDependencies.getLength() != 0)
1330 {
1331 // create error string for version mismatch
1332 ::rtl::OUString sVersion( RTL_CONSTASCII_USTRINGPARAM("%VERSION") );
1333 sal_Int32 nPos = m_noDependencyCurVer.indexOf( sVersion );
1334 if ( nPos >= 0 )
1335 {
1336 ::rtl::OUString sCurVersion( RTL_CONSTASCII_USTRINGPARAM( "${$OOO_BASE_DIR/program/" SAL_CONFIGFILE("version") ":Version:OOOPackageVersion}"));
1337 ::rtl::Bootstrap::expandMacros(sCurVersion);
1338 m_noDependencyCurVer = m_noDependencyCurVer.replaceAt( nPos, sVersion.getLength(), sCurVersion );
1339 }
1340
1341 b.append(m_noInstall);
1342 b.append(LF);
1343 b.append(m_noDependency);
1344 for (sal_Int32 i = 0;
1345 i < data.unsatisfiedDependencies.getLength(); ++i)
1346 {
1347 b.append(LF);
1348 b.appendAscii(RTL_CONSTASCII_STRINGPARAM(" "));
1349 // U+2003 EM SPACE would be better than two spaces,
1350 // but some fonts do not contain it
1351 b.append(
1352 confineToParagraph(
1353 data.unsatisfiedDependencies[i]));
1354 }
1355 b.append(LF);
1356 b.appendAscii(RTL_CONSTASCII_STRINGPARAM(" "));
1357 b.append(m_noDependencyCurVer);
1358 }
1359 break;
1360 }
1361 case SPECIFIC_ERROR:
1362 {
1363 UpdateDialog::SpecificError & data = m_specificErrors[ pos ];
1364 b.append(m_failure);
1365 b.append(LF);
1366 b.append( data.message.getLength() == 0 ? m_unknownError : data.message );
1367 break;
1368 }
1369 default:
1370 OSL_ASSERT(false);
1371 break;
1372 }
1373 }
1374
1375 if ( b.getLength() == 0 )
1376 b.append( m_noDescription );
1377
1378 showDescription( b.makeStringAndClear(), bInserted );
1379 return 0;
1380 }
1381
IMPL_LINK(UpdateDialog,allHandler,void *,EMPTYARG)1382 IMPL_LINK(UpdateDialog, allHandler, void *, EMPTYARG)
1383 {
1384 if (m_all.IsChecked())
1385 {
1386 m_update.Enable();
1387 m_updates.Enable();
1388 m_description.Enable();
1389 m_descriptions.Enable();
1390
1391 for (std::vector< UpdateDialog::Index* >::iterator i( m_ListboxEntries.begin() );
1392 i != m_ListboxEntries.end(); ++i )
1393 {
1394 if ( (*i)->m_bIgnored || ( (*i)->m_eKind != ENABLED_UPDATE ) )
1395 insertItem( (*i), SvLBoxButtonKind_disabledCheckbox );
1396 }
1397 }
1398 else
1399 {
1400 for ( sal_uInt16 i = 0; i < m_updates.getItemCount(); )
1401 {
1402 UpdateDialog::Index const * p = static_cast< UpdateDialog::Index const * >( m_updates.GetEntryData(i) );
1403 if ( p->m_bIgnored || ( p->m_eKind != ENABLED_UPDATE ) )
1404 {
1405 m_updates.RemoveEntry(i);
1406 } else {
1407 ++i;
1408 }
1409 }
1410
1411 if (m_updates.getItemCount() == 0)
1412 {
1413 clearDescription();
1414 m_update.Disable();
1415 m_updates.Disable();
1416 if (m_checking.IsVisible())
1417 m_description.Disable();
1418 else
1419 showDescription(m_noInstallable,false);
1420 }
1421 }
1422 return 0;
1423 }
1424
IMPL_LINK(UpdateDialog,okHandler,void *,EMPTYARG)1425 IMPL_LINK(UpdateDialog, okHandler, void *, EMPTYARG)
1426 {
1427 //If users are going to update a shared extension then we need
1428 //to warn them
1429 typedef ::std::vector<UpdateData>::const_iterator CIT;
1430 for (CIT i = m_enabledUpdates.begin(); i < m_enabledUpdates.end(); i++)
1431 {
1432 OSL_ASSERT(i->aInstalledPackage.is());
1433 //If the user has no write access to the shared folder then the update
1434 //for a shared extension is disable, that is it cannot be in m_enabledUpdates
1435 // OSL_ASSERT(isReadOnly(i->aInstalledPackage) == sal_False);
1436 #if 0
1437 // TODO: check!
1438 OSL_ASSERT(m_extensionManagerDialog.get());
1439 if (RET_CANCEL == m_extensionManagerDialog->continueUpdateForSharedExtension(
1440 this, i->aPackageManager))
1441 {
1442 EndDialog(RET_CANCEL);
1443 }
1444 #endif
1445 }
1446
1447
1448 for (sal_uInt16 i = 0; i < m_updates.getItemCount(); ++i) {
1449 UpdateDialog::Index const * p =
1450 static_cast< UpdateDialog::Index const * >(
1451 m_updates.GetEntryData(i));
1452 if (p->m_eKind == ENABLED_UPDATE && m_updates.IsChecked(i)) {
1453 m_updateData.push_back( m_enabledUpdates[ p->m_nIndex ] );
1454 }
1455 }
1456
1457 EndDialog(RET_OK);
1458 return 0;
1459 }
1460
IMPL_LINK(UpdateDialog,closeHandler,void *,EMPTYARG)1461 IMPL_LINK(UpdateDialog, closeHandler, void *, EMPTYARG) {
1462 m_thread->stop();
1463 EndDialog(RET_CANCEL);
1464 return 0;
1465 }
1466
IMPL_LINK(UpdateDialog,hyperlink_clicked,svt::FixedHyperlink *,pHyperlink)1467 IMPL_LINK( UpdateDialog, hyperlink_clicked, svt::FixedHyperlink*, pHyperlink )
1468 {
1469 ::rtl::OUString sURL;
1470 if ( pHyperlink )
1471 sURL = ::rtl::OUString( pHyperlink->GetURL() );
1472 if ( sURL.getLength() == 0 )
1473 return 0;
1474
1475 try
1476 {
1477 uno::Reference< com::sun::star::system::XSystemShellExecute > xSystemShellExecute(
1478 com::sun::star::system::SystemShellExecute::create( m_context ) );
1479 xSystemShellExecute->execute(
1480 sURL, ::rtl::OUString(), com::sun::star::system::SystemShellExecuteFlags::DEFAULTS);
1481 }
1482 catch (uno::Exception& )
1483 {
1484 }
1485
1486 return 1;
1487 }
1488