2017-11-12 17:55:20 +00:00
|
|
|
/*
|
|
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2017 CERN
|
2019-06-13 11:44:12 +00:00
|
|
|
* Copyright (C) 2019 KiCad Developers, see AUTHORS.txt for contributors.
|
2017-11-12 17:55:20 +00:00
|
|
|
* @author Maciej Suminski <maciej.suminski@cern.ch>
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 3
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, you may find one here:
|
|
|
|
* https://www.gnu.org/licenses/gpl-3.0.html
|
|
|
|
* or you may search the http://www.gnu.org website for the version 3 license,
|
|
|
|
* or you may write to the Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
|
|
|
*/
|
|
|
|
|
2018-07-27 20:47:51 +00:00
|
|
|
#include <symbol_tree_synchronizing_adapter.h>
|
2017-11-12 17:55:20 +00:00
|
|
|
#include <lib_manager.h>
|
2018-02-06 15:14:01 +00:00
|
|
|
#include <symbol_lib_table.h>
|
2017-11-12 17:55:20 +00:00
|
|
|
#include <class_libentry.h>
|
2019-06-05 19:15:57 +00:00
|
|
|
#include <tools/lib_control.h>
|
2017-11-12 17:55:20 +00:00
|
|
|
|
|
|
|
|
2019-06-05 19:15:57 +00:00
|
|
|
LIB_TREE_MODEL_ADAPTER::PTR SYMBOL_TREE_SYNCHRONIZING_ADAPTER::Create( LIB_EDIT_FRAME* aParent,
|
|
|
|
LIB_MANAGER* aLibMgr )
|
2017-11-12 17:55:20 +00:00
|
|
|
{
|
2019-06-05 19:15:57 +00:00
|
|
|
return PTR( new SYMBOL_TREE_SYNCHRONIZING_ADAPTER( aParent, aLibMgr ) );
|
2017-11-12 17:55:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-06-05 19:15:57 +00:00
|
|
|
SYMBOL_TREE_SYNCHRONIZING_ADAPTER::SYMBOL_TREE_SYNCHRONIZING_ADAPTER( LIB_EDIT_FRAME* aParent,
|
|
|
|
LIB_MANAGER* aLibMgr ) :
|
2020-05-31 21:42:04 +00:00
|
|
|
LIB_TREE_MODEL_ADAPTER( aParent, "pinned_symbol_libs" ),
|
2019-06-05 19:15:57 +00:00
|
|
|
m_frame( aParent ),
|
|
|
|
m_libMgr( aLibMgr ),
|
|
|
|
m_lastSyncHash( -1 )
|
2017-11-12 17:55:20 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-06-05 19:15:57 +00:00
|
|
|
TOOL_INTERACTIVE* SYMBOL_TREE_SYNCHRONIZING_ADAPTER::GetContextMenuTool()
|
|
|
|
{
|
|
|
|
return m_frame->GetToolManager()->GetTool<LIB_CONTROL>();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-07-27 20:47:51 +00:00
|
|
|
bool SYMBOL_TREE_SYNCHRONIZING_ADAPTER::IsContainer( const wxDataViewItem& aItem ) const
|
2017-11-12 17:55:20 +00:00
|
|
|
{
|
2018-07-27 20:47:51 +00:00
|
|
|
const LIB_TREE_NODE* node = ToNode( aItem );
|
2020-02-07 17:06:24 +00:00
|
|
|
return node ? node->m_Type == LIB_TREE_NODE::LIB : true;
|
2017-11-12 17:55:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-08-07 01:17:00 +00:00
|
|
|
#define PROGRESS_INTERVAL_MILLIS 120
|
2018-01-25 23:49:04 +00:00
|
|
|
|
2019-06-13 11:44:12 +00:00
|
|
|
void SYMBOL_TREE_SYNCHRONIZING_ADAPTER::Sync( bool aForce,
|
|
|
|
std::function<void( int, int, const wxString& )> aProgressCallback )
|
2017-11-12 17:55:20 +00:00
|
|
|
{
|
2018-01-29 00:51:07 +00:00
|
|
|
wxLongLong nextUpdate = wxGetUTCTimeMillis() + (PROGRESS_INTERVAL_MILLIS / 2);
|
2018-01-25 23:49:04 +00:00
|
|
|
|
2017-11-12 13:24:55 +00:00
|
|
|
int libMgrHash = m_libMgr->GetHash();
|
|
|
|
|
2017-11-14 16:21:38 +00:00
|
|
|
if( !aForce && m_lastSyncHash == libMgrHash )
|
2017-11-12 17:55:20 +00:00
|
|
|
return;
|
|
|
|
|
2017-11-12 13:24:55 +00:00
|
|
|
m_lastSyncHash = libMgrHash;
|
2017-11-22 11:06:17 +00:00
|
|
|
int i = 0, max = GetLibrariesCount();
|
2017-11-12 17:55:20 +00:00
|
|
|
|
|
|
|
// Process already stored libraries
|
2020-02-07 17:06:24 +00:00
|
|
|
for( auto it = m_tree.m_Children.begin(); it != m_tree.m_Children.end(); /* iteration inside */ )
|
2017-11-12 17:55:20 +00:00
|
|
|
{
|
2020-02-07 17:06:24 +00:00
|
|
|
const wxString& name = it->get()->m_Name;
|
2018-01-25 23:49:04 +00:00
|
|
|
|
|
|
|
if( wxGetUTCTimeMillis() > nextUpdate )
|
|
|
|
{
|
2018-01-29 00:51:07 +00:00
|
|
|
aProgressCallback( i, max, name );
|
2018-01-25 23:49:04 +00:00
|
|
|
nextUpdate = wxGetUTCTimeMillis() + PROGRESS_INTERVAL_MILLIS;
|
|
|
|
}
|
2017-11-12 17:55:20 +00:00
|
|
|
|
2019-06-13 11:44:12 +00:00
|
|
|
// There is a bug in LIB_MANAGER::LibraryExists() that uses the buffered modified
|
|
|
|
// libraries before the symbol library table which prevents the library from being
|
|
|
|
// removed from the tree control.
|
|
|
|
if( !m_libMgr->LibraryExists( name, true )
|
|
|
|
|| !m_frame->Prj().SchSymbolLibTable()->HasLibrary( name, true ) )
|
2017-11-12 17:55:20 +00:00
|
|
|
{
|
2017-11-12 13:24:55 +00:00
|
|
|
it = deleteLibrary( it );
|
2017-11-12 17:55:20 +00:00
|
|
|
continue;
|
|
|
|
}
|
2020-07-17 17:32:16 +00:00
|
|
|
else if( aForce || m_libMgr->GetLibraryHash( name ) != m_libHashes[name] )
|
2017-11-12 17:55:20 +00:00
|
|
|
{
|
2018-07-27 20:47:51 +00:00
|
|
|
updateLibrary( *(LIB_TREE_NODE_LIB*) it->get() );
|
2017-11-12 17:55:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
++it;
|
2018-01-29 00:51:07 +00:00
|
|
|
++i;
|
2017-11-12 17:55:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Look for new libraries
|
2020-03-31 01:09:01 +00:00
|
|
|
for( const wxString& libName : m_libMgr->GetLibraryNames() )
|
2017-11-12 17:55:20 +00:00
|
|
|
{
|
|
|
|
if( m_libHashes.count( libName ) == 0 )
|
2017-11-22 11:06:17 +00:00
|
|
|
{
|
2018-01-25 23:49:04 +00:00
|
|
|
if( wxGetUTCTimeMillis() > nextUpdate )
|
|
|
|
{
|
|
|
|
aProgressCallback( i++, max, libName );
|
|
|
|
nextUpdate = wxGetUTCTimeMillis() + PROGRESS_INTERVAL_MILLIS;
|
|
|
|
}
|
|
|
|
|
2018-02-06 15:14:01 +00:00
|
|
|
SYMBOL_LIB_TABLE_ROW* library = m_libMgr->GetLibrary( libName );
|
2020-03-31 01:09:01 +00:00
|
|
|
LIB_TREE_NODE_LIB& lib_node = DoAddLibraryNode( libName, library->GetDescr() );
|
2018-02-06 15:14:01 +00:00
|
|
|
|
2017-11-30 11:31:36 +00:00
|
|
|
updateLibrary( lib_node );
|
2017-11-22 11:06:17 +00:00
|
|
|
}
|
2017-11-12 17:55:20 +00:00
|
|
|
}
|
2018-08-05 11:56:02 +00:00
|
|
|
|
|
|
|
m_tree.AssignIntrinsicRanks();
|
2017-11-12 17:55:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-07-27 20:47:51 +00:00
|
|
|
int SYMBOL_TREE_SYNCHRONIZING_ADAPTER::GetLibrariesCount() const
|
2017-11-22 11:06:17 +00:00
|
|
|
{
|
2018-07-27 20:47:51 +00:00
|
|
|
int count = LIB_TREE_MODEL_ADAPTER::GetLibrariesCount();
|
2017-11-22 11:06:17 +00:00
|
|
|
|
2020-03-31 01:09:01 +00:00
|
|
|
for( const wxString& libName : m_libMgr->GetLibraryNames() )
|
2017-11-22 11:06:17 +00:00
|
|
|
{
|
|
|
|
if( m_libHashes.count( libName ) == 0 )
|
|
|
|
++count;
|
|
|
|
}
|
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-07-27 20:47:51 +00:00
|
|
|
void SYMBOL_TREE_SYNCHRONIZING_ADAPTER::updateLibrary( LIB_TREE_NODE_LIB& aLibNode )
|
2017-11-12 17:55:20 +00:00
|
|
|
{
|
2020-02-07 17:06:24 +00:00
|
|
|
auto hashIt = m_libHashes.find( aLibNode.m_Name );
|
2018-01-10 11:04:34 +00:00
|
|
|
|
|
|
|
if( hashIt == m_libHashes.end() )
|
2017-11-12 17:55:20 +00:00
|
|
|
{
|
2017-11-12 13:24:55 +00:00
|
|
|
// add a new library
|
2020-03-31 01:09:01 +00:00
|
|
|
for( LIB_PART* alias : m_libMgr->GetAliases( aLibNode.m_Name ) )
|
2018-07-30 13:18:37 +00:00
|
|
|
aLibNode.AddItem( alias );
|
2017-11-12 17:55:20 +00:00
|
|
|
}
|
2020-02-07 17:06:24 +00:00
|
|
|
else if( hashIt->second != m_libMgr->GetLibraryHash( aLibNode.m_Name ) )
|
2017-11-12 13:24:55 +00:00
|
|
|
{
|
2019-06-13 11:44:12 +00:00
|
|
|
// update an existing library
|
2020-02-07 17:06:24 +00:00
|
|
|
std::list<LIB_PART*> aliases = m_libMgr->GetAliases( aLibNode.m_Name );
|
2017-11-12 13:24:55 +00:00
|
|
|
|
|
|
|
// remove the common part from the aliases list
|
2020-02-07 17:06:24 +00:00
|
|
|
for( auto nodeIt = aLibNode.m_Children.begin(); nodeIt != aLibNode.m_Children.end(); /**/ )
|
2017-11-12 13:24:55 +00:00
|
|
|
{
|
|
|
|
auto aliasIt = std::find_if( aliases.begin(), aliases.end(),
|
2020-07-17 17:32:16 +00:00
|
|
|
[&] ( const LIB_PART* a )
|
|
|
|
{
|
|
|
|
return a->GetName() == (*nodeIt)->m_Name;
|
|
|
|
} );
|
2017-11-12 13:24:55 +00:00
|
|
|
|
|
|
|
if( aliasIt != aliases.end() )
|
|
|
|
{
|
|
|
|
// alias exists both in the component tree and the library manager,
|
2018-01-10 11:04:34 +00:00
|
|
|
// update only the node data
|
2018-07-27 20:47:51 +00:00
|
|
|
static_cast<LIB_TREE_NODE_LIB_ID*>( nodeIt->get() )->Update( *aliasIt );
|
2017-11-12 13:24:55 +00:00
|
|
|
aliases.erase( aliasIt );
|
|
|
|
++nodeIt;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// node does not exist in the library manager, remove the corresponding node
|
2020-02-07 17:06:24 +00:00
|
|
|
nodeIt = aLibNode.m_Children.erase( nodeIt );
|
2017-11-12 13:24:55 +00:00
|
|
|
}
|
|
|
|
}
|
2017-11-12 17:55:20 +00:00
|
|
|
|
2017-11-12 13:24:55 +00:00
|
|
|
// now the aliases list contains only new aliases that need to be added to the tree
|
2020-03-31 01:09:01 +00:00
|
|
|
for( LIB_PART* alias : aliases )
|
2018-07-30 13:18:37 +00:00
|
|
|
aLibNode.AddItem( alias );
|
2017-11-12 13:24:55 +00:00
|
|
|
}
|
2017-11-12 17:55:20 +00:00
|
|
|
|
2017-11-12 13:24:55 +00:00
|
|
|
aLibNode.AssignIntrinsicRanks();
|
2020-02-07 17:06:24 +00:00
|
|
|
m_libHashes[aLibNode.m_Name] = m_libMgr->GetLibraryHash( aLibNode.m_Name );
|
2017-11-12 17:55:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-07-27 20:47:51 +00:00
|
|
|
LIB_TREE_NODE::PTR_VECTOR::iterator SYMBOL_TREE_SYNCHRONIZING_ADAPTER::deleteLibrary(
|
|
|
|
LIB_TREE_NODE::PTR_VECTOR::iterator& aLibNodeIt )
|
2017-11-12 13:24:55 +00:00
|
|
|
{
|
2018-07-27 20:47:51 +00:00
|
|
|
LIB_TREE_NODE* node = aLibNodeIt->get();
|
2020-02-07 17:06:24 +00:00
|
|
|
m_libHashes.erase( node->m_Name );
|
|
|
|
auto it = m_tree.m_Children.erase( aLibNodeIt );
|
2017-11-12 13:24:55 +00:00
|
|
|
return it;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-07-27 20:47:51 +00:00
|
|
|
void SYMBOL_TREE_SYNCHRONIZING_ADAPTER::GetValue( wxVariant& aVariant, wxDataViewItem const& aItem,
|
|
|
|
unsigned int aCol ) const
|
2018-01-20 17:15:46 +00:00
|
|
|
{
|
2018-04-28 09:43:41 +00:00
|
|
|
if( IsFrozen() )
|
|
|
|
{
|
|
|
|
aVariant = wxEmptyString;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-03-31 01:09:01 +00:00
|
|
|
LIB_TREE_NODE* node = ToNode( aItem );
|
2018-01-20 17:15:46 +00:00
|
|
|
wxASSERT( node );
|
|
|
|
|
|
|
|
switch( aCol )
|
|
|
|
{
|
|
|
|
case 0:
|
2020-03-31 01:09:01 +00:00
|
|
|
if( m_frame->GetCurPart() && m_frame->GetCurPart()->GetLibId() == node->m_LibId )
|
|
|
|
node->m_Name = m_frame->GetCurPart()->GetLibId().GetLibItemName();
|
2020-02-07 17:06:24 +00:00
|
|
|
|
|
|
|
if( node->m_Pinned )
|
2020-02-16 09:21:36 +00:00
|
|
|
aVariant = GetPinningSymbol() + node->m_Name;
|
2020-03-31 01:09:01 +00:00
|
|
|
else
|
|
|
|
aVariant = node->m_Name;
|
2018-01-20 17:15:46 +00:00
|
|
|
|
2020-03-31 01:09:01 +00:00
|
|
|
// mark modified items with an asterisk
|
|
|
|
if( node->m_Type == LIB_TREE_NODE::LIB )
|
|
|
|
{
|
|
|
|
if( m_libMgr->IsLibraryModified( node->m_Name ) )
|
|
|
|
aVariant = aVariant.GetString() + " *";
|
|
|
|
}
|
|
|
|
else if( node->m_Type == LIB_TREE_NODE::LIBID )
|
|
|
|
{
|
|
|
|
if( m_libMgr->IsPartModified( node->m_Name, node->m_Parent->m_Name ) )
|
|
|
|
aVariant = aVariant.GetString() + " *";
|
|
|
|
}
|
2018-01-20 17:15:46 +00:00
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 1:
|
2020-03-31 01:09:01 +00:00
|
|
|
if( m_frame->GetCurPart() && m_frame->GetCurPart()->GetLibId() == node->m_LibId )
|
|
|
|
node->m_Desc = m_frame->GetCurPart()->GetDescription();
|
|
|
|
|
2020-02-07 17:06:24 +00:00
|
|
|
aVariant = node->m_Desc;
|
2020-08-29 23:19:05 +00:00
|
|
|
|
|
|
|
// Annotate that the library failed to load in the description column
|
|
|
|
if( node->m_Type == LIB_TREE_NODE::LIB )
|
|
|
|
{
|
|
|
|
if( !m_libMgr->IsLibraryLoaded( node->m_Name ) )
|
|
|
|
aVariant = _( "(failed to load) " ) + aVariant.GetString();
|
|
|
|
}
|
|
|
|
|
2018-01-20 17:15:46 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
default: // column == -1 is used for default Compare function
|
2020-02-07 17:06:24 +00:00
|
|
|
aVariant = node->m_Name;
|
2018-01-20 17:15:46 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-07-27 20:47:51 +00:00
|
|
|
bool SYMBOL_TREE_SYNCHRONIZING_ADAPTER::GetAttr( wxDataViewItem const& aItem, unsigned int aCol,
|
|
|
|
wxDataViewItemAttr& aAttr ) const
|
2017-11-12 17:55:20 +00:00
|
|
|
{
|
2018-04-28 09:43:41 +00:00
|
|
|
if( IsFrozen() )
|
|
|
|
return false;
|
|
|
|
|
2020-03-31 01:09:01 +00:00
|
|
|
LIB_TREE_NODE* node = ToNode( aItem );
|
2017-11-12 17:55:20 +00:00
|
|
|
wxCHECK( node, false );
|
|
|
|
|
2020-08-29 23:19:05 +00:00
|
|
|
// Mark both columns of unloaded libraries using grey text color (to look disabled)
|
|
|
|
if( node->m_Type == LIB_TREE_NODE::LIB && !m_libMgr->IsLibraryLoaded( node->m_Name ) )
|
|
|
|
{
|
|
|
|
aAttr.SetColour( wxSystemSettings::GetColour( wxSYS_COLOUR_GRAYTEXT ) );
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// The remaining attributes are only for the name column
|
|
|
|
if( aCol != 0 )
|
|
|
|
return false;
|
|
|
|
|
2020-02-07 17:06:24 +00:00
|
|
|
switch( node->m_Type )
|
2017-11-12 17:55:20 +00:00
|
|
|
{
|
2019-06-13 11:44:12 +00:00
|
|
|
case LIB_TREE_NODE::LIB:
|
|
|
|
// mark modified libs with bold font
|
2020-02-07 17:06:24 +00:00
|
|
|
aAttr.SetBold( m_libMgr->IsLibraryModified( node->m_Name ) );
|
2017-11-13 11:19:14 +00:00
|
|
|
|
2018-01-20 17:15:46 +00:00
|
|
|
#ifdef __WXGTK__
|
2019-06-13 11:44:12 +00:00
|
|
|
// The native wxGTK+ impl ignores background colour, so set the text colour instead.
|
|
|
|
// This works reasonably well in dark themes, and quite poorly in light ones....
|
2020-02-14 00:56:41 +00:00
|
|
|
if( node->m_Name == m_libMgr->GetCurrentLib() )
|
2019-06-13 11:44:12 +00:00
|
|
|
aAttr.SetColour( wxSystemSettings::GetColour( wxSYS_COLOUR_HIGHLIGHT ) );
|
2018-01-20 17:15:46 +00:00
|
|
|
#else
|
2019-06-13 11:44:12 +00:00
|
|
|
// mark the current library with background color
|
2020-02-07 17:06:24 +00:00
|
|
|
if( node->m_Name == m_libMgr->GetCurrentLib() )
|
2019-07-02 17:05:29 +00:00
|
|
|
{
|
|
|
|
aAttr.SetBackgroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_HIGHLIGHT ) );
|
|
|
|
aAttr.SetColour( wxSystemSettings::GetColour( wxSYS_COLOUR_LISTBOXHIGHLIGHTTEXT ) );
|
|
|
|
}
|
2018-01-20 17:15:46 +00:00
|
|
|
#endif
|
2019-06-13 11:44:12 +00:00
|
|
|
break;
|
2017-11-12 17:55:20 +00:00
|
|
|
|
2019-06-13 11:44:12 +00:00
|
|
|
case LIB_TREE_NODE::LIBID:
|
|
|
|
// mark modified part with bold font
|
2020-02-07 17:06:24 +00:00
|
|
|
aAttr.SetBold( m_libMgr->IsPartModified( node->m_Name, node->m_Parent->m_Name ) );
|
2017-11-12 17:55:20 +00:00
|
|
|
|
2019-06-13 11:44:12 +00:00
|
|
|
// mark aliases with italic font
|
2020-02-07 17:06:24 +00:00
|
|
|
aAttr.SetItalic( !node->m_IsRoot );
|
2017-11-13 11:19:14 +00:00
|
|
|
|
2018-01-20 17:15:46 +00:00
|
|
|
#ifdef __WXGTK__
|
2019-06-13 11:44:12 +00:00
|
|
|
// The native wxGTK+ impl ignores background colour, so set the text colour instead.
|
|
|
|
// This works reasonably well in dark themes, and quite poorly in light ones....
|
2020-02-14 00:56:41 +00:00
|
|
|
if( node->m_LibId == m_libMgr->GetCurrentLibId() )
|
2019-06-13 11:44:12 +00:00
|
|
|
aAttr.SetColour( wxSystemSettings::GetColour( wxSYS_COLOUR_HIGHLIGHT ) );
|
2018-01-20 17:15:46 +00:00
|
|
|
#else
|
2019-06-13 11:44:12 +00:00
|
|
|
// mark the current part with background color
|
2020-02-07 17:06:24 +00:00
|
|
|
if( node->m_LibId == m_libMgr->GetCurrentLibId() )
|
2019-07-02 17:05:29 +00:00
|
|
|
{
|
2019-06-13 11:44:12 +00:00
|
|
|
aAttr.SetBackgroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_HIGHLIGHT ) );
|
2019-07-02 17:05:29 +00:00
|
|
|
aAttr.SetColour( wxSystemSettings::GetColour( wxSYS_COLOUR_LISTBOXHIGHLIGHTTEXT ) );
|
|
|
|
}
|
2018-01-20 17:15:46 +00:00
|
|
|
#endif
|
2019-06-13 11:44:12 +00:00
|
|
|
break;
|
2017-11-12 17:55:20 +00:00
|
|
|
|
2019-06-13 11:44:12 +00:00
|
|
|
default:
|
|
|
|
return false;
|
2017-11-12 17:55:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|