/* -*- 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 #include #include #include #include #include #include #include #include #include #include #include using namespace ::com::sun::star::uno; using namespace ::com::sun::star::accessibility; #define MYTABMASK \ SvLBoxTabFlags( SvLBoxTabFlags::ADJUST_RIGHT | SvLBoxTabFlags::ADJUST_LEFT | SvLBoxTabFlags::ADJUST_CENTER | SvLBoxTabFlags::ADJUST_NUMERIC ) // SvTreeListBox callback void SvTabListBox::SetTabs() { SvTreeListBox::SetTabs(); if( mvTabList.empty() ) return; DBG_ASSERT(!mvTabList.empty(),"TabList ?"); // The tree listbox has now inserted its tabs into the list. Now we // fluff up the list with additional tabs and adjust the rightmost tab // of the tree listbox. // Picking the rightmost tab. // HACK for the explorer! If ViewParent != 0, the first tab of the tree // listbox is calculated by the tre listbox itself! This behavior is // necessary for ButtonsOnRoot, as the explorer does not know in this // case, which additional offset it need to add to the tabs in this mode // -- the tree listbox knows that, though! /* if( !pViewParent ) { SvLBoxTab* pFirstTab = (SvLBoxTab*)aTabs.GetObject( aTabs.Count()-1 ); pFirstTab->SetPos( pTabList[0].GetPos() ); pFirstTab->nFlags &= ~MYTABMASK; pFirstTab->nFlags |= pTabList[0].nFlags; } */ // append all other tabs to the list for( sal_uInt16 nCurTab = 1; nCurTab < sal_uInt16(mvTabList.size()); nCurTab++ ) { SvLBoxTab& rTab = mvTabList[nCurTab]; AddTab( rTab.GetPos(), rTab.nFlags ); } } void SvTabListBox::InitEntry(SvTreeListEntry* pEntry, const OUString& rStr, const Image& rColl, const Image& rExp, SvLBoxButtonKind eButtonKind) { SvTreeListBox::InitEntry(pEntry, rStr, rColl, rExp, eButtonKind); sal_Int32 nIndex = 0; // TODO: verify if nTabCount is always >0 here! const sal_uInt16 nCount = mvTabList.size() - 1; for( sal_uInt16 nToken = 0; nToken < nCount; nToken++ ) { const OUString aToken = GetToken(aCurEntry, nIndex); pEntry->AddItem(o3tl::make_unique(aToken)); } } SvTabListBox::SvTabListBox( vcl::Window* pParent, WinBits nBits ) : SvTreeListBox( pParent, nBits ) { SetHighlightRange(); // select full width } VCL_BUILDER_FACTORY_CONSTRUCTOR(SvTabListBox, WB_TABSTOP) SvTabListBox::~SvTabListBox() { disposeOnce(); } void SvTabListBox::dispose() { mvTabList.clear(); SvTreeListBox::dispose(); } void SvTabListBox::SetTabs(sal_uInt16 nTabs, long const pTabPositions[], MapUnit eMapUnit) { mvTabList.resize(nTabs); MapMode aMMSource( eMapUnit ); MapMode aMMDest( MapUnit::MapPixel ); for( sal_uInt16 nIdx = 0; nIdx < sal_uInt16(mvTabList.size()); nIdx++, pTabPositions++ ) { Size aSize( *pTabPositions, 0 ); aSize = LogicToLogic( aSize, &aMMSource, &aMMDest ); long nNewTab = aSize.Width(); mvTabList[nIdx].SetPos( nNewTab ); mvTabList[nIdx].nFlags=(SvLBoxTabFlags::ADJUST_LEFT| SvLBoxTabFlags::INV_ALWAYS); } SvTreeListBox::nTreeFlags |= SvTreeFlags::RECALCTABS; if( IsUpdateMode() ) Invalidate(); } void SvTabListBox::SetTab( sal_uInt16 nTab,long nValue,MapUnit eMapUnit ) { DBG_ASSERT(nTab= mvTabList.size() ) return; MapMode aMMSource( eMapUnit ); MapMode aMMDest( MapUnit::MapPixel ); Size aSize( nValue, 0 ); aSize = LogicToLogic( aSize, &aMMSource, &aMMDest ); nValue = aSize.Width(); mvTabList[ nTab ].SetPos( nValue ); SvTreeListBox::nTreeFlags |= SvTreeFlags::RECALCTABS; if( IsUpdateMode() ) Invalidate(); } SvTreeListEntry* SvTabListBox::InsertEntry( const OUString& rText, SvTreeListEntry* pParent, bool /*bChildrenOnDemand*/, sal_uLong nPos, void* pUserData, SvLBoxButtonKind ) { return InsertEntryToColumn( rText, pParent, nPos, 0xffff, pUserData ); } SvTreeListEntry* SvTabListBox::InsertEntry( const OUString& rText, const Image& rExpandedEntryBmp, const Image& rCollapsedEntryBmp, SvTreeListEntry* pParent, bool /*bChildrenOnDemand*/, sal_uLong nPos, void* pUserData, SvLBoxButtonKind ) { return InsertEntryToColumn( rText, rExpandedEntryBmp, rCollapsedEntryBmp, pParent, nPos, 0xffff, pUserData ); } SvTreeListEntry* SvTabListBox::InsertEntryToColumn(const OUString& rStr,SvTreeListEntry* pParent,sal_uLong nPos,sal_uInt16 nCol, void* pUser ) { OUString aStr; if( nCol != 0xffff ) { while( nCol ) { aStr += "\t"; nCol--; } } aStr += rStr; OUString aFirstStr( aStr ); sal_Int32 nEnd = aFirstStr.indexOf( '\t' ); if( nEnd != -1 ) { aFirstStr = aFirstStr.copy(0, nEnd); aCurEntry = aStr.copy(++nEnd); } else aCurEntry.clear(); return SvTreeListBox::InsertEntry( aFirstStr, pParent, false, nPos, pUser ); } SvTreeListEntry* SvTabListBox::InsertEntryToColumn( const OUString& rStr, const Image& rExpandedEntryBmp, const Image& rCollapsedEntryBmp, SvTreeListEntry* pParent,sal_uLong nPos,sal_uInt16 nCol, void* pUser ) { OUString aStr; if( nCol != 0xffff ) { while( nCol ) { aStr += "\t"; nCol--; } } aStr += rStr; OUString aFirstStr( aStr ); sal_Int32 nEnd = aFirstStr.indexOf('\t'); if (nEnd != -1) { aFirstStr = aFirstStr.copy(0, nEnd); aCurEntry = aStr.copy(++nEnd); } else aCurEntry.clear(); return SvTreeListBox::InsertEntry( aFirstStr, rExpandedEntryBmp, rCollapsedEntryBmp, pParent, false, nPos, pUser ); } SvTreeListEntry* SvTabListBox::InsertEntryToColumn( const OUString& rStr, sal_uLong nPos, sal_uInt16 nCol, void* pUser ) { return InsertEntryToColumn( rStr,nullptr,nPos, nCol, pUser ); } OUString SvTabListBox::GetEntryText( SvTreeListEntry* pEntry ) const { return GetEntryText( pEntry, 0xffff ); } OUString SvTabListBox::GetEntryText( SvTreeListEntry* pEntry, sal_uInt16 nCol ) { DBG_ASSERT(pEntry,"GetEntryText:Invalid Entry"); OUString aResult; if( pEntry ) { sal_uInt16 nCount = pEntry->ItemCount(); sal_uInt16 nCur = 0; while( nCur < nCount ) { const SvLBoxItem& rStr = pEntry->GetItem( nCur ); if (rStr.GetType() == SvLBoxItemType::String) { if( nCol == 0xffff ) { if (!aResult.isEmpty()) aResult += "\t"; aResult += static_cast(rStr).GetText(); } else { if( nCol == 0 ) return static_cast(rStr).GetText(); nCol--; } } nCur++; } } return aResult; } OUString SvTabListBox::GetEntryText( sal_uLong nPos, sal_uInt16 nCol ) const { SvTreeListEntry* pEntry = GetEntryOnPos( nPos ); return GetEntryText( pEntry, nCol ); } void SvTabListBox::SetEntryText(const OUString& rStr, sal_uLong nPos, sal_uInt16 nCol) { SvTreeListEntry* pEntry = SvTreeListBox::GetEntry( nPos ); SetEntryText( rStr, pEntry, nCol ); } void SvTabListBox::SetEntryText(const OUString& rStr, SvTreeListEntry* pEntry, sal_uInt16 nCol) { DBG_ASSERT(pEntry,"SetEntryText:Invalid Entry"); if( !pEntry ) return; OUString sOldText = GetEntryText(pEntry, nCol); if (sOldText == rStr) return; sal_Int32 nIndex = 0; const sal_uInt16 nTextColumn = nCol; const sal_uInt16 nCount = pEntry->ItemCount(); for (sal_uInt16 nCur = 0; nCur < nCount; ++nCur) { SvLBoxItem& rBoxItem = pEntry->GetItem( nCur ); if (rBoxItem.GetType() == SvLBoxItemType::String) { if (!nCol || nCol==0xFFFF) { const OUString aTemp(GetToken(rStr, nIndex)); static_cast(rBoxItem).SetText( aTemp ); if (!nCol && nIndex<0) break; } else { --nCol; } } } GetModel()->InvalidateEntry( pEntry ); std::unique_ptr pData( new TabListBoxEventData( pEntry, nTextColumn, sOldText ) ); CallEventListeners( VclEventId::TableCellNameChanged, pData.get() ); } OUString SvTabListBox::GetCellText( sal_uLong nPos, sal_uInt16 nCol ) const { SvTreeListEntry* pEntry = GetEntryOnPos( nPos ); DBG_ASSERT( pEntry, "SvTabListBox::GetCellText(): Invalid Entry" ); OUString aResult; if (pEntry && pEntry->ItemCount() > static_cast(nCol+1)) { const SvLBoxItem& rStr = pEntry->GetItem( nCol + 1 ); if (rStr.GetType() == SvLBoxItemType::String) aResult = static_cast(rStr).GetText(); } return aResult; } sal_uLong SvTabListBox::GetEntryPos( const OUString& rStr, sal_uInt16 nCol ) { sal_uLong nPos = 0; SvTreeListEntry* pEntry = First(); while( pEntry ) { OUString aStr( GetEntryText( pEntry, nCol )); if( aStr == rStr ) return nPos; pEntry = Next( pEntry ); nPos++; } return 0xffffffff; } sal_uLong SvTabListBox::GetEntryPos( const SvTreeListEntry* pEntry ) const { sal_uLong nPos = 0; SvTreeListEntry* pTmpEntry = First(); while( pTmpEntry ) { if ( pTmpEntry == pEntry ) return nPos; pTmpEntry = Next( pTmpEntry ); ++nPos; } return 0xffffffff; } // static OUString SvTabListBox::GetToken( const OUString &sStr, sal_Int32& nIndex ) { return sStr.getToken(0, '\t', nIndex); } OUString SvTabListBox::GetTabEntryText( sal_uLong nPos, sal_uInt16 nCol ) const { SvTreeListEntry* pEntry = SvTreeListBox::GetEntry( nPos ); DBG_ASSERT( pEntry, "GetTabEntryText(): Invalid entry " ); OUString aResult; if ( pEntry ) { sal_uInt16 nCount = pEntry->ItemCount(); sal_uInt16 nCur = ( 0 == nCol && IsCellFocusEnabled() ) ? GetCurrentTabPos() : 0; while( nCur < nCount ) { const SvLBoxItem& rBoxItem = pEntry->GetItem( nCur ); if (rBoxItem.GetType() == SvLBoxItemType::String) { if ( nCol == 0xffff ) { if (!aResult.isEmpty()) aResult += "\t"; aResult += static_cast(rBoxItem).GetText(); } else { if ( nCol == 0 ) { OUString sRet = static_cast(rBoxItem).GetText(); if ( sRet.isEmpty() ) sRet = SvtResId( STR_SVT_ACC_EMPTY_FIELD ); return sRet; } --nCol; } } ++nCur; } } return aResult; } SvTreeListEntry* SvTabListBox::GetEntryOnPos( sal_uLong _nEntryPos ) const { SvTreeListEntry* pEntry = nullptr; sal_uLong i, nPos = 0, nCount = GetLevelChildCount( nullptr ); for ( i = 0; i < nCount; ++i ) { SvTreeListEntry* pParent = GetEntry(i); if ( nPos == _nEntryPos ) { pEntry = pParent; break; } else { nPos++; pEntry = GetChildOnPos( pParent, _nEntryPos, nPos ); if ( pEntry ) break; } } return pEntry; } SvTreeListEntry* SvTabListBox::GetChildOnPos( SvTreeListEntry* _pParent, sal_uLong _nEntryPos, sal_uLong& _rPos ) const { sal_uLong i, nCount = GetLevelChildCount( _pParent ); for ( i = 0; i < nCount; ++i ) { SvTreeListEntry* pParent = GetEntry( _pParent, i ); if ( _rPos == _nEntryPos ) return pParent; else { _rPos++; SvTreeListEntry* pEntry = GetChildOnPos( pParent, _nEntryPos, _rPos ); if ( pEntry ) return pEntry; } } return nullptr; } void SvTabListBox::SetTabJustify( sal_uInt16 nTab, SvTabJustify eJustify) { DBG_ASSERT(nTab= mvTabList.size() ) return; SvLBoxTab& rTab = mvTabList[ nTab ]; SvLBoxTabFlags nFlags = rTab.nFlags; nFlags &= (~MYTABMASK); nFlags |= static_cast(eJustify); rTab.nFlags = nFlags; SvTreeListBox::nTreeFlags |= SvTreeFlags::RECALCTABS; if( IsUpdateMode() ) Invalidate(); } long SvTabListBox::GetLogicTab( sal_uInt16 nTab ) { if( SvTreeListBox::nTreeFlags & SvTreeFlags::RECALCTABS ) SetTabs(); DBG_ASSERT(nTabGetPos(); } namespace svt { struct SvHeaderTabListBoxImpl { VclPtr m_pHeaderBar; AccessibleFactoryAccess m_aFactoryAccess; SvHeaderTabListBoxImpl() : m_pHeaderBar( nullptr ) { } }; } SvHeaderTabListBox::SvHeaderTabListBox( vcl::Window* pParent, WinBits nWinStyle ) : SvTabListBox(pParent, nWinStyle) , m_bFirstPaint(true) , m_pImpl(new ::svt::SvHeaderTabListBoxImpl) , m_pAccessible(nullptr) { } SvHeaderTabListBox::~SvHeaderTabListBox() { disposeOnce(); } void SvHeaderTabListBox::dispose() { m_pImpl.reset(); SvTabListBox::dispose(); } void SvHeaderTabListBox::Paint( vcl::RenderContext& rRenderContext, const tools::Rectangle& rRect ) { if (m_bFirstPaint) { m_bFirstPaint = false; } SvTabListBox::Paint(rRenderContext, rRect); } void SvHeaderTabListBox::InitHeaderBar( HeaderBar* pHeaderBar ) { DBG_ASSERT( !m_pImpl->m_pHeaderBar, "header bar already initialized" ); DBG_ASSERT( pHeaderBar, "invalid header bar initialization" ); m_pImpl->m_pHeaderBar = pHeaderBar; SetScrolledHdl( LINK( this, SvHeaderTabListBox, ScrollHdl_Impl ) ); m_pImpl->m_pHeaderBar->SetCreateAccessibleHdl( LINK( this, SvHeaderTabListBox, CreateAccessibleHdl_Impl ) ); } bool SvHeaderTabListBox::IsItemChecked( SvTreeListEntry* pEntry, sal_uInt16 nCol ) { SvButtonState eState = SvButtonState::Unchecked; SvLBoxButton& rItem = static_cast( pEntry->GetItem( nCol + 1 ) ); if (rItem.GetType() == SvLBoxItemType::Button) { SvItemStateFlags nButtonFlags = rItem.GetButtonFlags(); eState = SvLBoxButtonData::ConvertToButtonState( nButtonFlags ); } return ( eState == SvButtonState::Checked ); } SvTreeListEntry* SvHeaderTabListBox::InsertEntryToColumn( const OUString& rStr, sal_uLong nPos, sal_uInt16 nCol, void* pUserData ) { SvTreeListEntry* pEntry = SvTabListBox::InsertEntryToColumn( rStr, nPos, nCol, pUserData ); RecalculateAccessibleChildren(); return pEntry; } SvTreeListEntry* SvHeaderTabListBox::InsertEntryToColumn( const OUString& rStr, SvTreeListEntry* pParent, sal_uLong nPos, sal_uInt16 nCol, void* pUserData ) { SvTreeListEntry* pEntry = SvTabListBox::InsertEntryToColumn( rStr, pParent, nPos, nCol, pUserData ); RecalculateAccessibleChildren(); return pEntry; } SvTreeListEntry* SvHeaderTabListBox::InsertEntryToColumn( const OUString& rStr, const Image& rExpandedEntryBmp, const Image& rCollapsedEntryBmp, SvTreeListEntry* pParent, sal_uLong nPos, sal_uInt16 nCol, void* pUserData ) { SvTreeListEntry* pEntry = SvTabListBox::InsertEntryToColumn( rStr, rExpandedEntryBmp, rCollapsedEntryBmp, pParent, nPos, nCol, pUserData ); RecalculateAccessibleChildren(); return pEntry; } sal_uLong SvHeaderTabListBox::Insert( SvTreeListEntry* pEnt, SvTreeListEntry* pPar, sal_uLong nPos ) { sal_uLong n = SvTabListBox::Insert( pEnt, pPar, nPos ); RecalculateAccessibleChildren(); return n; } sal_uLong SvHeaderTabListBox::Insert( SvTreeListEntry* pEntry, sal_uLong nRootPos ) { sal_uLong nPos = SvTabListBox::Insert( pEntry, nRootPos ); RecalculateAccessibleChildren(); return nPos; } void SvHeaderTabListBox::RemoveEntry( SvTreeListEntry const * _pEntry ) { GetModel()->Remove( _pEntry ); m_aAccessibleChildren.clear(); } void SvHeaderTabListBox::Clear() { SvTabListBox::Clear(); m_aAccessibleChildren.clear(); } IMPL_LINK_NOARG(SvHeaderTabListBox, ScrollHdl_Impl, SvTreeListBox*, void) { m_pImpl->m_pHeaderBar->SetOffset( -GetXOffset() ); } IMPL_LINK_NOARG(SvHeaderTabListBox, CreateAccessibleHdl_Impl, HeaderBar*, void) { vcl::Window* pParent = m_pImpl->m_pHeaderBar->GetAccessibleParentWindow(); DBG_ASSERT( pParent, "SvHeaderTabListBox..CreateAccessibleHdl_Impl - accessible parent not found" ); if ( pParent ) { css::uno::Reference< XAccessible > xAccParent = pParent->GetAccessible(); if ( xAccParent.is() ) { Reference< XAccessible > xAccessible = m_pImpl->m_aFactoryAccess.getFactory().createAccessibleBrowseBoxHeaderBar( xAccParent, *this, ::svt::BBTYPE_COLUMNHEADERBAR ); m_pImpl->m_pHeaderBar->SetAccessible( xAccessible ); } } } void SvHeaderTabListBox::RecalculateAccessibleChildren() { if ( !m_aAccessibleChildren.empty() ) { sal_uInt32 nCount = ( GetRowCount() + 1 ) * GetColumnCount(); if ( m_aAccessibleChildren.size() < nCount ) m_aAccessibleChildren.resize( nCount ); else { DBG_ASSERT( m_aAccessibleChildren.size() == nCount, "wrong children count" ); } } } bool SvHeaderTabListBox::IsCellCheckBox( long _nRow, sal_uInt16 _nColumn, TriState& _rState ) { bool bRet = false; SvTreeListEntry* pEntry = GetEntry( _nRow ); if ( pEntry ) { sal_uInt16 nItemCount = pEntry->ItemCount(); if ( nItemCount > ( _nColumn + 1 ) ) { SvLBoxItem& rItem = pEntry->GetItem( _nColumn + 1 ); if (rItem.GetType() == SvLBoxItemType::Button) { bRet = true; _rState = ( ( static_cast(rItem).GetButtonFlags() & SvItemStateFlags::UNCHECKED ) == SvItemStateFlags::NONE ) ? TRISTATE_TRUE : TRISTATE_FALSE; } } else { SAL_WARN( "svtools.contnr", "SvHeaderTabListBox::IsCellCheckBox(): column out of range" ); } } return bRet; } long SvHeaderTabListBox::GetRowCount() const { return GetEntryCount(); } sal_uInt16 SvHeaderTabListBox::GetColumnCount() const { return m_pImpl->m_pHeaderBar->GetItemCount(); } sal_Int32 SvHeaderTabListBox::GetCurrRow() const { sal_Int32 nRet = -1; SvTreeListEntry* pEntry = GetCurEntry(); if ( pEntry ) { sal_uLong nCount = GetEntryCount(); for ( sal_uLong i = 0; i < nCount; ++i ) { if ( pEntry == GetEntry(i) ) { nRet = i; break; } } } return nRet; } sal_uInt16 SvHeaderTabListBox::GetCurrColumn() const { sal_uInt16 nPos = GetCurrentTabPos() - 1; return nPos; } OUString SvHeaderTabListBox::GetRowDescription( sal_Int32 _nRow ) const { return GetEntryText( _nRow ); } OUString SvHeaderTabListBox::GetColumnDescription( sal_uInt16 _nColumn ) const { return m_pImpl->m_pHeaderBar->GetItemText( m_pImpl->m_pHeaderBar->GetItemId( _nColumn ) ); } bool SvHeaderTabListBox::HasRowHeader() const { return false; } bool SvHeaderTabListBox::GoToCell( sal_Int32 _nRow, sal_uInt16 _nColumn ) { bool bRet = IsCellFocusEnabled(); if ( bRet ) { // first set cursor to _nRow SetCursor( GetEntry( _nRow ), true ); // then set the focus into _nColumn bRet = SetCurrentTabPos( _nColumn ); } return bRet; } void SvHeaderTabListBox::SetNoSelection() { SvTreeListBox::SelectAll(false); } void SvHeaderTabListBox::SelectAll() { SvTreeListBox::SelectAll(true); } void SvHeaderTabListBox::SelectRow( long _nRow, bool _bSelect, bool ) { Select( GetEntry( _nRow ), _bSelect ); } void SvHeaderTabListBox::SelectColumn( sal_uInt16, bool ) { } sal_Int32 SvHeaderTabListBox::GetSelectedRowCount() const { return GetSelectionCount(); } sal_Int32 SvHeaderTabListBox::GetSelectedColumnCount() const { return 0; } bool SvHeaderTabListBox::IsRowSelected( long _nRow ) const { SvTreeListEntry* pEntry = GetEntry( _nRow ); return ( pEntry && IsSelected( pEntry ) ); } bool SvHeaderTabListBox::IsColumnSelected( long ) const { return false; } void SvHeaderTabListBox::GetAllSelectedRows( css::uno::Sequence< sal_Int32 >& ) const { } void SvHeaderTabListBox::GetAllSelectedColumns( css::uno::Sequence< sal_Int32 >& ) const { } bool SvHeaderTabListBox::IsCellVisible( sal_Int32, sal_uInt16 ) const { return true; } OUString SvHeaderTabListBox::GetAccessibleCellText( long _nRow, sal_uInt16 _nColumnPos ) const { return GetTabEntryText(_nRow, _nColumnPos); } tools::Rectangle SvHeaderTabListBox::calcHeaderRect( bool _bIsColumnBar, bool _bOnScreen ) { tools::Rectangle aRect; if ( _bIsColumnBar ) { vcl::Window* pParent = nullptr; if ( !_bOnScreen ) pParent = m_pImpl->m_pHeaderBar->GetAccessibleParentWindow(); aRect = m_pImpl->m_pHeaderBar->GetWindowExtentsRelative( pParent ); } return aRect; } tools::Rectangle SvHeaderTabListBox::calcTableRect( bool _bOnScreen ) { vcl::Window* pParent = nullptr; if ( !_bOnScreen ) pParent = GetAccessibleParentWindow(); tools::Rectangle aRect( GetWindowExtentsRelative( pParent ) ); return aRect; } tools::Rectangle SvHeaderTabListBox::GetFieldRectPixelAbs( sal_Int32 _nRow, sal_uInt16 _nColumn, bool _bIsHeader, bool _bOnScreen ) { DBG_ASSERT( !_bIsHeader || 0 == _nRow, "invalid parameters" ); tools::Rectangle aRect; SvTreeListEntry* pEntry = GetEntry( _nRow ); if ( pEntry ) { aRect = _bIsHeader ? calcHeaderRect( true, false ) : GetBoundingRect( pEntry ); Point aTopLeft = aRect.TopLeft(); DBG_ASSERT( m_pImpl->m_pHeaderBar->GetItemCount() > _nColumn, "invalid column" ); tools::Rectangle aItemRect = m_pImpl->m_pHeaderBar->GetItemRect( m_pImpl->m_pHeaderBar->GetItemId( _nColumn ) ); aTopLeft.setX( aItemRect.Left() ); Size aSize = aItemRect.GetSize(); aRect = tools::Rectangle( aTopLeft, aSize ); vcl::Window* pParent = nullptr; if ( !_bOnScreen ) pParent = GetAccessibleParentWindow(); aTopLeft = aRect.TopLeft(); aTopLeft += GetWindowExtentsRelative( pParent ).TopLeft(); aRect = tools::Rectangle( aTopLeft, aRect.GetSize() ); } return aRect; } Reference< XAccessible > SvHeaderTabListBox::CreateAccessibleCell( sal_Int32 _nRow, sal_uInt16 _nColumnPos ) { OSL_ENSURE( m_pAccessible, "Invalid call: Accessible is null" ); Reference< XAccessible > xChild; sal_Int32 nIndex = -1; if ( !AreChildrenTransient() ) { const sal_uInt16 nColumnCount = GetColumnCount(); // first call? -> initial list if ( m_aAccessibleChildren.empty() ) { sal_Int32 nCount = ( GetRowCount() + 1 ) * nColumnCount; m_aAccessibleChildren.assign( nCount, Reference< XAccessible >() ); } nIndex = ( _nRow * nColumnCount ) + _nColumnPos + nColumnCount; xChild = m_aAccessibleChildren[ nIndex ]; } if ( !xChild.is() ) { TriState eState = TRISTATE_INDET; bool bIsCheckBox = IsCellCheckBox( _nRow, _nColumnPos, eState ); if ( bIsCheckBox ) xChild = m_pImpl->m_aFactoryAccess.getFactory().createAccessibleCheckBoxCell( m_pAccessible->getHeaderBar(), *this, nullptr, _nRow, _nColumnPos, eState, false ); else xChild = m_pImpl->m_aFactoryAccess.getFactory().createAccessibleBrowseBoxTableCell( m_pAccessible->getHeaderBar(), *this, nullptr, _nRow, _nColumnPos, OFFSET_NONE ); // insert into list if ( !AreChildrenTransient() ) m_aAccessibleChildren[ nIndex ] = xChild; } return xChild; } Reference< XAccessible > SvHeaderTabListBox::CreateAccessibleRowHeader( sal_Int32 ) { Reference< XAccessible > xHeader; return xHeader; } Reference< XAccessible > SvHeaderTabListBox::CreateAccessibleColumnHeader( sal_uInt16 _nColumn ) { // first call? -> initial list if ( m_aAccessibleChildren.empty() ) { const sal_uInt16 nColumnCount = GetColumnCount(); sal_Int32 nCount = AreChildrenTransient() ? nColumnCount : ( GetRowCount() + 1 ) * nColumnCount; m_aAccessibleChildren.assign( nCount, Reference< XAccessible >() ); } // get header Reference< XAccessible > xChild = m_aAccessibleChildren[ _nColumn ]; // already exists? if ( !xChild.is() && m_pAccessible ) { // no -> create new header cell xChild = m_pImpl->m_aFactoryAccess.getFactory().createAccessibleBrowseBoxHeaderCell( _nColumn, m_pAccessible->getHeaderBar(), *this, nullptr, ::svt::BBTYPE_COLUMNHEADERCELL ); // insert into list m_aAccessibleChildren[ _nColumn ] = xChild; } return xChild; } sal_Int32 SvHeaderTabListBox::GetAccessibleControlCount() const { return -1; } Reference< XAccessible > SvHeaderTabListBox::CreateAccessibleControl( sal_Int32 ) { Reference< XAccessible > xControl; return xControl; } bool SvHeaderTabListBox::ConvertPointToControlIndex( sal_Int32&, const Point& ) { return false; } bool SvHeaderTabListBox::ConvertPointToCellAddress( sal_Int32&, sal_uInt16&, const Point& ) { return false; } bool SvHeaderTabListBox::ConvertPointToRowHeader( sal_Int32&, const Point& ) { return false; } bool SvHeaderTabListBox::ConvertPointToColumnHeader( sal_uInt16&, const Point& ) { return false; } OUString SvHeaderTabListBox::GetAccessibleObjectName( ::svt::AccessibleBrowseBoxObjType _eType, sal_Int32 _nPos ) const { OUString aRetText; switch( _eType ) { case ::svt::BBTYPE_BROWSEBOX: case ::svt::BBTYPE_TABLE: case ::svt::BBTYPE_COLUMNHEADERBAR: // should be empty now (see #i63983) aRetText.clear(); break; case ::svt::BBTYPE_TABLECELL: { // here we need a valid pos, we can not handle -1 if ( _nPos >= 0 ) { sal_uInt16 nColumnCount = GetColumnCount(); if (nColumnCount > 0) { sal_Int32 nRow = _nPos / nColumnCount; sal_uInt16 nColumn = static_cast< sal_uInt16 >( _nPos % nColumnCount ); aRetText = GetCellText( nRow, nColumn ); } } break; } case ::svt::BBTYPE_CHECKBOXCELL: { break; // checkbox cells have no name } case ::svt::BBTYPE_COLUMNHEADERCELL: { aRetText = m_pImpl->m_pHeaderBar->GetItemText( m_pImpl->m_pHeaderBar->GetItemId( static_cast(_nPos) ) ); break; } case ::svt::BBTYPE_ROWHEADERBAR: case ::svt::BBTYPE_ROWHEADERCELL: aRetText = "error"; break; default: OSL_FAIL("BrowseBox::GetAccessibleName: invalid enum!"); } return aRetText; } OUString SvHeaderTabListBox::GetAccessibleObjectDescription( ::svt::AccessibleBrowseBoxObjType _eType, sal_Int32 _nPos ) const { OUString aRetText; if( _eType == ::svt::BBTYPE_TABLECELL && _nPos != -1 ) { const OUString sVar1( "%1" ); const OUString sVar2( "%2" ); sal_uInt16 nColumnCount = GetColumnCount(); if (nColumnCount > 0) { sal_Int32 nRow = _nPos / nColumnCount; sal_uInt16 nColumn = static_cast< sal_uInt16 >( _nPos % nColumnCount ); OUString aText( SvtResId(STR_SVT_ACC_DESC_TABLISTBOX) ); aText = aText.replaceFirst( sVar1, OUString::number( nRow ) ); OUString sColHeader = m_pImpl->m_pHeaderBar->GetItemText( m_pImpl->m_pHeaderBar->GetItemId( nColumn ) ); if ( sColHeader.isEmpty() ) sColHeader = OUString::number( nColumn ); aText = aText.replaceFirst( sVar2, sColHeader ); aRetText = aText; } } return aRetText; } void SvHeaderTabListBox::FillAccessibleStateSet( ::utl::AccessibleStateSetHelper& _rStateSet, ::svt::AccessibleBrowseBoxObjType _eType ) const { switch( _eType ) { case ::svt::BBTYPE_BROWSEBOX: case ::svt::BBTYPE_TABLE: { _rStateSet.AddState( AccessibleStateType::FOCUSABLE ); if ( HasFocus() ) _rStateSet.AddState( AccessibleStateType::FOCUSED ); if ( IsActive() ) _rStateSet.AddState( AccessibleStateType::ACTIVE ); if ( IsEnabled() ) { _rStateSet.AddState( AccessibleStateType::ENABLED ); _rStateSet.AddState( AccessibleStateType::SENSITIVE ); } if ( IsReallyVisible() ) _rStateSet.AddState( AccessibleStateType::VISIBLE ); if ( _eType == ::svt::BBTYPE_TABLE ) { if ( AreChildrenTransient() ) _rStateSet.AddState( AccessibleStateType::MANAGES_DESCENDANTS ); _rStateSet.AddState( AccessibleStateType::MULTI_SELECTABLE ); } break; } case ::svt::BBTYPE_COLUMNHEADERBAR: { sal_Int32 nCurRow = GetCurrRow(); sal_uInt16 nCurColumn = GetCurrColumn(); if ( IsCellVisible( nCurRow, nCurColumn ) ) _rStateSet.AddState( AccessibleStateType::VISIBLE ); if ( IsEnabled() ) _rStateSet.AddState( AccessibleStateType::ENABLED ); _rStateSet.AddState( AccessibleStateType::TRANSIENT ); break; } case ::svt::BBTYPE_ROWHEADERCELL: case ::svt::BBTYPE_COLUMNHEADERCELL: { _rStateSet.AddState( AccessibleStateType::VISIBLE ); _rStateSet.AddState( AccessibleStateType::FOCUSABLE ); _rStateSet.AddState( AccessibleStateType::TRANSIENT ); if ( IsEnabled() ) _rStateSet.AddState( AccessibleStateType::ENABLED ); break; } default: break; } } void SvHeaderTabListBox::FillAccessibleStateSetForCell( ::utl::AccessibleStateSetHelper& _rStateSet, sal_Int32 _nRow, sal_uInt16 _nColumn ) const { _rStateSet.AddState( AccessibleStateType::SELECTABLE ); if ( AreChildrenTransient() ) _rStateSet.AddState( AccessibleStateType::TRANSIENT ); if ( IsCellVisible( _nRow, _nColumn ) ) { _rStateSet.AddState( AccessibleStateType::VISIBLE ); _rStateSet.AddState( AccessibleStateType::ENABLED ); } if ( IsRowSelected( _nRow ) ) { _rStateSet.AddState( AccessibleStateType::ACTIVE ); _rStateSet.AddState( AccessibleStateType::SELECTED ); } if ( IsEnabled() ) _rStateSet.AddState( AccessibleStateType::ENABLED ); } void SvHeaderTabListBox::GrabTableFocus() { GrabFocus(); } bool SvHeaderTabListBox::GetGlyphBoundRects( const Point& rOrigin, const OUString& rStr, int nIndex, int nLen, MetricVector& rVector ) { return Control::GetGlyphBoundRects( rOrigin, rStr, nIndex, nLen, rVector ); } tools::Rectangle SvHeaderTabListBox::GetWindowExtentsRelative( vcl::Window *pRelativeWindow ) const { return Control::GetWindowExtentsRelative( pRelativeWindow ); } void SvHeaderTabListBox::GrabFocus() { Control::GrabFocus(); } Reference< XAccessible > SvHeaderTabListBox::GetAccessible() { return Control::GetAccessible(); } vcl::Window* SvHeaderTabListBox::GetAccessibleParentWindow() const { return Control::GetAccessibleParentWindow(); } vcl::Window* SvHeaderTabListBox::GetWindowInstance() { return this; } Reference< XAccessible > SvHeaderTabListBox::CreateAccessible() { vcl::Window* pParent = GetAccessibleParentWindow(); DBG_ASSERT( pParent, "SvHeaderTabListBox::::CreateAccessible - accessible parent not found" ); Reference< XAccessible > xAccessible; if ( m_pAccessible ) xAccessible = m_pAccessible->getMyself(); if( pParent && !m_pAccessible ) { Reference< XAccessible > xAccParent = pParent->GetAccessible(); if ( xAccParent.is() ) { m_pAccessible = m_pImpl->m_aFactoryAccess.getFactory().createAccessibleTabListBox( xAccParent, *this ); if ( m_pAccessible ) xAccessible = m_pAccessible->getMyself(); } } return xAccessible; } tools::Rectangle SvHeaderTabListBox::GetFieldCharacterBounds(sal_Int32,sal_Int32,sal_Int32) { return tools::Rectangle(); } sal_Int32 SvHeaderTabListBox::GetFieldIndexAtPoint(sal_Int32 _nRow,sal_Int32 _nColumnPos,const Point& _rPoint) { OUString sText = GetAccessibleCellText( _nRow, static_cast< sal_uInt16 >( _nColumnPos ) ); MetricVector aRects; if ( GetGlyphBoundRects(Point(0,0), sText, 0, sText.getLength(), aRects) ) { sal_Int32 nPos = 0; for (auto const& rectangle : aRects) { if( rectangle.IsInside(_rPoint) ) return nPos; ++nPos; } } return -1; } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */