Files
libreoffice/tools/source/ref/errinf.cxx
Caolán McNamara ff82f13b68 drop unused error strings
includes ERRCODE_SFXMSG_STYLEREPLACE

which has the knock on effect that the flags argument
can be removed from a bunch of methods

Change-Id: I72b58bc2a19376bb4609e61aa44e71f734efb333
2017-02-08 12:18:41 +00:00

351 lines
9.5 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 <limits.h>
#include <tools/errinf.hxx>
#include <rtl/strbuf.hxx>
#include <osl/diagnose.h>
#include <vcl/window.hxx>
#include <vector>
class ErrorHandler;
namespace {
typedef void (* DisplayFnPtr)();
}
struct EDcrData
{
public:
std::vector<ErrorHandler*> errorHandlers;
std::vector<ErrorContext*> contexts;
DisplayFnPtr pDsp;
bool bIsWindowDsp;
DynamicErrorInfo *ppDcr[ERRCODE_DYNAMIC_COUNT];
sal_uInt16 nNextDcr;
EDcrData();
};
struct TheEDcrData: public rtl::Static<EDcrData, TheEDcrData> {};
class DynamicErrorInfo_Impl
{
sal_uIntPtr lErrId;
sal_uInt16 nMask;
void RegisterEDcr(DynamicErrorInfo *);
static void UnRegisterEDcr(DynamicErrorInfo const *);
static ErrorInfo *GetDynamicErrorInfo(sal_uIntPtr lId);
friend class DynamicErrorInfo;
friend class ErrorInfo;
};
EDcrData::EDcrData()
: pDsp(nullptr)
, bIsWindowDsp(false)
, nNextDcr(0)
{
for(DynamicErrorInfo*& rp : ppDcr)
rp = nullptr;
}
void DynamicErrorInfo_Impl::RegisterEDcr(DynamicErrorInfo *pDcr)
{
// Register dynamic identifier
EDcrData& rData = TheEDcrData::get();
lErrId = (((sal_uIntPtr)rData.nNextDcr + 1) << ERRCODE_DYNAMIC_SHIFT) +
pDcr->GetErrorCode();
if(rData.ppDcr[rData.nNextDcr])
{
delete rData.ppDcr[rData.nNextDcr];
}
rData.ppDcr[rData.nNextDcr] = pDcr;
if(++rData.nNextDcr>=ERRCODE_DYNAMIC_COUNT)
rData.nNextDcr=0;
}
void DynamicErrorInfo_Impl::UnRegisterEDcr(DynamicErrorInfo const *pDcr)
{
DynamicErrorInfo **ppDcr = TheEDcrData::get().ppDcr;
sal_uIntPtr lIdx = (((sal_uIntPtr)(*pDcr) & ERRCODE_DYNAMIC_MASK) >> ERRCODE_DYNAMIC_SHIFT) - 1;
DBG_ASSERT(ppDcr[lIdx]==pDcr,"ErrHdl: Error nicht gefunden");
if(ppDcr[lIdx]==pDcr)
ppDcr[lIdx]=nullptr;
}
ErrorInfo::~ErrorInfo()
{
}
ErrorInfo *ErrorInfo::GetErrorInfo(sal_uIntPtr lId)
{
if(lId & ERRCODE_DYNAMIC_MASK)
return DynamicErrorInfo_Impl::GetDynamicErrorInfo(lId);
else
return new ErrorInfo(lId);
}
DynamicErrorInfo::operator sal_uIntPtr() const
{
return pImpl->lErrId;
}
DynamicErrorInfo::DynamicErrorInfo(sal_uIntPtr lArgUserId, sal_uInt16 nMask)
: ErrorInfo(lArgUserId),
pImpl(new DynamicErrorInfo_Impl)
{
pImpl->RegisterEDcr(this);
pImpl->nMask=nMask;
}
DynamicErrorInfo::~DynamicErrorInfo()
{
DynamicErrorInfo_Impl::UnRegisterEDcr(this);
}
ErrorInfo* DynamicErrorInfo_Impl::GetDynamicErrorInfo(sal_uIntPtr lId)
{
sal_uIntPtr lIdx = ((lId & ERRCODE_DYNAMIC_MASK)>>ERRCODE_DYNAMIC_SHIFT)-1;
DynamicErrorInfo* pDcr = TheEDcrData::get().ppDcr[lIdx];
if(pDcr && (sal_uIntPtr)(*pDcr)==lId)
return pDcr;
else
return new ErrorInfo(lId & ~ERRCODE_DYNAMIC_MASK);
}
sal_uInt16 DynamicErrorInfo::GetDialogMask() const
{
return pImpl->nMask;
}
StringErrorInfo::StringErrorInfo(
sal_uIntPtr UserId, const OUString& aStringP, sal_uInt16 nMask)
: DynamicErrorInfo(UserId, nMask), aString(aStringP)
{
}
class ErrorHandler_Impl
{
public:
static bool CreateString(const ErrorInfo*, OUString&);
};
static void aDspFunc(const OUString &rErr, const OUString &rAction)
{
OStringBuffer aErr("Aktion: ");
aErr.append(OUStringToOString(rAction, RTL_TEXTENCODING_ASCII_US));
aErr.append(" Fehler: ");
aErr.append(OUStringToOString(rErr, RTL_TEXTENCODING_ASCII_US));
OSL_FAIL(aErr.getStr());
}
// FIXME: this is a horrible reverse dependency on VCL
struct ErrorContextImpl
{
vcl::Window *pWin; // should be VclPtr for strong lifecycle
};
ErrorContext::ErrorContext(vcl::Window *pWinP)
: pImpl( new ErrorContextImpl )
{
pImpl->pWin = pWinP;
TheEDcrData::get().contexts.insert(TheEDcrData::get().contexts.begin(), this);
}
ErrorContext::~ErrorContext()
{
auto &rContexts = TheEDcrData::get().contexts;
rContexts.erase( ::std::remove(rContexts.begin(), rContexts.end(), this), rContexts.end());
}
ErrorContext *ErrorContext::GetContext()
{
return TheEDcrData::get().contexts.empty() ? nullptr : TheEDcrData::get().contexts.front();
}
ErrorHandler::ErrorHandler()
{
EDcrData &rData = TheEDcrData::get();
rData.errorHandlers.insert(rData.errorHandlers.begin(), this);
if(!rData.pDsp)
RegisterDisplay(&aDspFunc);
}
ErrorHandler::~ErrorHandler()
{
auto &rErrorHandlers = TheEDcrData::get().errorHandlers;
rErrorHandlers.erase( ::std::remove(rErrorHandlers.begin(), rErrorHandlers.end(), this), rErrorHandlers.end());
}
vcl::Window* ErrorContext::GetParent()
{
return pImpl ? pImpl->pWin : nullptr;
}
void ErrorHandler::RegisterDisplay(WindowDisplayErrorFunc *aDsp)
{
EDcrData &rData = TheEDcrData::get();
rData.bIsWindowDsp = true;
rData.pDsp = reinterpret_cast< DisplayFnPtr >(aDsp);
}
void ErrorHandler::RegisterDisplay(BasicDisplayErrorFunc *aDsp)
{
EDcrData &rData = TheEDcrData::get();
rData.bIsWindowDsp = false;
rData.pDsp = reinterpret_cast< DisplayFnPtr >(aDsp);
}
/** Handles an error.
If nFlags is not set, the DynamicErrorInfo flags or the
resource flags will be used.
Thus:
1. nFlags,
2. Resource Flags
3. Dynamic Flags
4. Default ERRCODE_BUTTON_OK, ERRCODE_MSG_ERROR
@param lId error id
@param nFlags error flags.
@param bJustCreateString ???
@param rError ???
@return ???
*/
sal_uInt16 ErrorHandler::HandleError_Impl(
sal_uIntPtr lId, sal_uInt16 nFlags, bool bJustCreateString, OUString & rError)
{
OUString aErr;
OUString aAction;
if(!lId || lId == ERRCODE_ABORT)
return 0;
EDcrData &rData = TheEDcrData::get();
vcl::Window *pParent = nullptr;
ErrorInfo *pInfo = ErrorInfo::GetErrorInfo(lId);
if (!rData.contexts.empty())
{
rData.contexts.front()->GetString(pInfo->GetErrorCode(), aAction);
// Remove parent from context
for(ErrorContext *pCtx : rData.contexts)
if(pCtx->GetParent())
{
pParent=pCtx->GetParent();
break;
}
}
bool bWarning = ((lId & ERRCODE_WARNING_MASK) == ERRCODE_WARNING_MASK);
sal_uInt16 nErrFlags = ERRCODE_BUTTON_DEF_OK | ERRCODE_BUTTON_OK;
if (bWarning)
nErrFlags |= ERRCODE_MSG_WARNING;
else
nErrFlags |= ERRCODE_MSG_ERROR;
DynamicErrorInfo* pDynPtr=dynamic_cast<DynamicErrorInfo*>(pInfo);
if(pDynPtr)
{
sal_uInt16 nDynFlags = pDynPtr->GetDialogMask();
if( nDynFlags )
nErrFlags = nDynFlags;
}
if(ErrorHandler_Impl::CreateString(pInfo,aErr))
{
if (bJustCreateString)
{
rError = aErr;
return 1;
}
else
{
if(!rData.pDsp)
{
OStringBuffer aStr("Action: ");
aStr.append(OUStringToOString(aAction, RTL_TEXTENCODING_ASCII_US));
aStr.append("\nFehler: ");
aStr.append(OUStringToOString(aErr, RTL_TEXTENCODING_ASCII_US));
OSL_FAIL(aStr.getStr());
}
else
{
delete pInfo;
if(!rData.bIsWindowDsp)
{
(*reinterpret_cast<BasicDisplayErrorFunc*>(rData.pDsp))(aErr,aAction);
return 0;
}
else
{
if (nFlags != USHRT_MAX)
nErrFlags = nFlags;
return (*reinterpret_cast<WindowDisplayErrorFunc*>(rData.pDsp))(
pParent, nErrFlags, aErr, aAction);
}
}
}
}
OSL_FAIL("Error nicht behandelt");
// Error 1 is General Error in the Sfx
if(pInfo->GetErrorCode()!=1)
{
HandleError_Impl(1, USHRT_MAX, bJustCreateString, rError);
}
else
{
OSL_FAIL("Error 1 nicht gehandeled");
}
delete pInfo;
return 0;
}
// static
bool ErrorHandler::GetErrorString(sal_uIntPtr lId, OUString& rStr)
{
return (bool)HandleError_Impl( lId, USHRT_MAX, true, rStr );
}
/** Handles an error.
@see ErrorHandler::HandleError_Impl
*/
sal_uInt16 ErrorHandler::HandleError(sal_uIntPtr lId, sal_uInt16 nFlags)
{
OUString aDummy;
return HandleError_Impl( lId, nFlags, false, aDummy );
}
bool ErrorHandler_Impl::CreateString(const ErrorInfo* pInfo, OUString& rStr)
{
for(const ErrorHandler *pHdl : TheEDcrData::get().errorHandlers)
{
if(pHdl->CreateString(pInfo, rStr))
return true;
}
return false;
}
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */