blob: 3a5212b9367395f33e706f4e9da0d355b91743b4 [file] [log] [blame]
/**************************************************************
*
* 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
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*
*************************************************************/
// MARKER(update_precomp.py): autogen include statement, do not remove
#include "precompiled_vcl.hxx"
#include <tools/list.hxx>
#include <tools/debug.hxx>
#include <accel.h>
#include <vcl/accel.hxx>
#include <accmgr.hxx>
// =======================================================================
DECLARE_LIST( ImplAccelList, Accelerator* )
// =======================================================================
DBG_NAMEEX( Accelerator )
// =======================================================================
ImplAccelManager::~ImplAccelManager()
{
if ( mpAccelList )
delete mpAccelList;
if ( mpSequenceList )
delete mpSequenceList;
}
// -----------------------------------------------------------------------
sal_Bool ImplAccelManager::InsertAccel( Accelerator* pAccel )
{
if ( !mpAccelList )
mpAccelList = new ImplAccelList;
else
{
// Gibts den schon ?
if ( mpAccelList->GetPos( pAccel ) != LIST_ENTRY_NOTFOUND )
return sal_False;
}
// Am Anfang der Liste einfuegen
mpAccelList->Insert( pAccel, (sal_uLong)0 );
return sal_True;
}
// -----------------------------------------------------------------------
void ImplAccelManager::RemoveAccel( Accelerator* pAccel )
{
// Haben wir ueberhaupt eine Liste ?
if ( !mpAccelList )
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 ( mpSequenceList )
{
for (sal_uInt16 i = 0; i < pAccel->GetItemCount(); ++i)
{
Accelerator* pSubAccel = pAccel->GetAccel(pAccel->GetItemId(i));
if ( mpSequenceList->GetPos( pSubAccel ) != LIST_ENTRY_NOTFOUND )
{
EndSequence( true );
break;
}
}
}
// Raus damit
mpAccelList->Remove( pAccel );
}
// -----------------------------------------------------------------------
void ImplAccelManager::EndSequence( sal_Bool bCancel )
{
// Sind wir ueberhaupt in einer Sequenz ?
if ( !mpSequenceList )
return;
// Alle Deactivate-Handler der Acceleratoren in der Sequenz rufen
Accelerator* pTempAccel = mpSequenceList->First();
while( pTempAccel )
{
sal_Bool bDel = sal_False;
pTempAccel->mbIsCancel = bCancel;
pTempAccel->mpDel = &bDel;
pTempAccel->Deactivate();
if ( !bDel )
{
pTempAccel->mbIsCancel = sal_False;
pTempAccel->mpDel = NULL;
}
pTempAccel = mpSequenceList->Next();
}
// Sequenz-Liste loeschen
delete mpSequenceList;
mpSequenceList = NULL;
}
// -----------------------------------------------------------------------
sal_Bool ImplAccelManager::IsAccelKey( const KeyCode& rKeyCode, sal_uInt16 nRepeat )
{
Accelerator* pAccel;
// Haben wir ueberhaupt Acceleratoren ??
if ( !mpAccelList )
return sal_False;
if ( !mpAccelList->Count() )
return sal_False;
// Sind wir in einer Sequenz ?
if ( mpSequenceList )
{
pAccel = mpSequenceList->GetObject( 0 );
DBG_CHKOBJ( pAccel, Accelerator, NULL );
// Nicht Gefunden ?
if ( !pAccel )
{
// Sequenz abbrechen
FlushAccel();
return sal_False;
}
// Ist der Eintrag da drin ?
ImplAccelEntry* pEntry = pAccel->ImplGetAccelData( rKeyCode );
if ( pEntry )
{
Accelerator* pNextAccel = pEntry->mpAccel;
// Ist da ein Accelerator hinter ?
if ( pNextAccel )
{
DBG_CHKOBJ( pNextAccel, Accelerator, NULL );
mpSequenceList->Insert( pNextAccel, (sal_uLong)0 );
// Activate-Handler vom Neuen rufen
pNextAccel->Activate();
return sal_True;
}
else
{
// Hat ihn schon !
if ( pEntry->mbEnabled )
{
// Sequence beenden (Deactivate-Handler vorher rufen)
EndSequence();
// Dem Accelerator das aktuelle Item setzen
// und Handler rufen
sal_Bool bDel = sal_False;
pAccel->maCurKeyCode = rKeyCode;
pAccel->mnCurId = pEntry->mnId;
pAccel->mnCurRepeat = nRepeat;
pAccel->mpDel = &bDel;
pAccel->Select();
// Hat Accel den Aufruf ueberlebt
if ( !bDel )
{
DBG_CHKOBJ( pAccel, Accelerator, NULL );
pAccel->maCurKeyCode = KeyCode();
pAccel->mnCurId = 0;
pAccel->mnCurRepeat = 0;
pAccel->mpDel = NULL;
}
return sal_True;
}
else
{
// Sequenz abbrechen, weil Acceleraor disabled
// Taste wird weitergeleitet (ans System)
FlushAccel();
return sal_False;
}
}
}
else
{
// Sequenz abbrechen wegen falscher Taste
FlushAccel();
return sal_False;
}
}
// Durch die Liste der Acceleratoren wuehlen
pAccel = mpAccelList->First();
while ( pAccel )
{
DBG_CHKOBJ( pAccel, Accelerator, NULL );
// Ist der Eintrag da drin ?
ImplAccelEntry* pEntry = pAccel->ImplGetAccelData( rKeyCode );
if ( pEntry )
{
Accelerator* pNextAccel = pEntry->mpAccel;
// Ist da ein Accelerator hinter ?
if ( pNextAccel )
{
DBG_CHKOBJ( pNextAccel, Accelerator, NULL );
// Sequenz-Liste erzeugen
mpSequenceList = new ImplAccelList;
mpSequenceList->Insert( pAccel, (sal_uLong)0 );
mpSequenceList->Insert( pNextAccel, (sal_uLong)0 );
// Activate-Handler vom Neuen rufen
pNextAccel->Activate();
return sal_True;
}
else
{
// Hat ihn schon !
if ( pEntry->mbEnabled )
{
// Activate/Deactivate-Handler vorher rufen
pAccel->Activate();
pAccel->Deactivate();
// Dem Accelerator das aktuelle Item setzen
// und Handler rufen
sal_Bool bDel = sal_False;
pAccel->maCurKeyCode = rKeyCode;
pAccel->mnCurId = pEntry->mnId;
pAccel->mnCurRepeat = nRepeat;
pAccel->mpDel = &bDel;
pAccel->Select();
// Hat Accel den Aufruf ueberlebt
if ( !bDel )
{
DBG_CHKOBJ( pAccel, Accelerator, NULL );
pAccel->maCurKeyCode = KeyCode();
pAccel->mnCurId = 0;
pAccel->mnCurRepeat = 0;
pAccel->mpDel = NULL;
}
return sal_True;
}
else
return sal_False;
}
}
// Nicht gefunden, vielleicht im naechsten Accelerator
pAccel = mpAccelList->Next();
}
return sal_False;
}