|                                               Modular KiCad Blueprint Milestone B), major portions:
  *) When kicad.exe closes a project, close any open KIFACEs so that they cannot
     get disassociated from their true PROJECT.
  *) Allow loading eeschema library editor from kicad.exe
  *) Allow loading pcbnew library editor from kicad.exe
  *) Rename LIB_COMPONENT to LIB_PART.
  *) Add class PART_LIBS, and PART_LIB.
  *) Make PART_LIBS non-global, i.e. PROJECT specific.
  *) Implement "data on demand" for PART_LIBS
  *) Implement "data on demand" for schematic SEARCH_STACK.
  *) Use RSTRINGs to retain eeschema editor's notion of last library and part being edited.
  *) Get rid of library search on every SCH_COMPONENT::Draw() call, instead use
     a weak pointer.
  *) Remove all chdir() calls so projects don't need to be CWD.
  *) Romove APPEND support from OpenProjectFiles().
  *) Make OpenProjectFiles() robust, even for creating new projects.
  *) Load EESCHEMA colors in the KIWAY::OnKiwayStart() rather in window open,
     and save them in the .eeschema config file, not in the project file.
  *) Fix bug with wxDir() while accessing protected dirs in kicad.exe
  *) Consolidate template copying into PROJECT class, not in kicad.exe source.
  *) Generally untangle eeschema, making its libraries not global but rather
     held in the PROJECT.
 11 years ago     Modular KiCad Blueprint Milestone B), major portions:
  *) When kicad.exe closes a project, close any open KIFACEs so that they cannot
     get disassociated from their true PROJECT.
  *) Allow loading eeschema library editor from kicad.exe
  *) Allow loading pcbnew library editor from kicad.exe
  *) Rename LIB_COMPONENT to LIB_PART.
  *) Add class PART_LIBS, and PART_LIB.
  *) Make PART_LIBS non-global, i.e. PROJECT specific.
  *) Implement "data on demand" for PART_LIBS
  *) Implement "data on demand" for schematic SEARCH_STACK.
  *) Use RSTRINGs to retain eeschema editor's notion of last library and part being edited.
  *) Get rid of library search on every SCH_COMPONENT::Draw() call, instead use
     a weak pointer.
  *) Remove all chdir() calls so projects don't need to be CWD.
  *) Romove APPEND support from OpenProjectFiles().
  *) Make OpenProjectFiles() robust, even for creating new projects.
  *) Load EESCHEMA colors in the KIWAY::OnKiwayStart() rather in window open,
     and save them in the .eeschema config file, not in the project file.
  *) Fix bug with wxDir() while accessing protected dirs in kicad.exe
  *) Consolidate template copying into PROJECT class, not in kicad.exe source.
  *) Generally untangle eeschema, making its libraries not global but rather
     held in the PROJECT.
 11 years ago                                     |  | /*
 * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2004 Jean-Pierre Charras, jp.charras ar wanadoo.fr * Copyright (C) 2008 Wayne Stambaugh <stambaughw@gmail.com> * Copyright (C) 2004-2017 KiCad Developers, see AUTHORS.txt for contributors. * * 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 2 * 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: * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
 * or you may search the http://www.gnu.org website for the version 2 license,
 * or you may write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA */
/**
 * @file libarch.cpp * @brief Module for generation of component archive files. */
#include <fctsys.h>
#include <confirm.h>
#include <wildcards_and_files_ext.h>
#include <sch_edit_frame.h>
#include <symbol_lib_table.h>
#include <class_library.h>
#include <sch_component.h>
#include <sch_sheet.h>
bool SCH_EDIT_FRAME::CreateArchiveLibraryCacheFile( bool aUseCurrentSheetFilename ){    wxFileName fn;
    if( aUseCurrentSheetFilename )        fn = GetScreen()->GetFileName();    else        fn = g_RootSheet->GetScreen()->GetFileName();
    fn.SetName( fn.GetName() + "-cache" );    fn.SetExt( SchematicLibraryFileExtension );
    bool success = CreateArchiveLibrary( fn.GetFullPath() );
    // Update the schematic symbol library links.
    // because the lib cache has changed
    SCH_SCREENS schematic;    schematic.UpdateSymbolLinks();
    return success;}
bool SCH_EDIT_FRAME::CreateArchiveLibrary( const wxString& aFileName ){    wxString          tmp;    wxString          errorMsg;    SCH_SCREENS       screens;
    // Create a new empty library to archive components:
    std::unique_ptr<PART_LIB> archLib( new PART_LIB( LIBRARY_TYPE_EESCHEMA, aFileName ) );
    // Save symbols to file only when the library will be fully filled
    archLib->EnableBuffering();
    /* Examine all screens (not hierarchical sheets) used in the schematic and build a
     * library of unique symbols found in all screens.  Complex hierarchies are not a     * problem because we just want to know the library symbols used in the schematic     * not their reference.     */    for( SCH_SCREEN* screen = screens.GetFirst(); screen; screen = screens.GetNext() )    {        for( SCH_ITEM* item = screen->GetDrawItems(); item; item = item->Next() )        {            if( item->Type() != SCH_COMPONENT_T )                continue;
            LIB_PART* part = nullptr;            SCH_COMPONENT* component = (SCH_COMPONENT*) item;
            try            {                if( archLib->FindAlias( component->GetLibId() ) )                    continue;
                part = GetLibPart( component->GetLibId(), true );            }            catch( const IO_ERROR& )            {                // Queue up error messages for later.
                tmp.Printf( _( "Failed to add symbol \"%s\" to library file \"%s\"." ),                            component->GetLibId().GetUniStringLibItemName(), aFileName );
                // Don't bail out here.  Attempt to add as many of the symbols to the library
                // as possible.
            }            catch( ... )            {                tmp = _( "Unexpected exception occurred." );            }
            if( part )            {                // Use the full LIB_ID as the symbol name to prevent symbol name collisions.
                wxString oldName = part->GetName();                part->SetName( component->GetLibId().GetUniStringLibId() );
                // AddPart() does first clone the part before adding.
                archLib->AddPart( part );                part->SetName( oldName );            }            else            {                tmp.Printf( _( "Symbol %s not found in any library or cache." ),                            component->GetLibId().GetUniStringLibId() );            }
            if( !tmp.empty() )            {                if( errorMsg.empty() )                    errorMsg += tmp;                else                    errorMsg += "\n" + tmp;            }        }    }
    if( !errorMsg.empty() )    {        tmp.Printf( _( "Errors occurred creating symbol library %s." ), aFileName );        DisplayErrorMessage( this, tmp, errorMsg );    }
    archLib->EnableBuffering( false );
    try    {        archLib->Save( false );    }    catch( const IO_ERROR& ioe )    {        errorMsg.Printf( _( "Failed to save symbol library file \"%s\"" ), aFileName );        DisplayErrorMessage( this, errorMsg, ioe.What() );        return false;    }    catch( std::exception& error )    {        errorMsg.Printf( _( "Failed to save symbol library file \"%s\"" ), aFileName );        DisplayErrorMessage( this, errorMsg, error.what() );        return false;    }
    return true;}
 |