541 lines
17 KiB
C++
541 lines
17 KiB
C++
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
|
/*
|
|
* This file is part of the LibreOffice project.
|
|
*
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
*
|
|
* This file incorporates work covered by the following license notice:
|
|
*
|
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
|
* contributor license agreements. See the NOTICE file distributed
|
|
* with this work for additional information regarding copyright
|
|
* ownership. The ASF licenses this file to you under the Apache
|
|
* License, Version 2.0 (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.apache.org/licenses/LICENSE-2.0 .
|
|
*/
|
|
|
|
#include <string.h>
|
|
|
|
#include <comphelper/lok.hxx>
|
|
#include <comphelper/processfactory.hxx>
|
|
#include <comphelper/diagnose_ex.hxx>
|
|
#include <o3tl/string_view.hxx>
|
|
#include <unotools/resmgr.hxx>
|
|
#include <sal/log.hxx>
|
|
|
|
#include <configsettings.hxx>
|
|
#include <vcl/QueueInfo.hxx>
|
|
#include <vcl/cvtgrf.hxx>
|
|
#include <vcl/dockwin.hxx>
|
|
#include <vcl/fieldvalues.hxx>
|
|
#include <vcl/menu.hxx>
|
|
#include <vcl/print.hxx>
|
|
#include <vcl/settings.hxx>
|
|
#include <vcl/svapp.hxx>
|
|
#include <vcl/virdev.hxx>
|
|
#include <vcl/wrkwin.hxx>
|
|
#include <vcl/uitest/logger.hxx>
|
|
#include <salframe.hxx>
|
|
#include <scrwnd.hxx>
|
|
#include <helpwin.hxx>
|
|
#include <vcl/toolkit/dialog.hxx>
|
|
#include <salinst.hxx>
|
|
#include <salgdi.hxx>
|
|
#include <svdata.hxx>
|
|
#include <salsys.hxx>
|
|
#include <windowdev.hxx>
|
|
#include <units.hrc>
|
|
#include <print.h>
|
|
|
|
#include <config_features.h>
|
|
#include <basegfx/utils/systemdependentdata.hxx>
|
|
#include <mutex>
|
|
|
|
using namespace com::sun::star::uno;
|
|
|
|
namespace
|
|
{
|
|
struct private_aImplSVData :
|
|
public rtl::Static<ImplSVData, private_aImplSVData> {};
|
|
/// Default instance ensures that ImplSVData::mpHelpData is never null.
|
|
struct private_aImplSVHelpData :
|
|
public rtl::Static<ImplSVHelpData, private_aImplSVHelpData> {};
|
|
|
|
/// Default instance ensures that ImplSVData::mpWinData is never null.
|
|
struct private_aImplSVWinData :
|
|
public rtl::Static<ImplSVWinData, private_aImplSVWinData> {};
|
|
|
|
}
|
|
|
|
ImplSVData* ImplGetSVData() {
|
|
return &private_aImplSVData::get();
|
|
}
|
|
|
|
SalSystem* ImplGetSalSystem()
|
|
{
|
|
ImplSVData* pSVData = ImplGetSVData();
|
|
if( ! pSVData->mpSalSystem )
|
|
pSVData->mpSalSystem.reset( pSVData->mpDefInst->CreateSalSystem() );
|
|
return pSVData->mpSalSystem.get();
|
|
}
|
|
|
|
void ImplDeInitSVData()
|
|
{
|
|
ImplSVData* pSVData = ImplGetSVData();
|
|
|
|
// delete global instance data
|
|
pSVData->mpSettingsConfigItem.reset();
|
|
|
|
pSVData->mpDockingManager.reset();
|
|
|
|
pSVData->maCtrlData.maFieldUnitStrings.clear();
|
|
pSVData->maCtrlData.maCleanUnitStrings.clear();
|
|
pSVData->maPaperNames.clear();
|
|
}
|
|
|
|
namespace
|
|
{
|
|
typedef ::std::unordered_map< basegfx::SystemDependentData_SharedPtr, sal_uInt32 > EntryMap;
|
|
|
|
class SystemDependentDataBuffer final : public basegfx::SystemDependentDataManager
|
|
{
|
|
private:
|
|
std::mutex m_aMutex;
|
|
std::unique_ptr<AutoTimer> maTimer;
|
|
EntryMap maEntries;
|
|
|
|
DECL_LINK(implTimeoutHdl, Timer *, void);
|
|
|
|
public:
|
|
SystemDependentDataBuffer(const char* pDebugName)
|
|
: maTimer(std::make_unique<AutoTimer>(pDebugName))
|
|
{
|
|
maTimer->SetTimeout(1000);
|
|
maTimer->SetInvokeHandler(LINK(this, SystemDependentDataBuffer, implTimeoutHdl));
|
|
}
|
|
|
|
virtual ~SystemDependentDataBuffer() override
|
|
{
|
|
flushAll();
|
|
}
|
|
|
|
void startUsage(basegfx::SystemDependentData_SharedPtr& rData) override
|
|
{
|
|
std::unique_lock aGuard(m_aMutex);
|
|
EntryMap::iterator aFound(maEntries.find(rData));
|
|
|
|
if(aFound == maEntries.end())
|
|
{
|
|
if(maTimer && !maTimer->IsActive())
|
|
{
|
|
maTimer->Start();
|
|
}
|
|
|
|
maEntries[rData] = rData->calculateCombinedHoldCyclesInSeconds();
|
|
}
|
|
}
|
|
|
|
void endUsage(basegfx::SystemDependentData_SharedPtr& rData) override
|
|
{
|
|
// tdf#163428 prepare space for entry to hold to avoid early deletion
|
|
basegfx::SystemDependentData_SharedPtr aToBeDeletedEntry;
|
|
|
|
{
|
|
// lock m_aMutex in own scope
|
|
std::unique_lock aGuard(m_aMutex);
|
|
EntryMap::iterator aFound(maEntries.find(rData));
|
|
|
|
if(aFound != maEntries.end())
|
|
{
|
|
// tdf#163428 ensure to hold/not delete entry while m_aMutex is locked
|
|
aToBeDeletedEntry = aFound->first;
|
|
|
|
// remove from list
|
|
maEntries.erase(aFound);
|
|
}
|
|
}
|
|
|
|
// tdf#163428 aToBeDeletedEntry will be destructed, thus the
|
|
// entry referenced by SharedPtr may be deleted now
|
|
}
|
|
|
|
void touchUsage(basegfx::SystemDependentData_SharedPtr& rData) override
|
|
{
|
|
std::unique_lock aGuard(m_aMutex);
|
|
EntryMap::iterator aFound(maEntries.find(rData));
|
|
|
|
if(aFound != maEntries.end())
|
|
{
|
|
aFound->second = rData->calculateCombinedHoldCyclesInSeconds();
|
|
}
|
|
}
|
|
|
|
void flushAll() override
|
|
{
|
|
std::unique_lock aGuard(m_aMutex);
|
|
|
|
if(maTimer)
|
|
{
|
|
maTimer->Stop();
|
|
maTimer.reset();
|
|
}
|
|
|
|
maEntries.clear();
|
|
}
|
|
};
|
|
|
|
IMPL_LINK_NOARG(SystemDependentDataBuffer, implTimeoutHdl, Timer *, void)
|
|
{
|
|
// tdf#163428 prepare a temporary list for SystemDependentData_SharedPtr
|
|
// entries that need to be removed from the unordered_map, but do not need
|
|
// locking m_aMutex
|
|
::std::vector<basegfx::SystemDependentData_SharedPtr> aToBeDeletedEntries;
|
|
|
|
{
|
|
// lock m_aMutex in own scope
|
|
std::unique_lock aGuard(m_aMutex);
|
|
EntryMap::iterator aIter(maEntries.begin());
|
|
|
|
while(aIter != maEntries.end())
|
|
{
|
|
if(aIter->second)
|
|
{
|
|
aIter->second--;
|
|
++aIter;
|
|
}
|
|
else
|
|
{
|
|
// tdf#163428 Do not potentially directly delete SystemDependentData_SharedPtr
|
|
// entries in the unordered_map (maEntries). That can/would happen when the shared_ptr
|
|
// has only one reference left. This can cause problems, e.g. we have
|
|
// BufferedData_ModifiedBitmapEx which can contain a Bitmap with added
|
|
// DataBuffers itself, thus deleting this here *directly* would trigger e.g.
|
|
// endUsage above. That would then block when we would still hold m_aMutex.
|
|
// This can potentially be the case with other derivations of
|
|
// basegfx::SystemDependentData, too.
|
|
// To avoid this, protect the part that needs protection by the m_aMutex, that
|
|
// is the modification of the unordered_map itself. Cleaning up the list entries can be
|
|
// done after this, without holding the lock.
|
|
// Thus, remember the SystemDependentData_SharedPtr in a temporary list by adding
|
|
// a reference/shared_ptr to it to guarantee it gets not deleted when it gets removed
|
|
// from the unordered_map.
|
|
// Anyways, only locking while manipulating the list is better, destruction of
|
|
// objects may be expensive and hold m_aMutex longer than necessary.
|
|
aToBeDeletedEntries.push_back(aIter->first);
|
|
|
|
// remove from list. This decrements the shared_ptr, but delete is avoided
|
|
aIter = maEntries.erase(aIter);
|
|
}
|
|
}
|
|
|
|
if (maEntries.empty())
|
|
maTimer->Stop();
|
|
}
|
|
|
|
// tdf#163428 here aToBeDeletedEntries will be destroyed, the entries will be
|
|
// decremented and potentially deleted. These are of type SystemDependentData_SharedPtr,
|
|
// so we do not need to do anything explicitly here
|
|
}
|
|
}
|
|
|
|
basegfx::SystemDependentDataManager& ImplGetSystemDependentDataManager()
|
|
{
|
|
static SystemDependentDataBuffer aSystemDependentDataBuffer("vcl SystemDependentDataBuffer aSystemDependentDataBuffer");
|
|
|
|
return aSystemDependentDataBuffer;
|
|
}
|
|
|
|
/// Returns either the application window, or the default GL context window
|
|
vcl::Window* ImplGetDefaultWindow()
|
|
{
|
|
ImplSVData* pSVData = ImplGetSVData();
|
|
if (pSVData->maFrameData.mpAppWin)
|
|
return pSVData->maFrameData.mpAppWin;
|
|
else
|
|
return ImplGetDefaultContextWindow();
|
|
}
|
|
|
|
/// returns the default window created to hold the persistent VCL GL context.
|
|
vcl::Window *ImplGetDefaultContextWindow()
|
|
{
|
|
ImplSVData* pSVData = ImplGetSVData();
|
|
|
|
// Double check locking on mpDefaultWin.
|
|
if ( !pSVData->mpDefaultWin )
|
|
{
|
|
SolarMutexGuard aGuard;
|
|
|
|
if (!pSVData->mpDefaultWin && !pSVData->mbDeInit)
|
|
{
|
|
try
|
|
{
|
|
SAL_INFO( "vcl", "ImplGetDefaultWindow(): No AppWindow" );
|
|
|
|
pSVData->mpDefaultWin = VclPtr<WorkWindow>::Create(nullptr, WB_DEFAULTWIN);
|
|
pSVData->mpDefaultWin->SetText( u"VCL ImplGetDefaultWindow"_ustr );
|
|
}
|
|
catch (const css::uno::Exception&)
|
|
{
|
|
TOOLS_WARN_EXCEPTION("vcl", "unable to create Default Window");
|
|
}
|
|
}
|
|
}
|
|
|
|
return pSVData->mpDefaultWin;
|
|
}
|
|
|
|
const std::locale& ImplGetResLocale()
|
|
{
|
|
ImplSVData* pSVData = ImplGetSVData();
|
|
if (!pSVData->mbResLocaleSet || comphelper::LibreOfficeKit::isActive())
|
|
{
|
|
pSVData->maResLocale = Translate::Create("vcl");
|
|
pSVData->mbResLocaleSet = true;
|
|
}
|
|
return pSVData->maResLocale;
|
|
}
|
|
|
|
OUString VclResId(TranslateId aId)
|
|
{
|
|
return Translate::get(aId, ImplGetResLocale());
|
|
}
|
|
|
|
const FieldUnitStringList& ImplGetFieldUnits()
|
|
{
|
|
ImplSVData* pSVData = ImplGetSVData();
|
|
if( pSVData->maCtrlData.maFieldUnitStrings.empty() )
|
|
{
|
|
sal_uInt32 nUnits = std::size(SV_FUNIT_STRINGS);
|
|
pSVData->maCtrlData.maFieldUnitStrings.reserve( nUnits );
|
|
for (sal_uInt32 i = 0; i < nUnits; i++)
|
|
{
|
|
std::pair<OUString, FieldUnit> aElement(VclResId(SV_FUNIT_STRINGS[i].first), SV_FUNIT_STRINGS[i].second);
|
|
pSVData->maCtrlData.maFieldUnitStrings.push_back( aElement );
|
|
}
|
|
}
|
|
return pSVData->maCtrlData.maFieldUnitStrings;
|
|
}
|
|
|
|
namespace vcl
|
|
{
|
|
FieldUnit EnglishStringToMetric(std::u16string_view rEnglishMetricString)
|
|
{
|
|
sal_uInt32 nUnits = std::size(SV_FUNIT_STRINGS);
|
|
for (sal_uInt32 i = 0; i < nUnits; ++i)
|
|
{
|
|
if (o3tl::equalsAscii(rEnglishMetricString, SV_FUNIT_STRINGS[i].first.getId()))
|
|
return SV_FUNIT_STRINGS[i].second;
|
|
}
|
|
return FieldUnit::NONE;
|
|
}
|
|
}
|
|
|
|
const FieldUnitStringList& ImplGetCleanedFieldUnits()
|
|
{
|
|
ImplSVData* pSVData = ImplGetSVData();
|
|
if( pSVData->maCtrlData.maCleanUnitStrings.empty() )
|
|
{
|
|
const FieldUnitStringList& rUnits = ImplGetFieldUnits();
|
|
size_t nUnits = rUnits.size();
|
|
pSVData->maCtrlData.maCleanUnitStrings.reserve(nUnits);
|
|
for (size_t i = 0; i < nUnits; ++i)
|
|
{
|
|
OUString aUnit(rUnits[i].first);
|
|
aUnit = aUnit.replaceAll(" ", "");
|
|
aUnit = aUnit.toAsciiLowerCase();
|
|
std::pair<OUString, FieldUnit> aElement(aUnit, rUnits[i].second);
|
|
pSVData->maCtrlData.maCleanUnitStrings.push_back(aElement);
|
|
}
|
|
}
|
|
return pSVData->maCtrlData.maCleanUnitStrings;
|
|
}
|
|
|
|
DockingManager* ImplGetDockingManager()
|
|
{
|
|
ImplSVData* pSVData = ImplGetSVData();
|
|
if ( !pSVData->mpDockingManager )
|
|
pSVData->mpDockingManager.reset(new DockingManager());
|
|
|
|
return pSVData->mpDockingManager.get();
|
|
}
|
|
|
|
BlendFrameCache* ImplGetBlendFrameCache()
|
|
{
|
|
ImplSVData* pSVData = ImplGetSVData();
|
|
if ( !pSVData->mpBlendFrameCache)
|
|
pSVData->mpBlendFrameCache.reset( new BlendFrameCache() );
|
|
|
|
return pSVData->mpBlendFrameCache.get();
|
|
}
|
|
|
|
void LocaleConfigurationListener::ConfigurationChanged( utl::ConfigurationBroadcaster*, ConfigurationHints nHint )
|
|
{
|
|
AllSettings::LocaleSettingsChanged( nHint );
|
|
}
|
|
|
|
ImplSVWinData* CreateSVWinData()
|
|
{
|
|
if (!comphelper::LibreOfficeKit::isActive())
|
|
return nullptr;
|
|
|
|
ImplSVWinData* p = new ImplSVWinData;
|
|
|
|
ImplSVData* pSVData = ImplGetSVData();
|
|
assert(pSVData && pSVData->mpWinData);
|
|
|
|
p->mpFocusWin = pSVData->mpWinData->mpFocusWin;
|
|
return p;
|
|
}
|
|
|
|
void DestroySVWinData(ImplSVWinData* pData)
|
|
{
|
|
delete pData;
|
|
}
|
|
|
|
void SetSVWinData(ImplSVWinData* pSVWinData)
|
|
{
|
|
if (!comphelper::LibreOfficeKit::isActive())
|
|
return;
|
|
|
|
ImplSVData* pSVData = ImplGetSVData();
|
|
assert(pSVData != nullptr);
|
|
|
|
if (pSVData->mpWinData == pSVWinData)
|
|
return;
|
|
|
|
// If current one is the static, clean it up to avoid having lingering references.
|
|
if (pSVData->mpWinData == &private_aImplSVWinData::get())
|
|
{
|
|
pSVData->mpWinData->mpFocusWin.reset();
|
|
}
|
|
|
|
pSVData->mpWinData = pSVWinData;
|
|
if (pSVData->mpWinData == nullptr)
|
|
{
|
|
pSVData->mpWinData = &private_aImplSVWinData::get(); // Never leave it null.
|
|
}
|
|
}
|
|
|
|
ImplSVData::ImplSVData()
|
|
{
|
|
mpHelpData = &private_aImplSVHelpData::get();
|
|
mpWinData = &private_aImplSVWinData::get();
|
|
}
|
|
|
|
void ImplSVData::dropCaches()
|
|
{
|
|
// we are iterating over a map and doing erase while inside a loop which is doing erase
|
|
// hence we can't use clear() here
|
|
maGDIData.maScaleCache.remove_if([](const lru_scale_cache::key_value_pair_t&)
|
|
{ return true; });
|
|
|
|
maGDIData.maThemeDrawCommandsCache.clear();
|
|
maGDIData.maThemeImageCache.clear();
|
|
}
|
|
|
|
void ImplSVData::dumpState(rtl::OStringBuffer &rState)
|
|
{
|
|
rState.append("\nScaleCache:\t");
|
|
rState.append(static_cast<sal_Int32>(maGDIData.maScaleCache.size()));
|
|
rState.append("\t items:");
|
|
|
|
for (auto it = maGDIData.maScaleCache.begin();
|
|
it != maGDIData.maScaleCache.end(); ++it)
|
|
{
|
|
rState.append("\n\t");
|
|
rState.append(static_cast<sal_Int32>(it->first.maDestSize.Width()));
|
|
rState.append("x");
|
|
rState.append(static_cast<sal_Int32>(it->first.maDestSize.Height()));
|
|
}
|
|
}
|
|
|
|
ImplSVHelpData* CreateSVHelpData()
|
|
{
|
|
if (!comphelper::LibreOfficeKit::isActive())
|
|
return nullptr;
|
|
|
|
ImplSVHelpData* pNewData = new ImplSVHelpData;
|
|
|
|
// Set options set globally
|
|
ImplSVHelpData& aStaticHelpData = private_aImplSVHelpData::get();
|
|
pNewData->mbContextHelp = aStaticHelpData.mbContextHelp;
|
|
pNewData->mbExtHelp = aStaticHelpData.mbExtHelp;
|
|
pNewData->mbExtHelpMode = aStaticHelpData.mbExtHelpMode;
|
|
pNewData->mbOldBalloonMode = aStaticHelpData.mbOldBalloonMode;
|
|
pNewData->mbBalloonHelp = aStaticHelpData.mbBalloonHelp;
|
|
pNewData->mbQuickHelp = aStaticHelpData.mbQuickHelp;
|
|
|
|
return pNewData;
|
|
}
|
|
|
|
void DestroySVHelpData(ImplSVHelpData* pSVHelpData)
|
|
{
|
|
if (!comphelper::LibreOfficeKit::isActive())
|
|
return;
|
|
|
|
// Change the SVData's help date if necessary
|
|
if(ImplGetSVData()->mpHelpData == pSVHelpData)
|
|
{
|
|
ImplGetSVData()->mpHelpData = &private_aImplSVHelpData::get();
|
|
}
|
|
|
|
if(pSVHelpData)
|
|
{
|
|
ImplDestroyHelpWindow(*pSVHelpData, false);
|
|
delete pSVHelpData;
|
|
}
|
|
}
|
|
|
|
void SetSVHelpData(ImplSVHelpData* pSVHelpData)
|
|
{
|
|
if (!comphelper::LibreOfficeKit::isActive())
|
|
return;
|
|
|
|
ImplSVData* pSVData = ImplGetSVData();
|
|
if (pSVData->mpHelpData == pSVHelpData)
|
|
return;
|
|
|
|
// If current one is the static, clean it up to avoid having lingering references.
|
|
if (pSVData->mpHelpData == &private_aImplSVHelpData::get())
|
|
{
|
|
pSVData->mpHelpData->mpHelpWin.reset();
|
|
}
|
|
|
|
pSVData->mpHelpData = pSVHelpData;
|
|
if (pSVData->mpHelpData == nullptr)
|
|
{
|
|
pSVData->mpHelpData = &private_aImplSVHelpData::get(); // Never leave it null.
|
|
}
|
|
}
|
|
|
|
ImplSVHelpData& ImplGetSVHelpData()
|
|
{
|
|
ImplSVData* pSVData = ImplGetSVData();
|
|
if(pSVData->mpHelpData)
|
|
{
|
|
return *pSVData->mpHelpData;
|
|
}
|
|
else
|
|
{
|
|
return private_aImplSVHelpData::get();
|
|
}
|
|
}
|
|
|
|
ImplSVData::~ImplSVData() {}
|
|
|
|
ImplSVAppData::ImplSVAppData()
|
|
{
|
|
m_bUseSystemLoop = getenv("SAL_USE_SYSTEM_LOOP") != nullptr;
|
|
SAL_WARN_IF(m_bUseSystemLoop, "vcl.schedule", "Overriding to run LO on system event loop!");
|
|
}
|
|
|
|
ImplSVAppData::~ImplSVAppData() {}
|
|
ImplSVGDIData::~ImplSVGDIData() {}
|
|
ImplSVFrameData::~ImplSVFrameData() {}
|
|
ImplSVWinData::~ImplSVWinData() {}
|
|
ImplSVHelpData::~ImplSVHelpData() {}
|
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|