/* -*- 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 <accel.hxx>
#include <accmgr.hxx>
 
#include <algorithm>
 
ImplAccelManager::~ImplAccelManager()
{
}
 
bool ImplAccelManager::InsertAccel( Accelerator* pAccel )
{
    if ( !mxAccelList ) {
        mxAccelList.emplace();
    } else {
        for (Accelerator* i : *mxAccelList) {
            if ( i == pAccel ) {
                return false;
            }
        }
    }
 
    mxAccelList->insert( mxAccelList->begin(), pAccel );
    return true;
}
 
void ImplAccelManager::RemoveAccel( Accelerator const * pAccel )
{
    // do we have a list ?
    if ( !mxAccelList )
        return;
 
    //e.g. #i90599#. Someone starts typing a sequence in a dialog, but doesn't
    //end it, and then closes the dialog, deleting the accelerators. So if
    //we're removing an accelerator that a sub-accelerator which is in the
    //sequence list, throw away the entire sequence
    if ( mxSequenceList ) {
        for (sal_uInt16 i = 0; i < pAccel->GetItemCount(); ++i) {
            Accelerator* pSubAccel = pAccel->GetAccel( pAccel->GetItemId(i) );
            for (Accelerator* j : *mxSequenceList) {
                if ( j == pSubAccel ) {
                    EndSequence();
                    i = pAccel->GetItemCount();
                    break;
                }
            }
        }
    }
 
    // throw it away
    auto it = std::find(mxAccelList->begin(), mxAccelList->end(), pAccel);
    if (it != mxAccelList->end())
        mxAccelList->erase( it );
}
 
void ImplAccelManager::EndSequence()
{
    // are we in a list ?
    if ( !mxSequenceList )
        return;
 
    for (Accelerator* pTempAccel : *mxSequenceList)
    {
        pTempAccel->mpDel = nullptr;
    }
 
    // delete sequence-list
    mxSequenceList.reset();
}
 
bool ImplAccelManager::IsAccelKey( const vcl::KeyCode& rKeyCode )
{
    Accelerator* pAccel;
 
    // do we have accelerators ??
    if ( !mxAccelList )
        return false;
    if ( mxAccelList->empty() )
        return false;
 
    // are we in a sequence ?
    if ( mxSequenceList )
    {
        pAccel = mxSequenceList->empty() ? nullptr : (*mxSequenceList)[ 0 ];
 
        // not found ?
        if ( !pAccel )
        {
            // abort sequence
            FlushAccel();
            return false;
        }
 
        // can the entry be found ?
        ImplAccelEntry* pEntry = pAccel->ImplGetAccelData( rKeyCode );
        if ( pEntry )
        {
            Accelerator* pNextAccel = pEntry->mpAccel;
 
            // is an accelerator coupled ?
            if ( pNextAccel )
            {
 
                mxSequenceList->insert( mxSequenceList->begin(), pNextAccel );
 
                // call Activate-Handler of the new one
                pNextAccel->Activate();
                return true;
            }
            else
            {
                // it is there already !
                if ( pEntry->mbEnabled )
                {
                    // stop sequence (first call deactivate-handler)
                    EndSequence();
 
                    // set accelerator of the actual item
                    // and call the handler
                    bool bDel = false;
                    pAccel->mnCurId         = pEntry->mnId;
                    pAccel->mpDel           = &bDel;
                    pAccel->Select();
 
                    // did the accelerator survive the call
                    if ( !bDel )
                    {
                        pAccel->mnCurId         = 0;
                        pAccel->mpDel           = nullptr;
                    }
 
                    return true;
                }
                else
                {
                    // stop sequence as the accelerator was disabled
                    // transfer the key (to the system)
                    FlushAccel();
                    return false;
                }
            }
        }
        else
        {
            // wrong key => stop sequence
            FlushAccel();
            return false;
        }
    }
 
    // step through the list of accelerators
    for (Accelerator* i : *mxAccelList)
    {
        pAccel = i;
 
        // is the entry contained ?
        ImplAccelEntry* pEntry = pAccel->ImplGetAccelData( rKeyCode );
        if ( pEntry )
        {
            Accelerator* pNextAccel = pEntry->mpAccel;
 
            // is an accelerator assigned ?
            if ( pNextAccel )
            {
 
                // create sequence list
                mxSequenceList.emplace();
                mxSequenceList->insert( mxSequenceList->begin(), pAccel     );
                mxSequenceList->insert( mxSequenceList->begin(), pNextAccel );
 
                // call activate-Handler of the new one
                pNextAccel->Activate();
 
                return true;
            }
            else
            {
                // already assigned !
                if ( pEntry->mbEnabled )
                {
                    // first call activate/deactivate-Handler
                    pAccel->Activate();
 
                    // define accelerator of the actual item
                    // and call the handler
                    bool bDel = false;
                    pAccel->mnCurId         = pEntry->mnId;
                    pAccel->mpDel           = &bDel;
                    pAccel->Select();
 
                    // if the accelerator did survive the call
                    if ( !bDel )
                    {
                        pAccel->mnCurId         = 0;
                        pAccel->mpDel           = nullptr;
                    }
 
                    return true;
                }
                else
                    return false;
            }
        }
    }
 
    return false;
}
 
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */

V506 Pointer to local variable 'bDel' is stored outside the scope of this variable. Such a pointer will become invalid.

V506 Pointer to local variable 'bDel' is stored outside the scope of this variable. Such a pointer will become invalid.