diff --git a/3d-viewer/3d_cache/3d_cache.cpp b/3d-viewer/3d_cache/3d_cache.cpp
index 33c04d0fff..b016b15e73 100644
--- a/3d-viewer/3d_cache/3d_cache.cpp
+++ b/3d-viewer/3d_cache/3d_cache.cpp
@@ -202,12 +202,12 @@ S3D_CACHE::~S3D_CACHE()
 
 
 SCENEGRAPH* S3D_CACHE::load( const wxString& aModelFile, const wxString& aBasePath,
-                             S3D_CACHE_ENTRY** aCachePtr )
+                             S3D_CACHE_ENTRY** aCachePtr, const EMBEDDED_FILES* aEmbeddedFiles )
 {
     if( aCachePtr )
         *aCachePtr = nullptr;
 
-    wxString full3Dpath = m_FNResolver->ResolvePath( aModelFile, aBasePath );
+    wxString full3Dpath = m_FNResolver->ResolvePath( aModelFile, aBasePath, aEmbeddedFiles );
 
     if( full3Dpath.empty() )
     {
@@ -272,9 +272,9 @@ SCENEGRAPH* S3D_CACHE::load( const wxString& aModelFile, const wxString& aBasePa
 }
 
 
-SCENEGRAPH* S3D_CACHE::Load( const wxString& aModelFile, const wxString& aBasePath )
+SCENEGRAPH* S3D_CACHE::Load( const wxString& aModelFile, const wxString& aBasePath, const EMBEDDED_FILES* aEmbeddedFiles )
 {
-    return load( aModelFile, aBasePath );
+    return load( aModelFile, aBasePath, nullptr, aEmbeddedFiles );
 }
 
 
@@ -631,10 +631,11 @@ void S3D_CACHE::ClosePlugins()
 }
 
 
-S3DMODEL* S3D_CACHE::GetModel( const wxString& aModelFileName, const wxString& aBasePath )
+S3DMODEL* S3D_CACHE::GetModel( const wxString& aModelFileName, const wxString& aBasePath,
+                               const EMBEDDED_FILES* aEmbeddedFiles )
 {
     S3D_CACHE_ENTRY* cp = nullptr;
-    SCENEGRAPH*      sp = load( aModelFileName, aBasePath,&cp );
+    SCENEGRAPH*      sp = load( aModelFileName, aBasePath, &cp, aEmbeddedFiles );
 
     if( !sp )
         return nullptr;
diff --git a/3d-viewer/3d_cache/3d_cache.h b/3d-viewer/3d_cache/3d_cache.h
index 4abc62e78b..16268ab9a5 100644
--- a/3d-viewer/3d_cache/3d_cache.h
+++ b/3d-viewer/3d_cache/3d_cache.h
@@ -38,6 +38,7 @@
 #include <project.h>
 #include <wx/string.h>
 
+class  EMBEDDED_FILES;
 class  PGM_BASE;
 class  S3D_CACHE_ENTRY;
 class  SCENEGRAPH;
@@ -93,9 +94,10 @@ public:
      *
      * @param aModelFile is the partial or full path to the model to be loaded.
      * @param aBasePath is the path to search for any relative files
+     * @param aEmbeddedFiles is a pointer to the embedded files list.
      * @return true if the model was successfully loaded, otherwise false.
      */
-    SCENEGRAPH* Load( const wxString& aModelFile, const wxString& aBasePath );
+    SCENEGRAPH* Load( const wxString& aModelFile, const wxString& aBasePath, const EMBEDDED_FILES* aEmbeddedFiles );
 
     FILENAME_RESOLVER* GetResolver() noexcept;
 
@@ -123,9 +125,11 @@ public:
      * structure for display by a renderer.
      *
      * @param aModelFileName is the full path to the model to be loaded.
+     * @param aBasePath is the path to search for any relative files.
+     * @param aEmbeddedFiles is a pointer to the embedded files list.
      * @return is a pointer to the render data or NULL if not available.
      */
-    S3DMODEL* GetModel( const wxString& aModelFileName, const wxString& aBasePath );
+    S3DMODEL* GetModel( const wxString& aModelFileName, const wxString& aBasePath, const EMBEDDED_FILES* aEmbeddedFiles );
 
     /**
      * Delete up old cache files in cache directory.
@@ -165,7 +169,9 @@ private:
     bool saveCacheData( S3D_CACHE_ENTRY* aCacheItem );
 
     // the real load function (can supply a cache entry pointer to member functions)
-    SCENEGRAPH* load( const wxString& aModelFile, const wxString& aBasePath, S3D_CACHE_ENTRY** aCachePtr = nullptr );
+    SCENEGRAPH* load( const wxString& aModelFile, const wxString& aBasePath,
+                      S3D_CACHE_ENTRY** aCachePtr = nullptr,
+                      const EMBEDDED_FILES*   aEmbeddedFiles = nullptr );
 
     /// cache entries
     std::list< S3D_CACHE_ENTRY* > m_CacheList;
diff --git a/3d-viewer/3d_model_viewer/eda_3d_model_viewer.cpp b/3d-viewer/3d_model_viewer/eda_3d_model_viewer.cpp
index fc1b7eaf72..9b1824672b 100644
--- a/3d-viewer/3d_model_viewer/eda_3d_model_viewer.cpp
+++ b/3d-viewer/3d_model_viewer/eda_3d_model_viewer.cpp
@@ -156,7 +156,7 @@ void EDA_3D_MODEL_VIEWER::Set3DModel( const wxString& aModelPathName)
 
     if( m_cacheManager )
     {
-        const S3DMODEL* model = m_cacheManager->GetModel( aModelPathName, wxEmptyString );
+        const S3DMODEL* model = m_cacheManager->GetModel( aModelPathName, wxEmptyString, nullptr );
 
         if( model )
             Set3DModel( (const S3DMODEL &)*model );
diff --git a/3d-viewer/3d_model_viewer/eda_3d_model_viewer.h b/3d-viewer/3d_model_viewer/eda_3d_model_viewer.h
index a85fc66a06..161597dc4d 100644
--- a/3d-viewer/3d_model_viewer/eda_3d_model_viewer.h
+++ b/3d-viewer/3d_model_viewer/eda_3d_model_viewer.h
@@ -66,7 +66,9 @@ public:
     /**
      * Set this model to be displayed.
      *
-     * @param aModelPathName 3D model path name.
+     * N.B. This will not load a model from the internal cache.  Only from on disk.
+     *
+     * @param aModelPathName 3D model path name.  Must be a file on disk.
      */
     void Set3DModel( const wxString& aModelPathName );
 
diff --git a/3d-viewer/3d_rendering/opengl/create_scene.cpp b/3d-viewer/3d_rendering/opengl/create_scene.cpp
index f2faab5c69..3079dd4ed3 100644
--- a/3d-viewer/3d_rendering/opengl/create_scene.cpp
+++ b/3d-viewer/3d_rendering/opengl/create_scene.cpp
@@ -974,7 +974,7 @@ void RENDER_3D_OPENGL::load3dModels( REPORTER* aStatusReporter )
                 {
                     // It is not present, try get it from cache
                     const S3DMODEL* modelPtr =
-                            m_boardAdapter.Get3dCacheManager()->GetModel( fp_model.m_Filename, footprintBasePath );
+                            m_boardAdapter.Get3dCacheManager()->GetModel( fp_model.m_Filename, footprintBasePath, footprint );
 
                     // only add it if the return is not NULL
                     if( modelPtr )
diff --git a/3d-viewer/3d_rendering/raytracing/create_scene.cpp b/3d-viewer/3d_rendering/raytracing/create_scene.cpp
index a2cbc78c56..3ceacc0acd 100644
--- a/3d-viewer/3d_rendering/raytracing/create_scene.cpp
+++ b/3d-viewer/3d_rendering/raytracing/create_scene.cpp
@@ -1247,8 +1247,6 @@ void RENDER_3D_RAYTRACE_BASE::load3DModels( CONTAINER_3D& aDstContainer, bool aS
 
             // Get the list of model files for this model
             S3D_CACHE* cacheMgr = m_boardAdapter.Get3dCacheManager();
-            auto       sM       = fp->Models().begin();
-            auto       eM       = fp->Models().end();
 
             wxString                libraryName = fp->GetFPID().GetLibNickname();
 
@@ -1271,44 +1269,38 @@ void RENDER_3D_RAYTRACE_BASE::load3DModels( CONTAINER_3D& aDstContainer, bool aS
                 }
             }
 
-            while( sM != eM )
+            for( FP_3DMODEL& model : fp->Models() )
             {
-                if( ( static_cast<float>( sM->m_Opacity ) > FLT_EPSILON )
-                  && ( sM->m_Show && !sM->m_Filename.empty() ) )
+                // get it from cache
+                const S3DMODEL* modelPtr =
+                        cacheMgr->GetModel( model.m_Filename, footprintBasePath, fp );
+
+                // only add it if the return is not NULL.
+                if( modelPtr )
                 {
-                    // get it from cache
-                    const S3DMODEL* modelPtr =
-                            cacheMgr->GetModel( sM->m_Filename, footprintBasePath );
+                    glm::mat4 modelMatrix = fpMatrix;
 
-                    // only add it if the return is not NULL.
-                    if( modelPtr )
-                    {
-                        glm::mat4 modelMatrix = fpMatrix;
+                    modelMatrix = glm::translate( modelMatrix,
+                            SFVEC3F( model.m_Offset.x, model.m_Offset.y, model.m_Offset.z ) );
 
-                        modelMatrix = glm::translate( modelMatrix,
-                                SFVEC3F( sM->m_Offset.x, sM->m_Offset.y, sM->m_Offset.z ) );
+                    modelMatrix = glm::rotate( modelMatrix,
+                            (float) -( model.m_Rotation.z / 180.0f ) * glm::pi<float>(),
+                            SFVEC3F( 0.0f, 0.0f, 1.0f ) );
 
-                        modelMatrix = glm::rotate( modelMatrix,
-                                (float) -( sM->m_Rotation.z / 180.0f ) * glm::pi<float>(),
-                                SFVEC3F( 0.0f, 0.0f, 1.0f ) );
+                    modelMatrix = glm::rotate( modelMatrix,
+                            (float) -( model.m_Rotation.y / 180.0f ) * glm::pi<float>(),
+                            SFVEC3F( 0.0f, 1.0f, 0.0f ) );
 
-                        modelMatrix = glm::rotate( modelMatrix,
-                                (float) -( sM->m_Rotation.y / 180.0f ) * glm::pi<float>(),
-                                SFVEC3F( 0.0f, 1.0f, 0.0f ) );
+                    modelMatrix = glm::rotate( modelMatrix,
+                            (float) -( model.m_Rotation.x / 180.0f ) * glm::pi<float>(),
+                            SFVEC3F( 1.0f, 0.0f, 0.0f ) );
 
-                        modelMatrix = glm::rotate( modelMatrix,
-                                (float) -( sM->m_Rotation.x / 180.0f ) * glm::pi<float>(),
-                                SFVEC3F( 1.0f, 0.0f, 0.0f ) );
+                    modelMatrix = glm::scale( modelMatrix,
+                            SFVEC3F( model.m_Scale.x, model.m_Scale.y, model.m_Scale.z ) );
 
-                        modelMatrix = glm::scale( modelMatrix,
-                                SFVEC3F( sM->m_Scale.x, sM->m_Scale.y, sM->m_Scale.z ) );
-
-                        addModels( aDstContainer, modelPtr, modelMatrix, (float) sM->m_Opacity,
-                                   aSkipMaterialInformation, boardItem );
-                    }
+                    addModels( aDstContainer, modelPtr, modelMatrix, (float) model.m_Opacity,
+                               aSkipMaterialInformation, boardItem );
                 }
-
-                ++sM;
             }
         }
     }
diff --git a/3d-viewer/CMakeLists.txt b/3d-viewer/CMakeLists.txt
index 73db487ad9..23a2c75b03 100644
--- a/3d-viewer/CMakeLists.txt
+++ b/3d-viewer/CMakeLists.txt
@@ -91,7 +91,6 @@ set(3D-VIEWER_SRCS
     common_ogl/ogl_utils.cpp
     3d_fastmath.cpp
     3d_math.cpp
-    dialogs/3d_cache_dialogs.cpp
     dialogs/appearance_controls_3D.cpp
     dialogs/appearance_controls_3D_base.cpp
     dialogs/dialog_select_3d_model_base.cpp
diff --git a/3d-viewer/dialogs/3d_cache_dialogs.cpp b/3d-viewer/dialogs/3d_cache_dialogs.cpp
deleted file mode 100644
index 676726141f..0000000000
--- a/3d-viewer/dialogs/3d_cache_dialogs.cpp
+++ /dev/null
@@ -1,52 +0,0 @@
-/*
- * This program source code file is part of KiCad, a free EDA CAD application.
- *
- * Copyright (C) 2015-2016 Cirilo Bernardo <cirilo.bernardo@gmail.com>
- * Copyright (C) 2020-2021 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
- */
-
-#include <dialogs/dialog_configure_paths.h>
-
-#include "3d_info.h"
-#include "3d_cache.h"
-#include "3d_cache_dialogs.h"
-#include "dialog_select_3d_model.h"
-
-
-bool S3D::Select3DModel( wxWindow* aParent, S3D_CACHE* aCache, wxString& prevModelSelectDir,
-                         int& prevModelWildcard, FP_3DMODEL* aModel )
-{
-    if( nullptr == aModel )
-        return false;
-
-    DIALOG_SELECT_3DMODEL dm( aParent, aCache, aModel, prevModelSelectDir, prevModelWildcard );
-
-    // Use QuasiModal so that Configure3DPaths (and its help window) will work
-    return dm.ShowQuasiModal() == wxID_OK;
-}
-
-
-bool S3D::Configure3DPaths( wxWindow* aParent, FILENAME_RESOLVER* aResolver )
-{
-    DIALOG_CONFIGURE_PATHS dlg( aParent );
-
-    // Use QuasiModal so that HTML help window will work
-    return( dlg.ShowQuasiModal() == wxID_OK );
-}
diff --git a/3d-viewer/dialogs/3d_cache_dialogs.h b/3d-viewer/dialogs/3d_cache_dialogs.h
deleted file mode 100644
index 89412255f0..0000000000
--- a/3d-viewer/dialogs/3d_cache_dialogs.h
+++ /dev/null
@@ -1,40 +0,0 @@
-/*
- * This program source code file is part of KiCad, a free EDA CAD application.
- *
- * Copyright (C) 2015 Cirilo Bernardo <cirilo.bernardo@gmail.com>
- *
- * 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
- */
-
-#ifndef CACHE_DIALOGS_3D_H
-#define CACHE_DIALOGS_3D_H
-
-#include <wx/window.h>
-
-class S3D_CACHE;
-class FILENAME_RESOLVER;
-
-namespace S3D
-{
-    bool Select3DModel( wxWindow* aParent, S3D_CACHE* aCache, wxString& prevModelSelectDir,
-                        int& prevModelWildcard, FP_3DMODEL* aModel );
-
-    bool Configure3DPaths( wxWindow* aParent, FILENAME_RESOLVER* aResolver );
-}
-
-#endif  // CACHE_DIALOGS_3D_H
diff --git a/3d-viewer/dialogs/dialog_select_3d_model.cpp b/3d-viewer/dialogs/dialog_select_3d_model.cpp
index c1a1452c40..3c3d7f272f 100644
--- a/3d-viewer/dialogs/dialog_select_3d_model.cpp
+++ b/3d-viewer/dialogs/dialog_select_3d_model.cpp
@@ -29,9 +29,9 @@
 #include "project.h"
 #include "3d_cache/3d_info.h"
 #include "3d_cache/3d_cache.h"
-#include "3d_cache_dialogs.h"
 #include <3d_model_viewer/eda_3d_model_viewer.h>
 #include <common_ogl/ogl_attr_list.h>
+#include <dialogs/dialog_configure_paths.h>
 #include <filename_resolver.h>
 #include <pcbnew/footprint.h>
 #include <wx_filename.h>
@@ -197,7 +197,9 @@ void DIALOG_SELECT_3DMODEL::SetRootDir( wxCommandEvent& event )
 
 void DIALOG_SELECT_3DMODEL::Cfg3DPaths( wxCommandEvent& event )
 {
-    if( S3D::Configure3DPaths( this, m_resolver ) )
+    DIALOG_CONFIGURE_PATHS dlg( this );
+
+    if( dlg.ShowQuasiModal() == wxID_OK )
         updateDirChoiceList();
 }
 
diff --git a/3d-viewer/dialogs/dialog_select_3d_model.h b/3d-viewer/dialogs/dialog_select_3d_model.h
index 29c8d7fd1a..7445d9f184 100644
--- a/3d-viewer/dialogs/dialog_select_3d_model.h
+++ b/3d-viewer/dialogs/dialog_select_3d_model.h
@@ -44,6 +44,11 @@ public:
     void SetRootDir( wxCommandEvent& event ) override;
     void Cfg3DPaths( wxCommandEvent& event ) override;
 
+    bool IsEmbedded3DModel() const
+    {
+        return m_EmbedModelCb->IsChecked();
+    }
+
 private:
     void updateDirChoiceList( void );
 
diff --git a/3d-viewer/dialogs/dialog_select_3d_model_base.cpp b/3d-viewer/dialogs/dialog_select_3d_model_base.cpp
index 5e1825dfc0..620526b281 100644
--- a/3d-viewer/dialogs/dialog_select_3d_model_base.cpp
+++ b/3d-viewer/dialogs/dialog_select_3d_model_base.cpp
@@ -1,5 +1,5 @@
 ///////////////////////////////////////////////////////////////////////////
-// C++ code generated with wxFormBuilder (version 3.10.1-0-g8feb16b)
+// C++ code generated with wxFormBuilder (version 4.0.0-0-g0efcecf0)
 // http://www.wxformbuilder.org/
 //
 // PLEASE DO *NOT* EDIT THIS FILE!
@@ -35,6 +35,12 @@ DIALOG_SELECT_3D_MODEL_BASE::DIALOG_SELECT_3D_MODEL_BASE( wxWindow* parent, wxWi
 
 	bSizerLeft->Add( m_FileTree, 1, wxALL|wxEXPAND, 5 );
 
+	wxBoxSizer* bSizer7;
+	bSizer7 = new wxBoxSizer( wxHORIZONTAL );
+
+
+	bSizerLeft->Add( bSizer7, 0, wxEXPAND, 5 );
+
 
 	m_panelLeft->SetSizer( bSizerLeft );
 	m_panelLeft->Layout();
@@ -70,6 +76,15 @@ DIALOG_SELECT_3D_MODEL_BASE::DIALOG_SELECT_3D_MODEL_BASE( wxWindow* parent, wxWi
 
 	bSizerMain->Add( bSizerLower, 0, wxEXPAND, 5 );
 
+	wxBoxSizer* bSizer6;
+	bSizer6 = new wxBoxSizer( wxHORIZONTAL );
+
+	m_EmbedModelCb = new wxCheckBox( this, wxID_ANY, _("Embed Model"), wxDefaultPosition, wxDefaultSize, 0 );
+	bSizer6->Add( m_EmbedModelCb, 0, wxALL, 5 );
+
+
+	bSizer6->Add( 0, 0, 1, wxEXPAND, 5 );
+
 	m_sdbSizer = new wxStdDialogButtonSizer();
 	m_sdbSizerOK = new wxButton( this, wxID_OK );
 	m_sdbSizer->AddButton( m_sdbSizerOK );
@@ -77,7 +92,10 @@ DIALOG_SELECT_3D_MODEL_BASE::DIALOG_SELECT_3D_MODEL_BASE( wxWindow* parent, wxWi
 	m_sdbSizer->AddButton( m_sdbSizerCancel );
 	m_sdbSizer->Realize();
 
-	bSizerMain->Add( m_sdbSizer, 0, wxEXPAND|wxALL, 5 );
+	bSizer6->Add( m_sdbSizer, 0, wxEXPAND|wxALL, 5 );
+
+
+	bSizerMain->Add( bSizer6, 0, wxEXPAND, 5 );
 
 
 	this->SetSizer( bSizerMain );
diff --git a/3d-viewer/dialogs/dialog_select_3d_model_base.fbp b/3d-viewer/dialogs/dialog_select_3d_model_base.fbp
index 5d4b0626ec..b0293cf861 100644
--- a/3d-viewer/dialogs/dialog_select_3d_model_base.fbp
+++ b/3d-viewer/dialogs/dialog_select_3d_model_base.fbp
@@ -1,554 +1,659 @@
-<?xml version="1.0" encoding="UTF-8" standalone="yes" ?>
+<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
 <wxFormBuilder_Project>
-    <FileVersion major="1" minor="16" />
-    <object class="Project" expanded="1">
-        <property name="class_decoration">; </property>
-        <property name="code_generation">C++</property>
-        <property name="disconnect_events">1</property>
-        <property name="disconnect_mode">source_name</property>
-        <property name="disconnect_php_events">0</property>
-        <property name="disconnect_python_events">0</property>
-        <property name="embedded_files_path">res</property>
-        <property name="encoding">UTF-8</property>
-        <property name="event_generation">connect</property>
-        <property name="file">dialog_select_3d_model_base</property>
-        <property name="first_id">1000</property>
-        <property name="help_provider">none</property>
-        <property name="image_path_wrapper_function_name"></property>
-        <property name="indent_with_spaces"></property>
-        <property name="internationalize">1</property>
-        <property name="name">dialog_select_3d_model_base</property>
-        <property name="namespace"></property>
-        <property name="path">.</property>
-        <property name="precompiled_header"></property>
-        <property name="relative_path">1</property>
-        <property name="skip_lua_events">1</property>
-        <property name="skip_php_events">1</property>
-        <property name="skip_python_events">1</property>
-        <property name="ui_table">UI</property>
-        <property name="use_array_enum">0</property>
-        <property name="use_enum">0</property>
-        <property name="use_microsoft_bom">0</property>
-        <object class="Dialog" expanded="1">
-            <property name="aui_managed">0</property>
-            <property name="aui_manager_style">wxAUI_MGR_DEFAULT</property>
-            <property name="bg"></property>
-            <property name="center">wxBOTH</property>
-            <property name="context_help"></property>
-            <property name="context_menu">1</property>
-            <property name="enabled">1</property>
-            <property name="event_handler">impl_virtual</property>
-            <property name="extra_style"></property>
-            <property name="fg"></property>
-            <property name="font"></property>
-            <property name="hidden">0</property>
-            <property name="id">wxID_ANY</property>
-            <property name="maximum_size"></property>
-            <property name="minimum_size"></property>
-            <property name="name">DIALOG_SELECT_3D_MODEL_BASE</property>
-            <property name="pos"></property>
-            <property name="size">-1,-1</property>
-            <property name="style">wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER</property>
-            <property name="subclass">DIALOG_SHIM; dialog_shim.h; forward_declare</property>
-            <property name="title">Select 3D Model</property>
-            <property name="tooltip"></property>
-            <property name="two_step_creation">0</property>
-            <property name="window_extra_style"></property>
-            <property name="window_name"></property>
-            <property name="window_style"></property>
-            <object class="wxBoxSizer" expanded="1">
+  <FileVersion major="1" minor="17"/>
+  <object class="Project" expanded="true">
+    <property name="class_decoration">; </property>
+    <property name="code_generation">C++</property>
+    <property name="disconnect_events">1</property>
+    <property name="disconnect_mode">source_name</property>
+    <property name="disconnect_php_events">0</property>
+    <property name="disconnect_python_events">0</property>
+    <property name="embedded_files_path">res</property>
+    <property name="encoding">UTF-8</property>
+    <property name="event_generation">connect</property>
+    <property name="file">dialog_select_3d_model_base</property>
+    <property name="first_id">1000</property>
+    <property name="help_provider">none</property>
+    <property name="image_path_wrapper_function_name"></property>
+    <property name="indent_with_spaces"></property>
+    <property name="internationalize">1</property>
+    <property name="name">dialog_select_3d_model_base</property>
+    <property name="namespace"></property>
+    <property name="path">.</property>
+    <property name="precompiled_header"></property>
+    <property name="relative_path">1</property>
+    <property name="skip_lua_events">1</property>
+    <property name="skip_php_events">1</property>
+    <property name="skip_python_events">1</property>
+    <property name="ui_table">UI</property>
+    <property name="use_array_enum">0</property>
+    <property name="use_enum">0</property>
+    <property name="use_microsoft_bom">0</property>
+    <object class="Dialog" expanded="true">
+      <property name="aui_managed">0</property>
+      <property name="aui_manager_style">wxAUI_MGR_DEFAULT</property>
+      <property name="bg"></property>
+      <property name="center">wxBOTH</property>
+      <property name="context_help"></property>
+      <property name="context_menu">1</property>
+      <property name="drag_accept_files">0</property>
+      <property name="enabled">1</property>
+      <property name="event_handler">impl_virtual</property>
+      <property name="extra_style"></property>
+      <property name="fg"></property>
+      <property name="font"></property>
+      <property name="hidden">0</property>
+      <property name="id">wxID_ANY</property>
+      <property name="maximum_size"></property>
+      <property name="minimum_size"></property>
+      <property name="name">DIALOG_SELECT_3D_MODEL_BASE</property>
+      <property name="pos"></property>
+      <property name="size">-1,-1</property>
+      <property name="style">wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER</property>
+      <property name="subclass">DIALOG_SHIM; dialog_shim.h; forward_declare</property>
+      <property name="title">Select 3D Model</property>
+      <property name="tooltip"></property>
+      <property name="two_step_creation">0</property>
+      <property name="window_extra_style"></property>
+      <property name="window_name"></property>
+      <property name="window_style"></property>
+      <object class="wxBoxSizer" expanded="true">
+        <property name="minimum_size"></property>
+        <property name="name">bSizerMain</property>
+        <property name="orient">wxVERTICAL</property>
+        <property name="permission">none</property>
+        <object class="sizeritem" expanded="true">
+          <property name="border">5</property>
+          <property name="flag">wxEXPAND</property>
+          <property name="proportion">1</property>
+          <object class="wxBoxSizer" expanded="true">
+            <property name="minimum_size">-1,400</property>
+            <property name="name">bSizerUpper</property>
+            <property name="orient">wxHORIZONTAL</property>
+            <property name="permission">none</property>
+            <object class="sizeritem" expanded="true">
+              <property name="border">5</property>
+              <property name="flag">wxEXPAND</property>
+              <property name="proportion">1</property>
+              <object class="wxSplitterWindow" expanded="true">
+                <property name="BottomDockable">1</property>
+                <property name="LeftDockable">1</property>
+                <property name="RightDockable">1</property>
+                <property name="TopDockable">1</property>
+                <property name="aui_layer"></property>
+                <property name="aui_name"></property>
+                <property name="aui_position"></property>
+                <property name="aui_row"></property>
+                <property name="best_size"></property>
+                <property name="bg"></property>
+                <property name="caption"></property>
+                <property name="caption_visible">1</property>
+                <property name="center_pane">0</property>
+                <property name="close_button">1</property>
+                <property name="context_help"></property>
+                <property name="context_menu">1</property>
+                <property name="default_pane">0</property>
+                <property name="dock">Dock</property>
+                <property name="dock_fixed">0</property>
+                <property name="docking">Left</property>
+                <property name="drag_accept_files">0</property>
+                <property name="enabled">1</property>
+                <property name="fg"></property>
+                <property name="floatable">1</property>
+                <property name="font"></property>
+                <property name="gripper">0</property>
+                <property name="hidden">0</property>
+                <property name="id">wxID_ANY</property>
+                <property name="max_size"></property>
+                <property name="maximize_button">0</property>
+                <property name="maximum_size"></property>
+                <property name="min_pane_size">0</property>
+                <property name="min_size"></property>
+                <property name="minimize_button">0</property>
                 <property name="minimum_size"></property>
-                <property name="name">bSizerMain</property>
-                <property name="orient">wxVERTICAL</property>
-                <property name="permission">none</property>
-                <object class="sizeritem" expanded="1">
-                    <property name="border">5</property>
-                    <property name="flag">wxEXPAND</property>
-                    <property name="proportion">1</property>
-                    <object class="wxBoxSizer" expanded="1">
-                        <property name="minimum_size">-1,400</property>
-                        <property name="name">bSizerUpper</property>
-                        <property name="orient">wxHORIZONTAL</property>
-                        <property name="permission">none</property>
-                        <object class="sizeritem" expanded="1">
-                            <property name="border">5</property>
-                            <property name="flag">wxEXPAND</property>
-                            <property name="proportion">1</property>
-                            <object class="wxSplitterWindow" expanded="1">
-                                <property name="BottomDockable">1</property>
-                                <property name="LeftDockable">1</property>
-                                <property name="RightDockable">1</property>
-                                <property name="TopDockable">1</property>
-                                <property name="aui_layer"></property>
-                                <property name="aui_name"></property>
-                                <property name="aui_position"></property>
-                                <property name="aui_row"></property>
-                                <property name="best_size"></property>
-                                <property name="bg"></property>
-                                <property name="caption"></property>
-                                <property name="caption_visible">1</property>
-                                <property name="center_pane">0</property>
-                                <property name="close_button">1</property>
-                                <property name="context_help"></property>
-                                <property name="context_menu">1</property>
-                                <property name="default_pane">0</property>
-                                <property name="dock">Dock</property>
-                                <property name="dock_fixed">0</property>
-                                <property name="docking">Left</property>
-                                <property name="enabled">1</property>
-                                <property name="fg"></property>
-                                <property name="floatable">1</property>
-                                <property name="font"></property>
-                                <property name="gripper">0</property>
-                                <property name="hidden">0</property>
-                                <property name="id">wxID_ANY</property>
-                                <property name="max_size"></property>
-                                <property name="maximize_button">0</property>
-                                <property name="maximum_size"></property>
-                                <property name="min_pane_size">0</property>
-                                <property name="min_size"></property>
-                                <property name="minimize_button">0</property>
-                                <property name="minimum_size"></property>
-                                <property name="moveable">1</property>
-                                <property name="name">m_splitterWin</property>
-                                <property name="pane_border">1</property>
-                                <property name="pane_position"></property>
-                                <property name="pane_size"></property>
-                                <property name="permission">protected</property>
-                                <property name="pin_button">1</property>
-                                <property name="pos"></property>
-                                <property name="resize">Resizable</property>
-                                <property name="sashgravity">0.35</property>
-                                <property name="sashpos">300</property>
-                                <property name="sashsize">-1</property>
-                                <property name="show">1</property>
-                                <property name="size"></property>
-                                <property name="splitmode">wxSPLIT_VERTICAL</property>
-                                <property name="style">wxSP_3D</property>
-                                <property name="subclass">; ; forward_declare</property>
-                                <property name="toolbar_pane">0</property>
-                                <property name="tooltip"></property>
-                                <property name="window_extra_style"></property>
-                                <property name="window_name"></property>
-                                <property name="window_style"></property>
-                                <object class="splitteritem" expanded="1">
-                                    <object class="wxPanel" expanded="1">
-                                        <property name="BottomDockable">1</property>
-                                        <property name="LeftDockable">1</property>
-                                        <property name="RightDockable">1</property>
-                                        <property name="TopDockable">1</property>
-                                        <property name="aui_layer"></property>
-                                        <property name="aui_name"></property>
-                                        <property name="aui_position"></property>
-                                        <property name="aui_row"></property>
-                                        <property name="best_size"></property>
-                                        <property name="bg"></property>
-                                        <property name="caption"></property>
-                                        <property name="caption_visible">1</property>
-                                        <property name="center_pane">0</property>
-                                        <property name="close_button">1</property>
-                                        <property name="context_help"></property>
-                                        <property name="context_menu">1</property>
-                                        <property name="default_pane">0</property>
-                                        <property name="dock">Dock</property>
-                                        <property name="dock_fixed">0</property>
-                                        <property name="docking">Left</property>
-                                        <property name="enabled">1</property>
-                                        <property name="fg"></property>
-                                        <property name="floatable">1</property>
-                                        <property name="font"></property>
-                                        <property name="gripper">0</property>
-                                        <property name="hidden">0</property>
-                                        <property name="id">wxID_ANY</property>
-                                        <property name="max_size"></property>
-                                        <property name="maximize_button">0</property>
-                                        <property name="maximum_size"></property>
-                                        <property name="min_size"></property>
-                                        <property name="minimize_button">0</property>
-                                        <property name="minimum_size"></property>
-                                        <property name="moveable">1</property>
-                                        <property name="name">m_panelLeft</property>
-                                        <property name="pane_border">1</property>
-                                        <property name="pane_position"></property>
-                                        <property name="pane_size"></property>
-                                        <property name="permission">protected</property>
-                                        <property name="pin_button">1</property>
-                                        <property name="pos"></property>
-                                        <property name="resize">Resizable</property>
-                                        <property name="show">1</property>
-                                        <property name="size"></property>
-                                        <property name="subclass">; ; forward_declare</property>
-                                        <property name="toolbar_pane">0</property>
-                                        <property name="tooltip"></property>
-                                        <property name="window_extra_style"></property>
-                                        <property name="window_name"></property>
-                                        <property name="window_style">wxTAB_TRAVERSAL</property>
-                                        <object class="wxBoxSizer" expanded="1">
-                                            <property name="minimum_size"></property>
-                                            <property name="name">bSizerLeft</property>
-                                            <property name="orient">wxVERTICAL</property>
-                                            <property name="permission">none</property>
-                                            <object class="sizeritem" expanded="1">
-                                                <property name="border">5</property>
-                                                <property name="flag">wxALL|wxEXPAND</property>
-                                                <property name="proportion">1</property>
-                                                <object class="wxGenericDirCtrl" expanded="1">
-                                                    <property name="BottomDockable">1</property>
-                                                    <property name="LeftDockable">1</property>
-                                                    <property name="RightDockable">1</property>
-                                                    <property name="TopDockable">1</property>
-                                                    <property name="aui_layer"></property>
-                                                    <property name="aui_name"></property>
-                                                    <property name="aui_position"></property>
-                                                    <property name="aui_row"></property>
-                                                    <property name="best_size"></property>
-                                                    <property name="bg"></property>
-                                                    <property name="caption"></property>
-                                                    <property name="caption_visible">1</property>
-                                                    <property name="center_pane">0</property>
-                                                    <property name="close_button">1</property>
-                                                    <property name="context_help"></property>
-                                                    <property name="context_menu">1</property>
-                                                    <property name="default_pane">0</property>
-                                                    <property name="defaultfilter">0</property>
-                                                    <property name="defaultfolder"></property>
-                                                    <property name="dock">Dock</property>
-                                                    <property name="dock_fixed">0</property>
-                                                    <property name="docking">Left</property>
-                                                    <property name="enabled">1</property>
-                                                    <property name="fg"></property>
-                                                    <property name="filter"></property>
-                                                    <property name="floatable">1</property>
-                                                    <property name="font"></property>
-                                                    <property name="gripper">0</property>
-                                                    <property name="hidden">0</property>
-                                                    <property name="id">wxID_ANY</property>
-                                                    <property name="max_size"></property>
-                                                    <property name="maximize_button">0</property>
-                                                    <property name="maximum_size"></property>
-                                                    <property name="min_size"></property>
-                                                    <property name="minimize_button">0</property>
-                                                    <property name="minimum_size">300,-1</property>
-                                                    <property name="moveable">1</property>
-                                                    <property name="name">m_FileTree</property>
-                                                    <property name="pane_border">1</property>
-                                                    <property name="pane_position"></property>
-                                                    <property name="pane_size"></property>
-                                                    <property name="permission">protected</property>
-                                                    <property name="pin_button">1</property>
-                                                    <property name="pos"></property>
-                                                    <property name="resize">Resizable</property>
-                                                    <property name="show">1</property>
-                                                    <property name="show_hidden">0</property>
-                                                    <property name="size"></property>
-                                                    <property name="style">wxDIRCTRL_3D_INTERNAL|wxDIRCTRL_EDIT_LABELS|wxDIRCTRL_SELECT_FIRST|wxDIRCTRL_SHOW_FILTERS</property>
-                                                    <property name="subclass">; ; forward_declare</property>
-                                                    <property name="toolbar_pane">0</property>
-                                                    <property name="tooltip"></property>
-                                                    <property name="window_extra_style"></property>
-                                                    <property name="window_name"></property>
-                                                    <property name="window_style">wxBORDER_DEFAULT</property>
-                                                    <event name="OnDirctrlFileActivated">OnFileActivated</event>
-                                                    <event name="OnDirctrlSelectionChanged">OnSelectionChanged</event>
-                                                </object>
-                                            </object>
-                                        </object>
-                                    </object>
-                                </object>
-                                <object class="splitteritem" expanded="1">
-                                    <object class="wxPanel" expanded="1">
-                                        <property name="BottomDockable">1</property>
-                                        <property name="LeftDockable">1</property>
-                                        <property name="RightDockable">1</property>
-                                        <property name="TopDockable">1</property>
-                                        <property name="aui_layer"></property>
-                                        <property name="aui_name"></property>
-                                        <property name="aui_position"></property>
-                                        <property name="aui_row"></property>
-                                        <property name="best_size"></property>
-                                        <property name="bg"></property>
-                                        <property name="caption"></property>
-                                        <property name="caption_visible">1</property>
-                                        <property name="center_pane">0</property>
-                                        <property name="close_button">1</property>
-                                        <property name="context_help"></property>
-                                        <property name="context_menu">1</property>
-                                        <property name="default_pane">0</property>
-                                        <property name="dock">Dock</property>
-                                        <property name="dock_fixed">0</property>
-                                        <property name="docking">Left</property>
-                                        <property name="enabled">1</property>
-                                        <property name="fg"></property>
-                                        <property name="floatable">1</property>
-                                        <property name="font"></property>
-                                        <property name="gripper">0</property>
-                                        <property name="hidden">0</property>
-                                        <property name="id">wxID_ANY</property>
-                                        <property name="max_size"></property>
-                                        <property name="maximize_button">0</property>
-                                        <property name="maximum_size"></property>
-                                        <property name="min_size"></property>
-                                        <property name="minimize_button">0</property>
-                                        <property name="minimum_size"></property>
-                                        <property name="moveable">1</property>
-                                        <property name="name">m_pane3Dviewer</property>
-                                        <property name="pane_border">1</property>
-                                        <property name="pane_position"></property>
-                                        <property name="pane_size"></property>
-                                        <property name="permission">protected</property>
-                                        <property name="pin_button">1</property>
-                                        <property name="pos"></property>
-                                        <property name="resize">Resizable</property>
-                                        <property name="show">1</property>
-                                        <property name="size"></property>
-                                        <property name="subclass">; ; forward_declare</property>
-                                        <property name="toolbar_pane">0</property>
-                                        <property name="tooltip"></property>
-                                        <property name="window_extra_style"></property>
-                                        <property name="window_name"></property>
-                                        <property name="window_style">wxTAB_TRAVERSAL</property>
-                                        <object class="wxBoxSizer" expanded="1">
-                                            <property name="minimum_size"></property>
-                                            <property name="name">m_Sizer3Dviewer</property>
-                                            <property name="orient">wxVERTICAL</property>
-                                            <property name="permission">protected</property>
-                                        </object>
-                                    </object>
-                                </object>
-                            </object>
+                <property name="moveable">1</property>
+                <property name="name">m_splitterWin</property>
+                <property name="pane_border">1</property>
+                <property name="pane_position"></property>
+                <property name="pane_size"></property>
+                <property name="permission">protected</property>
+                <property name="pin_button">1</property>
+                <property name="pos"></property>
+                <property name="resize">Resizable</property>
+                <property name="sashgravity">0.35</property>
+                <property name="sashpos">300</property>
+                <property name="sashsize">-1</property>
+                <property name="show">1</property>
+                <property name="size"></property>
+                <property name="splitmode">wxSPLIT_VERTICAL</property>
+                <property name="style">wxSP_3D</property>
+                <property name="subclass">; ; forward_declare</property>
+                <property name="toolbar_pane">0</property>
+                <property name="tooltip"></property>
+                <property name="window_extra_style"></property>
+                <property name="window_name"></property>
+                <property name="window_style"></property>
+                <object class="splitteritem" expanded="true">
+                  <object class="wxPanel" expanded="true">
+                    <property name="BottomDockable">1</property>
+                    <property name="LeftDockable">1</property>
+                    <property name="RightDockable">1</property>
+                    <property name="TopDockable">1</property>
+                    <property name="aui_layer"></property>
+                    <property name="aui_name"></property>
+                    <property name="aui_position"></property>
+                    <property name="aui_row"></property>
+                    <property name="best_size"></property>
+                    <property name="bg"></property>
+                    <property name="caption"></property>
+                    <property name="caption_visible">1</property>
+                    <property name="center_pane">0</property>
+                    <property name="close_button">1</property>
+                    <property name="context_help"></property>
+                    <property name="context_menu">1</property>
+                    <property name="default_pane">0</property>
+                    <property name="dock">Dock</property>
+                    <property name="dock_fixed">0</property>
+                    <property name="docking">Left</property>
+                    <property name="drag_accept_files">0</property>
+                    <property name="enabled">1</property>
+                    <property name="fg"></property>
+                    <property name="floatable">1</property>
+                    <property name="font"></property>
+                    <property name="gripper">0</property>
+                    <property name="hidden">0</property>
+                    <property name="id">wxID_ANY</property>
+                    <property name="max_size"></property>
+                    <property name="maximize_button">0</property>
+                    <property name="maximum_size"></property>
+                    <property name="min_size"></property>
+                    <property name="minimize_button">0</property>
+                    <property name="minimum_size"></property>
+                    <property name="moveable">1</property>
+                    <property name="name">m_panelLeft</property>
+                    <property name="pane_border">1</property>
+                    <property name="pane_position"></property>
+                    <property name="pane_size"></property>
+                    <property name="permission">protected</property>
+                    <property name="pin_button">1</property>
+                    <property name="pos"></property>
+                    <property name="resize">Resizable</property>
+                    <property name="show">1</property>
+                    <property name="size"></property>
+                    <property name="subclass">; ; forward_declare</property>
+                    <property name="toolbar_pane">0</property>
+                    <property name="tooltip"></property>
+                    <property name="window_extra_style"></property>
+                    <property name="window_name"></property>
+                    <property name="window_style">wxTAB_TRAVERSAL</property>
+                    <object class="wxBoxSizer" expanded="true">
+                      <property name="minimum_size"></property>
+                      <property name="name">bSizerLeft</property>
+                      <property name="orient">wxVERTICAL</property>
+                      <property name="permission">none</property>
+                      <object class="sizeritem" expanded="false">
+                        <property name="border">5</property>
+                        <property name="flag">wxALL|wxEXPAND</property>
+                        <property name="proportion">1</property>
+                        <object class="wxGenericDirCtrl" expanded="false">
+                          <property name="BottomDockable">1</property>
+                          <property name="LeftDockable">1</property>
+                          <property name="RightDockable">1</property>
+                          <property name="TopDockable">1</property>
+                          <property name="aui_layer"></property>
+                          <property name="aui_name"></property>
+                          <property name="aui_position"></property>
+                          <property name="aui_row"></property>
+                          <property name="best_size"></property>
+                          <property name="bg"></property>
+                          <property name="caption"></property>
+                          <property name="caption_visible">1</property>
+                          <property name="center_pane">0</property>
+                          <property name="close_button">1</property>
+                          <property name="context_help"></property>
+                          <property name="context_menu">1</property>
+                          <property name="default_pane">0</property>
+                          <property name="defaultfilter">0</property>
+                          <property name="defaultfolder"></property>
+                          <property name="dock">Dock</property>
+                          <property name="dock_fixed">0</property>
+                          <property name="docking">Left</property>
+                          <property name="drag_accept_files">0</property>
+                          <property name="enabled">1</property>
+                          <property name="fg"></property>
+                          <property name="filter"></property>
+                          <property name="floatable">1</property>
+                          <property name="font"></property>
+                          <property name="gripper">0</property>
+                          <property name="hidden">0</property>
+                          <property name="id">wxID_ANY</property>
+                          <property name="max_size"></property>
+                          <property name="maximize_button">0</property>
+                          <property name="maximum_size"></property>
+                          <property name="min_size"></property>
+                          <property name="minimize_button">0</property>
+                          <property name="minimum_size">300,-1</property>
+                          <property name="moveable">1</property>
+                          <property name="name">m_FileTree</property>
+                          <property name="pane_border">1</property>
+                          <property name="pane_position"></property>
+                          <property name="pane_size"></property>
+                          <property name="permission">protected</property>
+                          <property name="pin_button">1</property>
+                          <property name="pos"></property>
+                          <property name="resize">Resizable</property>
+                          <property name="show">1</property>
+                          <property name="show_hidden">0</property>
+                          <property name="size"></property>
+                          <property name="style">wxDIRCTRL_3D_INTERNAL|wxDIRCTRL_EDIT_LABELS|wxDIRCTRL_SELECT_FIRST|wxDIRCTRL_SHOW_FILTERS</property>
+                          <property name="subclass">; ; forward_declare</property>
+                          <property name="toolbar_pane">0</property>
+                          <property name="tooltip"></property>
+                          <property name="window_extra_style"></property>
+                          <property name="window_name"></property>
+                          <property name="window_style">wxBORDER_DEFAULT</property>
+                          <event name="OnDirctrlFileActivated">OnFileActivated</event>
+                          <event name="OnDirctrlSelectionChanged">OnSelectionChanged</event>
                         </object>
+                      </object>
+                      <object class="sizeritem" expanded="true">
+                        <property name="border">5</property>
+                        <property name="flag">wxEXPAND</property>
+                        <property name="proportion">0</property>
+                        <object class="wxBoxSizer" expanded="true">
+                          <property name="minimum_size"></property>
+                          <property name="name">bSizer7</property>
+                          <property name="orient">wxHORIZONTAL</property>
+                          <property name="permission">none</property>
+                        </object>
+                      </object>
                     </object>
+                  </object>
                 </object>
-                <object class="sizeritem" expanded="1">
-                    <property name="border">5</property>
-                    <property name="flag">wxEXPAND</property>
-                    <property name="proportion">0</property>
-                    <object class="wxBoxSizer" expanded="1">
-                        <property name="minimum_size"></property>
-                        <property name="name">bSizerLower</property>
-                        <property name="orient">wxHORIZONTAL</property>
-                        <property name="permission">none</property>
-                        <object class="sizeritem" expanded="1">
-                            <property name="border">5</property>
-                            <property name="flag">wxALIGN_CENTER_VERTICAL|wxBOTTOM|wxLEFT|wxRIGHT|wxTOP</property>
-                            <property name="proportion">0</property>
-                            <object class="wxStaticText" expanded="1">
-                                <property name="BottomDockable">1</property>
-                                <property name="LeftDockable">1</property>
-                                <property name="RightDockable">1</property>
-                                <property name="TopDockable">1</property>
-                                <property name="aui_layer"></property>
-                                <property name="aui_name"></property>
-                                <property name="aui_position"></property>
-                                <property name="aui_row"></property>
-                                <property name="best_size"></property>
-                                <property name="bg"></property>
-                                <property name="caption"></property>
-                                <property name="caption_visible">1</property>
-                                <property name="center_pane">0</property>
-                                <property name="close_button">1</property>
-                                <property name="context_help"></property>
-                                <property name="context_menu">1</property>
-                                <property name="default_pane">0</property>
-                                <property name="dock">Dock</property>
-                                <property name="dock_fixed">0</property>
-                                <property name="docking">Left</property>
-                                <property name="enabled">1</property>
-                                <property name="fg"></property>
-                                <property name="floatable">1</property>
-                                <property name="font"></property>
-                                <property name="gripper">0</property>
-                                <property name="hidden">0</property>
-                                <property name="id">wxID_ANY</property>
-                                <property name="label">Available paths:</property>
-                                <property name="markup">0</property>
-                                <property name="max_size"></property>
-                                <property name="maximize_button">0</property>
-                                <property name="maximum_size"></property>
-                                <property name="min_size"></property>
-                                <property name="minimize_button">0</property>
-                                <property name="minimum_size"></property>
-                                <property name="moveable">1</property>
-                                <property name="name">m_stDirChoice</property>
-                                <property name="pane_border">1</property>
-                                <property name="pane_position"></property>
-                                <property name="pane_size"></property>
-                                <property name="permission">protected</property>
-                                <property name="pin_button">1</property>
-                                <property name="pos"></property>
-                                <property name="resize">Resizable</property>
-                                <property name="show">1</property>
-                                <property name="size"></property>
-                                <property name="style"></property>
-                                <property name="subclass">; ; forward_declare</property>
-                                <property name="toolbar_pane">0</property>
-                                <property name="tooltip"></property>
-                                <property name="window_extra_style"></property>
-                                <property name="window_name"></property>
-                                <property name="window_style"></property>
-                                <property name="wrap">-1</property>
-                            </object>
-                        </object>
-                        <object class="sizeritem" expanded="1">
-                            <property name="border">5</property>
-                            <property name="flag">wxTOP|wxBOTTOM|wxRIGHT</property>
-                            <property name="proportion">1</property>
-                            <object class="wxChoice" expanded="1">
-                                <property name="BottomDockable">1</property>
-                                <property name="LeftDockable">1</property>
-                                <property name="RightDockable">1</property>
-                                <property name="TopDockable">1</property>
-                                <property name="aui_layer"></property>
-                                <property name="aui_name"></property>
-                                <property name="aui_position"></property>
-                                <property name="aui_row"></property>
-                                <property name="best_size"></property>
-                                <property name="bg"></property>
-                                <property name="caption"></property>
-                                <property name="caption_visible">1</property>
-                                <property name="center_pane">0</property>
-                                <property name="choices"></property>
-                                <property name="close_button">1</property>
-                                <property name="context_help"></property>
-                                <property name="context_menu">1</property>
-                                <property name="default_pane">0</property>
-                                <property name="dock">Dock</property>
-                                <property name="dock_fixed">0</property>
-                                <property name="docking">Left</property>
-                                <property name="enabled">1</property>
-                                <property name="fg"></property>
-                                <property name="floatable">1</property>
-                                <property name="font"></property>
-                                <property name="gripper">0</property>
-                                <property name="hidden">0</property>
-                                <property name="id">wxID_ANY</property>
-                                <property name="max_size"></property>
-                                <property name="maximize_button">0</property>
-                                <property name="maximum_size"></property>
-                                <property name="min_size"></property>
-                                <property name="minimize_button">0</property>
-                                <property name="minimum_size"></property>
-                                <property name="moveable">1</property>
-                                <property name="name">m_dirChoices</property>
-                                <property name="pane_border">1</property>
-                                <property name="pane_position"></property>
-                                <property name="pane_size"></property>
-                                <property name="permission">protected</property>
-                                <property name="pin_button">1</property>
-                                <property name="pos"></property>
-                                <property name="resize">Resizable</property>
-                                <property name="selection">0</property>
-                                <property name="show">1</property>
-                                <property name="size"></property>
-                                <property name="style"></property>
-                                <property name="subclass">; ; forward_declare</property>
-                                <property name="toolbar_pane">0</property>
-                                <property name="tooltip"></property>
-                                <property name="validator_data_type"></property>
-                                <property name="validator_style">wxFILTER_NONE</property>
-                                <property name="validator_type">wxDefaultValidator</property>
-                                <property name="validator_variable"></property>
-                                <property name="window_extra_style"></property>
-                                <property name="window_name"></property>
-                                <property name="window_style"></property>
-                                <event name="OnChoice">SetRootDir</event>
-                            </object>
-                        </object>
-                        <object class="sizeritem" expanded="1">
-                            <property name="border">5</property>
-                            <property name="flag">wxALL|wxALIGN_CENTER_VERTICAL</property>
-                            <property name="proportion">0</property>
-                            <object class="wxButton" expanded="1">
-                                <property name="BottomDockable">1</property>
-                                <property name="LeftDockable">1</property>
-                                <property name="RightDockable">1</property>
-                                <property name="TopDockable">1</property>
-                                <property name="aui_layer"></property>
-                                <property name="aui_name"></property>
-                                <property name="aui_position"></property>
-                                <property name="aui_row"></property>
-                                <property name="auth_needed">0</property>
-                                <property name="best_size"></property>
-                                <property name="bg"></property>
-                                <property name="bitmap"></property>
-                                <property name="caption"></property>
-                                <property name="caption_visible">1</property>
-                                <property name="center_pane">0</property>
-                                <property name="close_button">1</property>
-                                <property name="context_help"></property>
-                                <property name="context_menu">1</property>
-                                <property name="current"></property>
-                                <property name="default">0</property>
-                                <property name="default_pane">0</property>
-                                <property name="disabled"></property>
-                                <property name="dock">Dock</property>
-                                <property name="dock_fixed">0</property>
-                                <property name="docking">Left</property>
-                                <property name="enabled">1</property>
-                                <property name="fg"></property>
-                                <property name="floatable">1</property>
-                                <property name="focus"></property>
-                                <property name="font"></property>
-                                <property name="gripper">0</property>
-                                <property name="hidden">0</property>
-                                <property name="id">wxID_ANY</property>
-                                <property name="label">Configure Paths</property>
-                                <property name="margins"></property>
-                                <property name="markup">0</property>
-                                <property name="max_size"></property>
-                                <property name="maximize_button">0</property>
-                                <property name="maximum_size"></property>
-                                <property name="min_size"></property>
-                                <property name="minimize_button">0</property>
-                                <property name="minimum_size"></property>
-                                <property name="moveable">1</property>
-                                <property name="name">m_cfgPathsButt</property>
-                                <property name="pane_border">1</property>
-                                <property name="pane_position"></property>
-                                <property name="pane_size"></property>
-                                <property name="permission">protected</property>
-                                <property name="pin_button">1</property>
-                                <property name="pos"></property>
-                                <property name="position"></property>
-                                <property name="pressed"></property>
-                                <property name="resize">Resizable</property>
-                                <property name="show">1</property>
-                                <property name="size"></property>
-                                <property name="style"></property>
-                                <property name="subclass">; ; forward_declare</property>
-                                <property name="toolbar_pane">0</property>
-                                <property name="tooltip"></property>
-                                <property name="validator_data_type"></property>
-                                <property name="validator_style">wxFILTER_NONE</property>
-                                <property name="validator_type">wxDefaultValidator</property>
-                                <property name="validator_variable"></property>
-                                <property name="window_extra_style"></property>
-                                <property name="window_name"></property>
-                                <property name="window_style"></property>
-                                <event name="OnButtonClick">Cfg3DPaths</event>
-                            </object>
-                        </object>
-                    </object>
-                </object>
-                <object class="sizeritem" expanded="1">
-                    <property name="border">5</property>
-                    <property name="flag">wxEXPAND|wxALL</property>
-                    <property name="proportion">0</property>
-                    <object class="wxStdDialogButtonSizer" expanded="1">
-                        <property name="Apply">0</property>
-                        <property name="Cancel">1</property>
-                        <property name="ContextHelp">0</property>
-                        <property name="Help">0</property>
-                        <property name="No">0</property>
-                        <property name="OK">1</property>
-                        <property name="Save">0</property>
-                        <property name="Yes">0</property>
-                        <property name="minimum_size"></property>
-                        <property name="name">m_sdbSizer</property>
-                        <property name="permission">protected</property>
+                <object class="splitteritem" expanded="false">
+                  <object class="wxPanel" expanded="false">
+                    <property name="BottomDockable">1</property>
+                    <property name="LeftDockable">1</property>
+                    <property name="RightDockable">1</property>
+                    <property name="TopDockable">1</property>
+                    <property name="aui_layer"></property>
+                    <property name="aui_name"></property>
+                    <property name="aui_position"></property>
+                    <property name="aui_row"></property>
+                    <property name="best_size"></property>
+                    <property name="bg"></property>
+                    <property name="caption"></property>
+                    <property name="caption_visible">1</property>
+                    <property name="center_pane">0</property>
+                    <property name="close_button">1</property>
+                    <property name="context_help"></property>
+                    <property name="context_menu">1</property>
+                    <property name="default_pane">0</property>
+                    <property name="dock">Dock</property>
+                    <property name="dock_fixed">0</property>
+                    <property name="docking">Left</property>
+                    <property name="drag_accept_files">0</property>
+                    <property name="enabled">1</property>
+                    <property name="fg"></property>
+                    <property name="floatable">1</property>
+                    <property name="font"></property>
+                    <property name="gripper">0</property>
+                    <property name="hidden">0</property>
+                    <property name="id">wxID_ANY</property>
+                    <property name="max_size"></property>
+                    <property name="maximize_button">0</property>
+                    <property name="maximum_size"></property>
+                    <property name="min_size"></property>
+                    <property name="minimize_button">0</property>
+                    <property name="minimum_size"></property>
+                    <property name="moveable">1</property>
+                    <property name="name">m_pane3Dviewer</property>
+                    <property name="pane_border">1</property>
+                    <property name="pane_position"></property>
+                    <property name="pane_size"></property>
+                    <property name="permission">protected</property>
+                    <property name="pin_button">1</property>
+                    <property name="pos"></property>
+                    <property name="resize">Resizable</property>
+                    <property name="show">1</property>
+                    <property name="size"></property>
+                    <property name="subclass">; ; forward_declare</property>
+                    <property name="toolbar_pane">0</property>
+                    <property name="tooltip"></property>
+                    <property name="window_extra_style"></property>
+                    <property name="window_name"></property>
+                    <property name="window_style">wxTAB_TRAVERSAL</property>
+                    <object class="wxBoxSizer" expanded="false">
+                      <property name="minimum_size"></property>
+                      <property name="name">m_Sizer3Dviewer</property>
+                      <property name="orient">wxVERTICAL</property>
+                      <property name="permission">protected</property>
                     </object>
+                  </object>
                 </object>
+              </object>
             </object>
+          </object>
         </object>
+        <object class="sizeritem" expanded="true">
+          <property name="border">5</property>
+          <property name="flag">wxEXPAND</property>
+          <property name="proportion">0</property>
+          <object class="wxBoxSizer" expanded="false">
+            <property name="minimum_size"></property>
+            <property name="name">bSizerLower</property>
+            <property name="orient">wxHORIZONTAL</property>
+            <property name="permission">none</property>
+            <object class="sizeritem" expanded="false">
+              <property name="border">5</property>
+              <property name="flag">wxALIGN_CENTER_VERTICAL|wxBOTTOM|wxLEFT|wxRIGHT|wxTOP</property>
+              <property name="proportion">0</property>
+              <object class="wxStaticText" expanded="false">
+                <property name="BottomDockable">1</property>
+                <property name="LeftDockable">1</property>
+                <property name="RightDockable">1</property>
+                <property name="TopDockable">1</property>
+                <property name="aui_layer"></property>
+                <property name="aui_name"></property>
+                <property name="aui_position"></property>
+                <property name="aui_row"></property>
+                <property name="best_size"></property>
+                <property name="bg"></property>
+                <property name="caption"></property>
+                <property name="caption_visible">1</property>
+                <property name="center_pane">0</property>
+                <property name="close_button">1</property>
+                <property name="context_help"></property>
+                <property name="context_menu">1</property>
+                <property name="default_pane">0</property>
+                <property name="dock">Dock</property>
+                <property name="dock_fixed">0</property>
+                <property name="docking">Left</property>
+                <property name="drag_accept_files">0</property>
+                <property name="enabled">1</property>
+                <property name="fg"></property>
+                <property name="floatable">1</property>
+                <property name="font"></property>
+                <property name="gripper">0</property>
+                <property name="hidden">0</property>
+                <property name="id">wxID_ANY</property>
+                <property name="label">Available paths:</property>
+                <property name="markup">0</property>
+                <property name="max_size"></property>
+                <property name="maximize_button">0</property>
+                <property name="maximum_size"></property>
+                <property name="min_size"></property>
+                <property name="minimize_button">0</property>
+                <property name="minimum_size"></property>
+                <property name="moveable">1</property>
+                <property name="name">m_stDirChoice</property>
+                <property name="pane_border">1</property>
+                <property name="pane_position"></property>
+                <property name="pane_size"></property>
+                <property name="permission">protected</property>
+                <property name="pin_button">1</property>
+                <property name="pos"></property>
+                <property name="resize">Resizable</property>
+                <property name="show">1</property>
+                <property name="size"></property>
+                <property name="style"></property>
+                <property name="subclass">; ; forward_declare</property>
+                <property name="toolbar_pane">0</property>
+                <property name="tooltip"></property>
+                <property name="window_extra_style"></property>
+                <property name="window_name"></property>
+                <property name="window_style"></property>
+                <property name="wrap">-1</property>
+              </object>
+            </object>
+            <object class="sizeritem" expanded="false">
+              <property name="border">5</property>
+              <property name="flag">wxTOP|wxBOTTOM|wxRIGHT</property>
+              <property name="proportion">1</property>
+              <object class="wxChoice" expanded="false">
+                <property name="BottomDockable">1</property>
+                <property name="LeftDockable">1</property>
+                <property name="RightDockable">1</property>
+                <property name="TopDockable">1</property>
+                <property name="aui_layer"></property>
+                <property name="aui_name"></property>
+                <property name="aui_position"></property>
+                <property name="aui_row"></property>
+                <property name="best_size"></property>
+                <property name="bg"></property>
+                <property name="caption"></property>
+                <property name="caption_visible">1</property>
+                <property name="center_pane">0</property>
+                <property name="choices"></property>
+                <property name="close_button">1</property>
+                <property name="context_help"></property>
+                <property name="context_menu">1</property>
+                <property name="default_pane">0</property>
+                <property name="dock">Dock</property>
+                <property name="dock_fixed">0</property>
+                <property name="docking">Left</property>
+                <property name="drag_accept_files">0</property>
+                <property name="enabled">1</property>
+                <property name="fg"></property>
+                <property name="floatable">1</property>
+                <property name="font"></property>
+                <property name="gripper">0</property>
+                <property name="hidden">0</property>
+                <property name="id">wxID_ANY</property>
+                <property name="max_size"></property>
+                <property name="maximize_button">0</property>
+                <property name="maximum_size"></property>
+                <property name="min_size"></property>
+                <property name="minimize_button">0</property>
+                <property name="minimum_size"></property>
+                <property name="moveable">1</property>
+                <property name="name">m_dirChoices</property>
+                <property name="pane_border">1</property>
+                <property name="pane_position"></property>
+                <property name="pane_size"></property>
+                <property name="permission">protected</property>
+                <property name="pin_button">1</property>
+                <property name="pos"></property>
+                <property name="resize">Resizable</property>
+                <property name="selection">0</property>
+                <property name="show">1</property>
+                <property name="size"></property>
+                <property name="style"></property>
+                <property name="subclass">; ; forward_declare</property>
+                <property name="toolbar_pane">0</property>
+                <property name="tooltip"></property>
+                <property name="validator_data_type"></property>
+                <property name="validator_style">wxFILTER_NONE</property>
+                <property name="validator_type">wxDefaultValidator</property>
+                <property name="validator_variable"></property>
+                <property name="window_extra_style"></property>
+                <property name="window_name"></property>
+                <property name="window_style"></property>
+                <event name="OnChoice">SetRootDir</event>
+              </object>
+            </object>
+            <object class="sizeritem" expanded="false">
+              <property name="border">5</property>
+              <property name="flag">wxALL|wxALIGN_CENTER_VERTICAL</property>
+              <property name="proportion">0</property>
+              <object class="wxButton" expanded="false">
+                <property name="BottomDockable">1</property>
+                <property name="LeftDockable">1</property>
+                <property name="RightDockable">1</property>
+                <property name="TopDockable">1</property>
+                <property name="aui_layer"></property>
+                <property name="aui_name"></property>
+                <property name="aui_position"></property>
+                <property name="aui_row"></property>
+                <property name="auth_needed">0</property>
+                <property name="best_size"></property>
+                <property name="bg"></property>
+                <property name="bitmap"></property>
+                <property name="caption"></property>
+                <property name="caption_visible">1</property>
+                <property name="center_pane">0</property>
+                <property name="close_button">1</property>
+                <property name="context_help"></property>
+                <property name="context_menu">1</property>
+                <property name="current"></property>
+                <property name="default">0</property>
+                <property name="default_pane">0</property>
+                <property name="disabled"></property>
+                <property name="dock">Dock</property>
+                <property name="dock_fixed">0</property>
+                <property name="docking">Left</property>
+                <property name="drag_accept_files">0</property>
+                <property name="enabled">1</property>
+                <property name="fg"></property>
+                <property name="floatable">1</property>
+                <property name="focus"></property>
+                <property name="font"></property>
+                <property name="gripper">0</property>
+                <property name="hidden">0</property>
+                <property name="id">wxID_ANY</property>
+                <property name="label">Configure Paths</property>
+                <property name="margins"></property>
+                <property name="markup">0</property>
+                <property name="max_size"></property>
+                <property name="maximize_button">0</property>
+                <property name="maximum_size"></property>
+                <property name="min_size"></property>
+                <property name="minimize_button">0</property>
+                <property name="minimum_size"></property>
+                <property name="moveable">1</property>
+                <property name="name">m_cfgPathsButt</property>
+                <property name="pane_border">1</property>
+                <property name="pane_position"></property>
+                <property name="pane_size"></property>
+                <property name="permission">protected</property>
+                <property name="pin_button">1</property>
+                <property name="pos"></property>
+                <property name="position"></property>
+                <property name="pressed"></property>
+                <property name="resize">Resizable</property>
+                <property name="show">1</property>
+                <property name="size"></property>
+                <property name="style"></property>
+                <property name="subclass">; ; forward_declare</property>
+                <property name="toolbar_pane">0</property>
+                <property name="tooltip"></property>
+                <property name="validator_data_type"></property>
+                <property name="validator_style">wxFILTER_NONE</property>
+                <property name="validator_type">wxDefaultValidator</property>
+                <property name="validator_variable"></property>
+                <property name="window_extra_style"></property>
+                <property name="window_name"></property>
+                <property name="window_style"></property>
+                <event name="OnButtonClick">Cfg3DPaths</event>
+              </object>
+            </object>
+          </object>
+        </object>
+        <object class="sizeritem" expanded="true">
+          <property name="border">5</property>
+          <property name="flag">wxEXPAND</property>
+          <property name="proportion">0</property>
+          <object class="wxBoxSizer" expanded="true">
+            <property name="minimum_size"></property>
+            <property name="name">bSizer6</property>
+            <property name="orient">wxHORIZONTAL</property>
+            <property name="permission">none</property>
+            <object class="sizeritem" expanded="true">
+              <property name="border">5</property>
+              <property name="flag">wxALL</property>
+              <property name="proportion">0</property>
+              <object class="wxCheckBox" expanded="true">
+                <property name="BottomDockable">1</property>
+                <property name="LeftDockable">1</property>
+                <property name="RightDockable">1</property>
+                <property name="TopDockable">1</property>
+                <property name="aui_layer"></property>
+                <property name="aui_name"></property>
+                <property name="aui_position"></property>
+                <property name="aui_row"></property>
+                <property name="best_size"></property>
+                <property name="bg"></property>
+                <property name="caption"></property>
+                <property name="caption_visible">1</property>
+                <property name="center_pane">0</property>
+                <property name="checked">0</property>
+                <property name="close_button">1</property>
+                <property name="context_help"></property>
+                <property name="context_menu">1</property>
+                <property name="default_pane">0</property>
+                <property name="dock">Dock</property>
+                <property name="dock_fixed">0</property>
+                <property name="docking">Left</property>
+                <property name="drag_accept_files">0</property>
+                <property name="enabled">1</property>
+                <property name="fg"></property>
+                <property name="floatable">1</property>
+                <property name="font"></property>
+                <property name="gripper">0</property>
+                <property name="hidden">0</property>
+                <property name="id">wxID_ANY</property>
+                <property name="label">Embed Model</property>
+                <property name="max_size"></property>
+                <property name="maximize_button">0</property>
+                <property name="maximum_size"></property>
+                <property name="min_size"></property>
+                <property name="minimize_button">0</property>
+                <property name="minimum_size"></property>
+                <property name="moveable">1</property>
+                <property name="name">m_EmbedModelCb</property>
+                <property name="pane_border">1</property>
+                <property name="pane_position"></property>
+                <property name="pane_size"></property>
+                <property name="permission">protected</property>
+                <property name="pin_button">1</property>
+                <property name="pos"></property>
+                <property name="resize">Resizable</property>
+                <property name="show">1</property>
+                <property name="size"></property>
+                <property name="style"></property>
+                <property name="subclass">; ; forward_declare</property>
+                <property name="toolbar_pane">0</property>
+                <property name="tooltip"></property>
+                <property name="validator_data_type"></property>
+                <property name="validator_style">wxFILTER_NONE</property>
+                <property name="validator_type">wxDefaultValidator</property>
+                <property name="validator_variable"></property>
+                <property name="window_extra_style"></property>
+                <property name="window_name"></property>
+                <property name="window_style"></property>
+              </object>
+            </object>
+            <object class="sizeritem" expanded="true">
+              <property name="border">5</property>
+              <property name="flag">wxEXPAND</property>
+              <property name="proportion">1</property>
+              <object class="spacer" expanded="true">
+                <property name="height">0</property>
+                <property name="permission">protected</property>
+                <property name="width">0</property>
+              </object>
+            </object>
+            <object class="sizeritem" expanded="true">
+              <property name="border">5</property>
+              <property name="flag">wxEXPAND|wxALL</property>
+              <property name="proportion">0</property>
+              <object class="wxStdDialogButtonSizer" expanded="true">
+                <property name="Apply">0</property>
+                <property name="Cancel">1</property>
+                <property name="ContextHelp">0</property>
+                <property name="Help">0</property>
+                <property name="No">0</property>
+                <property name="OK">1</property>
+                <property name="Save">0</property>
+                <property name="Yes">0</property>
+                <property name="minimum_size"></property>
+                <property name="name">m_sdbSizer</property>
+                <property name="permission">protected</property>
+              </object>
+            </object>
+          </object>
+        </object>
+      </object>
     </object>
+  </object>
 </wxFormBuilder_Project>
diff --git a/3d-viewer/dialogs/dialog_select_3d_model_base.h b/3d-viewer/dialogs/dialog_select_3d_model_base.h
index d0b50f2228..325d5b5b44 100644
--- a/3d-viewer/dialogs/dialog_select_3d_model_base.h
+++ b/3d-viewer/dialogs/dialog_select_3d_model_base.h
@@ -1,5 +1,5 @@
 ///////////////////////////////////////////////////////////////////////////
-// C++ code generated with wxFormBuilder (version 3.10.1-0-g8feb16b)
+// C++ code generated with wxFormBuilder (version 4.0.0-0-g0efcecf0)
 // http://www.wxformbuilder.org/
 //
 // PLEASE DO *NOT* EDIT THIS FILE!
@@ -26,6 +26,7 @@
 #include <wx/bitmap.h>
 #include <wx/image.h>
 #include <wx/icon.h>
+#include <wx/checkbox.h>
 #include <wx/dialog.h>
 
 ///////////////////////////////////////////////////////////////////////////
@@ -47,6 +48,7 @@ class DIALOG_SELECT_3D_MODEL_BASE : public DIALOG_SHIM
 		wxStaticText* m_stDirChoice;
 		wxChoice* m_dirChoices;
 		wxButton* m_cfgPathsButt;
+		wxCheckBox* m_EmbedModelCb;
 		wxStdDialogButtonSizer* m_sdbSizer;
 		wxButton* m_sdbSizerOK;
 		wxButton* m_sdbSizerCancel;
diff --git a/3d-viewer/dialogs/panel_preview_3d_model.h b/3d-viewer/dialogs/panel_preview_3d_model.h
index b60a56143c..b4c7e6f8c2 100644
--- a/3d-viewer/dialogs/panel_preview_3d_model.h
+++ b/3d-viewer/dialogs/panel_preview_3d_model.h
@@ -92,6 +92,12 @@ public:
      */
     void UpdateDummyFootprint( bool aRelaodRequired = true );
 
+    /**
+     * Get the dummy footprint that is used for previewing the 3D model.
+     * We use this to hold the temporary 3D model shapes.
+     */
+    FOOTPRINT* GetDummyFootprint() const { return m_dummyFootprint; }
+
 private:
     /**
      * Load 3D relevant settings from the user configuration
diff --git a/CMakeLists.txt b/CMakeLists.txt
index 56e8ca2316..d0b1f00972 100644
--- a/CMakeLists.txt
+++ b/CMakeLists.txt
@@ -809,6 +809,11 @@ include_directories( SYSTEM ${GLM_INCLUDE_DIR} )
 #
 find_package(ZLIB REQUIRED)
 
+#
+# Find Zstd library, required
+#
+find_package(ZSTD REQUIRED)
+
 #
 # Find libcurl, required
 #
diff --git a/cmake/BuildSteps/TokenList2DsnLexer.cmake b/cmake/BuildSteps/TokenList2DsnLexer.cmake
index de14e6b17d..6f6c4fd6c1 100644
--- a/cmake/BuildSteps/TokenList2DsnLexer.cmake
+++ b/cmake/BuildSteps/TokenList2DsnLexer.cmake
@@ -145,6 +145,7 @@ namespace ${enum}
         // these first few are negative special ones for syntax, and are
         // inherited from DSNLEXER.
         T_NONE          = DSN_NONE,
+        T_BAR           = DSN_BAR,          // Also called pipe: '|'
         T_COMMENT       = DSN_COMMENT,
         T_STRING_QUOTE  = DSN_STRING_QUOTE,
         T_QUOTE_DEF     = DSN_QUOTE_DEF,
diff --git a/cmake/FindZSTD.cmake b/cmake/FindZSTD.cmake
new file mode 100644
index 0000000000..98175e8a42
--- /dev/null
+++ b/cmake/FindZSTD.cmake
@@ -0,0 +1,41 @@
+# Copyright (c) Meta Platforms, Inc. and affiliates.
+#
+# Licensed 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.
+
+#
+# - Try to find Facebook zstd library
+# This will define
+# ZSTD_FOUND
+# ZSTD_INCLUDE_DIR
+# ZSTD_LIBRARY
+#
+
+find_path(ZSTD_INCLUDE_DIR NAMES zstd.h)
+
+find_library(ZSTD_LIBRARY_DEBUG NAMES zstdd zstd_staticd)
+find_library(ZSTD_LIBRARY_RELEASE NAMES zstd zstd_static)
+
+include(SelectLibraryConfigurations)
+SELECT_LIBRARY_CONFIGURATIONS(ZSTD)
+
+include(FindPackageHandleStandardArgs)
+FIND_PACKAGE_HANDLE_STANDARD_ARGS(
+    ZSTD DEFAULT_MSG
+    ZSTD_LIBRARY ZSTD_INCLUDE_DIR
+)
+
+if (ZSTD_FOUND)
+    message(STATUS "Found Zstd: ${ZSTD_LIBRARY}")
+endif()
+
+mark_as_advanced(ZSTD_INCLUDE_DIR ZSTD_LIBRARY)
\ No newline at end of file
diff --git a/common/CMakeLists.txt b/common/CMakeLists.txt
index bc0ec154f7..a6d56122ba 100644
--- a/common/CMakeLists.txt
+++ b/common/CMakeLists.txt
@@ -195,6 +195,8 @@ target_link_libraries( kicommon
     nlohmann_json
     fmt::fmt
     CURL::libcurl
+    picosha2
+    zstd
     ${wxWidgets_LIBRARIES}
     ${LIBGIT2_LIBRARIES}
 
@@ -267,6 +269,7 @@ target_include_directories( kicommon
         $<TARGET_PROPERTY:pegtl,INTERFACE_INCLUDE_DIRECTORIES>
         $<TARGET_PROPERTY:expected,INTERFACE_INCLUDE_DIRECTORIES>
         $<TARGET_PROPERTY:kiapi,INTERFACE_INCLUDE_DIRECTORIES>
+        $<TARGET_PROPERTY:picosha2,INTERFACE_INCLUDE_DIRECTORIES>
     )
 
 add_dependencies( kicommon pegtl version_header )
@@ -304,6 +307,7 @@ set( COMMON_DLG_SRCS
     dialogs/dialog_configure_paths_base.cpp
     dialogs/dialog_display_html_text_base.cpp
     dialogs/dialog_edit_library_tables.cpp
+    dialogs/dialog_embed_files.cpp
     dialogs/dialog_global_lib_table_config.cpp
     dialogs/dialog_global_lib_table_config_base.cpp
     dialogs/dialog_grid_settings.cpp
@@ -338,6 +342,8 @@ set( COMMON_DLG_SRCS
     dialogs/panel_color_settings.cpp
     dialogs/panel_common_settings.cpp
     dialogs/panel_common_settings_base.cpp
+    dialogs/panel_embedded_files.cpp
+    dialogs/panel_embedded_files_base.cpp
     dialogs/panel_gal_display_options.cpp
     dialogs/panel_hotkeys_editor.cpp
     dialogs/panel_image_editor.cpp
@@ -534,6 +540,7 @@ set( COMMON_SRCS
     eda_shape.cpp
     eda_text.cpp
     eda_tools.cpp
+    embedded_files.cpp
     env_paths.cpp
     executable_names.cpp
     filename_resolver.cpp
@@ -589,6 +596,7 @@ set( COMMON_SRCS
     tool/edit_constraints.cpp
     tool/edit_points.cpp
     tool/editor_conditions.cpp
+    tool/embed_tool.cpp
     tool/grid_helper.cpp
     tool/grid_menu.cpp
     tool/picker_tool.cpp
@@ -921,6 +929,17 @@ make_lexer_export(
     kicommon.h
     )
 
+# auto-generate embedded files lexer and keywords
+make_lexer_export(
+    kicommon
+    embedded_files.keywords
+    embedded_files_lexer.h
+    embedded_files_keywords.cpp
+    EMBEDDED_FILES_T
+    KICOMMON_API
+    kicommon.h
+)
+
 # This one gets made only when testing.
 # to build it, first enable #define STAND_ALONE at top of dsnlexer.cpp
 add_executable( dsntest EXCLUDE_FROM_ALL dsnlexer.cpp )
diff --git a/common/dialogs/dialog_embed_files.cpp b/common/dialogs/dialog_embed_files.cpp
new file mode 100644
index 0000000000..c39eaa1610
--- /dev/null
+++ b/common/dialogs/dialog_embed_files.cpp
@@ -0,0 +1,78 @@
+/*
+ * This program source code file is part of KiCad, a free EDA CAD application.
+ *
+ * Copyright (C) 2024 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 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, see <http://www.gnu.org/licenses/>.
+ */
+
+#include <wx/button.h>
+#include <wx/sizer.h>
+
+#include <dialogs/dialog_embed_files.h>
+
+
+DIALOG_EMBED_FILES::DIALOG_EMBED_FILES( wxWindow* aParent, const wxString& aTitle ) :
+        DIALOG_SHIM( aParent, wxID_ANY, aTitle, wxDefaultPosition, wxDefaultSize,
+                     wxDEFAULT_DIALOG_STYLE | wxRESIZE_BORDER ),
+        m_contentPanel( nullptr )
+{
+    // Construction delayed until after panel is installed
+}
+
+
+void DIALOG_EMBED_FILES::InstallPanel( wxPanel* aPanel )
+{
+    m_contentPanel = aPanel;
+
+    // Now perform the body of the constructor
+    auto mainSizer = new wxBoxSizer( wxVERTICAL );
+    SetSizer( mainSizer );
+
+    mainSizer->Add( m_contentPanel, 1, wxEXPAND|wxLEFT|wxTOP|wxRIGHT, 5 );
+    m_contentPanel->SetMinSize( FromDIP( wxSize( 1000, 600 ) ) );
+
+    auto sdbSizer = new wxStdDialogButtonSizer();
+    auto sdbSizerOK = new wxButton( this, wxID_OK );
+    sdbSizer->AddButton( sdbSizerOK );
+    auto sdbSizerCancel = new wxButton( this, wxID_CANCEL );
+    sdbSizer->AddButton( sdbSizerCancel );
+    sdbSizer->Realize();
+
+    mainSizer->Add( sdbSizer, 0, wxALL|wxEXPAND, 5 );
+
+    SetupStandardButtons();
+
+    finishDialogSettings();
+
+    // On some windows manager (Unity, XFCE), this dialog is not always raised, depending
+    // on how the dialog is run.
+    Raise();
+}
+
+
+bool DIALOG_EMBED_FILES::TransferDataToWindow()
+{
+    return m_contentPanel->TransferDataToWindow();
+}
+
+
+bool DIALOG_EMBED_FILES::TransferDataFromWindow()
+{
+    /**
+     * N.B. *do not* call wxDialog::TransferDataFromWindow() in the dialog code.
+     */
+    return m_contentPanel->TransferDataFromWindow();
+}
+
diff --git a/common/dialogs/dialog_page_settings.cpp b/common/dialogs/dialog_page_settings.cpp
index ccb56ce47c..03a32a8009 100644
--- a/common/dialogs/dialog_page_settings.cpp
+++ b/common/dialogs/dialog_page_settings.cpp
@@ -26,6 +26,8 @@
 #include <dialogs/dialog_page_settings.h>
 #include <eda_draw_frame.h>
 #include <eda_item.h>
+#include <embedded_files.h>
+#include <filename_resolver.h>
 #include <gr_basic.h>
 #include <kiface_base.h>
 #include <macros.h>
@@ -38,6 +40,7 @@
 #include <drawing_sheet/ds_painter.h>
 #include <string_utils.h>
 #include <widgets/std_bitmap_button.h>
+#include <widgets/filedlg_open_embed_file.h>
 #include <wx/valgen.h>
 #include <wx/tokenzr.h>
 #include <wx/filedlg.h>
@@ -75,7 +78,7 @@ static const wxString pageFmts[] =
                                     // to be recognized in code
 };
 
-DIALOG_PAGES_SETTINGS::DIALOG_PAGES_SETTINGS( EDA_DRAW_FRAME* aParent, double aIuPerMils,
+DIALOG_PAGES_SETTINGS::DIALOG_PAGES_SETTINGS( EDA_DRAW_FRAME* aParent, EMBEDDED_FILES* aEmbeddedFiles, double aIuPerMils,
                                               const VECTOR2D& aMaxUserSizeMils ) :
         DIALOG_PAGES_SETTINGS_BASE( aParent ),
         m_parent( aParent ),
@@ -83,6 +86,7 @@ DIALOG_PAGES_SETTINGS::DIALOG_PAGES_SETTINGS( EDA_DRAW_FRAME* aParent, double aI
         m_initialized( false ),
         m_pageBitmap( nullptr ),
         m_iuPerMils( aIuPerMils ),
+        m_embeddedFiles( aEmbeddedFiles ),
         m_customSizeX( aParent, m_userSizeXLabel, m_userSizeXCtrl, m_userSizeXUnits ),
         m_customSizeY( aParent, m_userSizeYLabel, m_userSizeYCtrl, m_userSizeYUnits )
 {
@@ -114,6 +118,10 @@ DIALOG_PAGES_SETTINGS::DIALOG_PAGES_SETTINGS( EDA_DRAW_FRAME* aParent, double aI
         m_staticTextTitleBlock->SetLabel( _( "Title Block" ) );
     }
 
+    m_filenameResolver = new FILENAME_RESOLVER;
+    m_filenameResolver->SetProject( &Prj() );
+    m_filenameResolver->SetProgramBase( &Pgm() );
+
     SetupStandardButtons();
 
     Centre();
@@ -467,7 +475,8 @@ bool DIALOG_PAGES_SETTINGS::SavePageSettings()
 
     if( fileName != BASE_SCREEN::m_DrawingSheetFileName )
     {
-        wxString fullFileName = DS_DATA_MODEL::ResolvePath( fileName, m_projectPath );
+
+        wxString fullFileName = m_filenameResolver->ResolvePath( fileName, m_projectPath, m_embeddedFiles );
 
         BASE_SCREEN::m_DrawingSheetFileName = fileName;
 
@@ -794,19 +803,30 @@ void DIALOG_PAGES_SETTINGS::OnWksFileSelection( wxCommandEvent& event )
     }
 
     // Display a file picker dialog
+    FILEDLG_OPEN_EMBED_FILE customize;
     wxFileDialog fileDialog( this, _( "Drawing Sheet File" ), path, name,
                              FILEEXT::DrawingSheetFileWildcard(),
                              wxFD_DEFAULT_STYLE | wxFD_FILE_MUST_EXIST );
 
+    if( m_embeddedFiles )
+        fileDialog.SetCustomizeHook( customize );
+
     if( fileDialog.ShowModal() != wxID_OK )
         return;
 
     wxString fileName = fileDialog.GetPath();
     wxString shortFileName;
 
-    // Try to use a project-relative path first:
-    if( !m_projectPath.IsEmpty() && fileName.StartsWith( m_projectPath ) )
+    if( m_embeddedFiles && customize.GetEmbed() )
     {
+        fn.Assign( fileName );
+        EMBEDDED_FILES::EMBEDDED_FILE* result = m_embeddedFiles->AddFile( fn, false );
+        shortFileName = result->GetLink();
+        fileName = m_embeddedFiles->GetTempFileName( result->name ).GetFullPath();
+    }
+    else if( !m_projectPath.IsEmpty() && fileName.StartsWith( m_projectPath ) )
+    {
+        // Try to use a project-relative path
         fn = wxFileName( fileName );
         fn.MakeRelativeTo( m_projectPath );
         shortFileName = fn.GetFullPath();
diff --git a/common/dialogs/panel_embedded_files.cpp b/common/dialogs/panel_embedded_files.cpp
new file mode 100644
index 0000000000..236ff009ac
--- /dev/null
+++ b/common/dialogs/panel_embedded_files.cpp
@@ -0,0 +1,314 @@
+/*
+ * This program source code file is part of KiCad, a free EDA CAD application.
+ *
+ * Copyright (C) 2024 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 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:
+ * http://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
+ */
+
+#include <bitmaps.h>
+#include <dialogs/panel_embedded_files.h>
+#include <embedded_files.h>
+#include <kidialog.h>
+#include <widgets/std_bitmap_button.h>
+#include <widgets/wx_grid.h>
+
+#include <wx/clipbrd.h>
+#include <wx/dirdlg.h>
+#include <wx/ffile.h>
+#include <wx/filedlg.h>
+#include <wx/filename.h>
+#include <wx/menu.h>
+
+PANEL_EMBEDDED_FILES::PANEL_EMBEDDED_FILES( wxWindow* parent, EMBEDDED_FILES* aFiles ) :
+    PANEL_EMBEDDED_FILES_BASE( parent ),
+    m_files( aFiles )
+{
+    m_localFiles = new EMBEDDED_FILES();
+
+    for( auto& [name, file] : m_files->EmbeddedFileMap() )
+    {
+        EMBEDDED_FILES::EMBEDDED_FILE* newFile = new EMBEDDED_FILES::EMBEDDED_FILE( *file );
+        m_localFiles->AddFile( newFile );
+    }
+
+    // Set up the standard buttons
+    m_delete_button->SetBitmap( KiBitmapBundle( BITMAPS::small_trash ) );
+    m_browse_button->SetBitmap( KiBitmapBundle( BITMAPS::small_folder ) );
+    m_files_grid->SetMargins( 0 - wxSYS_VSCROLL_X, 0 );
+    m_files_grid->EnableAlternateRowColors();
+}
+
+
+void PANEL_EMBEDDED_FILES::onSize( wxSizeEvent& event )
+{
+    resizeGrid();
+}
+
+
+void PANEL_EMBEDDED_FILES::resizeGrid()
+{
+    int panel_width = GetClientRect().GetWidth();
+    int first_width = m_files_grid->GetColSize( 0 );
+    int second_width = m_files_grid->GetColSize( 1 );
+
+    double ratio;
+
+    if( first_width + second_width > 0 )
+        ratio = (double)first_width / (double)( first_width + second_width );
+    else
+        ratio = 0.3;
+
+
+    m_files_grid->SetColSize( 0, panel_width * ratio );
+    m_files_grid->SetColSize( 1, panel_width * ( 1 - ratio ) );
+    Layout();
+}
+
+
+void PANEL_EMBEDDED_FILES::onGridRightClick( wxGridEvent& event )
+{
+    wxMenu menu;
+    menu.Append( wxID_COPY, _( "Copy Embedded Reference" ) );
+
+    menu.Bind( wxEVT_COMMAND_MENU_SELECTED,
+        [&]( wxCommandEvent& )
+        {
+            int row = event.GetRow();
+            if( row >= 0 && row < m_files_grid->GetNumberRows() )
+            {
+                wxString cellValue = m_files_grid->GetCellValue( row, 1 );
+
+                if( wxTheClipboard->Open() )
+                {
+                    wxTheClipboard->SetData( new wxTextDataObject( cellValue ) );
+                    wxTheClipboard->Close();
+                }
+            }
+        }, wxID_COPY );
+
+    PopupMenu( &menu );
+}
+
+
+bool PANEL_EMBEDDED_FILES::TransferDataToWindow()
+{
+    m_files_grid->ClearGrid();
+
+    if( m_files_grid->GetNumberRows() > 0 )
+        m_files_grid->DeleteRows( 0, m_files_grid->GetNumberRows() );
+
+    int ii = 0;
+    for( auto& [name, file] : m_localFiles->EmbeddedFileMap() )
+    {
+        while( m_files_grid->GetNumberRows() < ii + 1 )
+            m_files_grid->AppendRows( 1 );
+
+        m_files_grid->SetCellValue( ii, 0, name );
+        m_files_grid->SetCellValue( ii, 1, file->GetLink() );
+
+        ii++;
+    }
+
+    m_cbEmbedFonts->SetValue( m_files->GetAreFontsEmbedded() );
+
+    resizeGrid();
+
+    return true;
+}
+
+
+bool PANEL_EMBEDDED_FILES::TransferDataFromWindow()
+{
+    m_files->ClearEmbeddedFiles();
+
+    std::vector<EMBEDDED_FILES::EMBEDDED_FILE*> files;
+
+    for( auto it = m_localFiles->EmbeddedFileMap().begin(); it != m_localFiles->EmbeddedFileMap().end(); it++ )
+        files.push_back( it->second );
+
+    for( auto& file : files )
+    {
+        m_files->AddFile( file );
+        m_localFiles->RemoveFile( file->name, false );
+    }
+
+    m_files->SetAreFontsEmbedded( m_cbEmbedFonts->IsChecked() );
+
+    return true;
+}
+
+
+void PANEL_EMBEDDED_FILES::onAddEmbeddedFile( wxCommandEvent& event )
+{
+    wxFileDialog fileDialog( this, _( "Select a file to embed" ), wxEmptyString, wxEmptyString,
+                             _( "All files|*.*" ), wxFD_OPEN | wxFD_FILE_MUST_EXIST );
+
+    if( fileDialog.ShowModal() == wxID_OK )
+    {
+        wxFileName fileName( fileDialog.GetPath() );
+        wxString name = fileName.GetFullName();
+
+        if( m_localFiles->HasFile( name ) )
+        {
+            wxString msg = wxString::Format( _( "File '%s' already exists." ),
+                        name );
+
+            KIDIALOG errorDlg( m_parent, msg, _( "Confirmation" ),
+                                wxOK | wxCANCEL | wxICON_WARNING );
+            errorDlg.SetOKLabel( _( "Overwrite" ) );
+
+            if( errorDlg.ShowModal() != wxID_OK )
+                return;
+
+            for( int ii = 0; ii < m_files_grid->GetNumberRows(); ii++ )
+            {
+                if( m_files_grid->GetCellValue( ii, 0 ) == name )
+                {
+                    m_files_grid->DeleteRows( ii );
+                    break;
+                }
+            }
+        }
+
+        EMBEDDED_FILES::EMBEDDED_FILE* result = m_localFiles->AddFile( fileName, true );
+
+        if( !result )
+        {
+            wxString msg = wxString::Format( _( "Failed to add file '%s'." ),
+                        name );
+
+            KIDIALOG errorDlg( m_parent, msg, _( "Error" ), wxOK | wxICON_ERROR );
+            errorDlg.ShowModal();
+            return;
+        }
+
+        m_files_grid->AppendRows( 1 );
+        int ii = m_files_grid->GetNumberRows() - 1;
+        m_files_grid->SetCellValue( ii, 0, name );
+        m_files_grid->SetCellValue( ii, 1, result->GetLink() );
+    }
+}
+
+void PANEL_EMBEDDED_FILES::onDeleteEmbeddedFile( wxCommandEvent& event )
+{
+    int row = m_files_grid->GetGridCursorRow();
+
+    if( row < 0 )
+        return;
+
+    wxString name = m_files_grid->GetCellValue( row, 0 );
+
+    m_localFiles->RemoveFile( name );
+
+    m_files_grid->DeleteRows( row );
+
+    if( row < m_files_grid->GetNumberRows() )
+        m_files_grid->SetGridCursor( row, 0 );
+    else if( m_files_grid->GetNumberRows() > 0 )
+        m_files_grid->SetGridCursor( m_files_grid->GetNumberRows() - 1, 0 );
+}
+
+
+void PANEL_EMBEDDED_FILES::onExportFiles( wxCommandEvent& event )
+{
+    wxDirDialog dirDialog( this, _( "Select a directory to export files" ) );
+
+    if( dirDialog.ShowModal() != wxID_OK )
+        return;
+
+    wxString path = dirDialog.GetPath();
+
+    for( auto& [name, file] : m_localFiles->EmbeddedFileMap() )
+    {
+        wxFileName fileName( path, name );
+
+        if( fileName.FileExists() )
+        {
+            wxString msg = wxString::Format( _( "File '%s' already exists." ),
+                        fileName.GetFullName() );
+
+            KIDIALOG errorDlg( m_parent, msg, _( "Confirmation" ),
+                                wxOK | wxCANCEL | wxICON_WARNING );
+            errorDlg.SetOKCancelLabels( _( "Overwrite" ), _( "Skip" ) );
+            errorDlg.DoNotShowCheckbox( __FILE__, __LINE__ );
+
+            if( errorDlg.ShowModal() != wxID_OK )
+                continue;
+        }
+
+        bool skip_file = false;
+
+        while( 1 )
+        {
+            if( !fileName.IsDirWritable() )
+            {
+#ifndef __WXMAC__
+                wxString msg = wxString::Format( _( "Directory '%s' is not writable." ),
+                                                    fileName.GetFullName() );
+#else
+                wxString msg = wxString::Format( _( "Folder '%s' is not writable." ),
+                                                    fileName.GetPath() );
+#endif
+                // Don't set a 'do not show again' checkbox for this dialog
+                KIDIALOG errorDlg( m_parent, msg, _( "Error" ), wxYES_NO | wxCANCEL | wxICON_ERROR );
+                errorDlg.SetYesNoCancelLabels( _( "Retry" ), _( "Skip" ), _( "Cancel" ) );
+
+                int result = errorDlg.ShowModal();
+
+                if( result == wxID_CANCEL )
+                {
+                    return;
+                }
+                else if( result == wxID_NO )
+                {
+                    skip_file = true;
+                    break;
+                }
+            }
+            else
+            {
+                break;
+            }
+        }
+
+        if( skip_file )
+            continue;
+
+        wxFFile ffile( fileName.GetFullPath(), wxT( "w" ) );
+
+        if( !ffile.IsOpened() )
+        {
+            wxString msg = wxString::Format( _( "Failed to open file '%s'." ),
+                        fileName.GetFullName() );
+
+            KIDIALOG errorDlg( m_parent, msg, _( "Error" ), wxOK | wxICON_ERROR );
+            errorDlg.ShowModal();
+            continue;
+        }
+
+        if( !ffile.Write( file->decompressedData.data(), file->decompressedData.size() ) )
+        {
+            wxString msg = wxString::Format( _( "Failed to write file '%s'." ),
+                        fileName.GetFullName() );
+
+            KIDIALOG errorDlg( m_parent, msg, _( "Error" ), wxOK | wxICON_ERROR );
+            errorDlg.ShowModal();
+        }
+    }
+}
\ No newline at end of file
diff --git a/common/dialogs/panel_embedded_files.h b/common/dialogs/panel_embedded_files.h
new file mode 100644
index 0000000000..b5d184cdd7
--- /dev/null
+++ b/common/dialogs/panel_embedded_files.h
@@ -0,0 +1,58 @@
+/*
+ * This program source code file is part of KiCad, a free EDA CAD application.
+ *
+ * Copyright (C) 2023 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 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:
+ * http://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
+ */
+
+#ifndef PANEL_EMBEDDED_FILES_H
+#define PANEL_EMBEDDED_FILES_H
+
+#include "panel_embedded_files_base.h"
+
+class EMBEDDED_FILES;
+
+class PANEL_EMBEDDED_FILES : public PANEL_EMBEDDED_FILES_BASE
+{
+public:
+    PANEL_EMBEDDED_FILES( wxWindow* parent, EMBEDDED_FILES* aFiles );
+    ~PANEL_EMBEDDED_FILES() override {};
+
+    bool TransferDataFromWindow() override;
+    bool TransferDataToWindow() override;
+    bool GetEmbedFonts() const { return m_cbEmbedFonts->GetValue(); }
+
+protected:
+
+    void onGridRightClick( wxGridEvent& event ) override;
+    void onAddEmbeddedFile( wxCommandEvent& event ) override;
+    void onDeleteEmbeddedFile( wxCommandEvent& event ) override;
+    void onExportFiles( wxCommandEvent& event ) override;
+    void onSize( wxSizeEvent& event ) override;
+
+private:
+
+    void resizeGrid();
+
+    EMBEDDED_FILES* m_files;
+    EMBEDDED_FILES* m_localFiles;
+};
+
+
+#endif  // PANEL_EMBEDDED_FILES_H
\ No newline at end of file
diff --git a/common/dialogs/panel_embedded_files_base.cpp b/common/dialogs/panel_embedded_files_base.cpp
new file mode 100644
index 0000000000..77ff4830fe
--- /dev/null
+++ b/common/dialogs/panel_embedded_files_base.cpp
@@ -0,0 +1,118 @@
+///////////////////////////////////////////////////////////////////////////
+// C++ code generated with wxFormBuilder (version 4.2.1-0-g80c4cb6a-dirty)
+// http://www.wxformbuilder.org/
+//
+// PLEASE DO *NOT* EDIT THIS FILE!
+///////////////////////////////////////////////////////////////////////////
+
+#include "widgets/std_bitmap_button.h"
+#include "widgets/wx_grid.h"
+
+#include "panel_embedded_files_base.h"
+
+///////////////////////////////////////////////////////////////////////////
+
+PANEL_EMBEDDED_FILES_BASE::PANEL_EMBEDDED_FILES_BASE( wxWindow* parent, wxWindowID id, const wxPoint& pos, const wxSize& size, long style, const wxString& name ) : wxPanel( parent, id, pos, size, style, name )
+{
+	wxBoxSizer* bMainSizer;
+	bMainSizer = new wxBoxSizer( wxVERTICAL );
+
+	wxBoxSizer* m_global_sizer;
+	m_global_sizer = new wxBoxSizer( wxVERTICAL );
+
+	m_files_grid = new WX_GRID( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, 0 );
+
+	// Grid
+	m_files_grid->CreateGrid( 1, 2 );
+	m_files_grid->EnableEditing( false );
+	m_files_grid->EnableGridLines( true );
+	m_files_grid->EnableDragGridSize( false );
+	m_files_grid->SetMargins( 0, 0 );
+
+	// Columns
+	m_files_grid->SetColSize( 0, 440 );
+	m_files_grid->SetColSize( 1, 180 );
+	m_files_grid->EnableDragColMove( false );
+	m_files_grid->EnableDragColSize( true );
+	m_files_grid->SetColLabelValue( 0, _("Filename") );
+	m_files_grid->SetColLabelValue( 1, _("Internal Reference") );
+	m_files_grid->SetColLabelSize( 22 );
+	m_files_grid->SetColLabelAlignment( wxALIGN_CENTER, wxALIGN_CENTER );
+
+	// Rows
+	m_files_grid->AutoSizeRows();
+	m_files_grid->EnableDragRowSize( false );
+	m_files_grid->SetRowLabelSize( 0 );
+	m_files_grid->SetRowLabelAlignment( wxALIGN_CENTER, wxALIGN_CENTER );
+
+	// Label Appearance
+
+	// Cell Defaults
+	m_files_grid->SetDefaultCellAlignment( wxALIGN_LEFT, wxALIGN_CENTER );
+	m_global_sizer->Add( m_files_grid, 5, wxALL|wxEXPAND, 5 );
+
+
+	bMainSizer->Add( m_global_sizer, 1, wxEXPAND, 5 );
+
+	wxBoxSizer* bButtonsSizer;
+	bButtonsSizer = new wxBoxSizer( wxHORIZONTAL );
+
+	m_browse_button = new STD_BITMAP_BUTTON( this, wxID_ANY, wxNullBitmap, wxDefaultPosition, wxSize( -1,-1 ), wxBU_AUTODRAW|0 );
+	m_browse_button->SetToolTip( _("Add embedded file") );
+
+	bButtonsSizer->Add( m_browse_button, 0, wxTOP|wxBOTTOM|wxRIGHT, 5 );
+
+
+	bButtonsSizer->Add( 20, 0, 0, wxEXPAND, 5 );
+
+	m_delete_button = new STD_BITMAP_BUTTON( this, wxID_ANY, wxNullBitmap, wxDefaultPosition, wxSize( -1,-1 ), wxBU_AUTODRAW|0 );
+	m_delete_button->SetToolTip( _("Remove embedded file") );
+
+	bButtonsSizer->Add( m_delete_button, 0, wxTOP|wxBOTTOM|wxRIGHT, 5 );
+
+
+	bButtonsSizer->Add( 0, 0, 1, wxEXPAND, 5 );
+
+	m_export = new wxButton( this, wxID_ANY, _("&Export"), wxDefaultPosition, wxDefaultSize, 0 );
+	bButtonsSizer->Add( m_export, 0, wxALIGN_CENTER_VERTICAL|wxALL, 5 );
+
+
+	bMainSizer->Add( bButtonsSizer, 0, wxEXPAND|wxALL, 3 );
+
+	m_staticline1 = new wxStaticLine( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLI_HORIZONTAL );
+	bMainSizer->Add( m_staticline1, 0, wxEXPAND | wxALL, 5 );
+
+	wxBoxSizer* bSizer4;
+	bSizer4 = new wxBoxSizer( wxVERTICAL );
+
+	m_cbEmbedFonts = new wxCheckBox( this, wxID_ANY, _("Embed Fonts"), wxDefaultPosition, wxDefaultSize, 0 );
+	m_cbEmbedFonts->SetToolTip( _("Store a copy of all fonts used") );
+
+	bSizer4->Add( m_cbEmbedFonts, 0, wxALL, 5 );
+
+
+	bMainSizer->Add( bSizer4, 0, wxEXPAND, 5 );
+
+
+	this->SetSizer( bMainSizer );
+	this->Layout();
+	bMainSizer->Fit( this );
+
+	// Connect Events
+	this->Connect( wxEVT_SIZE, wxSizeEventHandler( PANEL_EMBEDDED_FILES_BASE::onSize ) );
+	m_files_grid->Connect( wxEVT_GRID_CELL_RIGHT_CLICK, wxGridEventHandler( PANEL_EMBEDDED_FILES_BASE::onGridRightClick ), NULL, this );
+	m_browse_button->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( PANEL_EMBEDDED_FILES_BASE::onAddEmbeddedFile ), NULL, this );
+	m_delete_button->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( PANEL_EMBEDDED_FILES_BASE::onDeleteEmbeddedFile ), NULL, this );
+	m_export->Connect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( PANEL_EMBEDDED_FILES_BASE::onExportFiles ), NULL, this );
+}
+
+PANEL_EMBEDDED_FILES_BASE::~PANEL_EMBEDDED_FILES_BASE()
+{
+	// Disconnect Events
+	this->Disconnect( wxEVT_SIZE, wxSizeEventHandler( PANEL_EMBEDDED_FILES_BASE::onSize ) );
+	m_files_grid->Disconnect( wxEVT_GRID_CELL_RIGHT_CLICK, wxGridEventHandler( PANEL_EMBEDDED_FILES_BASE::onGridRightClick ), NULL, this );
+	m_browse_button->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( PANEL_EMBEDDED_FILES_BASE::onAddEmbeddedFile ), NULL, this );
+	m_delete_button->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( PANEL_EMBEDDED_FILES_BASE::onDeleteEmbeddedFile ), NULL, this );
+	m_export->Disconnect( wxEVT_COMMAND_BUTTON_CLICKED, wxCommandEventHandler( PANEL_EMBEDDED_FILES_BASE::onExportFiles ), NULL, this );
+
+}
diff --git a/common/dialogs/panel_embedded_files_base.fbp b/common/dialogs/panel_embedded_files_base.fbp
new file mode 100644
index 0000000000..caef388d38
--- /dev/null
+++ b/common/dialogs/panel_embedded_files_base.fbp
@@ -0,0 +1,559 @@
+<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
+<wxFormBuilder_Project>
+  <FileVersion major="1" minor="18"/>
+  <object class="Project" expanded="true">
+    <property name="code_generation">C++</property>
+    <property name="cpp_class_decoration"></property>
+    <property name="cpp_disconnect_events">1</property>
+    <property name="cpp_event_generation">connect</property>
+    <property name="cpp_help_provider">none</property>
+    <property name="cpp_namespace"></property>
+    <property name="cpp_precompiled_header"></property>
+    <property name="cpp_use_array_enum">0</property>
+    <property name="cpp_use_enum">0</property>
+    <property name="embedded_files_path">res</property>
+    <property name="encoding">UTF-8</property>
+    <property name="file">panel_embedded_files_base</property>
+    <property name="first_id">1000</property>
+    <property name="internationalize">1</property>
+    <property name="lua_skip_events">1</property>
+    <property name="lua_ui_table">UI</property>
+    <property name="name">panel_embedded_files</property>
+    <property name="path">.</property>
+    <property name="php_disconnect_events">0</property>
+    <property name="php_disconnect_mode">source_name</property>
+    <property name="php_skip_events">1</property>
+    <property name="python_disconnect_events">0</property>
+    <property name="python_disconnect_mode">source_name</property>
+    <property name="python_image_path_wrapper_function_name"></property>
+    <property name="python_indent_with_spaces"></property>
+    <property name="python_skip_events">1</property>
+    <property name="relative_path">1</property>
+    <property name="use_microsoft_bom">0</property>
+    <property name="use_native_eol">0</property>
+    <object class="Panel" expanded="true">
+      <property name="aui_managed">0</property>
+      <property name="aui_manager_style">wxAUI_MGR_DEFAULT</property>
+      <property name="bg"></property>
+      <property name="context_help"></property>
+      <property name="context_menu">1</property>
+      <property name="drag_accept_files">0</property>
+      <property name="enabled">1</property>
+      <property name="event_handler">impl_virtual</property>
+      <property name="fg"></property>
+      <property name="font"></property>
+      <property name="hidden">0</property>
+      <property name="id">wxID_ANY</property>
+      <property name="maximum_size"></property>
+      <property name="minimum_size">-1,-1</property>
+      <property name="name">PANEL_EMBEDDED_FILES_BASE</property>
+      <property name="pos"></property>
+      <property name="size">-1,-1</property>
+      <property name="subclass">; forward_declare</property>
+      <property name="tooltip"></property>
+      <property name="two_step_creation">0</property>
+      <property name="window_extra_style"></property>
+      <property name="window_name"></property>
+      <property name="window_style">wxTAB_TRAVERSAL</property>
+      <event name="OnSize">onSize</event>
+      <object class="wxBoxSizer" expanded="true">
+        <property name="minimum_size"></property>
+        <property name="name">bMainSizer</property>
+        <property name="orient">wxVERTICAL</property>
+        <property name="permission">none</property>
+        <object class="sizeritem" expanded="true">
+          <property name="border">5</property>
+          <property name="flag">wxEXPAND</property>
+          <property name="proportion">1</property>
+          <object class="wxBoxSizer" expanded="false">
+            <property name="minimum_size"></property>
+            <property name="name">m_global_sizer</property>
+            <property name="orient">wxVERTICAL</property>
+            <property name="permission">none</property>
+            <object class="sizeritem" expanded="false">
+              <property name="border">5</property>
+              <property name="flag">wxALL|wxEXPAND</property>
+              <property name="proportion">5</property>
+              <object class="wxGrid" expanded="false">
+                <property name="BottomDockable">1</property>
+                <property name="LeftDockable">1</property>
+                <property name="RightDockable">1</property>
+                <property name="TopDockable">1</property>
+                <property name="aui_layer">0</property>
+                <property name="aui_name"></property>
+                <property name="aui_position">0</property>
+                <property name="aui_row">0</property>
+                <property name="autosize_cols">0</property>
+                <property name="autosize_rows">1</property>
+                <property name="best_size"></property>
+                <property name="bg"></property>
+                <property name="caption"></property>
+                <property name="caption_visible">1</property>
+                <property name="cell_bg"></property>
+                <property name="cell_font"></property>
+                <property name="cell_horiz_alignment">wxALIGN_LEFT</property>
+                <property name="cell_text"></property>
+                <property name="cell_vert_alignment">wxALIGN_CENTER</property>
+                <property name="center_pane">0</property>
+                <property name="close_button">1</property>
+                <property name="col_label_horiz_alignment">wxALIGN_CENTER</property>
+                <property name="col_label_size">22</property>
+                <property name="col_label_values">&quot;Filename&quot; &quot;Internal Reference&quot;</property>
+                <property name="col_label_vert_alignment">wxALIGN_CENTER</property>
+                <property name="cols">2</property>
+                <property name="column_sizes">440,180</property>
+                <property name="context_help"></property>
+                <property name="context_menu">1</property>
+                <property name="default_pane">0</property>
+                <property name="dock">Dock</property>
+                <property name="dock_fixed">1</property>
+                <property name="docking">Left</property>
+                <property name="drag_accept_files">0</property>
+                <property name="drag_col_move">0</property>
+                <property name="drag_col_size">1</property>
+                <property name="drag_grid_size">0</property>
+                <property name="drag_row_size">0</property>
+                <property name="editing">0</property>
+                <property name="enabled">1</property>
+                <property name="fg"></property>
+                <property name="floatable">0</property>
+                <property name="font"></property>
+                <property name="grid_line_color"></property>
+                <property name="grid_lines">1</property>
+                <property name="gripper">0</property>
+                <property name="hidden">0</property>
+                <property name="id">wxID_ANY</property>
+                <property name="label_bg"></property>
+                <property name="label_font"></property>
+                <property name="label_text"></property>
+                <property name="margin_height">0</property>
+                <property name="margin_width">0</property>
+                <property name="max_size"></property>
+                <property name="maximize_button">0</property>
+                <property name="maximum_size"></property>
+                <property name="min_size"></property>
+                <property name="minimize_button">0</property>
+                <property name="minimum_size">-1,-1</property>
+                <property name="moveable">0</property>
+                <property name="name">m_files_grid</property>
+                <property name="pane_border">1</property>
+                <property name="pane_position"></property>
+                <property name="pane_size"></property>
+                <property name="permission">protected</property>
+                <property name="pin_button">1</property>
+                <property name="pos"></property>
+                <property name="resize">Fixed</property>
+                <property name="row_label_horiz_alignment">wxALIGN_CENTER</property>
+                <property name="row_label_size">0</property>
+                <property name="row_label_values"></property>
+                <property name="row_label_vert_alignment">wxALIGN_CENTER</property>
+                <property name="row_sizes"></property>
+                <property name="rows">1</property>
+                <property name="show">1</property>
+                <property name="size"></property>
+                <property name="subclass">WX_GRID; widgets/wx_grid.h; forward_declare</property>
+                <property name="toolbar_pane">0</property>
+                <property name="tooltip"></property>
+                <property name="window_extra_style"></property>
+                <property name="window_name"></property>
+                <property name="window_style"></property>
+                <event name="OnGridCellRightClick">onGridRightClick</event>
+              </object>
+            </object>
+          </object>
+        </object>
+        <object class="sizeritem" expanded="true">
+          <property name="border">3</property>
+          <property name="flag">wxEXPAND|wxALL</property>
+          <property name="proportion">0</property>
+          <object class="wxBoxSizer" expanded="false">
+            <property name="minimum_size"></property>
+            <property name="name">bButtonsSizer</property>
+            <property name="orient">wxHORIZONTAL</property>
+            <property name="permission">none</property>
+            <object class="sizeritem" expanded="false">
+              <property name="border">5</property>
+              <property name="flag">wxTOP|wxBOTTOM|wxRIGHT</property>
+              <property name="proportion">0</property>
+              <object class="wxBitmapButton" expanded="false">
+                <property name="BottomDockable">1</property>
+                <property name="LeftDockable">1</property>
+                <property name="RightDockable">1</property>
+                <property name="TopDockable">1</property>
+                <property name="aui_layer">0</property>
+                <property name="aui_name"></property>
+                <property name="aui_position">0</property>
+                <property name="aui_row">0</property>
+                <property name="auth_needed">0</property>
+                <property name="best_size"></property>
+                <property name="bg"></property>
+                <property name="bitmap"></property>
+                <property name="caption"></property>
+                <property name="caption_visible">1</property>
+                <property name="center_pane">0</property>
+                <property name="close_button">1</property>
+                <property name="context_help"></property>
+                <property name="context_menu">1</property>
+                <property name="current"></property>
+                <property name="default">0</property>
+                <property name="default_pane">0</property>
+                <property name="disabled"></property>
+                <property name="dock">Dock</property>
+                <property name="dock_fixed">0</property>
+                <property name="docking">Left</property>
+                <property name="drag_accept_files">0</property>
+                <property name="enabled">1</property>
+                <property name="fg"></property>
+                <property name="floatable">1</property>
+                <property name="focus"></property>
+                <property name="font"></property>
+                <property name="gripper">0</property>
+                <property name="hidden">0</property>
+                <property name="id">wxID_ANY</property>
+                <property name="label">Add Embedded File</property>
+                <property name="margins"></property>
+                <property name="markup">0</property>
+                <property name="max_size"></property>
+                <property name="maximize_button">0</property>
+                <property name="maximum_size"></property>
+                <property name="min_size"></property>
+                <property name="minimize_button">0</property>
+                <property name="minimum_size"></property>
+                <property name="moveable">1</property>
+                <property name="name">m_browse_button</property>
+                <property name="pane_border">1</property>
+                <property name="pane_position"></property>
+                <property name="pane_size"></property>
+                <property name="permission">protected</property>
+                <property name="pin_button">1</property>
+                <property name="pos"></property>
+                <property name="position"></property>
+                <property name="pressed"></property>
+                <property name="resize">Resizable</property>
+                <property name="show">1</property>
+                <property name="size">-1,-1</property>
+                <property name="style"></property>
+                <property name="subclass">STD_BITMAP_BUTTON; widgets/std_bitmap_button.h; forward_declare</property>
+                <property name="toolbar_pane">0</property>
+                <property name="tooltip">Add embedded file</property>
+                <property name="validator_data_type"></property>
+                <property name="validator_style">wxFILTER_NONE</property>
+                <property name="validator_type">wxDefaultValidator</property>
+                <property name="validator_variable"></property>
+                <property name="window_extra_style"></property>
+                <property name="window_name"></property>
+                <property name="window_style"></property>
+                <event name="OnButtonClick">onAddEmbeddedFile</event>
+              </object>
+            </object>
+            <object class="sizeritem" expanded="false">
+              <property name="border">5</property>
+              <property name="flag">wxEXPAND</property>
+              <property name="proportion">0</property>
+              <object class="spacer" expanded="false">
+                <property name="height">0</property>
+                <property name="permission">protected</property>
+                <property name="width">20</property>
+              </object>
+            </object>
+            <object class="sizeritem" expanded="false">
+              <property name="border">5</property>
+              <property name="flag">wxTOP|wxBOTTOM|wxRIGHT</property>
+              <property name="proportion">0</property>
+              <object class="wxBitmapButton" expanded="false">
+                <property name="BottomDockable">1</property>
+                <property name="LeftDockable">1</property>
+                <property name="RightDockable">1</property>
+                <property name="TopDockable">1</property>
+                <property name="aui_layer">0</property>
+                <property name="aui_name"></property>
+                <property name="aui_position">0</property>
+                <property name="aui_row">0</property>
+                <property name="auth_needed">0</property>
+                <property name="best_size"></property>
+                <property name="bg"></property>
+                <property name="bitmap"></property>
+                <property name="caption"></property>
+                <property name="caption_visible">1</property>
+                <property name="center_pane">0</property>
+                <property name="close_button">1</property>
+                <property name="context_help"></property>
+                <property name="context_menu">1</property>
+                <property name="current"></property>
+                <property name="default">0</property>
+                <property name="default_pane">0</property>
+                <property name="disabled"></property>
+                <property name="dock">Dock</property>
+                <property name="dock_fixed">0</property>
+                <property name="docking">Left</property>
+                <property name="drag_accept_files">0</property>
+                <property name="enabled">1</property>
+                <property name="fg"></property>
+                <property name="floatable">1</property>
+                <property name="focus"></property>
+                <property name="font"></property>
+                <property name="gripper">0</property>
+                <property name="hidden">0</property>
+                <property name="id">wxID_ANY</property>
+                <property name="label">Delete SelectedFile</property>
+                <property name="margins"></property>
+                <property name="markup">0</property>
+                <property name="max_size"></property>
+                <property name="maximize_button">0</property>
+                <property name="maximum_size"></property>
+                <property name="min_size"></property>
+                <property name="minimize_button">0</property>
+                <property name="minimum_size"></property>
+                <property name="moveable">1</property>
+                <property name="name">m_delete_button</property>
+                <property name="pane_border">1</property>
+                <property name="pane_position"></property>
+                <property name="pane_size"></property>
+                <property name="permission">protected</property>
+                <property name="pin_button">1</property>
+                <property name="pos"></property>
+                <property name="position"></property>
+                <property name="pressed"></property>
+                <property name="resize">Resizable</property>
+                <property name="show">1</property>
+                <property name="size">-1,-1</property>
+                <property name="style"></property>
+                <property name="subclass">STD_BITMAP_BUTTON; widgets/std_bitmap_button.h; forward_declare</property>
+                <property name="toolbar_pane">0</property>
+                <property name="tooltip">Remove embedded file</property>
+                <property name="validator_data_type"></property>
+                <property name="validator_style">wxFILTER_NONE</property>
+                <property name="validator_type">wxDefaultValidator</property>
+                <property name="validator_variable"></property>
+                <property name="window_extra_style"></property>
+                <property name="window_name"></property>
+                <property name="window_style"></property>
+                <event name="OnButtonClick">onDeleteEmbeddedFile</event>
+              </object>
+            </object>
+            <object class="sizeritem" expanded="false">
+              <property name="border">5</property>
+              <property name="flag">wxEXPAND</property>
+              <property name="proportion">1</property>
+              <object class="spacer" expanded="false">
+                <property name="height">0</property>
+                <property name="permission">protected</property>
+                <property name="width">0</property>
+              </object>
+            </object>
+            <object class="sizeritem" expanded="false">
+              <property name="border">5</property>
+              <property name="flag">wxALIGN_CENTER_VERTICAL|wxALL</property>
+              <property name="proportion">0</property>
+              <object class="wxButton" expanded="false">
+                <property name="BottomDockable">1</property>
+                <property name="LeftDockable">1</property>
+                <property name="RightDockable">1</property>
+                <property name="TopDockable">1</property>
+                <property name="aui_layer">0</property>
+                <property name="aui_name"></property>
+                <property name="aui_position">0</property>
+                <property name="aui_row">0</property>
+                <property name="auth_needed">0</property>
+                <property name="best_size"></property>
+                <property name="bg"></property>
+                <property name="bitmap"></property>
+                <property name="caption"></property>
+                <property name="caption_visible">1</property>
+                <property name="center_pane">0</property>
+                <property name="close_button">1</property>
+                <property name="context_help"></property>
+                <property name="context_menu">1</property>
+                <property name="current"></property>
+                <property name="default">0</property>
+                <property name="default_pane">0</property>
+                <property name="disabled"></property>
+                <property name="dock">Dock</property>
+                <property name="dock_fixed">0</property>
+                <property name="docking">Left</property>
+                <property name="drag_accept_files">0</property>
+                <property name="enabled">1</property>
+                <property name="fg"></property>
+                <property name="floatable">1</property>
+                <property name="focus"></property>
+                <property name="font"></property>
+                <property name="gripper">0</property>
+                <property name="hidden">0</property>
+                <property name="id">wxID_ANY</property>
+                <property name="label">&amp;Export</property>
+                <property name="margins"></property>
+                <property name="markup">0</property>
+                <property name="max_size"></property>
+                <property name="maximize_button">0</property>
+                <property name="maximum_size"></property>
+                <property name="min_size"></property>
+                <property name="minimize_button">0</property>
+                <property name="minimum_size"></property>
+                <property name="moveable">1</property>
+                <property name="name">m_export</property>
+                <property name="pane_border">1</property>
+                <property name="pane_position"></property>
+                <property name="pane_size"></property>
+                <property name="permission">protected</property>
+                <property name="pin_button">1</property>
+                <property name="pos"></property>
+                <property name="position"></property>
+                <property name="pressed"></property>
+                <property name="resize">Resizable</property>
+                <property name="show">1</property>
+                <property name="size"></property>
+                <property name="style"></property>
+                <property name="subclass">; ; forward_declare</property>
+                <property name="toolbar_pane">0</property>
+                <property name="tooltip"></property>
+                <property name="validator_data_type"></property>
+                <property name="validator_style">wxFILTER_NONE</property>
+                <property name="validator_type">wxDefaultValidator</property>
+                <property name="validator_variable"></property>
+                <property name="window_extra_style"></property>
+                <property name="window_name"></property>
+                <property name="window_style"></property>
+                <event name="OnButtonClick">onExportFiles</event>
+              </object>
+            </object>
+          </object>
+        </object>
+        <object class="sizeritem" expanded="true">
+          <property name="border">5</property>
+          <property name="flag">wxEXPAND | wxALL</property>
+          <property name="proportion">0</property>
+          <object class="wxStaticLine" expanded="true">
+            <property name="BottomDockable">1</property>
+            <property name="LeftDockable">1</property>
+            <property name="RightDockable">1</property>
+            <property name="TopDockable">1</property>
+            <property name="aui_layer">0</property>
+            <property name="aui_name"></property>
+            <property name="aui_position">0</property>
+            <property name="aui_row">0</property>
+            <property name="best_size"></property>
+            <property name="bg"></property>
+            <property name="caption"></property>
+            <property name="caption_visible">1</property>
+            <property name="center_pane">0</property>
+            <property name="close_button">1</property>
+            <property name="context_help"></property>
+            <property name="context_menu">1</property>
+            <property name="default_pane">0</property>
+            <property name="dock">Dock</property>
+            <property name="dock_fixed">0</property>
+            <property name="docking">Left</property>
+            <property name="drag_accept_files">0</property>
+            <property name="enabled">1</property>
+            <property name="fg"></property>
+            <property name="floatable">1</property>
+            <property name="font"></property>
+            <property name="gripper">0</property>
+            <property name="hidden">0</property>
+            <property name="id">wxID_ANY</property>
+            <property name="max_size"></property>
+            <property name="maximize_button">0</property>
+            <property name="maximum_size"></property>
+            <property name="min_size"></property>
+            <property name="minimize_button">0</property>
+            <property name="minimum_size"></property>
+            <property name="moveable">1</property>
+            <property name="name">m_staticline1</property>
+            <property name="pane_border">1</property>
+            <property name="pane_position"></property>
+            <property name="pane_size"></property>
+            <property name="permission">protected</property>
+            <property name="pin_button">1</property>
+            <property name="pos"></property>
+            <property name="resize">Resizable</property>
+            <property name="show">1</property>
+            <property name="size"></property>
+            <property name="style">wxLI_HORIZONTAL</property>
+            <property name="subclass">; ; forward_declare</property>
+            <property name="toolbar_pane">0</property>
+            <property name="tooltip"></property>
+            <property name="window_extra_style"></property>
+            <property name="window_name"></property>
+            <property name="window_style"></property>
+          </object>
+        </object>
+        <object class="sizeritem" expanded="true">
+          <property name="border">5</property>
+          <property name="flag">wxEXPAND</property>
+          <property name="proportion">0</property>
+          <object class="wxBoxSizer" expanded="true">
+            <property name="minimum_size"></property>
+            <property name="name">bSizer4</property>
+            <property name="orient">wxVERTICAL</property>
+            <property name="permission">none</property>
+            <object class="sizeritem" expanded="true">
+              <property name="border">5</property>
+              <property name="flag">wxALL</property>
+              <property name="proportion">0</property>
+              <object class="wxCheckBox" expanded="true">
+                <property name="BottomDockable">1</property>
+                <property name="LeftDockable">1</property>
+                <property name="RightDockable">1</property>
+                <property name="TopDockable">1</property>
+                <property name="aui_layer">0</property>
+                <property name="aui_name"></property>
+                <property name="aui_position">0</property>
+                <property name="aui_row">0</property>
+                <property name="best_size"></property>
+                <property name="bg"></property>
+                <property name="caption"></property>
+                <property name="caption_visible">1</property>
+                <property name="center_pane">0</property>
+                <property name="checked">0</property>
+                <property name="close_button">1</property>
+                <property name="context_help"></property>
+                <property name="context_menu">1</property>
+                <property name="default_pane">0</property>
+                <property name="dock">Dock</property>
+                <property name="dock_fixed">0</property>
+                <property name="docking">Left</property>
+                <property name="drag_accept_files">0</property>
+                <property name="enabled">1</property>
+                <property name="fg"></property>
+                <property name="floatable">1</property>
+                <property name="font"></property>
+                <property name="gripper">0</property>
+                <property name="hidden">0</property>
+                <property name="id">wxID_ANY</property>
+                <property name="label">Embed Fonts</property>
+                <property name="max_size"></property>
+                <property name="maximize_button">0</property>
+                <property name="maximum_size"></property>
+                <property name="min_size"></property>
+                <property name="minimize_button">0</property>
+                <property name="minimum_size"></property>
+                <property name="moveable">1</property>
+                <property name="name">m_cbEmbedFonts</property>
+                <property name="pane_border">1</property>
+                <property name="pane_position"></property>
+                <property name="pane_size"></property>
+                <property name="permission">protected</property>
+                <property name="pin_button">1</property>
+                <property name="pos"></property>
+                <property name="resize">Resizable</property>
+                <property name="show">1</property>
+                <property name="size"></property>
+                <property name="style"></property>
+                <property name="subclass">; ; forward_declare</property>
+                <property name="toolbar_pane">0</property>
+                <property name="tooltip">Store a copy of all fonts used</property>
+                <property name="validator_data_type"></property>
+                <property name="validator_style">wxFILTER_NONE</property>
+                <property name="validator_type">wxDefaultValidator</property>
+                <property name="validator_variable"></property>
+                <property name="window_extra_style"></property>
+                <property name="window_name"></property>
+                <property name="window_style"></property>
+              </object>
+            </object>
+          </object>
+        </object>
+      </object>
+    </object>
+  </object>
+</wxFormBuilder_Project>
diff --git a/common/dialogs/panel_embedded_files_base.h b/common/dialogs/panel_embedded_files_base.h
new file mode 100644
index 0000000000..22270be115
--- /dev/null
+++ b/common/dialogs/panel_embedded_files_base.h
@@ -0,0 +1,64 @@
+///////////////////////////////////////////////////////////////////////////
+// C++ code generated with wxFormBuilder (version 4.2.1-0-g80c4cb6a-dirty)
+// http://www.wxformbuilder.org/
+//
+// PLEASE DO *NOT* EDIT THIS FILE!
+///////////////////////////////////////////////////////////////////////////
+
+#pragma once
+
+#include <wx/artprov.h>
+#include <wx/xrc/xmlres.h>
+#include <wx/intl.h>
+class STD_BITMAP_BUTTON;
+class WX_GRID;
+
+#include <wx/colour.h>
+#include <wx/settings.h>
+#include <wx/string.h>
+#include <wx/font.h>
+#include <wx/grid.h>
+#include <wx/gdicmn.h>
+#include <wx/sizer.h>
+#include <wx/bmpbuttn.h>
+#include <wx/bitmap.h>
+#include <wx/image.h>
+#include <wx/icon.h>
+#include <wx/button.h>
+#include <wx/statline.h>
+#include <wx/checkbox.h>
+#include <wx/panel.h>
+
+///////////////////////////////////////////////////////////////////////////
+
+///////////////////////////////////////////////////////////////////////////////
+/// Class PANEL_EMBEDDED_FILES_BASE
+///////////////////////////////////////////////////////////////////////////////
+class PANEL_EMBEDDED_FILES_BASE : public wxPanel
+{
+	private:
+
+	protected:
+		WX_GRID* m_files_grid;
+		STD_BITMAP_BUTTON* m_browse_button;
+		STD_BITMAP_BUTTON* m_delete_button;
+		wxButton* m_export;
+		wxStaticLine* m_staticline1;
+		wxCheckBox* m_cbEmbedFonts;
+
+		// Virtual event handlers, override them in your derived class
+		virtual void onSize( wxSizeEvent& event ) { event.Skip(); }
+		virtual void onGridRightClick( wxGridEvent& event ) { event.Skip(); }
+		virtual void onAddEmbeddedFile( wxCommandEvent& event ) { event.Skip(); }
+		virtual void onDeleteEmbeddedFile( wxCommandEvent& event ) { event.Skip(); }
+		virtual void onExportFiles( wxCommandEvent& event ) { event.Skip(); }
+
+
+	public:
+
+		PANEL_EMBEDDED_FILES_BASE( wxWindow* parent, wxWindowID id = wxID_ANY, const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( -1,-1 ), long style = wxTAB_TRAVERSAL, const wxString& name = wxEmptyString );
+
+		~PANEL_EMBEDDED_FILES_BASE();
+
+};
+
diff --git a/common/drawing_sheet/ds_data_model.cpp b/common/drawing_sheet/ds_data_model.cpp
index 370ba9064b..8dc2f9f27d 100644
--- a/common/drawing_sheet/ds_data_model.cpp
+++ b/common/drawing_sheet/ds_data_model.cpp
@@ -145,33 +145,3 @@ DS_DATA_ITEM* DS_DATA_MODEL::GetItem( unsigned aIdx ) const
         return nullptr;
 }
 
-
-const wxString DS_DATA_MODEL::ResolvePath( const wxString& aPath, const wxString& aProjectPath )
-{
-    wxString fullFileName = ExpandEnvVarSubstitutions( aPath, nullptr );
-
-    if( fullFileName.IsEmpty() )
-        return fullFileName;
-
-    wxFileName fn = fullFileName;
-
-    if( fn.IsAbsolute() )
-        return fullFileName;
-
-    // the path is not absolute: search it in project path, and then in kicad valid paths
-    if( !aProjectPath.IsEmpty() )
-    {
-        fn.MakeAbsolute( aProjectPath );
-
-        if( wxFileExists( fn.GetFullPath() ) )
-            return fn.GetFullPath();
-    }
-
-    fn = fullFileName;
-    wxString name = Kiface().KifaceSearch().FindValidPath( fn.GetFullName() );
-
-    if( !name.IsEmpty() )
-        fullFileName = name;
-
-    return fullFileName;
-}
diff --git a/common/dsnlexer.cpp b/common/dsnlexer.cpp
index 89c4d87631..73d0a4cfcb 100644
--- a/common/dsnlexer.cpp
+++ b/common/dsnlexer.cpp
@@ -276,6 +276,9 @@ const char* DSNLEXER::Syntax( int aTok )
     case DSN_EOF:
         ret = "end of input";
         break;
+    case DSN_BAR:
+        ret = "|";
+        break;
     default:
         ret = "???";
     }
@@ -379,6 +382,15 @@ void DSNLEXER::NeedRIGHT()
 }
 
 
+void DSNLEXER::NeedBAR()
+{
+    int tok = NextTok();
+
+    if( tok != DSN_BAR )
+        Expecting( DSN_BAR );
+}
+
+
 int DSNLEXER::NeedSYMBOL()
 {
     int tok = NextTok();
@@ -452,7 +464,7 @@ inline bool isDigit( char cc )
 ///< @return true if @a cc is an s-expression separator character.
 inline bool isSep( char cc )
 {
-    return isSpace( cc ) || cc=='(' || cc==')';
+    return isSpace( cc ) || cc == '(' || cc == ')' || cc == '|';
 }
 
 
@@ -597,6 +609,14 @@ L_read:
         goto exit;
     }
 
+    if( *cur == '|' )
+    {
+        curText = *cur;
+        curTok = DSN_BAR;
+        head = cur+1;
+        goto exit;
+    }
+
     // Non-specctraMode, understands and deciphers escaped \, \r, \n, and \".
     // Strips off leading and trailing double quotes
     if( !specctraMode )
diff --git a/common/eda_doc.cpp b/common/eda_doc.cpp
index 88f9c21dbb..78e2504831 100644
--- a/common/eda_doc.cpp
+++ b/common/eda_doc.cpp
@@ -25,6 +25,7 @@
 #include <pgm_base.h>
 #include <common.h>
 #include <confirm.h>
+#include <embedded_files.h>
 #include <gestfich.h>
 #include <settings/common_settings.h>
 
@@ -58,7 +59,8 @@ static const wxFileTypeInfo EDAfallbacks[] =
 };
 
 
-bool GetAssociatedDocument( wxWindow* aParent, const wxString& aDocName, PROJECT* aProject, SEARCH_STACK* aPaths )
+bool GetAssociatedDocument( wxWindow* aParent, const wxString& aDocName, PROJECT* aProject,
+                            SEARCH_STACK* aPaths, EMBEDDED_FILES* aFiles )
 {
     wxString      docname;
     wxString      fullfilename;
@@ -74,8 +76,48 @@ bool GetAssociatedDocument( wxWindow* aParent, const wxString& aDocName, PROJECT
         wxURI     uri( docname );
         wxLogNull logNo; // Disable log messages
 
-        if( uri.HasScheme() && wxLaunchDefaultBrowser( docname ) )
-            return true;
+        if( uri.HasScheme() )
+        {
+            wxString scheme = uri.GetScheme().Lower();
+
+            if( scheme != FILEEXT::KiCadUriPrefix )
+            {
+                if( wxLaunchDefaultBrowser( docname ) )
+                    return true;
+            }
+            else
+            {
+                if( !aFiles )
+                {
+                    wxLogTrace( wxT( "KICAD_EMBED" ),
+                                wxT( "No EMBEDDED_FILES object provided for kicad_embed URI" ) );
+                    return false;
+                }
+
+                if( !docname.starts_with( FILEEXT::KiCadUriPrefix + "://" ) )
+                {
+                    wxLogTrace( wxT( "KICAD_EMBED" ),
+                                wxT( "Invalid kicad_embed URI '%s'" ), docname );
+                    return false;
+                }
+
+                docname = docname.Mid( 14 );
+
+                wxFileName temp_file = aFiles->GetTempFileName( docname );
+
+                if( !temp_file.IsOk() )
+                {
+                    wxLogTrace( wxT( "KICAD_EMBED" ),
+                                wxT( "Failed to get temp file '%s' for kicad_embed URI" ), docname );
+                    return false;
+                }
+
+                wxLogTrace( wxT( "KICAD_EMBED" ),
+                            wxT( "Opening embedded file '%s' as '%s'" ),
+                            docname, temp_file.GetFullPath() );
+                docname = temp_file.GetFullPath();
+            }
+        }
     }
 
 #ifdef __WINDOWS__
diff --git a/common/embedded_files.cpp b/common/embedded_files.cpp
new file mode 100644
index 0000000000..c7b7aa4727
--- /dev/null
+++ b/common/embedded_files.cpp
@@ -0,0 +1,511 @@
+/*
+ * This program source code file is part of KiCad, a free EDA CAD application.
+ *
+ * Copyright (C) 2024 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 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, see <http://www.gnu.org/licenses/>.
+ */
+
+#include <wx/base64.h>
+#include <wx/debug.h>
+#include <wx/file.h>
+#include <wx/filename.h>
+#include <wx/log.h>
+#include <wx/mstream.h>
+#include <wx/wfstream.h>
+
+#include <map>
+#include <memory>
+#include <sstream>
+
+#include <zstd.h>
+
+#include <embedded_files.h>
+#include <kiid.h>
+#include <paths.h>
+
+
+
+EMBEDDED_FILES::EMBEDDED_FILE* EMBEDDED_FILES::AddFile( const wxFileName& aName, bool aOverwrite )
+{
+    if( HasFile( aName.GetFullName() ) )
+    {
+        if( !aOverwrite )
+            return m_files[aName.GetFullName()];
+
+        m_files.erase( aName.GetFullName() );
+    }
+
+    wxFFileInputStream file( aName.GetFullPath() );
+    wxMemoryBuffer     buffer;
+
+    if( !file.IsOk() )
+        return nullptr;
+
+    wxFileOffset length = file.GetLength();
+
+    std::unique_ptr<EMBEDDED_FILE> efile = std::make_unique<EMBEDDED_FILE>();
+    efile->name = aName.GetFullName();
+    efile->decompressedData.resize( length );
+
+    wxString ext = aName.GetExt().Upper();
+
+    // Handle some common file extensions
+    if( ext == "STP" || ext == "STPZ" || ext == "STEP" || ext == "WRL" || ext == "WRZ" )
+    {
+        efile->type = EMBEDDED_FILE::FILE_TYPE::MODEL;
+    }
+    else if( ext == "WOFF" || ext == "WOFF2" || ext == "TTF" || ext == "OTF" )
+    {
+        efile->type = EMBEDDED_FILE::FILE_TYPE::FONT;
+    }
+    else if( ext == "PDF" )
+    {
+        efile->type = EMBEDDED_FILE::FILE_TYPE::DATASHEET;
+    }
+    else if( ext == "KICAD_WKS" )
+    {
+        efile->type = EMBEDDED_FILE::FILE_TYPE::WORKSHEET;
+    }
+
+    if( !efile->decompressedData.data() )
+        return nullptr;
+
+    char* data = efile->decompressedData.data();
+    wxFileOffset total_read = 0;
+
+    while( !file.Eof() && total_read < length )
+    {
+        file.Read( data, length - total_read );
+
+        size_t read = file.LastRead();
+        data += read;
+        total_read += read;
+    }
+
+    if( CompressAndEncode( *efile ) != RETURN_CODE::OK )
+        return nullptr;
+
+    efile->is_valid = true;
+
+    m_files[aName.GetFullName()] = efile.release();
+
+    return m_files[aName.GetFullName()];
+}
+
+
+void EMBEDDED_FILES::AddFile( EMBEDDED_FILE* aFile )
+{
+    m_files.insert( { aFile->name, aFile } );
+}
+
+// Remove a file from the collection
+void EMBEDDED_FILES::RemoveFile( const wxString& name, bool aErase )
+{
+    auto it = m_files.find( name );
+
+    if( it != m_files.end() )
+    {
+        m_files.erase( it );
+
+        if( aErase )
+            delete it->second;
+    }
+}
+
+
+void EMBEDDED_FILES::ClearEmbeddedFonts()
+{
+    for( auto it = m_files.begin(); it != m_files.end(); )
+    {
+        if( it->second->type == EMBEDDED_FILE::FILE_TYPE::FONT )
+        {
+            delete it->second;
+            it = m_files.erase( it );
+        }
+        else
+        {
+            ++it;
+        }
+    }
+}
+
+
+// Write the collection of files to a disk file in the specified format
+void EMBEDDED_FILES::WriteEmbeddedFiles( OUTPUTFORMATTER& aOut, int aNestLevel,
+                                         bool aWriteData ) const
+{
+    ssize_t MIME_BASE64_LENGTH = 76;
+    aOut.Print( aNestLevel, "(embedded_files\n" );
+
+    for( const auto& [name, entry] : m_files )
+    {
+        const EMBEDDED_FILE& file = *entry;
+
+        aOut.Print( aNestLevel + 1, "(file\n" );
+        aOut.Print( aNestLevel + 2, "(name \"%s\")\n", file.name.c_str().AsChar() );
+
+        const char* type = nullptr;
+
+        switch( file.type )
+        {
+        case EMBEDDED_FILE::FILE_TYPE::DATASHEET:
+            type = "datasheet";
+            break;
+        case EMBEDDED_FILE::FILE_TYPE::FONT:
+            type = "font";
+            break;
+        case EMBEDDED_FILE::FILE_TYPE::MODEL:
+            type = "model";
+            break;
+        case EMBEDDED_FILE::FILE_TYPE::WORKSHEET:
+            type = "worksheet";
+            break;
+        default:
+            type = "other";
+            break;
+        }
+
+        aOut.Print( aNestLevel + 2, "(type %s)\n", type );
+
+        if( aWriteData )
+        {
+            aOut.Print( 2, "(data\n" );
+
+            size_t first = 0;
+
+            while( first < file.compressedEncodedData.length() )
+            {
+                ssize_t remaining = file.compressedEncodedData.length() - first;
+                int     length = std::min( remaining, MIME_BASE64_LENGTH );
+
+                std::string_view view( file.compressedEncodedData.data() + first, length );
+
+                aOut.Print( aNestLevel + 3, "%1s%.*s%s\n", first ? "" : "|", length, view.data(),
+                            remaining == length ? "|" : "" );
+                first += MIME_BASE64_LENGTH;
+            }
+            aOut.Print( aNestLevel + 2, ")\n" ); // Close data
+        }
+
+        aOut.Print( aNestLevel + 2, "(checksum \"%s\")\n", file.data_sha.c_str() );
+        aOut.Print( aNestLevel + 1, ")\n" ); // Close file
+    }
+
+    aOut.Print( aNestLevel, ")\n" ); // Close embedded_files
+}
+
+// Compress and Base64 encode data
+EMBEDDED_FILES::RETURN_CODE EMBEDDED_FILES::CompressAndEncode( EMBEDDED_FILE& aFile )
+{
+    std::vector<char> compressedData;
+    size_t estCompressedSize = ZSTD_compressBound( aFile.decompressedData.size() );
+    compressedData.resize( estCompressedSize );
+    size_t compressedSize = ZSTD_compress( compressedData.data(), estCompressedSize,
+                                           aFile.decompressedData.data(),
+                                           aFile.decompressedData.size(), 15 );
+
+    if( ZSTD_isError( compressedSize ) )
+    {
+        compressedData.clear();
+        return RETURN_CODE::OUT_OF_MEMORY;
+    }
+
+    const size_t dstLen = wxBase64EncodedSize( compressedSize );
+    aFile.compressedEncodedData.resize( dstLen );
+    size_t retval = wxBase64Encode( aFile.compressedEncodedData.data(), dstLen,
+                                          compressedData.data(), compressedSize );
+    if( retval != dstLen )
+    {
+        aFile.compressedEncodedData.clear();
+        return RETURN_CODE::OUT_OF_MEMORY;
+    }
+
+    picosha2::hash256_hex_string( aFile.decompressedData, aFile.data_sha );
+
+    return RETURN_CODE::OK;
+}
+
+// Decompress and Base64 decode data
+EMBEDDED_FILES::RETURN_CODE EMBEDDED_FILES::DecompressAndDecode( EMBEDDED_FILE& aFile )
+{
+    std::vector<char> compressedData;
+    size_t            compressedSize = wxBase64DecodedSize( aFile.compressedEncodedData.size() );
+
+    if( compressedSize == 0 )
+    {
+        wxLogTrace( wxT( "KICAD_EMBED" ),
+                    wxT( "%s:%s:%d\n * Base64DecodedSize failed for file '%s' with size %zu" ),
+                    __FILE__, __FUNCTION__, __LINE__, aFile.name, aFile.compressedEncodedData.size() );
+        return RETURN_CODE::OUT_OF_MEMORY;
+    }
+
+    compressedData.resize( compressedSize );
+    void* compressed = compressedData.data();
+
+    // The return value from wxBase64Decode is the actual size of the decoded data avoiding
+    // the modulo 4 padding of the base64 encoding
+    compressedSize = wxBase64Decode( compressed, compressedSize, aFile.compressedEncodedData );
+
+    unsigned long long estDecompressedSize = ZSTD_getFrameContentSize( compressed, compressedSize );
+
+    if( estDecompressedSize > 1e9 ) // Limit to 1GB
+        return RETURN_CODE::OUT_OF_MEMORY;
+
+    if( estDecompressedSize == ZSTD_CONTENTSIZE_ERROR
+        || estDecompressedSize == ZSTD_CONTENTSIZE_UNKNOWN )
+    {
+        return RETURN_CODE::OUT_OF_MEMORY;
+    }
+
+    aFile.decompressedData.resize( estDecompressedSize );
+    void* decompressed = aFile.decompressedData.data();
+
+    size_t decompressedSize = ZSTD_decompress( decompressed, estDecompressedSize,
+                                               compressed, compressedSize );
+
+    if( ZSTD_isError( decompressedSize ) )
+    {
+        wxLogTrace( wxT( "KICAD_EMBED" ),
+                    wxT( "%s:%s:%d\n * ZSTD_decompress failed with error '%s'" ),
+                    __FILE__, __FUNCTION__, __LINE__, ZSTD_getErrorName( decompressedSize ) );
+        aFile.decompressedData.clear();
+        return RETURN_CODE::OUT_OF_MEMORY;
+    }
+
+    aFile.decompressedData.resize( decompressedSize );
+
+    std::string new_sha;
+    picosha2::hash256_hex_string( aFile.decompressedData, new_sha );
+
+    if( new_sha != aFile.data_sha )
+    {
+        wxLogTrace( wxT( "KICAD_EMBED" ),
+                    wxT( "%s:%s:%d\n * Checksum error in embedded file '%s'" ),
+                    __FILE__, __FUNCTION__, __LINE__, aFile.name );
+        aFile.decompressedData.clear();
+        return RETURN_CODE::CHECKSUM_ERROR;
+    }
+
+    return RETURN_CODE::OK;
+}
+
+// Parsing method
+void EMBEDDED_FILES_PARSER::ParseEmbedded( EMBEDDED_FILES* aFiles )
+{
+    if( !aFiles )
+        THROW_PARSE_ERROR( "No embedded files object provided", CurSource(), CurLine(),
+                           CurLineNumber(), CurOffset() );
+
+    using namespace EMBEDDED_FILES_T;
+
+    std::unique_ptr<EMBEDDED_FILES::EMBEDDED_FILE> file( nullptr );
+
+    for( T token = NextTok(); token != T_RIGHT; token = NextTok() )
+    {
+        if( token != T_LEFT )
+            Expecting( T_LEFT );
+
+        token = NextTok();
+
+        if( token != T_file )
+            Expecting( "file" );
+
+        if( file )
+        {
+            if( !file->compressedEncodedData.empty() )
+            {
+                EMBEDDED_FILES::DecompressAndDecode( *file );
+
+                if( !file->Validate() )
+                    THROW_PARSE_ERROR( "Checksum error in embedded file " + file->name, CurSource(),
+                                        CurLine(), CurLineNumber(), CurOffset() );
+            }
+
+            aFiles->AddFile( file.release() );
+        }
+
+        file = std::unique_ptr<EMBEDDED_FILES::EMBEDDED_FILE>( nullptr );
+
+
+        for( token = NextTok(); token != T_RIGHT; token = NextTok() )
+        {
+            if( token != T_LEFT )
+                Expecting( T_LEFT );
+
+            token = NextTok();
+
+            switch( token )
+            {
+
+            case T_checksum:
+                NeedSYMBOLorNUMBER();
+
+                if( !IsSymbol( token ) )
+                    Expecting( "checksum data" );
+
+                file->data_sha = CurStr();
+                NeedRIGHT();
+                break;
+
+            case T_data:
+                NeedBAR();
+                token = NextTok();
+
+                file->compressedEncodedData.reserve( 1 << 17 );
+
+                while( token != T_BAR )
+                {
+                    if( !IsSymbol( token ) )
+                        Expecting( "base64 file data" );
+
+                    file->compressedEncodedData += CurStr();
+                    token = NextTok();
+                }
+
+                file->compressedEncodedData.shrink_to_fit();
+
+                NeedRIGHT();
+                break;
+
+            case T_name:
+
+                if( file )
+                {
+                    wxLogTrace( wxT( "KICAD_EMBED" ),
+                                wxT( "Duplicate 'name' tag in embedded file %s" ), file->name );
+                }
+
+                NeedSYMBOLorNUMBER();
+
+                file = std::make_unique<EMBEDDED_FILES::EMBEDDED_FILE>();
+                file->name = CurStr();
+                NeedRIGHT();
+
+                break;
+
+            case T_type:
+
+                token = NextTok();
+
+                switch( token )
+                {
+                case T_datasheet:
+                    file->type = EMBEDDED_FILES::EMBEDDED_FILE::FILE_TYPE::DATASHEET;
+                    break;
+                case T_font:
+                    file->type = EMBEDDED_FILES::EMBEDDED_FILE::FILE_TYPE::FONT;
+                    break;
+                case T_model:
+                    file->type = EMBEDDED_FILES::EMBEDDED_FILE::FILE_TYPE::MODEL;
+                    break;
+                case T_worksheet:
+                    file->type = EMBEDDED_FILES::EMBEDDED_FILE::FILE_TYPE::WORKSHEET;
+                    break;
+                case T_other:
+                    file->type = EMBEDDED_FILES::EMBEDDED_FILE::FILE_TYPE::OTHER;
+                    break;
+                default:
+                    Expecting( "datasheet, font, model, worksheet or other" );
+                    break;
+                }
+                NeedRIGHT();
+                break;
+
+            default:
+                Expecting( "checksum, data or name" );
+            }
+        }
+    }
+
+    // Add the last file in the collection
+    if( file )
+    {
+        if( !file->compressedEncodedData.empty() )
+        {
+            EMBEDDED_FILES::DecompressAndDecode( *file );
+
+            if( !file->Validate() )
+                THROW_PARSE_ERROR( "Checksum error in embedded file " + file->name, CurSource(),
+                                    CurLine(), CurLineNumber(), CurOffset() );
+        }
+
+        aFiles->AddFile( file.release() );
+    }
+}
+
+
+wxFileName EMBEDDED_FILES::GetTempFileName( const wxString& aName ) const
+{
+    wxFileName cacheFile;
+
+    auto it = m_files.find( aName );
+
+    if( it == m_files.end() )
+        return cacheFile;
+
+    cacheFile.AssignDir( PATHS::GetUserCachePath() );
+    cacheFile.AppendDir( wxT( "embed" ) );
+
+    if( !PATHS::EnsurePathExists( cacheFile.GetFullPath() ) )
+    {
+        wxLogTrace( wxT( "KICAD_EMBED" ),
+                    wxT( "%s:%s:%d\n * failed to create embed cache directory '%s'" ),
+                    __FILE__, __FUNCTION__, __LINE__, cacheFile.GetPath() );
+
+        cacheFile.SetPath( wxFileName::GetTempDir() );
+    }
+
+    wxFileName inputName( aName );
+
+    // Store the cache file name using the data SHA to allow for shared data between
+    // multiple projects using the same files as well as deconflicting files with the same name
+    cacheFile.SetName( "kicad_embedded_" + it->second->data_sha );
+    cacheFile.SetExt( inputName.GetExt() );
+
+    if( cacheFile.FileExists() && cacheFile.IsFileReadable() )
+        return cacheFile;
+
+    wxFFileOutputStream out( cacheFile.GetFullPath() );
+
+    if( !out.IsOk() )
+    {
+        cacheFile.Clear();
+        return cacheFile;
+    }
+
+    out.Write( it->second->decompressedData.data(), it->second->decompressedData.size() );
+
+    return cacheFile;
+}
+
+
+const std::vector<wxString>* EMBEDDED_FILES::GetFontFiles() const
+{
+    return &m_fontFiles;
+}
+
+
+const std::vector<wxString>* EMBEDDED_FILES::UpdateFontFiles()
+{
+    m_fontFiles.clear();
+
+    for( const auto& [name, entry] : m_files )
+    {
+        if( entry->type == EMBEDDED_FILE::FILE_TYPE::FONT )
+            m_fontFiles.push_back( GetTempFileName( name ).GetFullPath() );
+    }
+
+    return &m_fontFiles;
+}
\ No newline at end of file
diff --git a/common/embedded_files.keywords b/common/embedded_files.keywords
new file mode 100644
index 0000000000..ebc07f6eb2
--- /dev/null
+++ b/common/embedded_files.keywords
@@ -0,0 +1,13 @@
+checksum
+data
+datasheet
+embedded_files
+embedded_fonts
+file
+font
+model
+name
+other
+reference
+type
+worksheet
\ No newline at end of file
diff --git a/common/filename_resolver.cpp b/common/filename_resolver.cpp
index 10f9637d5c..4de8f9910e 100644
--- a/common/filename_resolver.cpp
+++ b/common/filename_resolver.cpp
@@ -27,10 +27,12 @@
 #include <sstream>
 
 #include <wx/log.h>
+#include <wx/uri.h>
 #include <pgm_base.h>
 #include <trace_helpers.h>
 
 #include <common.h>
+#include <embedded_files.h>
 #include <env_vars.h>
 #include <filename_resolver.h>
 #include <confirm.h>
@@ -241,7 +243,8 @@ bool FILENAME_RESOLVER::UpdatePathList( const std::vector< SEARCH_PATH >& aPathL
 }
 
 
-wxString FILENAME_RESOLVER::ResolvePath( const wxString& aFileName, const wxString& aWorkingPath )
+wxString FILENAME_RESOLVER::ResolvePath( const wxString& aFileName, const wxString& aWorkingPath,
+                                         const EMBEDDED_FILES* aFiles )
 {
     std::lock_guard<std::mutex> lock( mutex_resolver );
 
@@ -260,6 +263,32 @@ wxString FILENAME_RESOLVER::ResolvePath( const wxString& aFileName, const wxStri
     // for getenv().
     tname = ExpandEnvVarSubstitutions( tname, m_project );
 
+    // Check to see if the file is a URI for an embedded file.
+    if( tname.StartsWith( FILEEXT::KiCadUriPrefix + "://" ) )
+    {
+        if( !aFiles )
+        {
+            wxLogTrace( wxT( "KICAD_EMBED" ),
+                        wxT( "No EMBEDDED_FILES object provided for kicad_embed URI" ) );
+            return wxEmptyString;
+        }
+
+        wxString path = tname.Mid( 14 );
+        wxFileName temp_file = aFiles->GetTempFileName( path );
+
+        if( !temp_file.IsOk() )
+        {
+            wxLogTrace( wxT( "KICAD_EMBED" ),
+                        wxT( "Failed to get temp file '%s' for kicad_embed URI" ), path );
+            return wxEmptyString;
+        }
+
+        wxLogTrace( wxT( "KICAD_EMBED" ), wxT( "Opening embedded file '%s' as '%s'" ),
+                    tname, temp_file.GetFullPath() );
+
+        return temp_file.GetFullPath();
+    }
+
     wxFileName tmpFN( tname );
 
     // this case covers full paths, leading expanded vars, and paths relative to the current
@@ -688,11 +717,23 @@ bool FILENAME_RESOLVER::ValidateFileName( const wxString& aFileName, bool& hasAl
     //    ALIAS:relative/path
     // 2. ALIAS is a UTF string excluding wxT( "{}[]()%~<>\"='`;:.,&?/\\|$" )
     // 3. The relative path must be a valid relative path for the platform
+    // 4. We allow a URI for embedded files, but only if it has a name
+
     hasAlias = false;
 
     if( aFileName.empty() )
         return false;
 
+    if( aFileName.StartsWith( wxT( "file://" ) )
+        || aFileName.StartsWith( FILEEXT::KiCadUriPrefix + "://" ) )
+    {
+        size_t prefixLength = aFileName.StartsWith( wxT( "file://" ) ) ? 7 : 14;
+        if( aFileName.length() > prefixLength && aFileName[prefixLength] != '/' )
+            return true;
+        else
+            return false;
+    }
+
     wxString filename = aFileName;
     wxString lpath;
     size_t aliasStart = aFileName.StartsWith( ':' ) ? 1 : 0;
diff --git a/common/font/font.cpp b/common/font/font.cpp
index a4012399e0..ad36d7706c 100644
--- a/common/font/font.cpp
+++ b/common/font/font.cpp
@@ -143,7 +143,7 @@ FONT* FONT::getDefaultFont()
 }
 
 
-FONT* FONT::GetFont( const wxString& aFontName, bool aBold, bool aItalic )
+FONT* FONT::GetFont( const wxString& aFontName, bool aBold, bool aItalic, const std::vector<wxString>* aEmbeddedFiles )
 {
     if( aFontName.empty() || aFontName.StartsWith( KICAD_FONT_NAME ) )
         return getDefaultFont();
@@ -156,7 +156,7 @@ FONT* FONT::GetFont( const wxString& aFontName, bool aBold, bool aItalic )
         font = s_fontMap[key];
 
     if( !font )
-        font = OUTLINE_FONT::LoadFont( aFontName, aBold, aItalic );
+        font = OUTLINE_FONT::LoadFont( aFontName, aBold, aItalic, aEmbeddedFiles );
 
     if( !font )
         font = getDefaultFont();
diff --git a/common/font/fontconfig.cpp b/common/font/fontconfig.cpp
index 8d3a0b6b2b..1e76177f75 100644
--- a/common/font/fontconfig.cpp
+++ b/common/font/fontconfig.cpp
@@ -27,6 +27,7 @@
 #include <macros.h>
 #include <cstdint>
 #include <reporter.h>
+#include <embedded_files.h>
 
 #ifdef __WIN32__
 #define WIN32_LEAN_AND_MEAN
@@ -196,14 +197,25 @@ std::string FONTCONFIG::getFamilyStringByLang( FONTCONFIG_PAT& aPat, const wxStr
 }
 
 
-FONTCONFIG::FF_RESULT FONTCONFIG::FindFont( const wxString &aFontName, wxString &aFontFile,
-                                            int& aFaceIndex, bool aBold, bool aItalic )
+FONTCONFIG::FF_RESULT FONTCONFIG::FindFont( const wxString& aFontName, wxString& aFontFile,
+                                            int& aFaceIndex, bool aBold, bool aItalic,
+                                            const std::vector<wxString>* aEmbeddedFiles )
 {
     FF_RESULT retval = FF_RESULT::FF_ERROR;
 
     if( !g_fcInitSuccess )
         return retval;
 
+    FcConfig* config = FcConfigGetCurrent();
+
+    if( aEmbeddedFiles )
+    {
+        for( const auto& file : *aEmbeddedFiles )
+        {
+            FcConfigAppFontAddFile( config, (const FcChar8*) file.c_str().AsChar() );
+        }
+    }
+
     wxString qualifiedFontName = aFontName;
 
     wxScopedCharBuffer const fcBuffer = qualifiedFontName.ToUTF8();
@@ -218,11 +230,11 @@ FONTCONFIG::FF_RESULT FONTCONFIG::FindFont( const wxString &aFontName, wxString
 
     FcPatternAddString( pat, FC_FAMILY, (FcChar8*) fcBuffer.data() );
 
-    FcConfigSubstitute( nullptr, pat, FcMatchPattern );
+    FcConfigSubstitute( config, pat, FcMatchPattern );
     FcDefaultSubstitute( pat );
 
     FcResult   r = FcResultNoMatch;
-    FcPattern* font = FcFontMatch( nullptr, pat, &r );
+    FcPattern* font = FcFontMatch( config, pat, &r );
 
     wxString fontName;
 
@@ -342,18 +354,29 @@ FONTCONFIG::FF_RESULT FONTCONFIG::FindFont( const wxString &aFontName, wxString
 }
 
 
-void FONTCONFIG::ListFonts( std::vector<std::string>& aFonts, const std::string& aDesiredLang )
+void FONTCONFIG::ListFonts( std::vector<std::string>& aFonts, const std::string& aDesiredLang,
+                            const std::vector<wxString>* aEmbeddedFiles, bool aForce )
 {
     if( !g_fcInitSuccess )
         return;
 
     // be sure to cache bust if the language changed
-    if( m_fontInfoCache.empty() || m_fontCacheLastLang != aDesiredLang )
+    if( m_fontInfoCache.empty() || m_fontCacheLastLang != aDesiredLang || aForce )
     {
+        FcConfig* config = FcConfigGetCurrent();
+
+        if( aEmbeddedFiles )
+        {
+            for( const auto& file : *aEmbeddedFiles )
+            {
+                FcConfigAppFontAddFile( config, (const FcChar8*) file.c_str().AsChar() );
+            }
+        }
+
         FcPattern*   pat = FcPatternCreate();
         FcObjectSet* os = FcObjectSetBuild( FC_FAMILY, FC_FAMILYLANG, FC_STYLE, FC_LANG, FC_FILE,
                                             FC_OUTLINE, nullptr );
-        FcFontSet*   fs = FcFontList( nullptr, pat, os );
+        FcFontSet*   fs = FcFontList( config, pat, os );
 
         for( int i = 0; fs && i < fs->nfont; ++i )
         {
diff --git a/common/font/outline_font.cpp b/common/font/outline_font.cpp
index 85a8337481..eb92518154 100644
--- a/common/font/outline_font.cpp
+++ b/common/font/outline_font.cpp
@@ -29,6 +29,10 @@
 #include <geometry/shape_poly_set.h>
 #include <font/fontconfig.h>
 #include <font/outline_font.h>
+#include <ft2build.h>
+#include FT_FREETYPE_H
+#include FT_SFNT_NAMES_H
+#include FT_TRUETYPE_TABLES_H
 #include FT_GLYPH_H
 #include FT_BBOX_H
 #include <trigo.h>
@@ -53,7 +57,33 @@ OUTLINE_FONT::OUTLINE_FONT() :
 }
 
 
-OUTLINE_FONT* OUTLINE_FONT::LoadFont( const wxString& aFontName, bool aBold, bool aItalic )
+OUTLINE_FONT::EMBEDDING_PERMISSION OUTLINE_FONT::GetEmbeddingPermission() const
+{
+    TT_OS2* os2 = reinterpret_cast<TT_OS2*>( FT_Get_Sfnt_Table( m_face, FT_SFNT_OS2 ) );
+
+    // If this table isn't present, we can't assume anything
+    if( !os2 )
+        return EMBEDDING_PERMISSION::RESTRICTED;
+
+    if( os2->fsType == FT_FSTYPE_INSTALLABLE_EMBEDDING ) // This allows the font to be exported from KiCad
+        return EMBEDDING_PERMISSION::INSTALLABLE;
+
+    if( os2->fsType & FT_FSTYPE_BITMAP_EMBEDDING_ONLY ) // We don't support bitmap fonts, so this disables embedding
+        return EMBEDDING_PERMISSION::RESTRICTED;
+
+    if( os2->fsType & FT_FSTYPE_EDITABLE_EMBEDDING ) // This allows us to use the font in KiCad but not export
+        return EMBEDDING_PERMISSION::EDITABLE;
+
+    if( os2->fsType & FT_FSTYPE_PREVIEW_AND_PRINT_EMBEDDING ) // This is not actually supported by KiCad ATM(2024)
+        return EMBEDDING_PERMISSION::PRINT_PREVIEW_ONLY;
+
+    // Anything else that is not explicitly enabled we treat as restricted.
+    return EMBEDDING_PERMISSION::RESTRICTED;
+}
+
+
+OUTLINE_FONT* OUTLINE_FONT::LoadFont( const wxString& aFontName, bool aBold, bool aItalic,
+                                      const std::vector<wxString>* aEmbeddedFiles )
 {
     std::unique_ptr<OUTLINE_FONT> font = std::make_unique<OUTLINE_FONT>();
 
@@ -61,7 +91,9 @@ OUTLINE_FONT* OUTLINE_FONT::LoadFont( const wxString& aFontName, bool aBold, boo
     int      faceIndex;
     using fc = fontconfig::FONTCONFIG;
 
-    fc::FF_RESULT retval = Fontconfig()->FindFont( aFontName, fontFile, faceIndex, aBold, aItalic );
+
+    fc::FF_RESULT retval = Fontconfig()->FindFont( aFontName, fontFile, faceIndex, aBold, aItalic,
+                                                   aEmbeddedFiles );
 
     if( retval == fc::FF_RESULT::FF_ERROR )
         return nullptr;
diff --git a/common/io/altium/altium_binary_parser.cpp b/common/io/altium/altium_binary_parser.cpp
index 9654c6d1e1..bac282e7eb 100644
--- a/common/io/altium/altium_binary_parser.cpp
+++ b/common/io/altium/altium_binary_parser.cpp
@@ -149,30 +149,6 @@ ALTIUM_COMPOUND_FILE::DecodeIntLibStream( const CFB::COMPOUND_FILE_ENTRY& cfe )
 }
 
 
-std::map<wxString, wxString> ALTIUM_COMPOUND_FILE::ListLibFootprints()
-{
-    if( m_libFootprintDirNameCache.empty() )
-        cacheLibFootprintNames();
-
-    return m_libFootprintDirNameCache;
-}
-
-
-std::tuple<wxString, const CFB::COMPOUND_FILE_ENTRY*>
-ALTIUM_COMPOUND_FILE::FindLibFootprintDirName( const wxString& aFpUnicodeName )
-{
-    if( m_libFootprintNameCache.empty() )
-        cacheLibFootprintNames();
-
-    auto it = m_libFootprintNameCache.find( aFpUnicodeName );
-
-    if( it == m_libFootprintNameCache.end() )
-        return { wxEmptyString, nullptr };
-
-    return { it->first, it->second };
-}
-
-
 const CFB::COMPOUND_FILE_ENTRY*
 ALTIUM_COMPOUND_FILE::FindStreamSingleLevel( const CFB::COMPOUND_FILE_ENTRY* aEntry,
                                              const std::string aName, const bool aIsStream ) const
@@ -333,53 +309,6 @@ ALTIUM_COMPOUND_FILE::FindStream( const std::vector<std::string>& aStreamPath )
 }
 
 
-void ALTIUM_COMPOUND_FILE::cacheLibFootprintNames()
-{
-    m_libFootprintDirNameCache.clear();
-    m_libFootprintNameCache.clear();
-
-    if( !m_reader )
-        return;
-
-    const CFB::COMPOUND_FILE_ENTRY* root = m_reader->GetRootEntry();
-
-    if( !root )
-        return;
-
-    m_reader->EnumFiles( root, 1,
-                        [this]( const CFB::COMPOUND_FILE_ENTRY* tentry, const CFB::utf16string& dir,
-                            int level ) -> int
-                        {
-                            if( m_reader->IsStream( tentry ) )
-                                return 0;
-
-                            m_reader->EnumFiles( tentry, 1,
-                                        [&]( const CFB::COMPOUND_FILE_ENTRY* entry, const CFB::utf16string&, int ) -> int
-                                        {
-                                            std::wstring fileName = UTF16ToWstring( entry->name, entry->nameLen );
-
-                                            if( m_reader->IsStream( entry ) && fileName == L"Parameters" )
-                                            {
-                                                ALTIUM_BINARY_PARSER                parametersReader( *this, entry );
-                                                std::map<wxString, wxString> parameterProperties =
-                                                        parametersReader.ReadProperties();
-
-                                                wxString key = ALTIUM_PROPS_UTILS::ReadString(
-                                                        parameterProperties, wxT( "PATTERN" ), wxT( "" ) );
-                                                wxString fpName = ALTIUM_PROPS_UTILS::ReadUnicodeString(
-                                                        parameterProperties, wxT( "PATTERN" ), wxT( "" ) );
-
-                                                m_libFootprintDirNameCache[key] = fpName;
-                                                m_libFootprintNameCache[fpName] = tentry;
-                                            }
-
-                                            return 0;
-                                        } );
-                            return 0;
-                         } );
-}
-
-
 ALTIUM_BINARY_PARSER::ALTIUM_BINARY_PARSER( const ALTIUM_COMPOUND_FILE&     aFile,
                               const CFB::COMPOUND_FILE_ENTRY* aEntry )
 {
diff --git a/common/io/altium/altium_binary_parser.h b/common/io/altium/altium_binary_parser.h
index 6e2796daaf..a0ed8c693d 100644
--- a/common/io/altium/altium_binary_parser.h
+++ b/common/io/altium/altium_binary_parser.h
@@ -38,6 +38,7 @@
 #include <wx/mstream.h>
 #include <wx/zstream.h>
 #include <math/vector2d.h>
+#include <math/vector3.h>
 
 namespace CFB
 {
@@ -63,8 +64,11 @@ public:
     const CFB::COMPOUND_FILE_ENTRY* m_pinsSymbolLineWidth;
 };
 
+
 class ALTIUM_COMPOUND_FILE
 {
+    friend class ALTIUM_PCB_COMPOUND_FILE;
+
 public:
     /**
      * Open a CFB file. Constructor might throw an IO_ERROR.
@@ -90,10 +94,6 @@ public:
 
     std::unique_ptr<ALTIUM_COMPOUND_FILE> DecodeIntLibStream( const CFB::COMPOUND_FILE_ENTRY& cfe );
 
-    std::map<wxString, wxString> ListLibFootprints();
-
-    std::tuple<wxString, const CFB::COMPOUND_FILE_ENTRY*> FindLibFootprintDirName( const wxString& aFpUnicodeName );
-
     const CFB::COMPOUND_FILE_ENTRY* FindStream( const std::vector<std::string>& aStreamPath ) const;
 
     const CFB::COMPOUND_FILE_ENTRY* FindStream( const CFB::COMPOUND_FILE_ENTRY* aStart, const std::vector<std::string>& aStreamPath ) const;
@@ -108,13 +108,8 @@ public:
 
 private:
 
-    void cacheLibFootprintNames();
-
     std::unique_ptr<CFB::CompoundFileReader> m_reader;
     std::vector<char>                        m_buffer;
-
-    std::map<wxString, const CFB::COMPOUND_FILE_ENTRY*> m_libFootprintNameCache;
-    std::map<wxString, wxString>                        m_libFootprintDirNameCache;
 };
 
 
diff --git a/common/pcb.keywords b/common/pcb.keywords
index 204393627b..7088b48926 100644
--- a/common/pcb.keywords
+++ b/common/pcb.keywords
@@ -106,6 +106,8 @@ edge_connector
 edge_plating
 edge_width
 effects
+embedded_fonts
+embedded_files
 enabled
 end
 epsilon_r
diff --git a/common/tool/actions.cpp b/common/tool/actions.cpp
index 0945ac8a91..eee6d9ee2e 100644
--- a/common/tool/actions.cpp
+++ b/common/tool/actions.cpp
@@ -1184,6 +1184,26 @@ TOOL_ACTION ACTIONS::pluginsReload( TOOL_ACTION_ARGS()
         .Tooltip( _( "Reload all python plugins and refresh plugin menus" ) )
         .Icon( BITMAPS::reload ) );
 
+// Embedding Files
+
+TOOL_ACTION ACTIONS::embeddedFiles( TOOL_ACTION_ARGS()
+        .Name( "common.Embed.embededFile" )
+        .Scope( AS_GLOBAL )
+        .FriendlyName( _( "Embedded Files" ) )
+        .Tooltip( _( "Manage embedded files" ) ) );
+
+TOOL_ACTION ACTIONS::removeFile( TOOL_ACTION_ARGS()
+        .Name( "common.Embed.removeFile" )
+        .Scope( AS_GLOBAL )
+        .FriendlyName( _( "Remove File" ) )
+        .Tooltip( _( "Remove an embedded file" ) ) );
+
+TOOL_ACTION ACTIONS::extractFile( TOOL_ACTION_ARGS()
+        .Name( "common.Embed.extractFile" )
+        .Scope( AS_GLOBAL )
+        .FriendlyName( _( "Extract File" ) )
+        .Tooltip( _( "Extract an embedded file" ) ) );
+
 // System-wide selection Events
 
 const TOOL_EVENT EVENTS::PointSelectedEvent( TC_MESSAGE, TA_ACTION, "common.Interactive.pointSelected" );
diff --git a/common/tool/embed_tool.cpp b/common/tool/embed_tool.cpp
new file mode 100644
index 0000000000..48f7e93582
--- /dev/null
+++ b/common/tool/embed_tool.cpp
@@ -0,0 +1,92 @@
+/*
+ * This program source code file is part of KiCad, a free EDA CAD application.
+ *
+ * Copyright (C) 2024 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 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, see <http://www.gnu.org/licenses/>.
+ */
+
+#include <dialogs/dialog_embed_files.h>
+#include <dialogs/panel_embedded_files.h>
+#include <eda_draw_frame.h>
+#include <eda_item.h>
+#include <embedded_files.h>
+#include <tool/actions.h>
+#include <wx/debug.h>
+#include <wx/filedlg.h>
+
+#include <tool/embed_tool.h>
+
+
+EMBED_TOOL::EMBED_TOOL( const std::string& aName ) :
+        TOOL_INTERACTIVE( aName )
+{
+}
+
+
+EMBED_TOOL::EMBED_TOOL() :
+        TOOL_INTERACTIVE( "common.Embed" )
+{
+}
+
+
+bool EMBED_TOOL::Init()
+{
+
+    m_files = getModel<EDA_ITEM>()->GetEmbeddedFiles();
+
+    return true;
+}
+
+
+void EMBED_TOOL::Reset( RESET_REASON aReason )
+{
+    m_files = getModel<EDA_ITEM>()->GetEmbeddedFiles();
+
+}
+
+int EMBED_TOOL::AddFile( const TOOL_EVENT& aEvent )
+{
+    wxString name = aEvent.Parameter<wxString>();
+    m_files->AddFile( name, false );
+
+    return 1;
+}
+
+int EMBED_TOOL::RemoveFile( const TOOL_EVENT& aEvent )
+{
+    wxString name = aEvent.Parameter<wxString>();
+    m_files->RemoveFile( name );
+
+    return 1;
+}
+
+std::vector<wxString> EMBED_TOOL::GetFileList()
+{
+    std::vector<wxString> list;
+
+    for( auto& [name, file] : m_files->EmbeddedFileMap() )
+    {
+        list.push_back( name );
+    }
+
+    return list;
+}
+
+void EMBED_TOOL::setTransitions()
+{
+    Go( &EMBED_TOOL::AddFile, ACTIONS::embeddedFiles.MakeEvent() );
+    Go( &EMBED_TOOL::RemoveFile, ACTIONS::removeFile.MakeEvent() );
+}
+
diff --git a/common/widgets/grid_text_button_helpers.cpp b/common/widgets/grid_text_button_helpers.cpp
index 9fbe221d9a..908885cb0c 100644
--- a/common/widgets/grid_text_button_helpers.cpp
+++ b/common/widgets/grid_text_button_helpers.cpp
@@ -22,12 +22,14 @@
  * 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA
  */
 
+#include <wx/checkbox.h>
 #include <wx/combo.h>
 #include <wx/filedlg.h>
 #include <wx/dirdlg.h>
 #include <wx/textctrl.h>
 
 #include <bitmaps.h>
+#include <embedded_files.h>
 #include <kiway.h>
 #include <kiway_player.h>
 #include <kiway_express.h>
@@ -37,6 +39,7 @@
 #include <env_paths.h>
 #include <pgm_base.h>
 #include <widgets/wx_grid.h>
+#include <widgets/filedlg_open_embed_file.h>
 #include <widgets/grid_text_button_helpers.h>
 #include <eda_doc.h>
 
@@ -323,16 +326,25 @@ void GRID_CELL_FPID_EDITOR::Create( wxWindow* aParent, wxWindowID aId,
 class TEXT_BUTTON_URL : public wxComboCtrl
 {
 public:
-    TEXT_BUTTON_URL( wxWindow* aParent, DIALOG_SHIM* aParentDlg, SEARCH_STACK* aSearchStack ) :
+    TEXT_BUTTON_URL( wxWindow* aParent, DIALOG_SHIM* aParentDlg, SEARCH_STACK* aSearchStack, EMBEDDED_FILES* aFiles ) :
             wxComboCtrl( aParent, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize,
                          wxTE_PROCESS_ENTER | wxBORDER_NONE ),
             m_dlg( aParentDlg ),
-            m_searchStack( aSearchStack )
+            m_searchStack( aSearchStack ),
+            m_files( aFiles )
     {
-        SetButtonBitmaps( KiBitmapBundle( BITMAPS::www ) );
+        UpdateButtonBitmaps();
 
         // win32 fix, avoids drawing the "native dropdown caret"
         Customize( wxCC_IFLAG_HAS_NONSTANDARD_BUTTON );
+
+        // Bind event to handle text changes
+        Bind(wxEVT_TEXT, &TEXT_BUTTON_URL::OnTextChange, this);
+    }
+
+    ~TEXT_BUTTON_URL()
+    {
+        Unbind(wxEVT_TEXT, &TEXT_BUTTON_URL::OnTextChange, this);
     }
 
 protected:
@@ -345,19 +357,59 @@ protected:
     {
         wxString filename = GetValue();
 
-        if( !filename.IsEmpty() && filename != wxT( "~" ) )
-            GetAssociatedDocument( m_dlg, GetValue(), &m_dlg->Prj(), m_searchStack );
+        if (filename.IsEmpty() || filename == wxT("~"))
+        {
+            FILEDLG_OPEN_EMBED_FILE customize;
+            wxFileDialog openFileDialog( this, _( "Open file" ), "", "", "All files (*.*)|*.*",
+                                         wxFD_OPEN | wxFD_FILE_MUST_EXIST );
+            openFileDialog.SetCustomizeHook( customize );
+
+            if( openFileDialog.ShowModal() == wxID_OK )
+            {
+                filename = openFileDialog.GetPath();
+                wxFileName fn( filename );
+
+                if( customize.GetEmbed() )
+                {
+                    EMBEDDED_FILES::EMBEDDED_FILE* result = m_files->AddFile( fn, false );
+                    SetValue( result->GetLink() );
+                }
+                else
+                {
+                    SetValue( "file://" + filename );
+                }
+            }
+        }
+        else
+        {
+            GetAssociatedDocument(m_dlg, GetValue(), &m_dlg->Prj(), m_searchStack, m_files);
+        }
+    }
+
+    void OnTextChange(wxCommandEvent& event)
+    {
+        UpdateButtonBitmaps();
+        event.Skip(); // Ensure that other handlers can process this event too
+    }
+
+    void UpdateButtonBitmaps()
+    {
+        if (GetValue().IsEmpty())
+            SetButtonBitmaps(KiBitmapBundle(BITMAPS::small_folder));
+        else
+            SetButtonBitmaps(KiBitmapBundle(BITMAPS::www));
     }
 
     DIALOG_SHIM* m_dlg;
     SEARCH_STACK* m_searchStack;
+    EMBEDDED_FILES* m_files;
 };
 
 
 void GRID_CELL_URL_EDITOR::Create( wxWindow* aParent, wxWindowID aId,
                                    wxEvtHandler* aEventHandler )
 {
-    m_control = new TEXT_BUTTON_URL( aParent, m_dlg, m_searchStack );
+    m_control = new TEXT_BUTTON_URL( aParent, m_dlg, m_searchStack, m_files );
     WX_GRID::CellEditorSetMargins( Combo() );
 
 #if wxUSE_VALIDATORS
diff --git a/common/widgets/wx_grid.cpp b/common/widgets/wx_grid.cpp
index 50d4470895..88dbee458e 100644
--- a/common/widgets/wx_grid.cpp
+++ b/common/widgets/wx_grid.cpp
@@ -154,7 +154,7 @@ class WX_GRID_ALT_ROW_COLOR_PROVIDER : public wxGridCellAttrProvider
 {
 public:
     WX_GRID_ALT_ROW_COLOR_PROVIDER( const wxColor& aBaseColor ) : wxGridCellAttrProvider(),
-        m_attrOdd( new wxGridCellAttr() )
+        m_attrEven( new wxGridCellAttr() )
     {
         UpdateColors( aBaseColor );
     }
@@ -165,7 +165,7 @@ public:
         // Choose the default color, taking into account if the dark mode theme is enabled
         wxColor rowColor = aBaseColor.ChangeLightness( KIPLATFORM::UI::IsDarkTheme() ? 105 : 95 );
 
-        m_attrOdd->SetBackgroundColour( rowColor );
+        m_attrEven->SetBackgroundColour( rowColor );
     }
 
 
@@ -174,20 +174,20 @@ public:
     {
         wxGridCellAttrPtr cellAttr( wxGridCellAttrProvider::GetAttr( row, col, kind ) );
 
-        // Just pass through the cell attribute on even rows
-        if( row % 2 )
+        // Just pass through the cell attribute on odd rows (start normal to allow for the header row)
+        if( !( row % 2 ) )
             return cellAttr.release();
 
         if( !cellAttr )
         {
-            cellAttr = m_attrOdd;
+            cellAttr = m_attrEven;
         }
         else
         {
             if( !cellAttr->HasBackgroundColour() )
             {
                 cellAttr = cellAttr->Clone();
-                cellAttr->SetBackgroundColour( m_attrOdd->GetBackgroundColour() );
+                cellAttr->SetBackgroundColour( m_attrEven->GetBackgroundColour() );
             }
         }
 
@@ -195,7 +195,7 @@ public:
     }
 
 private:
-    wxGridCellAttrPtr m_attrOdd;
+    wxGridCellAttrPtr m_attrEven;
 };
 
 
diff --git a/common/wildcards_and_files_ext.cpp b/common/wildcards_and_files_ext.cpp
index 7b261d6e2b..cc6c1cbe46 100644
--- a/common/wildcards_and_files_ext.cpp
+++ b/common/wildcards_and_files_ext.cpp
@@ -202,6 +202,8 @@ const std::string FILEEXT::XaoFileExtension( "xao" );
 
 const wxString FILEEXT::GerberFileExtensionsRegex( "(gbr|gko|pho|(g[tb][alops])|(gm?\\d\\d*)|(gp[tb]))" );
 
+const std::string FILEEXT::KiCadUriPrefix( "kicad-embed" );
+
 
 bool FILEEXT::IsGerberFileExtension( const wxString& ext )
 {
diff --git a/eeschema/dialogs/dialog_eeschema_page_settings.cpp b/eeschema/dialogs/dialog_eeschema_page_settings.cpp
index fa2c13adc9..fa6d32dc4f 100644
--- a/eeschema/dialogs/dialog_eeschema_page_settings.cpp
+++ b/eeschema/dialogs/dialog_eeschema_page_settings.cpp
@@ -26,9 +26,9 @@
 #include <schematic.h>
 #include <eeschema_settings.h>
 
-DIALOG_EESCHEMA_PAGE_SETTINGS::DIALOG_EESCHEMA_PAGE_SETTINGS( EDA_DRAW_FRAME* aParent,
+DIALOG_EESCHEMA_PAGE_SETTINGS::DIALOG_EESCHEMA_PAGE_SETTINGS( EDA_DRAW_FRAME* aParent, EMBEDDED_FILES* aEmbeddedFiles,
                                                               VECTOR2I        aMaxUserSizeMils ) :
-        DIALOG_PAGES_SETTINGS( aParent, schIUScale.IU_PER_MILS, aMaxUserSizeMils )
+        DIALOG_PAGES_SETTINGS( aParent, aEmbeddedFiles, schIUScale.IU_PER_MILS, aMaxUserSizeMils )
 {
 }
 
diff --git a/eeschema/dialogs/dialog_eeschema_page_settings.h b/eeschema/dialogs/dialog_eeschema_page_settings.h
index c87f85a43b..06f729de77 100644
--- a/eeschema/dialogs/dialog_eeschema_page_settings.h
+++ b/eeschema/dialogs/dialog_eeschema_page_settings.h
@@ -22,11 +22,13 @@
 
 #include <dialogs/dialog_page_settings.h>
 
+class EMBEDDED_FILES;
+
 
 class DIALOG_EESCHEMA_PAGE_SETTINGS : public DIALOG_PAGES_SETTINGS
 {
 public:
-    DIALOG_EESCHEMA_PAGE_SETTINGS( EDA_DRAW_FRAME* aParent, VECTOR2I aMaxUserSizeMils );
+    DIALOG_EESCHEMA_PAGE_SETTINGS( EDA_DRAW_FRAME* aParent, EMBEDDED_FILES* aEmbeddedFiles, VECTOR2I aMaxUserSizeMils );
     virtual ~DIALOG_EESCHEMA_PAGE_SETTINGS();
 
 private:
diff --git a/eeschema/dialogs/dialog_label_properties.cpp b/eeschema/dialogs/dialog_label_properties.cpp
index 725d94f680..262679f39a 100644
--- a/eeschema/dialogs/dialog_label_properties.cpp
+++ b/eeschema/dialogs/dialog_label_properties.cpp
@@ -108,7 +108,7 @@ DIALOG_LABEL_PROPERTIES::DIALOG_LABEL_PROPERTIES( SCH_EDIT_FRAME* aParent, SCH_L
     m_grid->SetDefaultRowSize( m_grid->GetDefaultRowSize() + 4 );
 
     m_grid->SetTable( m_fields );
-    m_grid->PushEventHandler( new FIELDS_GRID_TRICKS( m_grid, this,
+    m_grid->PushEventHandler( new FIELDS_GRID_TRICKS( m_grid, this, nullptr,
                                                       [&]( wxCommandEvent& aEvent )
                                                       {
                                                           OnAddField( aEvent );
diff --git a/eeschema/dialogs/dialog_lib_symbol_properties.cpp b/eeschema/dialogs/dialog_lib_symbol_properties.cpp
index 16fb0c3cee..73750f43a3 100644
--- a/eeschema/dialogs/dialog_lib_symbol_properties.cpp
+++ b/eeschema/dialogs/dialog_lib_symbol_properties.cpp
@@ -40,6 +40,7 @@
 
 #include <dialog_sim_model.h>
 
+#include <panel_embedded_files.h>
 #include <dialog_lib_symbol_properties.h>
 #include <settings/settings_manager.h>
 #include <symbol_editor_settings.h>
@@ -63,11 +64,14 @@ DIALOG_LIB_SYMBOL_PROPERTIES::DIALOG_LIB_SYMBOL_PROPERTIES( SYMBOL_EDIT_FRAME* a
     m_delayedFocusColumn( -1 ),
     m_delayedFocusPage( -1 )
 {
+    m_embeddedFiles = new PANEL_EMBEDDED_FILES( m_NoteBook, m_libEntry );
+    m_NoteBook->AddPage( m_embeddedFiles, _( "Embedded Files" ) );
+
     // Give a bit more room for combobox editors
     m_grid->SetDefaultRowSize( m_grid->GetDefaultRowSize() + 4 );
     m_fields = new FIELDS_GRID_TABLE( this, aParent, m_grid, m_libEntry );
     m_grid->SetTable( m_fields );
-    m_grid->PushEventHandler( new FIELDS_GRID_TRICKS( m_grid, this,
+    m_grid->PushEventHandler( new FIELDS_GRID_TRICKS( m_grid, this, aLibEntry,
                                                       [&]( wxCommandEvent& aEvent )
                                                       {
                                                           OnAddField( aEvent );
@@ -79,7 +83,7 @@ DIALOG_LIB_SYMBOL_PROPERTIES::DIALOG_LIB_SYMBOL_PROPERTIES( SYMBOL_EDIT_FRAME* a
     m_grid->ShowHideColumns( cfg->m_EditSymbolVisibleColumns );
 
     wxGridCellAttr* attr = new wxGridCellAttr;
-    attr->SetEditor( new GRID_CELL_URL_EDITOR( this, PROJECT_SCH::SchSearchS( &Prj() ) ) );
+    attr->SetEditor( new GRID_CELL_URL_EDITOR( this, PROJECT_SCH::SchSearchS( &Prj() ), aLibEntry ) );
     m_grid->SetAttr( DATASHEET_FIELD, FDC_VALUE, attr );
 
     m_SymbolNameCtrl->SetValidator( FIELD_VALIDATOR( VALUE_FIELD ) );
@@ -257,6 +261,8 @@ bool DIALOG_LIB_SYMBOL_PROPERTIES::TransferDataToWindow()
 
     m_NoteBook->SetSelection( (unsigned) m_lastOpenedPage );
 
+    m_embeddedFiles->TransferDataToWindow();
+
     return true;
 }
 
@@ -469,6 +475,7 @@ bool DIALOG_LIB_SYMBOL_PROPERTIES::TransferDataFromWindow()
     // occurs.
     m_Parent->SetShowDeMorgan( m_hasAlternateBodyStyles->GetValue() );
 
+    m_embeddedFiles->TransferDataFromWindow();
     return true;
 }
 
diff --git a/eeschema/dialogs/dialog_lib_symbol_properties.h b/eeschema/dialogs/dialog_lib_symbol_properties.h
index 07b6be6978..fb55767d23 100644
--- a/eeschema/dialogs/dialog_lib_symbol_properties.h
+++ b/eeschema/dialogs/dialog_lib_symbol_properties.h
@@ -32,6 +32,7 @@
 
 class SYMBOL_EDIT_FRAME;
 class LIB_SYMBOL;
+class PANEL_EMBEDDED_FILES;
 class WX_GRID;
 
 
@@ -93,6 +94,8 @@ public:
     std::bitset<64>    m_shownColumns;
     wxSize             m_size;
 
+    PANEL_EMBEDDED_FILES* m_embeddedFiles;
+
 private:
     static int m_lastOpenedPage;    // To remember the last notebook selection
 
diff --git a/eeschema/dialogs/dialog_schematic_setup.cpp b/eeschema/dialogs/dialog_schematic_setup.cpp
index 8834a90e5a..10140d818d 100644
--- a/eeschema/dialogs/dialog_schematic_setup.cpp
+++ b/eeschema/dialogs/dialog_schematic_setup.cpp
@@ -30,6 +30,7 @@
 #include <erc/erc_item.h>
 #include <panel_text_variables.h>
 #include <panel_bom_presets.h>
+#include <panel_embedded_files.h>
 #include <project/project_file.h>
 #include <project/net_settings.h>
 #include <settings/settings_manager.h>
@@ -121,6 +122,16 @@ DIALOG_SCHEMATIC_SETUP::DIALOG_SCHEMATIC_SETUP( SCH_EDIT_FRAME* aFrame ) :
                 return new PANEL_TEXT_VARIABLES( aParent, &Prj() );
             }, _( "Text Variables" ) );
 
+
+    m_treebook->AddPage( new wxPanel( GetTreebook() ), _( "Schematic Data" ) );
+
+    m_embeddedFilesPage = m_treebook->GetPageCount();
+    m_treebook->AddLazySubPage(
+            [this]( wxWindow* aParent ) -> wxWindow*
+            {
+                return new PANEL_EMBEDDED_FILES( aParent, &m_frame->Schematic() );
+            }, _( "Embedded Files" ) );
+
     for( size_t i = 0; i < m_treebook->GetPageCount(); ++i )
         m_treebook->ExpandNode( i );
 
diff --git a/eeschema/dialogs/dialog_schematic_setup.h b/eeschema/dialogs/dialog_schematic_setup.h
index 7e2fe413e0..1f4b5ab84c 100644
--- a/eeschema/dialogs/dialog_schematic_setup.h
+++ b/eeschema/dialogs/dialog_schematic_setup.h
@@ -56,6 +56,7 @@ protected:
     size_t                    m_busesPage;
     size_t                    m_severitiesPage;
     size_t                    m_netclassesPage;
+    size_t                    m_embeddedFilesPage;
 };
 
 
diff --git a/eeschema/dialogs/dialog_sheet_properties.cpp b/eeschema/dialogs/dialog_sheet_properties.cpp
index a15ad4d0c4..057ecf33c2 100644
--- a/eeschema/dialogs/dialog_sheet_properties.cpp
+++ b/eeschema/dialogs/dialog_sheet_properties.cpp
@@ -67,7 +67,7 @@ DIALOG_SHEET_PROPERTIES::DIALOG_SHEET_PROPERTIES( SCH_EDIT_FRAME* aParent, SCH_S
     m_grid->SetDefaultRowSize( m_grid->GetDefaultRowSize() + 4 );
 
     m_grid->SetTable( m_fields );
-    m_grid->PushEventHandler( new FIELDS_GRID_TRICKS( m_grid, this,
+    m_grid->PushEventHandler( new FIELDS_GRID_TRICKS( m_grid, this, &aParent->Schematic(),
                                                       [&]( wxCommandEvent& aEvent )
                                                       {
                                                           OnAddField( aEvent );
diff --git a/eeschema/dialogs/dialog_symbol_fields_table.cpp b/eeschema/dialogs/dialog_symbol_fields_table.cpp
index a7dfe97db7..1e1667d867 100644
--- a/eeschema/dialogs/dialog_symbol_fields_table.cpp
+++ b/eeschema/dialogs/dialog_symbol_fields_table.cpp
@@ -343,7 +343,8 @@ void DIALOG_SYMBOL_FIELDS_TABLE::SetupColumnProperties( int aCol )
     else if( m_dataModel->GetColFieldName( aCol ) == GetCanonicalFieldName( DATASHEET_FIELD ) )
     {
         // set datasheet column viewer button
-        attr->SetEditor( new GRID_CELL_URL_EDITOR( this, PROJECT_SCH::SchSearchS( &Prj() ) ) );
+        attr->SetEditor( new GRID_CELL_URL_EDITOR( this, PROJECT_SCH::SchSearchS( &Prj() ),
+                                                   &m_parent->Schematic() ) );
         m_grid->SetColAttr( aCol, attr );
     }
     else if( m_dataModel->ColIsQuantity( aCol ) || m_dataModel->ColIsItemNumber( aCol ) )
diff --git a/eeschema/dialogs/dialog_symbol_properties.cpp b/eeschema/dialogs/dialog_symbol_properties.cpp
index a26d20f3b9..14a17d8ba1 100644
--- a/eeschema/dialogs/dialog_symbol_properties.cpp
+++ b/eeschema/dialogs/dialog_symbol_properties.cpp
@@ -334,7 +334,7 @@ DIALOG_SYMBOL_PROPERTIES::DIALOG_SYMBOL_PROPERTIES( SCH_EDIT_FRAME* aParent,
     m_pinGrid->SetDefaultRowSize( m_pinGrid->GetDefaultRowSize() + 4 );
 
     m_fieldsGrid->SetTable( m_fields );
-    m_fieldsGrid->PushEventHandler( new FIELDS_GRID_TRICKS( m_fieldsGrid, this,
+    m_fieldsGrid->PushEventHandler( new FIELDS_GRID_TRICKS( m_fieldsGrid, this, &aParent->Schematic(),
                                                             [&]( wxCommandEvent& aEvent )
                                                             {
                                                                 OnAddField( aEvent );
diff --git a/eeschema/eeschema_config.cpp b/eeschema/eeschema_config.cpp
index 6226561554..6e399fc868 100644
--- a/eeschema/eeschema_config.cpp
+++ b/eeschema/eeschema_config.cpp
@@ -26,6 +26,7 @@
 #include <kiway.h>
 #include <symbol_edit_frame.h>
 #include <dialogs/panel_gal_display_options.h>
+#include <filename_resolver.h>
 #include <pgm_base.h>
 #include <project/project_file.h>
 #include <project/project_local_settings.h>
@@ -69,14 +70,6 @@ bool SCH_EDIT_FRAME::LoadProjectSettings()
 
     BASE_SCREEN::m_DrawingSheetFileName = settings.m_SchDrawingSheetFileName;
 
-    // Load the drawing sheet from the filename stored in BASE_SCREEN::m_DrawingSheetFileName.
-    // If empty, or not existing, the default drawing sheet is loaded.
-    wxString filename = DS_DATA_MODEL::ResolvePath( BASE_SCREEN::m_DrawingSheetFileName,
-                                                    Prj().GetProjectPath() );
-
-    if( !DS_DATA_MODEL::GetTheInstance().LoadDrawingSheet( filename, nullptr ) )
-        ShowInfoBarError( _( "Error loading drawing sheet." ), true );
-
     PROJECT_LOCAL_SETTINGS& localSettings = Prj().GetLocalSettings();
 
     EE_SELECTION_TOOL* selTool = GetToolManager()->GetTool<EE_SELECTION_TOOL>();
@@ -87,6 +80,26 @@ bool SCH_EDIT_FRAME::LoadProjectSettings()
 }
 
 
+void SCH_EDIT_FRAME::LoadDrawingSheet()
+{
+    // Load the drawing sheet from the filename stored in BASE_SCREEN::m_DrawingSheetFileName.
+    // If empty, or not existing, the default drawing sheet is loaded.
+
+    SCHEMATIC_SETTINGS& settings = Schematic().Settings();
+    FILENAME_RESOLVER resolver;
+    resolver.SetProject( &Prj() );
+    resolver.SetProgramBase( &Pgm() );
+
+    wxString filename = resolver.ResolvePath( settings.m_SchDrawingSheetFileName,
+                                              Prj().GetProjectPath(),
+                                              Schematic().GetEmbeddedFiles() );
+    wxString msg;
+
+    if( !DS_DATA_MODEL::GetTheInstance().LoadDrawingSheet( filename, &msg ) )
+        ShowInfoBarError( msg, true );
+}
+
+
 void SCH_EDIT_FRAME::ShowSchematicSetupDialog( const wxString& aInitialPage )
 {
     SCH_SCREENS screens( Schematic().Root() );
@@ -177,15 +190,20 @@ void SCH_EDIT_FRAME::saveProjectSettings()
 
     if( !BASE_SCREEN::m_DrawingSheetFileName.IsEmpty() )
     {
-        wxFileName layoutfn( DS_DATA_MODEL::ResolvePath( BASE_SCREEN::m_DrawingSheetFileName,
-                                                         Prj().GetProjectPath() ) );
+        FILENAME_RESOLVER resolve;
+        resolve.SetProject( &Prj() );
+        resolve.SetProgramBase( &Pgm() );
+
+        wxFileName layoutfn( resolve.ResolvePath( BASE_SCREEN::m_DrawingSheetFileName,
+                                                  Prj().GetProjectPath(),
+                                                  Schematic().GetEmbeddedFiles() ) );
 
         bool success = true;
 
         if( !layoutfn.IsAbsolute() )
             success = layoutfn.MakeAbsolute( Prj().GetProjectPath() );
 
-        if( success && layoutfn.IsOk() && !layoutfn.FileExists() )
+        if( success && layoutfn.IsOk() && !layoutfn.FileExists() && layoutfn.HasName() )
         {
             if( layoutfn.DirExists() && layoutfn.IsDirWritable() )
                 DS_DATA_MODEL::GetTheInstance().Save( layoutfn.GetFullPath() );
diff --git a/eeschema/eeschema_jobs_handler.cpp b/eeschema/eeschema_jobs_handler.cpp
index 0ad6b7beef..5176cf8d3b 100644
--- a/eeschema/eeschema_jobs_handler.cpp
+++ b/eeschema/eeschema_jobs_handler.cpp
@@ -37,6 +37,7 @@
 #include <memory>
 #include <connection_graph.h>
 #include "eeschema_helpers.h"
+#include <filename_resolver.h>
 #include <kiway.h>
 #include <sch_painter.h>
 #include <locale_io.h>
@@ -111,9 +112,14 @@ void EESCHEMA_JOBS_HANDLER::InitRenderSettings( SCH_RENDER_SETTINGS* aRenderSett
     auto loadSheet =
             [&]( const wxString& path ) -> bool
             {
-                wxString absolutePath = DS_DATA_MODEL::ResolvePath( path,
-                                                                    aSch->Prj().GetProjectPath() );
                 wxString msg;
+                FILENAME_RESOLVER resolve;
+                resolve.SetProject( &aSch->Prj() );
+                resolve.SetProgramBase( &Pgm() );
+
+                wxString absolutePath = resolve.ResolvePath( path,
+                                                            wxGetCwd(),
+                                                            aSch->GetEmbeddedFiles() );
 
                 if( !DS_DATA_MODEL::GetTheInstance().LoadDrawingSheet( absolutePath, &msg ) )
                 {
diff --git a/eeschema/fields_grid_table.cpp b/eeschema/fields_grid_table.cpp
index 17583d1fda..679a2e1ceb 100644
--- a/eeschema/fields_grid_table.cpp
+++ b/eeschema/fields_grid_table.cpp
@@ -21,6 +21,7 @@
  * 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA
  */
 
+#include <embedded_files.h>
 #include <kiway.h>
 #include <kiway_player.h>
 #include <dialog_shim.h>
@@ -238,8 +239,16 @@ void FIELDS_GRID_TABLE::initGrid( WX_GRID* aGrid )
     fpIdEditor->SetValidator( m_nonUrlValidator );
     m_footprintAttr->SetEditor( fpIdEditor );
 
+    EMBEDDED_FILES* files = nullptr;
+
+    if( m_frame->GetFrameType() == FRAME_SCH )
+        files = m_frame->GetScreen()->Schematic();
+    else if( m_frame->GetFrameType() == FRAME_SCH_SYMBOL_EDITOR || m_frame->GetFrameType() == FRAME_SCH_VIEWER )
+        files = m_part;
+
     m_urlAttr = new wxGridCellAttr;
-    GRID_CELL_URL_EDITOR* urlEditor = new GRID_CELL_URL_EDITOR( m_dialog, PROJECT_SCH::SchSearchS( &m_frame->Prj() ) );
+    GRID_CELL_URL_EDITOR* urlEditor =
+            new GRID_CELL_URL_EDITOR( m_dialog, PROJECT_SCH::SchSearchS( &m_frame->Prj() ), files );
     urlEditor->SetValidator( m_urlValidator );
     m_urlAttr->SetEditor( urlEditor );
 
@@ -292,6 +301,9 @@ void FIELDS_GRID_TABLE::initGrid( WX_GRID* aGrid )
     SCH_EDIT_FRAME* editFrame = dynamic_cast<SCH_EDIT_FRAME*>( m_frame );
     wxArrayString   existingNetclasses;
 
+    wxArrayString            fonts;
+    std::vector<std::string> fontNames;
+
     if( editFrame )
     {
         // Load the combobox with existing existingNetclassNames
@@ -302,15 +314,30 @@ void FIELDS_GRID_TABLE::initGrid( WX_GRID* aGrid )
 
         for( const auto& [ name, netclass ] : settings->m_NetClasses )
             existingNetclasses.push_back( name );
+
+        // We don't need to re-cache the embedded fonts when looking at symbols in the schematic editor
+        // because the fonts are all available in the schematic.
+        const std::vector<wxString>* fontFiles = nullptr;
+
+        if( m_frame->GetScreen() && m_frame->GetScreen()->Schematic() )
+            fontFiles = m_frame->GetScreen()->Schematic()->GetEmbeddedFiles()->GetFontFiles();
+
+        Fontconfig()->ListFonts( fontNames, std::string( Pgm().GetLanguageTag().utf8_str() ),
+                                fontFiles, false );
+    }
+    else
+    {
+        const std::vector<wxString>* fontFiles = m_part->GetEmbeddedFiles()->UpdateFontFiles();
+
+        // If there are font files embedded, we want to re-cache our fonts for each symbol that we
+        // are looking at in the symbol editor.
+        Fontconfig()->ListFonts( fontNames, std::string( Pgm().GetLanguageTag().utf8_str() ),
+                                fontFiles, !fontFiles->empty() );
     }
 
     m_netclassAttr = new wxGridCellAttr;
     m_netclassAttr->SetEditor( new GRID_CELL_COMBOBOX( existingNetclasses ) );
 
-    wxArrayString            fonts;
-    std::vector<std::string> fontNames;
-    Fontconfig()->ListFonts( fontNames, std::string( Pgm().GetLanguageTag().utf8_str() ) );
-
     for( const std::string& name : fontNames )
         fonts.Add( wxString( name ) );
 
@@ -931,8 +958,9 @@ void FIELDS_GRID_TRICKS::doPopupSelection( wxCommandEvent& event )
     else if (event.GetId() == MYID_SHOW_DATASHEET )
     {
         wxString datasheet_uri = m_grid->GetCellValue( DATASHEET_FIELD, FDC_VALUE );
+
         GetAssociatedDocument( m_dlg, datasheet_uri, &m_dlg->Prj(),
-                               PROJECT_SCH::SchSearchS( &m_dlg->Prj() ) );
+                               PROJECT_SCH::SchSearchS( &m_dlg->Prj() ), m_files );
     }
     else
     {
diff --git a/eeschema/fields_grid_table.h b/eeschema/fields_grid_table.h
index 4a0a00661c..1f4fd1a95c 100644
--- a/eeschema/fields_grid_table.h
+++ b/eeschema/fields_grid_table.h
@@ -31,22 +31,25 @@
 
 class SCH_BASE_FRAME;
 class DIALOG_SHIM;
+class EMBEDDED_FILES;
 class SCH_LABEL_BASE;
 
 
 class FIELDS_GRID_TRICKS : public GRID_TRICKS
 {
 public:
-    FIELDS_GRID_TRICKS( WX_GRID* aGrid, DIALOG_SHIM* aDialog,
+    FIELDS_GRID_TRICKS( WX_GRID* aGrid, DIALOG_SHIM* aDialog, EMBEDDED_FILES* aFiles,
                         std::function<void( wxCommandEvent& )> aAddHandler ) :
         GRID_TRICKS( aGrid, std::move( aAddHandler ) ),
-        m_dlg( aDialog )
+        m_dlg( aDialog ),
+        m_files( aFiles )
     {}
 
 protected:
     void showPopupMenu( wxMenu& menu, wxGridEvent& aEvent ) override;
     void doPopupSelection( wxCommandEvent& event ) override;
     DIALOG_SHIM* m_dlg;
+    EMBEDDED_FILES* m_files;
 };
 
 
diff --git a/eeschema/files-io.cpp b/eeschema/files-io.cpp
index 8565d65058..622ed5cc64 100644
--- a/eeschema/files-io.cpp
+++ b/eeschema/files-io.cpp
@@ -517,6 +517,11 @@ bool SCH_EDIT_FRAME::OpenProjectFiles( const std::vector<wxString>& aFileSet, in
             }
         }
 
+        // After the schematic is successfully loaded, we load the drawing sheet.
+        // This allows us to use the drawing sheet embedded in the schematic (if any)
+        // instead of the default one.
+        LoadDrawingSheet();
+
         schematic.PruneOrphanedSymbolInstances( Prj().GetProjectName(), sheetList );
         schematic.PruneOrphanedSheetInstances( Prj().GetProjectName(), sheetList );
 
diff --git a/eeschema/lib_symbol.cpp b/eeschema/lib_symbol.cpp
index ec902d139f..0c8311af6e 100644
--- a/eeschema/lib_symbol.cpp
+++ b/eeschema/lib_symbol.cpp
@@ -24,6 +24,7 @@
  * 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA
  */
 
+#include <font/outline_font.h>
 #include <sch_draw_panel.h>
 #include <plotters/plotter.h>
 #include <sch_screen.h>
@@ -1872,3 +1873,50 @@ double LIB_SYMBOL::Similarity( const SCH_ITEM& aOther ) const
 
     return similarity;
 }
+
+
+EMBEDDED_FILES* LIB_SYMBOL::GetEmbeddedFiles()
+{
+    return static_cast<EMBEDDED_FILES*>( this );
+}
+
+
+const EMBEDDED_FILES* LIB_SYMBOL::GetEmbeddedFiles() const
+{
+    return static_cast<const EMBEDDED_FILES*>( this );
+}
+
+
+void LIB_SYMBOL::EmbedFonts()
+{
+    using OUTLINE_FONT = KIFONT::OUTLINE_FONT;
+    using EMBEDDING_PERMISSION = OUTLINE_FONT::EMBEDDING_PERMISSION;
+
+    std::set<OUTLINE_FONT*> fonts;
+
+    for( SCH_ITEM& item : m_drawings )
+    {
+        if( item.Type() == SCH_TEXT_T )
+        {
+            auto* text = static_cast<SCH_TEXT*>( &item );
+
+            if( auto* font = text->GetFont(); font && !font->IsStroke() )
+            {
+                auto* outline = static_cast<OUTLINE_FONT*>( font );
+                auto permission = outline->GetEmbeddingPermission();
+
+                if( permission == EMBEDDING_PERMISSION::EDITABLE
+                    || permission == EMBEDDING_PERMISSION::INSTALLABLE )
+                {
+                    fonts.insert( outline );
+                }
+            }
+        }
+    }
+
+    for( auto* font : fonts )
+    {
+        auto file = GetEmbeddedFiles()->AddFile( font->GetFileName(), false );
+        file->type = EMBEDDED_FILES::EMBEDDED_FILE::FILE_TYPE::FONT;
+    }
+}
\ No newline at end of file
diff --git a/eeschema/lib_symbol.h b/eeschema/lib_symbol.h
index fd1cd6201a..b8036f61c7 100644
--- a/eeschema/lib_symbol.h
+++ b/eeschema/lib_symbol.h
@@ -27,6 +27,7 @@
 #ifndef LIB_SYMBOL_H
 #define LIB_SYMBOL_H
 
+#include <embedded_files.h>
 #include <symbol.h>
 #include <sch_field.h>
 #include <sch_pin.h>
@@ -73,7 +74,7 @@ struct LIB_SYMBOL_UNIT
  * A library symbol object is typically saved and loaded in a symbol library file (.lib).
  * Library symbols are different from schematic symbols.
  */
-class LIB_SYMBOL : public SYMBOL, public LIB_TREE_ITEM
+class LIB_SYMBOL : public SYMBOL, public LIB_TREE_ITEM, public EMBEDDED_FILES
 {
 public:
     LIB_SYMBOL( const wxString& aName, LIB_SYMBOL* aParent = nullptr,
@@ -333,6 +334,11 @@ public:
         return GetValueField().GetText();
     }
 
+    EMBEDDED_FILES* GetEmbeddedFiles() override;
+    const EMBEDDED_FILES* GetEmbeddedFiles() const;
+
+    void EmbedFonts() override;
+
     void RunOnChildren( const std::function<void( SCH_ITEM* )>& aFunction ) override;
 
     /**
diff --git a/eeschema/sch_edit_frame.cpp b/eeschema/sch_edit_frame.cpp
index e72eb36b7e..c857a2a3a6 100644
--- a/eeschema/sch_edit_frame.cpp
+++ b/eeschema/sch_edit_frame.cpp
@@ -66,6 +66,7 @@
 #include <tool/action_toolbar.h>
 #include <tool/common_control.h>
 #include <tool/common_tools.h>
+#include <tool/embed_tool.h>
 #include <tool/picker_tool.h>
 #include <tool/properties_tool.h>
 #include <tool/selection.h>
@@ -368,6 +369,7 @@ SCH_EDIT_FRAME::SCH_EDIT_FRAME( KIWAY* aKiway, wxWindow* aParent ) :
     }
 
     LoadProjectSettings();
+    LoadDrawingSheet();
 
     view->SetLayerVisible( LAYER_ERC_ERR, cfg->m_Appearance.show_erc_errors );
     view->SetLayerVisible( LAYER_ERC_WARN, cfg->m_Appearance.show_erc_warnings );
@@ -524,6 +526,7 @@ void SCH_EDIT_FRAME::setupTools()
     m_toolManager->RegisterTool( new EE_POINT_EDITOR );
     m_toolManager->RegisterTool( new SCH_NAVIGATE_TOOL );
     m_toolManager->RegisterTool( new PROPERTIES_TOOL );
+    m_toolManager->RegisterTool( new EMBED_TOOL );
     m_toolManager->InitTools();
 
     // Run the selection tool, it is supposed to be always active
diff --git a/eeschema/sch_edit_frame.h b/eeschema/sch_edit_frame.h
index 5632447021..260f4f3679 100644
--- a/eeschema/sch_edit_frame.h
+++ b/eeschema/sch_edit_frame.h
@@ -159,6 +159,11 @@ public:
      */
     bool LoadProjectSettings();
 
+    /**
+     * Load the drawing sheet file.
+     */
+    void LoadDrawingSheet();
+
     void ShowSchematicSetupDialog( const wxString& aInitialPage = wxEmptyString );
 
     void LoadSettings( APP_SETTINGS_BASE* aCfg ) override;
diff --git a/eeschema/sch_file_versions.h b/eeschema/sch_file_versions.h
index 4c948c328b..d55bc39d98 100644
--- a/eeschema/sch_file_versions.h
+++ b/eeschema/sch_file_versions.h
@@ -49,7 +49,8 @@
 //#define SEXPR_SYMBOL_LIB_FILE_VERSION  20220914   // Symbol unit display names.
 //#define SEXPR_SYMBOL_LIB_FILE_VERSION  20220914   // Don't save property ID
 //#define SEXPR_SYMBOL_LIB_FILE_VERSION  20230620   // ki_description -> Description Field
-#define   SEXPR_SYMBOL_LIB_FILE_VERSION  20231120   // generator_version; V8 cleanups
+//#define SEXPR_SYMBOL_LIB_FILE_VERSION  20231120   // generator_version; V8 cleanups
+#define  SEXPR_SYMBOL_LIB_FILE_VERSION  20240529   // Embedded Files
 
 /**
  * Schematic file version.
@@ -105,4 +106,5 @@
 //#define SEXPR_SCHEMATIC_FILE_VERSION 20231120  // generator_version; V8 cleanups
 //#define SEXPR_SCHEMATIC_FILE_VERSION 20240101  // Tables.
 //#define SEXPR_SCHEMATIC_FILE_VERSION 20240417  // Rule areas
-#define   SEXPR_SCHEMATIC_FILE_VERSION 20240602  // Sheet attributes
+//#define SEXPR_SCHEMATIC_FILE_VERSION 20240602  // Sheet attributes
+#define   SEXPR_SCHEMATIC_FILE_VERSION 20240620  // Embedded Files
diff --git a/eeschema/sch_io/kicad_sexpr/sch_io_kicad_sexpr.cpp b/eeschema/sch_io/kicad_sexpr/sch_io_kicad_sexpr.cpp
index f8e8251777..9885f7f0cc 100644
--- a/eeschema/sch_io/kicad_sexpr/sch_io_kicad_sexpr.cpp
+++ b/eeschema/sch_io/kicad_sexpr/sch_io_kicad_sexpr.cpp
@@ -372,6 +372,14 @@ void SCH_IO_KICAD_SEXPR::Format( SCH_SHEET* aSheet )
 
     wxCHECK( screen, /* void */ );
 
+    // If we've requested to embed the fonts in the schematic, do so.
+    // Otherwise, clear the embedded fonts from the schematic.  Embedded
+    // fonts will be used if available
+    if( m_schematic->GetAreFontsEmbedded() )
+        m_schematic->EmbedFonts();
+    else
+        m_schematic->GetEmbeddedFiles()->ClearEmbeddedFonts();
+
     m_out->Print( 0, "(kicad_sch (version %d) (generator \"eeschema\") (generator_version \"%s\")\n\n",
                   SEXPR_SCHEMATIC_FILE_VERSION, GetMajorMinorVersion().c_str().AsChar() );
 
@@ -477,7 +485,7 @@ void SCH_IO_KICAD_SEXPR::Format( SCH_SHEET* aSheet )
         case SCH_SHAPE_T:
             saveShape( static_cast<SCH_SHAPE*>( item ), 1 );
             break;
-        
+
         case SCH_RULE_AREA_T:
             saveRuleArea( static_cast<SCH_RULE_AREA*>( item ), 1 );
             break;
@@ -509,6 +517,13 @@ void SCH_IO_KICAD_SEXPR::Format( SCH_SHEET* aSheet )
 
         instances.emplace_back( aSheet->GetRootInstance() );
         saveInstances( instances, 1 );
+
+        m_out->Print( 1, "(embedded_fonts %s)\n",
+                      m_schematic->GetAreFontsEmbedded() ? "yes" : "no" );
+
+        // Save any embedded files
+        if( !m_schematic->GetEmbeddedFiles()->IsEmpty() )
+            m_schematic->WriteEmbeddedFiles( *m_out, 1, true );
     }
 
     m_out->Print( 0, ")\n" );
@@ -527,6 +542,14 @@ void SCH_IO_KICAD_SEXPR::Format( EE_SELECTION* aSelection, SCH_SHEET_PATH* aSele
     m_schematic = &aSchematic;
     m_out = aFormatter;
 
+    // If we've requested to embed the fonts in the schematic, do so.
+    // Otherwise, clear the embedded fonts from the schematic.  Embedded
+    // fonts will be used if available
+    if( m_schematic->GetAreFontsEmbedded() )
+        m_schematic->EmbedFonts();
+    else
+        m_schematic->GetEmbeddedFiles()->ClearEmbeddedFonts();
+
     size_t                          i;
     SCH_ITEM*                       item;
     std::map<wxString, LIB_SYMBOL*> libSymbols;
@@ -564,7 +587,7 @@ void SCH_IO_KICAD_SEXPR::Format( EE_SELECTION* aSelection, SCH_SHEET_PATH* aSele
         for( const std::pair<const wxString, LIB_SYMBOL*>& libSymbol : libSymbols )
         {
             SCH_IO_KICAD_SEXPR_LIB_CACHE::SaveSymbol( libSymbol.second, *m_out, 1,
-                                                      libSymbol.first );
+                                                      libSymbol.first, false );
         }
 
         m_out->Print( 0, ")\n\n" );
diff --git a/eeschema/sch_io/kicad_sexpr/sch_io_kicad_sexpr_lib_cache.cpp b/eeschema/sch_io/kicad_sexpr/sch_io_kicad_sexpr_lib_cache.cpp
index 13d5c10a66..b9e8f8a550 100644
--- a/eeschema/sch_io/kicad_sexpr/sch_io_kicad_sexpr_lib_cache.cpp
+++ b/eeschema/sch_io/kicad_sexpr/sch_io_kicad_sexpr_lib_cache.cpp
@@ -130,7 +130,7 @@ void SCH_IO_KICAD_SEXPR_LIB_CACHE::Save( const std::optional<bool>& aOpt )
 
 
 void SCH_IO_KICAD_SEXPR_LIB_CACHE::SaveSymbol( LIB_SYMBOL* aSymbol, OUTPUTFORMATTER& aFormatter,
-                                               int aNestLevel, const wxString& aLibName )
+                                               int aNestLevel, const wxString& aLibName, bool aIncludeData )
 {
     wxCHECK_RET( aSymbol, "Invalid LIB_SYMBOL pointer." );
 
@@ -138,6 +138,15 @@ void SCH_IO_KICAD_SEXPR_LIB_CACHE::SaveSymbol( LIB_SYMBOL* aSymbol, OUTPUTFORMAT
     wxCHECK2( wxLocale::GetInfo( wxLOCALE_DECIMAL_POINT, wxLOCALE_CAT_NUMBER ) == ".",
               LOCALE_IO toggle );
 
+
+    // If we've requested to embed the fonts in the symbol, do so.
+    // Otherwise, clear the embedded fonts from the symbol.  Embedded
+    // fonts will be used if available
+    if( aSymbol->GetAreFontsEmbedded() )
+        aSymbol->EmbedFonts();
+    else
+        aSymbol->GetEmbeddedFiles()->ClearEmbeddedFonts();
+
     int nextFreeFieldId = MANDATORY_FIELDS;
     std::vector<SCH_FIELD*> fields;
     std::string name = aFormatter.Quotew( aSymbol->GetLibId().GetLibItemName().wx_str() );
@@ -262,6 +271,12 @@ void SCH_IO_KICAD_SEXPR_LIB_CACHE::SaveSymbol( LIB_SYMBOL* aSymbol, OUTPUTFORMAT
 
             aFormatter.Print( aNestLevel + 1, ")\n" );
         }
+
+        aFormatter.Print( aNestLevel + 1, "(embedded_fonts %s)\n",
+                          aSymbol->GetAreFontsEmbedded() ? "yes" : "no" );
+
+        if( !aSymbol->EmbeddedFileMap().empty() )
+            aSymbol->WriteEmbeddedFiles( aFormatter, aNestLevel + 1, aIncludeData );
     }
     else
     {
diff --git a/eeschema/sch_io/kicad_sexpr/sch_io_kicad_sexpr_lib_cache.h b/eeschema/sch_io/kicad_sexpr/sch_io_kicad_sexpr_lib_cache.h
index 6cc8fe7a32..8f4ef30dec 100644
--- a/eeschema/sch_io/kicad_sexpr/sch_io_kicad_sexpr_lib_cache.h
+++ b/eeschema/sch_io/kicad_sexpr/sch_io_kicad_sexpr_lib_cache.h
@@ -51,8 +51,8 @@ public:
 
     void DeleteSymbol( const wxString& aName ) override;
 
-    static void SaveSymbol( LIB_SYMBOL* aSymbol, OUTPUTFORMATTER& aFormatter,
-                            int aNestLevel = 0, const wxString& aLibName = wxEmptyString );
+    static void SaveSymbol( LIB_SYMBOL* aSymbol, OUTPUTFORMATTER& aFormatter, int aNestLevel = 0,
+                            const wxString& aLibName = wxEmptyString, bool aIncludeData = true );
 
     void SetFileFormatVersionAtLoad( int aVersion ) { m_fileFormatVersionAtLoad = aVersion; }
     int GetFileFormatVersionAtLoad()  const { return m_fileFormatVersionAtLoad; }
diff --git a/eeschema/sch_io/kicad_sexpr/sch_io_kicad_sexpr_parser.cpp b/eeschema/sch_io/kicad_sexpr/sch_io_kicad_sexpr_parser.cpp
index 5ce1b91a3d..357dae58ff 100644
--- a/eeschema/sch_io/kicad_sexpr/sch_io_kicad_sexpr_parser.cpp
+++ b/eeschema/sch_io/kicad_sexpr/sch_io_kicad_sexpr_parser.cpp
@@ -31,6 +31,7 @@
 #include <fmt/format.h>
 #define wxUSE_BASE64 1
 #include <wx/base64.h>
+#include <wx/log.h>
 #include <wx/mstream.h>
 #include <wx/tokenzr.h>
 
@@ -39,6 +40,8 @@
 #include <sch_pin.h>
 #include <math/util.h>                           // KiROUND, Clamp
 #include <font/font.h>
+#include <font/fontconfig.h>
+#include <pgm_base.h>
 #include <string_utils.h>
 #include <sch_bitmap.h>
 #include <sch_bus_entry.h>
@@ -274,6 +277,13 @@ LIB_SYMBOL* SCH_IO_KICAD_SEXPR_PARSER::ParseSymbol( LIB_SYMBOL_MAP& aSymbolLibMa
         }
     }
 
+    for( auto& [text, params] : m_fontTextMap )
+    {
+        text->SetFont( KIFONT::FONT::GetFont( std::get<0>( params ), std::get<1>( params ),
+                                              std::get<2>( params ),
+                                              newSymbol->GetEmbeddedFiles()->UpdateFontFiles() ) );
+    }
+
     return newSymbol;
 }
 
@@ -534,6 +544,31 @@ LIB_SYMBOL* SCH_IO_KICAD_SEXPR_PARSER::parseLibSymbol( LIB_SYMBOL_MAP& aSymbolLi
             symbol->AddDrawItem( item, false );
             break;
 
+        case T_embedded_fonts:
+        {
+            symbol->SetAreFontsEmbedded( parseBool() );
+            NeedRIGHT();
+            break;
+        }
+
+        case T_embedded_files:
+        {
+            EMBEDDED_FILES_PARSER embeddedFilesParser( reader );
+            embeddedFilesParser.SyncLineReaderWith( *this );
+
+            try
+            {
+                embeddedFilesParser.ParseEmbedded( symbol->GetEmbeddedFiles() );
+            }
+            catch( const IO_ERROR& e )
+            {
+                wxLogError( e.What() );
+            }
+
+            SyncLineReaderWith( embeddedFilesParser );
+            break;
+        }
+
         default:
             Expecting( "pin_names, pin_numbers, arc, bezier, circle, pin, polyline, "
                        "rectangle, or text" );
@@ -742,7 +777,7 @@ void SCH_IO_KICAD_SEXPR_PARSER::parseEDA_TEXT( EDA_TEXT* aText, bool aConvertOve
             }
 
             if( !faceName.IsEmpty() )
-                aText->SetFont( KIFONT::FONT::GetFont( faceName, bold, italic ) );
+                m_fontTextMap[aText] = { faceName, bold, italic };
 
             break;
 
@@ -2755,10 +2790,48 @@ void SCH_IO_KICAD_SEXPR_PARSER::ParseSchematic( SCH_SHEET* aSheet, bool aIsCopya
             parseBusAlias( screen );
             break;
 
+        case T_embedded_fonts:
+        {
+            SCHEMATIC* schematic = aSheet->Schematic();
+
+            if( !schematic )
+                THROW_PARSE_ERROR( _( "No schematic object" ), CurSource(), CurLine(),
+                                   CurLineNumber(), CurOffset() );
+
+            schematic->GetEmbeddedFiles()->SetAreFontsEmbedded( parseBool() );
+            NeedRIGHT();
+            break;
+        }
+
+        case T_embedded_files:
+        {
+            SCHEMATIC* schematic = aSheet->Schematic();
+
+            if( !schematic )
+                THROW_PARSE_ERROR( _( "No schematic object" ), CurSource(), CurLine(),
+                                   CurLineNumber(), CurOffset() );
+
+            EMBEDDED_FILES_PARSER embeddedFilesParser( reader );
+            embeddedFilesParser.SyncLineReaderWith( *this );
+
+            try
+            {
+                embeddedFilesParser.ParseEmbedded( schematic->GetEmbeddedFiles() );
+            }
+            catch( const PARSE_ERROR& e )
+            {
+                wxLogError( e.What() );
+            }
+
+            SyncLineReaderWith( embeddedFilesParser );
+            break;
+        }
+
+
         default:
-            Expecting( "symbol, paper, page, title_block, bitmap, sheet, junction, no_connect, "
-                       "bus_entry, line, bus, text, label, class_label, global_label, "
-                       "hierarchical_label, symbol_instances, rule_area, or bus_alias" );
+            Expecting( "bitmap, bus, bus_alias, bus_entry, class_label, embedded_files, global_label, "
+                       "hierarchical_label, junction, label, line, no_connect, page, paper, rule_area, "
+                       "sheet, symbol, symbol_instances, text, title_block" );
         }
     }
 
@@ -2771,6 +2844,26 @@ void SCH_IO_KICAD_SEXPR_PARSER::ParseSchematic( SCH_SHEET* aSheet, bool aIsCopya
     }
 
     screen->UpdateLocalLibSymbolLinks();
+    screen->FixupEmbeddedData();
+
+    SCHEMATIC* schematic = aSheet->Schematic();
+
+    if( !schematic )
+        THROW_PARSE_ERROR( _( "No schematic object" ), CurSource(), CurLine(),
+                            CurLineNumber(), CurOffset() );
+
+    for( auto& [text, params] : m_fontTextMap )
+    {
+        text->SetFont( KIFONT::FONT::GetFont( std::get<0>( params ), std::get<1>( params ),
+                                              std::get<2>( params ),
+                                              schematic->GetEmbeddedFiles()->UpdateFontFiles() ) );
+    }
+
+    // When loading the schematic, take a moment to cache the fonts so that the font
+    // picker can show the embedded fonts immediately.
+    std::vector<std::string> fontNames;
+    Fontconfig()->ListFonts( fontNames, std::string( Pgm().GetLanguageTag().utf8_str() ),
+                             schematic->GetEmbeddedFiles()->GetFontFiles(), true );
 
     if( m_requiredVersion < 20200828 )
         screen->SetLegacySymbolInstanceData();
diff --git a/eeschema/sch_io/kicad_sexpr/sch_io_kicad_sexpr_parser.h b/eeschema/sch_io/kicad_sexpr/sch_io_kicad_sexpr_parser.h
index 2fec010d02..ca26e2b0c1 100644
--- a/eeschema/sch_io/kicad_sexpr/sch_io_kicad_sexpr_parser.h
+++ b/eeschema/sch_io/kicad_sexpr/sch_io_kicad_sexpr_parser.h
@@ -244,6 +244,8 @@ private:
 
     std::set<KIID>     m_uuids;
 
+    std::map<EDA_TEXT*, std::tuple<wxString, bool, bool>> m_fontTextMap;
+
     PROGRESS_REPORTER* m_progressReporter;  // optional; may be nullptr
     const LINE_READER* m_lineReader;        // for progress reporting
     unsigned           m_lastProgressLine;
diff --git a/eeschema/sch_screen.cpp b/eeschema/sch_screen.cpp
index d7cca68758..b2b87eeeb0 100644
--- a/eeschema/sch_screen.cpp
+++ b/eeschema/sch_screen.cpp
@@ -1488,6 +1488,28 @@ void SCH_SCREEN::AddLibSymbol( LIB_SYMBOL* aLibSymbol )
 }
 
 
+void SCH_SCREEN::FixupEmbeddedData()
+{
+    SCHEMATIC* schematic = Schematic();
+
+    for( auto& [name, libSym] : m_libSymbols )
+    {
+        for( auto& [filename, embeddedFile] : libSym->EmbeddedFileMap() )
+        {
+            EMBEDDED_FILES::EMBEDDED_FILE* file = schematic->GetEmbeddedFile( filename );
+
+            if( file )
+            {
+                embeddedFile->compressedEncodedData = file->compressedEncodedData;
+                embeddedFile->decompressedData = file->decompressedData;
+                embeddedFile->data_sha = file->data_sha;
+                embeddedFile->is_valid = file->is_valid;
+            }
+        }
+    }
+}
+
+
 void SCH_SCREEN::AddBusAlias( std::shared_ptr<BUS_ALIAS> aAlias )
 {
     m_aliases.insert( aAlias );
diff --git a/eeschema/sch_screen.h b/eeschema/sch_screen.h
index cf5bf97257..a54a995837 100644
--- a/eeschema/sch_screen.h
+++ b/eeschema/sch_screen.h
@@ -490,6 +490,13 @@ public:
      */
     void AddLibSymbol( LIB_SYMBOL* aLibSymbol );
 
+    /**
+     * After loading a file from disk, the library symbols do not yet contain the full
+     * data for their embedded files, only a reference.  This iterates over all lib symbols
+     * in the schematic and updates the library symbols with the full data.
+    */
+    void FixupEmbeddedData();
+
     /**
      * Add a bus alias definition (and transfers ownership of the pointer).
      */
diff --git a/eeschema/schematic.cpp b/eeschema/schematic.cpp
index 6f11d39897..2bd422d5fc 100644
--- a/eeschema/schematic.cpp
+++ b/eeschema/schematic.cpp
@@ -24,18 +24,21 @@
 #include <core/kicad_algo.h>
 #include <ee_collectors.h>
 #include <erc/erc_settings.h>
-#include <sch_marker.h>
+#include <font/outline_font.h>
+#include <netlist_exporter_spice.h>
 #include <project.h>
-#include <project/project_file.h>
 #include <project/net_settings.h>
+#include <project/project_file.h>
 #include <schematic.h>
 #include <sch_junction.h>
+#include <sch_label.h>
 #include <sch_line.h>
+#include <sch_marker.h>
 #include <sch_screen.h>
 #include <sim/spice_settings.h>
-#include <sch_label.h>
 #include <sim/spice_value.h>
-#include <netlist_exporter_spice.h>
+
+#include <wx/log.h>
 
 bool SCHEMATIC::m_IsSchematicExists = false;
 
@@ -846,3 +849,59 @@ void SCHEMATIC::ResolveERCExclusionsPostUpdate()
             RootScreen()->Append( marker );
     }
 }
+
+
+EMBEDDED_FILES* SCHEMATIC::GetEmbeddedFiles()
+{
+    return static_cast<EMBEDDED_FILES*>( this );
+}
+
+
+const EMBEDDED_FILES* SCHEMATIC::GetEmbeddedFiles() const
+{
+    return static_cast<const EMBEDDED_FILES*>( this );
+}
+
+
+void SCHEMATIC::EmbedFonts()
+{
+    std::set<KIFONT::OUTLINE_FONT*> fonts;
+
+    SCH_SHEET_LIST sheetList = BuildUnorderedSheetList();
+
+    for( const SCH_SHEET_PATH& sheet : sheetList )
+    {
+        for( SCH_ITEM* item : sheet.LastScreen()->Items() )
+        {
+            if( EDA_TEXT* text = dynamic_cast<EDA_TEXT*>( item ) )
+            {
+                KIFONT::FONT* font = text->GetFont();
+
+                if( !font || font->IsStroke() )
+                    continue;
+
+                using EMBEDDING_PERMISSION = KIFONT::OUTLINE_FONT::EMBEDDING_PERMISSION;
+                auto* outline = static_cast<KIFONT::OUTLINE_FONT*>( font );
+
+                if( outline->GetEmbeddingPermission() == EMBEDDING_PERMISSION::EDITABLE
+                    || outline->GetEmbeddingPermission() == EMBEDDING_PERMISSION::INSTALLABLE )
+                {
+                    fonts.insert( outline );
+                }
+            }
+        }
+    }
+
+    for( KIFONT::OUTLINE_FONT* font : fonts )
+    {
+        auto file = GetEmbeddedFiles()->AddFile( font->GetFileName(), false );
+
+        if( !file )
+        {
+            wxLogTrace( "EMBED", "Failed to add font file: %s", font->GetFileName() );
+            continue;
+        }
+
+        file->type = EMBEDDED_FILES::EMBEDDED_FILE::FILE_TYPE::FONT;
+    }
+}
\ No newline at end of file
diff --git a/eeschema/schematic.h b/eeschema/schematic.h
index 004466e3d1..59f5473cce 100644
--- a/eeschema/schematic.h
+++ b/eeschema/schematic.h
@@ -21,6 +21,7 @@
 #define KICAD_SCHEMATIC_H
 
 #include <eda_item.h>
+#include <embedded_files.h>
 #include <sch_sheet_path.h>
 #include <schematic_settings.h>
 
@@ -71,7 +72,7 @@ public:
  * Right now, Eeschema can have only one schematic open at a time, but this could change.
  * Please keep this possibility in mind when adding to this object.
  */
-class SCHEMATIC : public SCHEMATIC_IFACE, public EDA_ITEM
+class SCHEMATIC : public SCHEMATIC_IFACE, public EDA_ITEM, public EMBEDDED_FILES
 {
 public:
     SCHEMATIC( PROJECT* aPrj );
@@ -161,6 +162,9 @@ public:
 
     std::vector<SCH_MARKER*> ResolveERCExclusions();
 
+    EMBEDDED_FILES* GetEmbeddedFiles() override;
+    const EMBEDDED_FILES* GetEmbeddedFiles() const;
+
     /**
      * Return a pointer to a bus alias object for the given label, or null if one
      * doesn't exist.
@@ -305,6 +309,11 @@ public:
      */
     void RemoveAllListeners();
 
+    /**
+     * Embed fonts in the schematic
+     */
+    void EmbedFonts() override;
+
     /**
      * True if a SCHEMATIC exists, false if not
      */
diff --git a/eeschema/schematic.keywords b/eeschema/schematic.keywords
index fb05e65b46..9ff47e346e 100644
--- a/eeschema/schematic.keywords
+++ b/eeschema/schematic.keywords
@@ -16,6 +16,7 @@ bus_alias
 bus_entry
 cells
 center
+checksum
 circle
 clock
 clock_low
@@ -38,12 +39,15 @@ do_not_autoplace
 dot
 edge_clock_high
 effects
+embedded_fonts
+embedded_files
 end
 extends
 external
 exclude_from_sim
 face
 fields_autoplaced
+file
 fill
 font
 footprint
diff --git a/eeschema/symbol_editor/symbol_edit_frame.cpp b/eeschema/symbol_editor/symbol_edit_frame.cpp
index c9bcf707bc..7787c8461e 100644
--- a/eeschema/symbol_editor/symbol_edit_frame.cpp
+++ b/eeschema/symbol_editor/symbol_edit_frame.cpp
@@ -51,6 +51,7 @@
 #include <tool/common_control.h>
 #include <tool/common_tools.h>
 #include <tool/editor_conditions.h>
+#include <tool/embed_tool.h>
 #include <tool/library_editor_control.h>
 #include <tool/picker_tool.h>
 #include <tool/properties_tool.h>
@@ -395,6 +396,7 @@ void SYMBOL_EDIT_FRAME::setupTools()
     m_toolManager->RegisterTool( new LIBRARY_EDITOR_CONTROL );
     m_toolManager->RegisterTool( new SYMBOL_EDITOR_CONTROL );
     m_toolManager->RegisterTool( new PROPERTIES_TOOL );
+    m_toolManager->RegisterTool( new EMBED_TOOL );
     m_toolManager->InitTools();
 
     // Run the selection tool, it is supposed to be always active
diff --git a/eeschema/tools/ee_inspection_tool.cpp b/eeschema/tools/ee_inspection_tool.cpp
index d345e54b9e..0aac98fff3 100644
--- a/eeschema/tools/ee_inspection_tool.cpp
+++ b/eeschema/tools/ee_inspection_tool.cpp
@@ -496,6 +496,7 @@ int EE_INSPECTION_TOOL::RunSimulation( const TOOL_EVENT& aEvent )
 int EE_INSPECTION_TOOL::ShowDatasheet( const TOOL_EVENT& aEvent )
 {
     wxString datasheet;
+    EMBEDDED_FILES* files = nullptr;
 
     if( m_frame->IsType( FRAME_SCH_SYMBOL_EDITOR ) )
     {
@@ -505,6 +506,7 @@ int EE_INSPECTION_TOOL::ShowDatasheet( const TOOL_EVENT& aEvent )
             return 0;
 
         datasheet = symbol->GetDatasheetField().GetText();
+        files = symbol;
     }
     else if( m_frame->IsType( FRAME_SCH_VIEWER ) )
     {
@@ -514,6 +516,7 @@ int EE_INSPECTION_TOOL::ShowDatasheet( const TOOL_EVENT& aEvent )
             return 0;
 
         datasheet = entry->GetDatasheetField().GetText();
+        files = entry;
     }
     else if( m_frame->IsType( FRAME_SCH ) )
     {
@@ -528,6 +531,7 @@ int EE_INSPECTION_TOOL::ShowDatasheet( const TOOL_EVENT& aEvent )
         // Use GetShownText() to resolve any text variables, but don't allow adding extra text
         // (ie: the field name)
         datasheet = field->GetShownText( &symbol->Schematic()->CurrentSheet(), false );
+        files = symbol->Schematic();
     }
 
     if( datasheet.IsEmpty() || datasheet == wxS( "~" ) )
@@ -537,7 +541,7 @@ int EE_INSPECTION_TOOL::ShowDatasheet( const TOOL_EVENT& aEvent )
     else
     {
         GetAssociatedDocument( m_frame, datasheet, &m_frame->Prj(),
-                               PROJECT_SCH::SchSearchS( &m_frame->Prj() ) );
+                               PROJECT_SCH::SchSearchS( &m_frame->Prj() ), files );
     }
 
     return 0;
diff --git a/eeschema/tools/sch_editor_control.cpp b/eeschema/tools/sch_editor_control.cpp
index b35005294c..c47575242f 100644
--- a/eeschema/tools/sch_editor_control.cpp
+++ b/eeschema/tools/sch_editor_control.cpp
@@ -200,7 +200,7 @@ int SCH_EDITOR_CONTROL::PageSetup( const TOOL_EVENT& aEvent )
     undoCmd.SetDescription( _( "Page Settings" ) );
     m_frame->SaveCopyInUndoList( undoCmd, UNDO_REDO::PAGESETTINGS, false, false );
 
-    DIALOG_EESCHEMA_PAGE_SETTINGS dlg( m_frame, VECTOR2I( MAX_PAGE_SIZE_EESCHEMA_MILS,
+    DIALOG_EESCHEMA_PAGE_SETTINGS dlg( m_frame, m_frame->Schematic().GetEmbeddedFiles(), VECTOR2I( MAX_PAGE_SIZE_EESCHEMA_MILS,
                                                           MAX_PAGE_SIZE_EESCHEMA_MILS ) );
     dlg.SetWksFileName( BASE_SCREEN::m_DrawingSheetFileName );
 
diff --git a/eeschema/tools/sch_navigate_tool.cpp b/eeschema/tools/sch_navigate_tool.cpp
index 545202d8b5..9c792c8c0d 100644
--- a/eeschema/tools/sch_navigate_tool.cpp
+++ b/eeschema/tools/sch_navigate_tool.cpp
@@ -87,7 +87,7 @@ void SCH_NAVIGATE_TOOL::HypertextCommand( const wxString& href )
         menu.Append( 1, wxString::Format( _( "Open %s" ), href ) );
 
         if( m_frame->GetPopupMenuSelectionFromUser( menu ) == 1 )
-            GetAssociatedDocument( m_frame, href, &m_frame->Prj() );
+            GetAssociatedDocument( m_frame, href, &m_frame->Prj(), nullptr, &m_frame->Schematic() );
     }
 }
 
diff --git a/eeschema/widgets/sch_properties_panel.cpp b/eeschema/widgets/sch_properties_panel.cpp
index 260ba62724..eb20395438 100644
--- a/eeschema/widgets/sch_properties_panel.cpp
+++ b/eeschema/widgets/sch_properties_panel.cpp
@@ -30,6 +30,8 @@
 #include <properties/property_mgr.h>
 #include <sch_commit.h>
 #include <sch_edit_frame.h>
+#include <symbol_edit_frame.h>
+#include <symbol_viewer_frame.h>
 #include <schematic.h>
 #include <settings/color_settings.h>
 #include <string_utils.h>
@@ -218,10 +220,31 @@ void SCH_PROPERTIES_PANEL::OnLanguageChanged( wxCommandEvent& aEvent )
 void SCH_PROPERTIES_PANEL::updateFontList()
 {
     wxPGChoices fonts;
+    const std::vector<wxString>* fontFiles = nullptr;
+
+    if( m_frame->GetFrameType() == FRAME_SCH && m_frame->GetScreen() && m_frame->GetScreen()->Schematic() )
+    {
+        fontFiles = m_frame->GetScreen()->Schematic()->GetEmbeddedFiles()->GetFontFiles();
+    }
+    else if( m_frame->GetFrameType() == FRAME_SCH_SYMBOL_EDITOR )
+    {
+        SYMBOL_EDIT_FRAME* symbolFrame = static_cast<SYMBOL_EDIT_FRAME*>( m_frame );
+
+        if( symbolFrame->GetCurSymbol() )
+            fontFiles = symbolFrame->GetCurSymbol()->GetEmbeddedFiles()->UpdateFontFiles();
+    }
+    else if( m_frame->GetFrameType() == FRAME_SCH_VIEWER )
+    {
+        SYMBOL_VIEWER_FRAME* symbolFrame = static_cast<SYMBOL_VIEWER_FRAME*>( m_frame );
+
+        if( symbolFrame->GetSelectedSymbol() )
+            fontFiles = symbolFrame->GetSelectedSymbol()->GetEmbeddedFiles()->UpdateFontFiles();
+    }
 
     // Regnerate font names
     std::vector<std::string> fontNames;
-    Fontconfig()->ListFonts( fontNames, std::string( Pgm().GetLanguageTag().utf8_str() ) );
+    Fontconfig()->ListFonts( fontNames, std::string( Pgm().GetLanguageTag().utf8_str() ),
+                             fontFiles );
 
     fonts.Add( _( "Default Font" ), -1 );
     fonts.Add( KICAD_FONT_NAME, -2 );
diff --git a/include/dialogs/dialog_embed_files.h b/include/dialogs/dialog_embed_files.h
new file mode 100644
index 0000000000..95cec29651
--- /dev/null
+++ b/include/dialogs/dialog_embed_files.h
@@ -0,0 +1,43 @@
+/*
+ * This program source code file is part of KiCad, a free EDA CAD application.
+ *
+ * Copyright (C) 2024 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 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, see <http://www.gnu.org/licenses/>.
+ */
+
+
+#ifndef DIALOG_EMBED_FILES_H
+#define DIALOG_EMBED_FILES_H
+
+#include <dialog_shim.h>
+#include <wx/panel.h>
+
+
+class DIALOG_EMBED_FILES : public DIALOG_SHIM
+{
+public:
+    DIALOG_EMBED_FILES( wxWindow* aParent, const wxString& aTitle );
+
+    void InstallPanel( wxPanel* aPanel );
+
+    bool TransferDataToWindow() override;
+    bool TransferDataFromWindow() override;
+
+protected:
+    wxPanel* m_contentPanel;
+};
+
+
+#endif //DIALOG_EMBED_FILES_H
\ No newline at end of file
diff --git a/include/dialogs/dialog_page_settings.h b/include/dialogs/dialog_page_settings.h
index d78f5d98b8..ef9328ace9 100644
--- a/include/dialogs/dialog_page_settings.h
+++ b/include/dialogs/dialog_page_settings.h
@@ -29,6 +29,8 @@
 class DS_DATA_MODEL;
 class EDA_DRAW_FRAME;
 class BASE_SCREEN;
+class EMBEDDED_FILES;
+class FILENAME_RESOLVER;
 
 /*!
  * DIALOG_PAGES_SETTINGS class declaration
@@ -37,8 +39,8 @@ class BASE_SCREEN;
 class DIALOG_PAGES_SETTINGS: public DIALOG_PAGES_SETTINGS_BASE
 {
 public:
-    DIALOG_PAGES_SETTINGS( EDA_DRAW_FRAME* aParent, double aIuPerMils,
-                           const VECTOR2D& aMaxUserSizeMils );
+    DIALOG_PAGES_SETTINGS( EDA_DRAW_FRAME* aParent, EMBEDDED_FILES* aEmbeddedFiles,
+                           double aIuPerMils, const VECTOR2D& aMaxUserSizeMils );
     virtual ~DIALOG_PAGES_SETTINGS();
 
     const wxString GetWksFileName()
@@ -130,7 +132,9 @@ protected:
     TITLE_BLOCK     m_tb;                    /// Temporary title block (basic inscriptions).
     DS_DATA_MODEL*  m_drawingSheet; // the alternate and temporary drawing sheet shown by the
                                     // dialog when the initial one is replaced by a new one
-    double          m_iuPerMils;
+    double             m_iuPerMils;
+    EMBEDDED_FILES*    m_embeddedFiles; // the embedded files reference from the parent
+    FILENAME_RESOLVER* m_filenameResolver;
 
 private:
     UNIT_BINDER m_customSizeX;
diff --git a/include/dsnlexer.h b/include/dsnlexer.h
index 805a996fcc..33df25ccb9 100644
--- a/include/dsnlexer.h
+++ b/include/dsnlexer.h
@@ -57,7 +57,8 @@ struct KICOMMON_API KEYWORD
  */
 enum DSN_SYNTAX_T
 {
-    DSN_NONE         = -11,
+    DSN_NONE         = -12,
+    DSN_BAR          = -11, // Also called pipe '|'
     DSN_COMMENT      = -10,
     DSN_STRING_QUOTE = -9,
     DSN_QUOTE_DEF    = -8,
@@ -381,6 +382,13 @@ public:
      */
     void NeedRIGHT();
 
+    /**
+     * Call #NextTok() and then verifies that the token read in is a #DSN_BAR.
+     *
+     * @throw IO_ERROR if the next token is not a #DSN_BAR
+     */
+    void NeedBAR();
+
     /**
      * Return the C string representation of a #DSN_T value.
      */
diff --git a/include/eda_doc.h b/include/eda_doc.h
index 1d986155a3..98c48f1b4c 100644
--- a/include/eda_doc.h
+++ b/include/eda_doc.h
@@ -31,6 +31,8 @@
 #ifndef __INCLUDE__EDA_DOC_H__
 #define __INCLUDE__EDA_DOC_H__ 1
 
+class EMBEDDED_FILES;
+
 /**
  * Open a document (file) with the suitable browser.
  *
@@ -43,7 +45,7 @@
  * @param aPaths Additional paths to search for local disk datasheet files
 */
 bool GetAssociatedDocument( wxWindow* aParent, const wxString& aDocName, PROJECT* aProject,
-                            SEARCH_STACK* aPaths = nullptr );
+                            SEARCH_STACK* aPaths = nullptr, EMBEDDED_FILES* aFiles = nullptr );
 
 
 #endif /* __INCLUDE__EDA_DOC_H__ */
diff --git a/include/eda_item.h b/include/eda_item.h
index 384fa10976..f9cb57d9cc 100644
--- a/include/eda_item.h
+++ b/include/eda_item.h
@@ -52,6 +52,7 @@ enum class INSPECT_RESULT
 class UNITS_PROVIDER;
 class EDA_DRAW_FRAME;
 class MSG_PANEL_ITEM;
+class EMBEDDED_FILES;
 
 namespace google { namespace protobuf { class Any; } }
 
@@ -440,6 +441,8 @@ public:
 
     virtual void ViewGetLayers( int aLayers[], int& aCount ) const override;
 
+    virtual EMBEDDED_FILES* GetEmbeddedFiles() { return nullptr; }
+
 #if defined(DEBUG)
 
     /**
diff --git a/include/embedded_files.h b/include/embedded_files.h
new file mode 100644
index 0000000000..6bfac91e95
--- /dev/null
+++ b/include/embedded_files.h
@@ -0,0 +1,243 @@
+/*
+ * This program source code file is part of KiCad, a free EDA CAD application.
+ *
+ * Copyright (C) 2024 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 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, see <http://www.gnu.org/licenses/>.
+ */
+
+#ifndef EMBEDDED_FILES_H
+#define EMBEDDED_FILES_H
+
+#include <map>
+
+#include <wx/string.h>
+#include <wx/filename.h>
+
+#include <embedded_files_lexer.h>
+#include <wildcards_and_files_ext.h>
+#include <richio.h>
+#include <picosha2.h>
+
+class EMBEDDED_FILES
+{
+public:
+    struct EMBEDDED_FILE
+    {
+        enum class FILE_TYPE
+        {
+            FONT,
+            MODEL,
+            WORKSHEET,
+            DATASHEET,
+            OTHER
+        };
+
+        EMBEDDED_FILE() :
+                type( FILE_TYPE::OTHER ),
+                is_valid( false )
+        {}
+
+        bool Validate()
+        {
+            std::string new_sha;
+            picosha2::hash256_hex_string( decompressedData, new_sha );
+
+            is_valid = ( new_sha == data_sha );
+            return is_valid;
+        }
+
+        wxString GetLink() const
+        {
+            return wxString::Format( "%s://%s", FILEEXT::KiCadUriPrefix, name );
+        }
+
+        wxString          name;
+        FILE_TYPE         type;
+        std::string       compressedEncodedData;
+        std::vector<char> decompressedData;
+        std::string       data_sha;
+        bool              is_valid;
+    };
+
+    enum class RETURN_CODE : int
+    {
+        OK,                  // Success
+        FILE_NOT_FOUND,      // File not found on disk
+        PERMISSIONS_ERROR,   // Could not read/write file
+        FILE_ALREADY_EXISTS, // File already exists in the collection
+        OUT_OF_MEMORY,       // Could not allocate memory
+        CHECKSUM_ERROR,      // Checksum in file does not match data
+    };
+
+    EMBEDDED_FILES() = default;
+
+    ~EMBEDDED_FILES()
+    {
+        for( auto& file : m_files )
+            delete file.second;
+    }
+
+    /**
+     * Loads a file from disk and adds it to the collection.
+     * @param aName is the name of the file to load.
+     * @param aOverwrite is true if the file should be overwritten if it already exists.
+    */
+    EMBEDDED_FILE* AddFile( const wxFileName& aName, bool aOverwrite );
+
+    /**
+     * Appends a file to the collection.
+    */
+    void AddFile( EMBEDDED_FILE* aFile );
+
+    /**
+     * Removes a file from the collection and frees the memory.
+     * @param aName is the name of the file to remove.
+    */
+    void RemoveFile( const wxString& name, bool aErase = true );
+
+    /**
+     * Output formatter for the embedded files.
+     * @param aOut is the output formatter.
+     * @param aNestLevel is the current indentation level.
+     * @param aWriteData is true if the actual data should be written.  This is false when writing an element
+     *                   that is already embedded in a file that itself has embedded files (boards, schematics, etc.)
+    */
+    void WriteEmbeddedFiles( OUTPUTFORMATTER& aOut, int aNestLevel, bool aWriteData ) const;
+
+    /**
+     * Returns the link for an embedded file.
+     * @param aFile is the file to get the link for.
+     * @return the link for the file to be used in a hyperlink.
+    */
+    wxString GetEmbeddedFileLink( const EMBEDDED_FILE& aFile ) const
+    {
+        return aFile.GetLink();
+    }
+
+    bool HasFile( const wxString& name ) const
+    {
+        wxFileName fileName( name );
+
+        return m_files.find( fileName.GetFullName() ) != m_files.end();
+    }
+
+    bool IsEmpty() const
+    {
+        return m_files.empty();
+    }
+
+    /**
+     * Helper function to get a list of fonts for fontconfig to add to the library.
+     *
+     * This is neccesary because EMBEDDED_FILES lives in common at the moment and
+     * fontconfig is in libkicommon.  This will create the cache files in the KiCad
+     * cache directory (if they do not already exist) and return the temp files names
+    */
+    const std::vector<wxString>* UpdateFontFiles();
+
+    /**
+     * If we just need the cached version of the font files, we can use this function which
+     * is const and will not update the font files.
+    */
+    const std::vector<wxString>* GetFontFiles() const;
+
+    /**
+     * Removes all embedded fonts from the collection
+    */
+    void ClearEmbeddedFonts();
+
+    /**
+     * Takes data from the #decompressedData buffer and compresses it using ZSTD
+     * into the #compressedEncodedData buffer. The data is then Base64 encoded.
+     *
+     * This call is used when adding a new file to the collection from disk
+    */
+    static RETURN_CODE  CompressAndEncode( EMBEDDED_FILE& aFile );
+
+    /**
+     * Takes data from the #compressedEncodedData buffer and Base64 decodes it.
+     * The data is then decompressed using ZSTD and stored in the #decompressedData buffer.
+     *
+     * This call is used when loading the embedded files using the parsers.
+    */
+    static RETURN_CODE  DecompressAndDecode( EMBEDDED_FILE& aFile );
+
+    /**
+     * Returns the embedded file with the given name or nullptr if it does not exist.
+    */
+    EMBEDDED_FILE* GetEmbeddedFile( const wxString& aName ) const
+    {
+        auto it = m_files.find( aName );
+
+        return it == m_files.end() ? nullptr : it->second;
+    }
+
+    const std::map<wxString, EMBEDDED_FILE*>& EmbeddedFileMap() const
+    {
+        return m_files;
+    }
+
+    wxFileName GetTempFileName( const wxString& aName ) const;
+
+    wxFileName GetTempFileName( EMBEDDED_FILE* aFile ) const;
+
+    void ClearEmbeddedFiles( bool aDeleteFiles = true )
+    {
+        for( auto& file : m_files )
+        {
+            if( aDeleteFiles )
+                delete file.second;
+        }
+
+        m_files.clear();
+    }
+
+    virtual void EmbedFonts() {};
+
+    void SetAreFontsEmbedded( bool aEmbedFonts )
+    {
+        m_embedFonts = aEmbedFonts;
+    }
+
+    bool GetAreFontsEmbedded() const
+    {
+        return m_embedFonts;
+    }
+
+protected:
+    bool            m_embedFonts;        // If set, fonts will be embedded in the element on save
+                                         // Otherwise, font files embedded in the element will be
+                                         // removed on save
+
+
+private:
+    std::map<wxString, EMBEDDED_FILE*> m_files;
+    std::vector<wxString>              m_fontFiles;
+};
+
+
+
+class EMBEDDED_FILES_PARSER : public EMBEDDED_FILES_LEXER
+{
+public:
+    EMBEDDED_FILES_PARSER( LINE_READER* aReader ) :
+            EMBEDDED_FILES_LEXER( aReader )
+    {
+    }
+
+    void ParseEmbedded( EMBEDDED_FILES* aFiles );
+
+};
+#endif // EMBEDDED_FILES_H
\ No newline at end of file
diff --git a/include/filename_resolver.h b/include/filename_resolver.h
index d38a74333e..eb41e18bb0 100644
--- a/include/filename_resolver.h
+++ b/include/filename_resolver.h
@@ -36,6 +36,7 @@
 
 class PROJECT;
 class PGM_BASE;
+class EMBEDDED_FILES;
 
 struct SEARCH_PATH
 {
@@ -99,8 +100,10 @@ public:
      *
      * @param aFileName The configured file path to resolve
      * @param aWorkingPath The current working path for relative path resolutions
+     * @param aFiles The embedded files object to use for embedded file resolution
      */
-    wxString ResolvePath( const wxString& aFileName, const wxString& aWorkingPath );
+    wxString ResolvePath( const wxString& aFileName, const wxString& aWorkingPath,
+                          const EMBEDDED_FILES* aFiles );
 
     /**
      * Produce a relative path based on the existing search directories or returns the same path
diff --git a/include/font/font.h b/include/font/font.h
index 197375ee1f..0fbab48f32 100644
--- a/include/font/font.h
+++ b/include/font/font.h
@@ -141,7 +141,7 @@ public:
     virtual bool IsItalic() const  { return false; }
 
     static FONT* GetFont( const wxString& aFontName = wxEmptyString, bool aBold = false,
-                          bool aItalic = false );
+                          bool aItalic = false, const std::vector<wxString>* aEmbeddedFiles = nullptr );
     static bool IsStroke( const wxString& aFontName );
 
     const wxString& GetName() const { return m_fontName; };
diff --git a/include/font/fontconfig.h b/include/font/fontconfig.h
index 9ad5514749..cda8d9f48d 100644
--- a/include/font/fontconfig.h
+++ b/include/font/fontconfig.h
@@ -59,14 +59,18 @@ public:
      *
      * A return value of false indicates a serious error in the font system.
      */
-    FF_RESULT FindFont( const wxString& aFontName, wxString& aFontFile, int& aFaceIndex, bool aBold, bool aItalic );
+    FF_RESULT FindFont( const wxString& aFontName, wxString& aFontFile, int& aFaceIndex, bool aBold,
+                        bool aItalic, const std::vector<wxString>* aEmbeddedFiles = nullptr );
 
     /**
      * List the current available font families.
      *
      * @param aDesiredLang The desired language of font name to report back if available, otherwise it will fallback
+     * @param aEmbeddedFiles A list of embedded to use for searching fonts, if nullptr, this is not used
+     * @param aForce If true, force rebuilding the font cache
      */
-    void ListFonts( std::vector<std::string>& aFonts, const std::string& aDesiredLang );
+    void ListFonts( std::vector<std::string>& aFonts, const std::string& aDesiredLang,
+                    const std::vector<wxString>* aEmbeddedFiles = nullptr, bool aForce = false );
 
     /**
      * Set the reporter to use for reporting font substitution warnings.
diff --git a/include/font/outline_font.h b/include/font/outline_font.h
index 782aa556d9..9a9e943959 100644
--- a/include/font/outline_font.h
+++ b/include/font/outline_font.h
@@ -36,10 +36,11 @@
 #endif
 #include FT_FREETYPE_H
 #include FT_OUTLINE_H
-//#include <gal/opengl/opengl_freetype.h>
+
 #include <font/font.h>
 #include <font/glyph.h>
 #include <font/outline_decomposer.h>
+#include <embedded_files.h>
 
 #include <mutex>
 
@@ -51,6 +52,16 @@ namespace KIFONT
 class GAL_API OUTLINE_FONT : public FONT
 {
 public:
+
+    enum class EMBEDDING_PERMISSION
+    {
+        INSTALLABLE,
+        EDITABLE,
+        PRINT_PREVIEW_ONLY,
+        RESTRICTED,
+        INVALID
+    };
+
     OUTLINE_FONT();
 
     bool IsOutline() const override { return true; }
@@ -75,11 +86,16 @@ public:
         m_fakeItal = true;
     }
 
+    const wxString& GetFileName() const { return m_fontFileName; }
+
+    EMBEDDING_PERMISSION GetEmbeddingPermission() const;
+
     /**
      * Load an outline font. TrueType (.ttf) and OpenType (.otf) are supported.
      * @param aFontFileName is the (platform-specific) fully qualified name of the font file
      */
-    static OUTLINE_FONT* LoadFont( const wxString& aFontFileName, bool aBold, bool aItalic );
+    static OUTLINE_FONT* LoadFont( const wxString& aFontFileName, bool aBold, bool aItalic,
+                                   const std::vector<wxString>* aEmbeddedFiles );
 
     /**
      * Compute the distance (interline) between 2 lines of text (for multiline texts).  This is
diff --git a/include/tool/actions.h b/include/tool/actions.h
index e83a4f67e1..1d110842a1 100644
--- a/include/tool/actions.h
+++ b/include/tool/actions.h
@@ -228,6 +228,11 @@ public:
     // API
     static TOOL_ACTION pluginsReload;
 
+    // Embedding Files
+    static TOOL_ACTION embeddedFiles;
+    static TOOL_ACTION extractFile;
+    static TOOL_ACTION removeFile;
+
     ///< Cursor control event types
     enum CURSOR_EVENT_TYPE
     {
diff --git a/include/tool/embed_tool.h b/include/tool/embed_tool.h
new file mode 100644
index 0000000000..6939e4675e
--- /dev/null
+++ b/include/tool/embed_tool.h
@@ -0,0 +1,57 @@
+/*
+ * This program source code file is part of KiCad, a free EDA CAD application.
+ *
+ * Copyright (C) 2024 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 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, see <http://www.gnu.org/licenses/>.
+ */
+
+#ifndef EMBED_TOOL_H
+#define EMBED_TOOL_H
+
+#include <tool/tool_interactive.h>
+
+class wxFileName;
+class EMBEDDED_FILES;
+
+class EMBED_TOOL : public TOOL_INTERACTIVE
+{
+public:
+    EMBED_TOOL();
+
+    EMBED_TOOL( const std::string& aName );
+
+    virtual ~EMBED_TOOL() = default;
+
+    /// @copydoc TOOL_INTERACTIVE::Init()
+    bool Init() override;
+
+    /// @copydoc TOOL_INTERACTIVE::Reset()
+    void Reset( RESET_REASON aReason ) override;
+
+    int AddFile( const TOOL_EVENT& aEvent );
+
+    int RemoveFile( const TOOL_EVENT& aEvent );
+
+    std::vector<wxString> GetFileList();
+protected:
+
+    ///< @copydoc TOOL_INTERACTIVE::setTransitions();
+    void setTransitions() override;
+
+private:
+    EMBEDDED_FILES* m_files;
+};
+
+#endif /* EMBED_TOOL_H */
diff --git a/include/widgets/filedlg_open_embed_file.h b/include/widgets/filedlg_open_embed_file.h
new file mode 100644
index 0000000000..0657815d3b
--- /dev/null
+++ b/include/widgets/filedlg_open_embed_file.h
@@ -0,0 +1,55 @@
+/*
+ * This program source code file is part of KiCad, a free EDA CAD application.
+ *
+ * Copyright (C) 2024 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 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, see <http://www.gnu.org/licenses/>.
+ */
+
+#ifndef KICAD_FILEDLG_OPEN_EMBED_FILE_H
+#define KICAD_FILEDLG_OPEN_EMBED_FILE_H
+
+#include <wx/wx.h>
+#include <wx/filedlgcustomize.h>
+
+
+class FILEDLG_OPEN_EMBED_FILE : public wxFileDialogCustomizeHook
+{
+public:
+    FILEDLG_OPEN_EMBED_FILE( bool aDefaultEmbed = true ) :
+            m_embed( aDefaultEmbed )
+    {};
+
+    virtual void AddCustomControls( wxFileDialogCustomize& customizer ) override
+    {
+        m_cb = customizer.AddCheckBox( _( "Embed File" ) );
+        m_cb->SetValue( m_embed );
+    }
+
+    virtual void TransferDataFromCustomControls() override
+    {
+        m_embed = m_cb->GetValue();
+    }
+
+    bool GetEmbed() const { return m_embed; }
+
+private:
+    bool m_embed;
+
+    wxFileDialogCheckBox* m_cb = nullptr;
+
+    wxDECLARE_NO_COPY_CLASS( FILEDLG_OPEN_EMBED_FILE );
+};
+
+#endif //KICAD_FILEDLG_OPEN_EMBED_FILE_H
diff --git a/include/widgets/grid_text_button_helpers.h b/include/widgets/grid_text_button_helpers.h
index f3d9469f30..0153b82c90 100644
--- a/include/widgets/grid_text_button_helpers.h
+++ b/include/widgets/grid_text_button_helpers.h
@@ -35,6 +35,7 @@
 class wxGrid;
 class WX_GRID;
 class DIALOG_SHIM;
+class EMBEDDED_FILES;
 
 
 class GRID_CELL_TEXT_BUTTON : public wxGridCellEditor
@@ -118,8 +119,9 @@ protected:
 class GRID_CELL_URL_EDITOR : public GRID_CELL_TEXT_BUTTON
 {
 public:
-    GRID_CELL_URL_EDITOR( DIALOG_SHIM* aParent, SEARCH_STACK* aSearchStack = nullptr ) :
-            m_dlg( aParent ), m_searchStack( aSearchStack )
+    GRID_CELL_URL_EDITOR( DIALOG_SHIM* aParent, SEARCH_STACK* aSearchStack = nullptr,
+                          EMBEDDED_FILES* aFiles = nullptr ) :
+            m_dlg( aParent ), m_searchStack( aSearchStack ), m_files( aFiles )
     { }
 
     wxGridCellEditor* Clone() const override
@@ -132,6 +134,7 @@ public:
 protected:
     DIALOG_SHIM* m_dlg;
     SEARCH_STACK* m_searchStack;
+    EMBEDDED_FILES* m_files;
 };
 
 
diff --git a/include/wildcards_and_files_ext.h b/include/wildcards_and_files_ext.h
index 6686e8a84e..e7df1c4cc1 100644
--- a/include/wildcards_and_files_ext.h
+++ b/include/wildcards_and_files_ext.h
@@ -192,6 +192,8 @@ public:
 
     static const wxString GerberFileExtensionsRegex;
 
+    static const std::string KiCadUriPrefix;
+
     /**
      * @}
      */
diff --git a/pagelayout_editor/tools/pl_editor_control.cpp b/pagelayout_editor/tools/pl_editor_control.cpp
index 00a63ec184..097dd8af70 100644
--- a/pagelayout_editor/tools/pl_editor_control.cpp
+++ b/pagelayout_editor/tools/pl_editor_control.cpp
@@ -90,7 +90,7 @@ int PL_EDITOR_CONTROL::PageSetup( const TOOL_EVENT& aEvent )
 {
     m_frame->SaveCopyInUndoList();
 
-    DIALOG_PAGES_SETTINGS dlg( m_frame, drawSheetIUScale.IU_PER_MILS,
+    DIALOG_PAGES_SETTINGS dlg( m_frame, nullptr, drawSheetIUScale.IU_PER_MILS,
                                VECTOR2I( MAX_PAGE_SIZE_EESCHEMA_MILS,
                                          MAX_PAGE_SIZE_EESCHEMA_MILS ) );
     dlg.SetWksFileName( m_frame->GetCurrentFileName() );
diff --git a/pcbnew/board.cpp b/pcbnew/board.cpp
index a97bc24d8d..9b08b1330f 100644
--- a/pcbnew/board.cpp
+++ b/pcbnew/board.cpp
@@ -38,6 +38,7 @@
 #include <connectivity/connectivity_data.h>
 #include <convert_shape_list_to_polygon.h>
 #include <footprint.h>
+#include <font/outline_font.h>
 #include <lset.h>
 #include <pcb_base_frame.h>
 #include <pcb_track.h>
@@ -82,7 +83,8 @@ BOARD::BOARD() :
         m_project( nullptr ),
         m_userUnits( EDA_UNITS::MILLIMETRES ),
         m_designSettings( new BOARD_DESIGN_SETTINGS( nullptr, "board.design_settings" ) ),
-        m_NetInfo( this )
+        m_NetInfo( this ),
+        m_embedFonts( false )
 {
     // A too small value do not allow connecting 2 shapes (i.e. segments) not exactly connected
     // A too large value do not allow safely connecting 2 shapes like very short segments.
@@ -957,6 +959,26 @@ void BOARD::CacheTriangulation( PROGRESS_REPORTER* aReporter, const std::vector<
 }
 
 
+void BOARD::FixupEmbeddedData()
+{
+    for( FOOTPRINT* footprint : m_footprints )
+    {
+        for( auto& [filename, embeddedFile] : footprint->EmbeddedFileMap() )
+        {
+            EMBEDDED_FILES::EMBEDDED_FILE* file = GetEmbeddedFile( filename );
+
+            if( file )
+            {
+                embeddedFile->compressedEncodedData = file->compressedEncodedData;
+                embeddedFile->decompressedData = file->decompressedData;
+                embeddedFile->data_sha = file->data_sha;
+                embeddedFile->is_valid = file->is_valid;
+            }
+        }
+    }
+}
+
+
 void BOARD::Add( BOARD_ITEM* aBoardItem, ADD_MODE aMode, bool aSkipConnectivity )
 {
     if( aBoardItem == nullptr )
@@ -2498,6 +2520,56 @@ bool BOARD::GetBoardPolygonOutlines( SHAPE_POLY_SET& aOutlines,
 }
 
 
+EMBEDDED_FILES* BOARD::GetEmbeddedFiles()
+{
+    if( IsFootprintHolder() )
+        return static_cast<EMBEDDED_FILES*>( GetFirstFootprint() );
+
+    return static_cast<EMBEDDED_FILES*>( this );
+}
+
+
+const EMBEDDED_FILES* BOARD::GetEmbeddedFiles() const
+{
+    if( IsFootprintHolder() )
+        return static_cast<const EMBEDDED_FILES*>( GetFirstFootprint() );
+
+    return static_cast<const EMBEDDED_FILES*>( this );
+}
+
+
+void BOARD::EmbedFonts()
+{
+    std::set<KIFONT::OUTLINE_FONT*> fonts;
+
+    for( BOARD_ITEM* item : Drawings() )
+    {
+        if( EDA_TEXT* text = dynamic_cast<EDA_TEXT*>( item ) )
+        {
+            KIFONT::FONT* font = text->GetFont();
+
+            if( !font || font->IsStroke() )
+                continue;
+
+            using EMBEDDING_PERMISSION = KIFONT::OUTLINE_FONT::EMBEDDING_PERMISSION;
+            auto* outline = static_cast<KIFONT::OUTLINE_FONT*>( font );
+
+            if( outline->GetEmbeddingPermission() == EMBEDDING_PERMISSION::EDITABLE
+                || outline->GetEmbeddingPermission() == EMBEDDING_PERMISSION::INSTALLABLE )
+            {
+                fonts.insert( outline );
+            }
+        }
+    }
+
+    for( KIFONT::OUTLINE_FONT* font : fonts )
+    {
+        auto file = GetEmbeddedFiles()->AddFile( font->GetFileName(), false );
+        file->type = EMBEDDED_FILES::EMBEDDED_FILE::FILE_TYPE::FONT;
+    }
+}
+
+
 const std::vector<PAD*> BOARD::GetPads() const
 {
     std::vector<PAD*> allPads;
diff --git a/pcbnew/board.h b/pcbnew/board.h
index 2d81ab2814..aab89f5335 100644
--- a/pcbnew/board.h
+++ b/pcbnew/board.h
@@ -27,6 +27,7 @@
 
 #include <board_item_container.h>
 #include <board_stackup_manager/board_stackup.h>
+#include <embedded_files.h>
 #include <common.h> // Needed for stl hash extensions
 #include <convert_shape_list_to_polygon.h> // for OUTLINE_ERROR_HANDLER
 #include <hash.h>
@@ -284,7 +285,7 @@ enum class BOARD_USE
 /**
  * Information pertinent to a Pcbnew printed circuit board.
  */
-class BOARD : public BOARD_ITEM_CONTAINER
+class BOARD : public BOARD_ITEM_CONTAINER, public EMBEDDED_FILES
 {
 public:
     static inline bool ClassOf( const EDA_ITEM* aItem )
@@ -426,6 +427,13 @@ public:
      */
     void FinalizeBulkRemove( std::vector<BOARD_ITEM*>& aRemovedItems );
 
+    /**
+     * After loading a file from disk, the footprints do not yet contain the full
+     * data for their embedded files, only a reference.  This iterates over all footprints
+     * in the board and updates them with the full embedded data.
+    */
+    void FixupEmbeddedData();
+
     void CacheTriangulation( PROGRESS_REPORTER* aReporter = nullptr,
                              const std::vector<ZONE*>& aZones = {} );
 
@@ -1246,6 +1254,14 @@ public:
     bool LegacyTeardrops() const { return m_legacyTeardrops; }
     void SetLegacyTeardrops( bool aFlag ) { m_legacyTeardrops = aFlag; }
 
+    EMBEDDED_FILES* GetEmbeddedFiles() override;
+    const EMBEDDED_FILES* GetEmbeddedFiles() const;
+
+    /**
+     * Finds all fonts used in the board and embeds them in the file if permissions allow
+    */
+    void EmbedFonts() override;
+
     // --------- Item order comparators ---------
 
     struct cmp_items
@@ -1359,6 +1375,8 @@ private:
     NETINFO_LIST                 m_NetInfo;         // net info list (name, design constraints...
 
     std::vector<BOARD_LISTENER*> m_listeners;
+
+    bool                         m_embedFonts;
 };
 
 
diff --git a/pcbnew/dialogs/dialog_board_setup.cpp b/pcbnew/dialogs/dialog_board_setup.cpp
index b10989b3fb..bc354c02d8 100644
--- a/pcbnew/dialogs/dialog_board_setup.cpp
+++ b/pcbnew/dialogs/dialog_board_setup.cpp
@@ -31,6 +31,7 @@
 #include <dialog_import_settings.h>
 #include <pcb_io/pcb_io.h>
 #include <pcb_io/pcb_io_mgr.h>
+#include <panel_embedded_files.h>
 #include <dialogs/panel_setup_severities.h>
 #include <dialogs/panel_setup_rules.h>
 #include <dialogs/panel_setup_teardrops.h>
@@ -203,6 +204,14 @@ DIALOG_BOARD_SETUP::DIALOG_BOARD_SETUP( PCB_EDIT_FRAME* aFrame ) :
                                                    board->GetDesignSettings().m_DRCSeverities );
             }, _( "Violation Severity" ) );
 
+    m_treebook->AddPage( new wxPanel( GetTreebook() ), _( "Board Data" ) );
+    m_embeddedFilesPage = m_treebook->GetPageCount();
+    m_treebook->AddLazySubPage(
+            [this]( wxWindow* aParent ) -> wxWindow*
+            {
+                return new PANEL_EMBEDDED_FILES( aParent, m_frame->GetBoard() );
+            }, _( "Embedded Files" ) );
+
     for( size_t i = 0; i < m_treebook->GetPageCount(); ++i )
         m_treebook->ExpandNode( i );
 
diff --git a/pcbnew/dialogs/dialog_board_setup.h b/pcbnew/dialogs/dialog_board_setup.h
index ad1c066dd7..c0fa517b2e 100644
--- a/pcbnew/dialogs/dialog_board_setup.h
+++ b/pcbnew/dialogs/dialog_board_setup.h
@@ -73,6 +73,7 @@ private:
     size_t m_netclassesPage;
     size_t m_customRulesPage;
     size_t m_severitiesPage;
+    size_t m_embeddedFilesPage;
 };
 
 
diff --git a/pcbnew/dialogs/dialog_footprint_properties.cpp b/pcbnew/dialogs/dialog_footprint_properties.cpp
index 89ec820b87..ce79df7c0c 100644
--- a/pcbnew/dialogs/dialog_footprint_properties.cpp
+++ b/pcbnew/dialogs/dialog_footprint_properties.cpp
@@ -42,8 +42,8 @@
 #include <widgets/text_ctrl_eval.h>
 #include <widgets/std_bitmap_button.h>
 #include <settings/settings_manager.h>
+#include <panel_embedded_files.h>
 #include <panel_fp_properties_3d_model.h>
-#include <dialogs/3d_cache_dialogs.h>
 #include <dialogs/panel_preview_3d_model.h>
 #include <dialog_footprint_properties.h>
 
@@ -75,6 +75,9 @@ DIALOG_FOOTPRINT_PROPERTIES::DIALOG_FOOTPRINT_PROPERTIES( PCB_EDIT_FRAME* aParen
     m_3dPanel = new PANEL_FP_PROPERTIES_3D_MODEL( m_frame, m_footprint, this, m_NoteBook );
     m_NoteBook->AddPage( m_3dPanel, _("3D Models"), false );
 
+    m_embeddedFiles = new PANEL_EMBEDDED_FILES( m_NoteBook, m_footprint );
+    m_NoteBook->AddPage( m_embeddedFiles, _( "Embedded Files" ) );
+
     // Configure display origin transforms
     m_posX.SetCoordType( ORIGIN_TRANSFORMS::ABS_X_COORD );
     m_posY.SetCoordType( ORIGIN_TRANSFORMS::ABS_Y_COORD );
@@ -261,6 +264,9 @@ bool DIALOG_FOOTPRINT_PROPERTIES::TransferDataToWindow()
     if( !m_3dPanel->TransferDataToWindow() )
         return false;
 
+    if( !m_embeddedFiles->TransferDataToWindow() )
+        return false;
+
     // Footprint Fields
     for( PCB_FIELD* srcField : m_footprint->GetFields() )
     {
@@ -494,6 +500,9 @@ bool DIALOG_FOOTPRINT_PROPERTIES::TransferDataFromWindow()
     if( !m_3dPanel->TransferDataFromWindow() )
         return false;
 
+    if( !m_embeddedFiles->TransferDataFromWindow() )
+        return false;
+
     auto view = m_frame->GetCanvas()->GetView();
     BOARD_COMMIT commit( m_frame );
     commit.Modify( m_footprint );
diff --git a/pcbnew/dialogs/dialog_footprint_properties.h b/pcbnew/dialogs/dialog_footprint_properties.h
index 1d1070a826..d4e2b4421a 100644
--- a/pcbnew/dialogs/dialog_footprint_properties.h
+++ b/pcbnew/dialogs/dialog_footprint_properties.h
@@ -36,6 +36,7 @@
 
 class PCB_EDIT_FRAME;
 class PANEL_FP_PROPERTIES_3D_MODEL;
+class PANEL_EMBEDDED_FILES;
 
 class DIALOG_FOOTPRINT_PROPERTIES: public DIALOG_FOOTPRINT_PROPERTIES_BASE
 {
@@ -106,6 +107,7 @@ private:
 
     wxSize                           m_gridSize;
     wxSize                           m_lastRequestedSize;
+    PANEL_EMBEDDED_FILES*            m_embeddedFiles;
 };
 
 
diff --git a/pcbnew/dialogs/dialog_footprint_properties_fp_editor.cpp b/pcbnew/dialogs/dialog_footprint_properties_fp_editor.cpp
index 8361f71319..3b50b05529 100644
--- a/pcbnew/dialogs/dialog_footprint_properties_fp_editor.cpp
+++ b/pcbnew/dialogs/dialog_footprint_properties_fp_editor.cpp
@@ -45,7 +45,6 @@
 #include "filename_resolver.h"
 #include <pgm_base.h>
 #include "dialogs/panel_preview_3d_model.h"
-#include "dialogs/3d_cache_dialogs.h"
 #include <settings/settings_manager.h>
 #include <tool/tool_manager.h>
 #include <tools/pcb_selection_tool.h>
diff --git a/pcbnew/dialogs/dialog_global_edit_text_and_graphics.cpp b/pcbnew/dialogs/dialog_global_edit_text_and_graphics.cpp
index ca79598122..d347c5bc9f 100644
--- a/pcbnew/dialogs/dialog_global_edit_text_and_graphics.cpp
+++ b/pcbnew/dialogs/dialog_global_edit_text_and_graphics.cpp
@@ -397,7 +397,8 @@ void DIALOG_GLOBAL_EDIT_TEXT_AND_GRAPHICS::processItem( BOARD_COMMIT& aCommit, B
                 if( !text->GetFontName().IsEmpty() )
                 {
                     text->SetFont( KIFONT::FONT::GetFont( text->GetFontName(), text->IsBold(),
-                                                          text->IsItalic() ) );
+                                                          text->IsItalic(),
+                                                          m_parent->GetBoard()->GetEmbeddedFiles()->GetFontFiles() ) );
                 }
             }
 
diff --git a/pcbnew/dialogs/panel_fp_properties_3d_model.cpp b/pcbnew/dialogs/panel_fp_properties_3d_model.cpp
index 336e3b2e69..c27e1dec1b 100644
--- a/pcbnew/dialogs/panel_fp_properties_3d_model.cpp
+++ b/pcbnew/dialogs/panel_fp_properties_3d_model.cpp
@@ -27,6 +27,7 @@
 #include <panel_fp_properties_3d_model.h>
 
 #include <3d_viewer/eda_3d_viewer_frame.h>
+#include <dialogs/dialog_configure_paths.h>
 #include <env_vars.h>
 #include <bitmaps.h>
 #include <widgets/grid_icon_text_helpers.h>
@@ -35,18 +36,21 @@
 #include <widgets/std_bitmap_button.h>
 #include <footprint.h>
 #include <fp_lib_table.h>
+#include <footprint.h>
 #include <footprint_edit_frame.h>
 #include <footprint_editor_settings.h>
 #include <dialog_footprint_properties_fp_editor.h>
 #include "filename_resolver.h"
 #include <pgm_base.h>
 #include <kiplatform/ui.h>
-#include "dialogs/panel_preview_3d_model.h"
-#include "dialogs/3d_cache_dialogs.h"
+#include <dialogs/panel_preview_3d_model.h>
+#include <dialogs/dialog_select_3d_model.h>
 #include <settings/settings_manager.h>
-#include <wx/defs.h>
 #include <project_pcb.h>
 
+#include <wx/defs.h>
+#include <wx/msgdlg.h>
+
 enum MODELS_TABLE_COLUMNS
 {
     COL_PROBLEM  = 0,
@@ -147,6 +151,13 @@ bool PANEL_FP_PROPERTIES_3D_MODEL::TransferDataFromWindow()
     if( !m_modelsGrid->CommitPendingChanges() )
         return false;
 
+    FOOTPRINT* fp = m_previewPane->GetDummyFootprint();
+
+    for( const auto& [name, file] : fp->EmbeddedFileMap() )
+    {
+        if( !m_footprint->HasFile( name ) )
+            m_footprint->AddFile( new EMBEDDED_FILES::EMBEDDED_FILE( *file ) );
+    }
     return true;
 }
 
@@ -294,14 +305,16 @@ void PANEL_FP_PROPERTIES_3D_MODEL::OnAdd3DModel( wxCommandEvent&  )
 
     int selected = m_modelsGrid->GetGridCursorRow();
 
-    PROJECT&   prj = m_frame->Prj();
-    FP_3DMODEL model;
+    PROJECT&           prj = m_frame->Prj();
+    FP_3DMODEL         model;
+    S3D_CACHE*         cache = PROJECT_PCB::Get3DCacheManager( &prj );
+    FILENAME_RESOLVER* res = cache->GetResolver();
 
     wxString initialpath = prj.GetRString( PROJECT::VIEWER_3D_PATH );
     wxString sidx = prj.GetRString( PROJECT::VIEWER_3D_FILTER_INDEX );
     int      filter = 0;
 
-    // If the PROJECT::VIEWER_3D_PATH hasn't been set yet, use the KICAD7_3DMODEL_DIR environment
+    // If the PROJECT::VIEWER_3D_PATH hasn't been set yet, use the 3DMODEL_DIR environment
     // variable and fall back to the project path if necessary.
     if( initialpath.IsEmpty() )
     {
@@ -321,8 +334,13 @@ void PANEL_FP_PROPERTIES_3D_MODEL::OnAdd3DModel( wxCommandEvent&  )
             filter = (int) tmp;
     }
 
-    if( !S3D::Select3DModel( m_parentDialog, PROJECT_PCB::Get3DCacheManager( &m_frame->Prj() ), initialpath, filter, &model )
-        || model.m_Filename.empty() )
+
+    DIALOG_SELECT_3DMODEL dm( m_parentDialog, cache, &model, initialpath, filter );
+
+    // Use QuasiModal so that Configure3DPaths (and its help window) will work
+    int retval = dm.ShowQuasiModal();
+
+    if( retval != wxID_OK || model.m_Filename.empty() )
     {
         if( selected >= 0 )
         {
@@ -333,10 +351,47 @@ void PANEL_FP_PROPERTIES_3D_MODEL::OnAdd3DModel( wxCommandEvent&  )
         return;
     }
 
+    if( dm.IsEmbedded3DModel() )
+    {
+        wxString libraryName = m_footprint->GetFPID().GetLibNickname();
+        const FP_LIB_TABLE_ROW* fpRow = nullptr;
+
+        wxString footprintBasePath = wxEmptyString;
+
+        try
+        {
+            fpRow = PROJECT_PCB::PcbFootprintLibs( &m_frame->Prj() )->FindRow( libraryName, false );
+
+            if( fpRow )
+                footprintBasePath = fpRow->GetFullURI( true );
+        }
+        catch( ... )
+        {
+            // if libraryName is not found in table, do nothing
+        }
+
+
+        wxString fullPath = res->ResolvePath( model.m_Filename, footprintBasePath, nullptr );
+        wxFileName fname( fullPath );
+
+        EMBEDDED_FILES::EMBEDDED_FILE* result = m_previewPane->GetDummyFootprint()->AddFile( fname, false );
+
+        if( !result )
+        {
+
+            wxString msg = wxString::Format( _( "Error adding 3D model" ) );
+            wxMessageBox( msg, _( "Error" ), wxICON_ERROR | wxOK, this );
+            return;
+        }
+
+        model.m_Filename = result->GetLink();
+    }
+
+
     prj.SetRString( PROJECT::VIEWER_3D_PATH, initialpath );
     sidx = wxString::Format( wxT( "%i" ), filter );
     prj.SetRString( PROJECT::VIEWER_3D_FILTER_INDEX, sidx );
-    FILENAME_RESOLVER* res = PROJECT_PCB::Get3DCacheManager( &m_frame->Prj() )->GetResolver();
+
     wxString alias;
     wxString shortPath;
     wxString filename = model.m_Filename;
@@ -466,7 +521,7 @@ MODEL_VALIDATE_ERRORS PANEL_FP_PROPERTIES_3D_MODEL::validateModelExists( const w
     if( fpRow )
         footprintBasePath = fpRow->GetFullURI( true );
 
-    wxString fullPath = resolv->ResolvePath( aFilename, footprintBasePath );
+    wxString fullPath = resolv->ResolvePath( aFilename, footprintBasePath, m_footprint );
 
     if( fullPath.IsEmpty() )
         return MODEL_VALIDATE_ERRORS::RESOLVE_FAIL;
@@ -480,7 +535,9 @@ MODEL_VALIDATE_ERRORS PANEL_FP_PROPERTIES_3D_MODEL::validateModelExists( const w
 
 void PANEL_FP_PROPERTIES_3D_MODEL::Cfg3DPath( wxCommandEvent& event )
 {
-    if( S3D::Configure3DPaths( this, PROJECT_PCB::Get3DCacheManager( &m_frame->Prj() )->GetResolver() ) )
+    DIALOG_CONFIGURE_PATHS dlg( this );
+
+    if( dlg.ShowQuasiModal() == wxID_OK )
         m_previewPane->UpdateDummyFootprint();
 }
 
diff --git a/pcbnew/exporters/export_idf.cpp b/pcbnew/exporters/export_idf.cpp
index af161c954c..9fb17b3dd6 100644
--- a/pcbnew/exporters/export_idf.cpp
+++ b/pcbnew/exporters/export_idf.cpp
@@ -437,7 +437,7 @@ static void idf_export_footprint( BOARD* aPcb, FOOTPRINT* aFootprint, IDF3_BOARD
             continue;
         }
 
-        idfFile.Assign( resolver->ResolvePath( sM->m_Filename, footprintBasePath ) );
+        idfFile.Assign( resolver->ResolvePath( sM->m_Filename, footprintBasePath, aFootprint ) );
         idfExt = idfFile.GetExt();
 
         if( idfExt.Cmp( wxT( "idf" ) ) && idfExt.Cmp( wxT( "IDF" ) ) )
diff --git a/pcbnew/exporters/exporter_vrml.cpp b/pcbnew/exporters/exporter_vrml.cpp
index 3c6e789fa6..4f70168c02 100644
--- a/pcbnew/exporters/exporter_vrml.cpp
+++ b/pcbnew/exporters/exporter_vrml.cpp
@@ -1046,7 +1046,7 @@ void EXPORTER_PCB_VRML::ExportVrmlFootprint( FOOTPRINT* aFootprint, std::ostream
             continue;
         }
 
-        SGNODE* mod3d = (SGNODE*) m_Cache3Dmodels->Load( sM->m_Filename, footprintBasePath );
+        SGNODE* mod3d = (SGNODE*) m_Cache3Dmodels->Load( sM->m_Filename, footprintBasePath, aFootprint );
 
         if( nullptr == mod3d )
         {
@@ -1112,7 +1112,7 @@ void EXPORTER_PCB_VRML::ExportVrmlFootprint( FOOTPRINT* aFootprint, std::ostream
             aOutputFile->precision( m_precision );
 
             wxFileName srcFile =
-                    m_Cache3Dmodels->GetResolver()->ResolvePath( sM->m_Filename, wxEmptyString );
+                    m_Cache3Dmodels->GetResolver()->ResolvePath( sM->m_Filename, wxEmptyString, aFootprint );
             wxFileName dstFile;
             dstFile.SetPath( m_Subdir3DFpModels );
             dstFile.SetName( srcFile.GetName() );
diff --git a/pcbnew/exporters/step/exporter_step.cpp b/pcbnew/exporters/step/exporter_step.cpp
index ed5f1b7af8..6274bc831d 100644
--- a/pcbnew/exporters/step/exporter_step.cpp
+++ b/pcbnew/exporters/step/exporter_step.cpp
@@ -333,7 +333,7 @@ bool EXPORTER_STEP::buildFootprint3DShapes( FOOTPRINT* aFootprint, VECTOR2D aOri
             continue;
 
         std::vector<wxString> searchedPaths;
-        wxString mname = m_resolver->ResolvePath( fp_model.m_Filename, footprintBasePath );
+        wxString mname = m_resolver->ResolvePath( fp_model.m_Filename, footprintBasePath, aFootprint );
 
 
         if( mname.empty() || !wxFileName::FileExists( mname ) )
diff --git a/pcbnew/files.cpp b/pcbnew/files.cpp
index ca03b5428f..bd135a35c7 100644
--- a/pcbnew/files.cpp
+++ b/pcbnew/files.cpp
@@ -387,6 +387,7 @@ bool PCB_EDIT_FRAME::Files_io_from_id( int id )
             return false;
 
         LoadProjectSettings();
+        LoadDrawingSheet();
 
         onBoardLoaded();
 
@@ -935,6 +936,7 @@ bool PCB_EDIT_FRAME::OpenProjectFiles( const std::vector<wxString>& aFileSet, in
 
     // Load project settings after setting up board; some of them depend on the nets list
     LoadProjectSettings();
+    LoadDrawingSheet();
 
     // Syncs the UI (appearance panel, etc) with the loaded board and project
     onBoardLoaded();
diff --git a/pcbnew/footprint.cpp b/pcbnew/footprint.cpp
index 7b457fcf28..8d4f0f17da 100644
--- a/pcbnew/footprint.cpp
+++ b/pcbnew/footprint.cpp
@@ -25,35 +25,39 @@
  */
 #include <magic_enum.hpp>
 
-#include <core/mirror.h>
-#include <confirm.h>
-#include <refdes_utils.h>
-#include <bitmaps.h>
 #include <unordered_set>
-#include <string_utils.h>
-#include <pcb_edit_frame.h>
+
+#include <bitmaps.h>
 #include <board.h>
 #include <board_design_settings.h>
+#include <confirm.h>
+#include <convert_basic_shapes_to_polygon.h>
+#include <convert_shape_list_to_polygon.h>
+#include <core/mirror.h>
+#include <drc/drc_item.h>
+#include <embedded_files.h>
+#include <font/font.h>
+#include <font/outline_font.h>
+#include <footprint.h>
+#include <geometry/convex_hull.h>
+#include <geometry/shape_segment.h>
+#include <geometry/shape_simple.h>
+#include <i18n_utility.h>
 #include <lset.h>
 #include <macros.h>
 #include <pad.h>
-#include <pcb_marker.h>
-#include <pcb_group.h>
-#include <pcb_track.h>
 #include <pcb_dimension.h>
+#include <pcb_edit_frame.h>
+#include <pcb_field.h>
+#include <pcb_group.h>
+#include <pcb_marker.h>
 #include <pcb_reference_image.h>
 #include <pcb_textbox.h>
-#include <pcb_field.h>
-#include <footprint.h>
-#include <zone.h>
+#include <pcb_track.h>
+#include <refdes_utils.h>
+#include <string_utils.h>
 #include <view/view.h>
-#include <i18n_utility.h>
-#include <drc/drc_item.h>
-#include <geometry/shape_segment.h>
-#include <geometry/shape_simple.h>
-#include <convert_shape_list_to_polygon.h>
-#include <geometry/convex_hull.h>
-#include "convert_basic_shapes_to_polygon.h"
+#include <zone.h>
 
 #include <google/protobuf/any.pb.h>
 #include <api/board/board_types.pb.h>
@@ -79,6 +83,7 @@ FOOTPRINT::FOOTPRINT( BOARD* parent ) :
     m_lastEditTime = 0;
     m_zoneConnection          = ZONE_CONNECTION::INHERITED;
     m_fileFormatVersionAtLoad = 0;
+    m_embedFonts = false;
 
     // These are the mandatory fields for the editor to work
     for( int i = 0; i < MANDATORY_FIELDS; i++ )
@@ -118,6 +123,7 @@ FOOTPRINT::FOOTPRINT( const FOOTPRINT& aFootprint ) :
     m_lastEditTime = aFootprint.m_lastEditTime;
     m_link         = aFootprint.m_link;
     m_path         = aFootprint.m_path;
+    m_embedFonts   = aFootprint.m_embedFonts;
 
     m_cachedBoundingBox              = aFootprint.m_cachedBoundingBox;
     m_boundingBoxCacheTimeStamp      = aFootprint.m_boundingBoxCacheTimeStamp;
@@ -198,6 +204,9 @@ FOOTPRINT::FOOTPRINT( const FOOTPRINT& aFootprint ) :
         }
     }
 
+    for( auto& [ name, file ] : aFootprint.EmbeddedFileMap() )
+        AddFile( new EMBEDDED_FILES::EMBEDDED_FILE( *file ) );
+
     // Copy auxiliary data
     m_3D_Drawings   = aFootprint.m_3D_Drawings;
     m_libDescription = aFootprint.m_libDescription;
@@ -3702,6 +3711,39 @@ void FOOTPRINT::TransformFPShapesToPolySet( SHAPE_POLY_SET& aBuffer, PCB_LAYER_I
 }
 
 
+void FOOTPRINT::EmbedFonts()
+{
+    using OUTLINE_FONT = KIFONT::OUTLINE_FONT;
+    using EMBEDDING_PERMISSION = OUTLINE_FONT::EMBEDDING_PERMISSION;
+
+    std::set<OUTLINE_FONT*> fonts;
+
+    for( BOARD_ITEM* item : GraphicalItems() )
+    {
+        if( auto* text = dynamic_cast<EDA_TEXT*>( item ) )
+        {
+            if( auto* font = text->GetFont(); font && !font->IsStroke() )
+            {
+                auto* outline = static_cast<OUTLINE_FONT*>( font );
+                auto permission = outline->GetEmbeddingPermission();
+
+                if( permission == EMBEDDING_PERMISSION::EDITABLE
+                    || permission == EMBEDDING_PERMISSION::INSTALLABLE )
+                {
+                    fonts.insert( outline );
+                }
+            }
+        }
+    }
+
+    for( auto* font : fonts )
+    {
+        auto file = GetEmbeddedFiles()->AddFile( font->GetFileName(), false );
+        file->type = EMBEDDED_FILES::EMBEDDED_FILE::FILE_TYPE::FONT;
+    }
+}
+
+
 static struct FOOTPRINT_DESC
 {
     FOOTPRINT_DESC()
diff --git a/pcbnew/footprint.h b/pcbnew/footprint.h
index 6331c90585..197d5c69de 100644
--- a/pcbnew/footprint.h
+++ b/pcbnew/footprint.h
@@ -32,6 +32,7 @@
 #include <board_item_container.h>
 #include <board_item.h>
 #include <collectors.h>
+#include <embedded_files.h>
 #include <layer_ids.h> // ALL_LAYERS definition.
 #include <lset.h>
 #include <lib_id.h>
@@ -103,7 +104,7 @@ public:
 };
 
 
-class FOOTPRINT : public BOARD_ITEM_CONTAINER
+class FOOTPRINT : public BOARD_ITEM_CONTAINER, public EMBEDDED_FILES
 {
 public:
     FOOTPRINT( BOARD* parent );
@@ -968,6 +969,18 @@ public:
     std::shared_ptr<SHAPE> GetEffectiveShape( PCB_LAYER_ID aLayer = UNDEFINED_LAYER,
                                               FLASHING aFlash = FLASHING::DEFAULT ) const override;
 
+    EMBEDDED_FILES* GetEmbeddedFiles() override
+    {
+        return static_cast<EMBEDDED_FILES*>( this );
+    }
+
+    const EMBEDDED_FILES* GetEmbeddedFiles() const
+    {
+        return static_cast<const EMBEDDED_FILES*>( this );
+    }
+
+    void EmbedFonts() override;
+
     double Similarity( const BOARD_ITEM& aOther ) const override;
 
     bool operator==( const BOARD_ITEM& aOther ) const override;
diff --git a/pcbnew/footprint_edit_frame.cpp b/pcbnew/footprint_edit_frame.cpp
index 4eeb28871a..1ce6b5f0b6 100644
--- a/pcbnew/footprint_edit_frame.cpp
+++ b/pcbnew/footprint_edit_frame.cpp
@@ -20,6 +20,7 @@
  * with this program.  If not, see <http://www.gnu.org/licenses/>.
  */
 
+#include "tool/embed_tool.h"
 #include "tools/convert_tool.h"
 #include "tools/drawing_tool.h"
 #include "tools/edit_tool.h"
@@ -1117,6 +1118,7 @@ void FOOTPRINT_EDIT_FRAME::setupTools()
     m_toolManager->RegisterTool( new CONVERT_TOOL );
     m_toolManager->RegisterTool( new SCRIPTING_TOOL );
     m_toolManager->RegisterTool( new PROPERTIES_TOOL );
+    m_toolManager->RegisterTool( new EMBED_TOOL );
 
     for( TOOL_BASE* tool : m_toolManager->Tools() )
     {
@@ -1281,6 +1283,7 @@ void FOOTPRINT_EDIT_FRAME::setupUIConditions()
 
     CURRENT_EDIT_TOOL( ACTIONS::deleteTool );
     CURRENT_EDIT_TOOL( ACTIONS::measureTool );
+    CURRENT_EDIT_TOOL( ACTIONS::embeddedFiles );
     CURRENT_EDIT_TOOL( PCB_ACTIONS::placePad );
     CURRENT_EDIT_TOOL( PCB_ACTIONS::drawLine );
     CURRENT_EDIT_TOOL( PCB_ACTIONS::drawRectangle );
diff --git a/pcbnew/pcb_edit_frame.cpp b/pcbnew/pcb_edit_frame.cpp
index 4ac4a1e04b..e36b8e5915 100644
--- a/pcbnew/pcb_edit_frame.cpp
+++ b/pcbnew/pcb_edit_frame.cpp
@@ -62,6 +62,7 @@
 #include <tool/action_toolbar.h>
 #include <tool/common_control.h>
 #include <tool/common_tools.h>
+#include <tool/embed_tool.h>
 #include <tool/properties_tool.h>
 #include <tool/selection.h>
 #include <tool/zoom_tool.h>
@@ -705,6 +706,7 @@ void PCB_EDIT_FRAME::setupTools()
     m_toolManager->RegisterTool( new GENERATOR_TOOL );
     m_toolManager->RegisterTool( new SCRIPTING_TOOL );
     m_toolManager->RegisterTool( new PROPERTIES_TOOL );
+    m_toolManager->RegisterTool( new EMBED_TOOL );
     m_toolManager->InitTools();
 
     for( TOOL_BASE* tool : m_toolManager->Tools() )
@@ -992,6 +994,7 @@ void PCB_EDIT_FRAME::setupUIConditions()
                                                            .Enable( isDRCIdle ) )
 
     // These tools edit the board, so they must be disabled during some operations
+    CURRENT_EDIT_TOOL( ACTIONS::embeddedFiles );
     CURRENT_EDIT_TOOL( ACTIONS::deleteTool );
     CURRENT_EDIT_TOOL( PCB_ACTIONS::placeFootprint );
     CURRENT_EDIT_TOOL( PCB_ACTIONS::routeSingleTrack);
diff --git a/pcbnew/pcb_edit_frame.h b/pcbnew/pcb_edit_frame.h
index 48850564f3..1d7a3ffcd5 100644
--- a/pcbnew/pcb_edit_frame.h
+++ b/pcbnew/pcb_edit_frame.h
@@ -219,6 +219,11 @@ public:
 
     void SaveSettings( APP_SETTINGS_BASE* aCfg ) override;
 
+    /**
+     * Load the drawing sheet file.
+     */
+    void LoadDrawingSheet();
+
     /**
      * Get the last path for a particular type.
      *
diff --git a/pcbnew/pcb_fields_grid_table.cpp b/pcbnew/pcb_fields_grid_table.cpp
index e3a8d1615d..0200e3452b 100644
--- a/pcbnew/pcb_fields_grid_table.cpp
+++ b/pcbnew/pcb_fields_grid_table.cpp
@@ -21,13 +21,18 @@
  * 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA
  */
 
-#include <kiway_player.h>
-#include <project.h>
-#include <pcb_fields_grid_table.h>
-#include <widgets/grid_combobox.h>
-#include <trigo.h>
-#include <pcb_base_frame.h>
+#include <board.h>
 #include <footprint.h>
+#include <footprint_edit_frame.h>
+#include <kiway.h>
+#include <kiway_player.h>
+#include <pcb_fields_grid_table.h>
+#include <pcb_base_frame.h>
+#include <pcb_edit_frame.h>
+#include <project.h>
+#include <trigo.h>
+#include <widgets/grid_combobox.h>
+
 #include "grid_layer_box_helpers.h"
 #include <widgets/grid_text_button_helpers.h>
 #include <widgets/grid_text_helpers.h>
@@ -95,8 +100,36 @@ PCB_FIELDS_GRID_TABLE::PCB_FIELDS_GRID_TABLE( PCB_BASE_FRAME* aFrame, DIALOG_SHI
     fpIdEditor->SetValidator( m_nonUrlValidator );
     m_footprintAttr->SetEditor( fpIdEditor );
 
+    EMBEDDED_FILES* files = nullptr;
+
+    // In the case of the footprint editor, we need to distinguish between the footprint
+    // in the library where the embedded files are stored with the footprint and the footprint
+    // from the board where the embedded files are stored with the board.
+    if( m_frame->GetFrameType() == FRAME_FOOTPRINT_EDITOR )
+    {
+        FOOTPRINT_EDIT_FRAME* fpFrame = static_cast<FOOTPRINT_EDIT_FRAME*>( m_frame );
+
+        if( fpFrame->IsCurrentFPFromBoard() )
+        {
+            PCB_EDIT_FRAME* pcbframe = (PCB_EDIT_FRAME*) m_frame->Kiway().Player( FRAME_PCB_EDITOR, false );
+
+            if( pcbframe != nullptr )       // happens when the board editor is not active (or closed)
+            {
+                files = pcbframe->GetBoard();
+            }
+        }
+        else
+        {
+            files = fpFrame->GetBoard()->GetFirstFootprint();
+        }
+    }
+    else if( m_frame->GetFrameType() == FRAME_PCB_EDITOR )
+    {
+        files = static_cast<PCB_EDIT_FRAME*>( m_frame )->GetBoard();
+    }
+
     m_urlAttr = new wxGridCellAttr;
-    GRID_CELL_URL_EDITOR* urlEditor = new GRID_CELL_URL_EDITOR( m_dialog );
+    GRID_CELL_URL_EDITOR* urlEditor = new GRID_CELL_URL_EDITOR( m_dialog, nullptr, files );
     urlEditor->SetValidator( m_urlValidator );
     m_urlAttr->SetEditor( urlEditor );
 
diff --git a/pcbnew/pcb_io/altium/CMakeLists.txt b/pcbnew/pcb_io/altium/CMakeLists.txt
index 171c6f085b..86aa41d70a 100644
--- a/pcbnew/pcb_io/altium/CMakeLists.txt
+++ b/pcbnew/pcb_io/altium/CMakeLists.txt
@@ -4,6 +4,7 @@
 set( ALTIUM2PCBNEW_SRCS
     altium_parser_pcb.cpp
     altium_pcb.cpp
+    altium_pcb_compound_file.cpp
     altium_rule_transformer.cpp
     pcb_io_altium_circuit_maker.cpp
     pcb_io_altium_circuit_studio.cpp
diff --git a/pcbnew/pcb_io/altium/altium_parser_pcb.cpp b/pcbnew/pcb_io/altium/altium_parser_pcb.cpp
index 4d50d7ad6e..a69e810fbc 100644
--- a/pcbnew/pcb_io/altium/altium_parser_pcb.cpp
+++ b/pcbnew/pcb_io/altium/altium_parser_pcb.cpp
@@ -462,6 +462,9 @@ AMODEL::AMODEL( ALTIUM_BINARY_PARSER& aReader )
     rotation.y = ALTIUM_PROPS_UTILS::ReadDouble( properties, wxT( "ROTY" ), 0. );
     rotation.z = ALTIUM_PROPS_UTILS::ReadDouble( properties, wxT( "ROTZ" ), 0. );
 
+    z_offset = ALTIUM_PROPS_UTILS::ReadDouble( properties, wxT( "DZ" ), 0. );
+    checksum = ALTIUM_PROPS_UTILS::ReadInt( properties, wxT( "CHECKSUM" ), 0 );
+
     if( aReader.HasParsingError() )
         THROW_IO_ERROR( wxT( "Model stream was not parsed correctly" ) );
 }
@@ -704,7 +707,7 @@ ACOMPONENTBODY6::ACOMPONENTBODY6( ALTIUM_BINARY_PARSER& aReader )
 
     rotation = ALTIUM_PROPS_UTILS::ReadDouble( properties, wxT( "MODEL.2D.ROTATION" ), 0. );
 
-    bodyOpacity = ALTIUM_PROPS_UTILS::ReadDouble( properties, wxT( "BODYOPACITY3D" ), 1. );
+    body_opacity_3d = ALTIUM_PROPS_UTILS::ReadDouble( properties, wxT( "BODYOPACITY3D" ), 1. );
 
     aReader.SkipSubrecord();
 
diff --git a/pcbnew/pcb_io/altium/altium_parser_pcb.h b/pcbnew/pcb_io/altium/altium_parser_pcb.h
index e5dcfd4929..5450db753b 100644
--- a/pcbnew/pcb_io/altium/altium_parser_pcb.h
+++ b/pcbnew/pcb_io/altium/altium_parser_pcb.h
@@ -460,6 +460,8 @@ struct AMODEL
     bool     isEmbedded;
 
     VECTOR3D rotation;
+    double   z_offset;
+    int32_t  checksum;
 
     explicit AMODEL( ALTIUM_BINARY_PARSER& aReader );
 };
@@ -592,14 +594,37 @@ struct ACOMPONENTBODY6
 {
     uint16_t             component;
 
-    wxString             modelName;
+    wxString             body_name;
+    int                  kind;
+    int                  subpolyindex;
+    int                  unionindex;
+    int                  arc_resolution;
+    bool                 is_shape_based;
+    int                  cavity_height;
+    int                  standoff_height;
+    int                  overall_height;
+    int                  body_projection;
+    int                  body_color_3d;
+    int                  body_opacity_3d;
+    wxString             identifier;
+    wxString             texture;
+    int                  texture_center_x;
+    int                  texture_center_y;
+    int                  texture_size_x;
+    int                  texture_size_y;
+    int                  texture_rotation;
+
     wxString             modelId;
+    wxString             modelChecksum;
     bool                 modelIsEmbedded;
+    wxString             modelName;
+    int                  modelType;
+    int                  modelSource;
+    int                  modelSnapCount;
 
     VECTOR3D             modelPosition;
     VECTOR3D             modelRotation;
     double               rotation;
-    double               bodyOpacity;
 
     explicit ACOMPONENTBODY6( ALTIUM_BINARY_PARSER& aReader );
 };
diff --git a/pcbnew/pcb_io/altium/altium_pcb.cpp b/pcbnew/pcb_io/altium/altium_pcb.cpp
index 8c82926774..66dda9f162 100644
--- a/pcbnew/pcb_io/altium/altium_pcb.cpp
+++ b/pcbnew/pcb_io/altium/altium_pcb.cpp
@@ -24,6 +24,7 @@
 
 #include "altium_pcb.h"
 #include "altium_parser_pcb.h"
+#include <altium_pcb_compound_file.h>
 #include <io/altium/altium_binary_parser.h>
 #include <io/altium/altium_parser_utils.h>
 
@@ -313,114 +314,114 @@ void ALTIUM_PCB::checkpoint()
     }
 }
 
-void ALTIUM_PCB::Parse( const ALTIUM_COMPOUND_FILE&                  altiumPcbFile,
+void ALTIUM_PCB::Parse( const ALTIUM_PCB_COMPOUND_FILE&                  altiumPcbFile,
                         const std::map<ALTIUM_PCB_DIR, std::string>& aFileMapping )
 {
     // this vector simply declares in which order which functions to call.
     const std::vector<std::tuple<bool, ALTIUM_PCB_DIR, PARSE_FUNCTION_POINTER_fp>> parserOrder = {
         { true, ALTIUM_PCB_DIR::FILE_HEADER,
-          [this]( const ALTIUM_COMPOUND_FILE& aFile, auto fileHeader )
+          [this]( const ALTIUM_PCB_COMPOUND_FILE& aFile, auto fileHeader )
           {
               this->ParseFileHeader( aFile, fileHeader );
           } },
         { true, ALTIUM_PCB_DIR::BOARD6,
-          [this]( const ALTIUM_COMPOUND_FILE& aFile, auto fileHeader )
+          [this]( const ALTIUM_PCB_COMPOUND_FILE& aFile, auto fileHeader )
           {
               this->ParseBoard6Data( aFile, fileHeader );
           } },
         { false, ALTIUM_PCB_DIR::EXTENDPRIMITIVEINFORMATION,
-          [this]( const ALTIUM_COMPOUND_FILE& aFile, auto fileHeader )
+          [this]( const ALTIUM_PCB_COMPOUND_FILE& aFile, auto fileHeader )
           {
               this->ParseExtendedPrimitiveInformationData( aFile, fileHeader );
           } },
         { true, ALTIUM_PCB_DIR::COMPONENTS6,
-          [this]( const ALTIUM_COMPOUND_FILE& aFile, auto fileHeader )
+          [this]( const ALTIUM_PCB_COMPOUND_FILE& aFile, auto fileHeader )
           {
               this->ParseComponents6Data( aFile, fileHeader );
           } },
         { false, ALTIUM_PCB_DIR::MODELS,
-          [this, aFileMapping]( const ALTIUM_COMPOUND_FILE& aFile, auto fileHeader )
+          [this, aFileMapping]( const ALTIUM_PCB_COMPOUND_FILE& aFile, auto fileHeader )
           {
               std::vector<std::string> dir{ aFileMapping.at( ALTIUM_PCB_DIR::MODELS ) };
               this->ParseModelsData( aFile, fileHeader, dir );
           } },
         { true, ALTIUM_PCB_DIR::COMPONENTBODIES6,
-          [this]( const ALTIUM_COMPOUND_FILE& aFile, auto fileHeader )
+          [this]( const ALTIUM_PCB_COMPOUND_FILE& aFile, auto fileHeader )
           {
               this->ParseComponentsBodies6Data( aFile, fileHeader );
           } },
         { true, ALTIUM_PCB_DIR::NETS6,
-          [this]( const ALTIUM_COMPOUND_FILE& aFile, auto fileHeader )
+          [this]( const ALTIUM_PCB_COMPOUND_FILE& aFile, auto fileHeader )
           {
               this->ParseNets6Data( aFile, fileHeader );
           } },
         { true, ALTIUM_PCB_DIR::CLASSES6,
-          [this]( const ALTIUM_COMPOUND_FILE& aFile, auto fileHeader )
+          [this]( const ALTIUM_PCB_COMPOUND_FILE& aFile, auto fileHeader )
           {
               this->ParseClasses6Data( aFile, fileHeader );
           } },
         { true, ALTIUM_PCB_DIR::RULES6,
-          [this]( const ALTIUM_COMPOUND_FILE& aFile, auto fileHeader )
+          [this]( const ALTIUM_PCB_COMPOUND_FILE& aFile, auto fileHeader )
           {
               this->ParseRules6Data( aFile, fileHeader );
           } },
         { true, ALTIUM_PCB_DIR::DIMENSIONS6,
-          [this]( const ALTIUM_COMPOUND_FILE& aFile, auto fileHeader )
+          [this]( const ALTIUM_PCB_COMPOUND_FILE& aFile, auto fileHeader )
           {
               this->ParseDimensions6Data( aFile, fileHeader );
           } },
         { true, ALTIUM_PCB_DIR::POLYGONS6,
-          [this]( const ALTIUM_COMPOUND_FILE& aFile, auto fileHeader )
+          [this]( const ALTIUM_PCB_COMPOUND_FILE& aFile, auto fileHeader )
           {
               this->ParsePolygons6Data( aFile, fileHeader );
           } },
         { true, ALTIUM_PCB_DIR::ARCS6,
-          [this]( const ALTIUM_COMPOUND_FILE& aFile, auto fileHeader )
+          [this]( const ALTIUM_PCB_COMPOUND_FILE& aFile, auto fileHeader )
           {
               this->ParseArcs6Data( aFile, fileHeader );
           } },
         { true, ALTIUM_PCB_DIR::PADS6,
-          [this]( const ALTIUM_COMPOUND_FILE& aFile, auto fileHeader )
+          [this]( const ALTIUM_PCB_COMPOUND_FILE& aFile, auto fileHeader )
           {
               this->ParsePads6Data( aFile, fileHeader );
           } },
         { true, ALTIUM_PCB_DIR::VIAS6,
-          [this]( const ALTIUM_COMPOUND_FILE& aFile, auto fileHeader )
+          [this]( const ALTIUM_PCB_COMPOUND_FILE& aFile, auto fileHeader )
           {
               this->ParseVias6Data( aFile, fileHeader );
           } },
         { true, ALTIUM_PCB_DIR::TRACKS6,
-          [this]( const ALTIUM_COMPOUND_FILE& aFile, auto fileHeader )
+          [this]( const ALTIUM_PCB_COMPOUND_FILE& aFile, auto fileHeader )
           {
               this->ParseTracks6Data( aFile, fileHeader );
           } },
         { false, ALTIUM_PCB_DIR::WIDESTRINGS6,
-          [this]( const ALTIUM_COMPOUND_FILE& aFile, auto fileHeader )
+          [this]( const ALTIUM_PCB_COMPOUND_FILE& aFile, auto fileHeader )
           {
               this->ParseWideStrings6Data( aFile, fileHeader );
           } },
         { true, ALTIUM_PCB_DIR::TEXTS6,
-          [this]( const ALTIUM_COMPOUND_FILE& aFile, auto fileHeader )
+          [this]( const ALTIUM_PCB_COMPOUND_FILE& aFile, auto fileHeader )
           {
               this->ParseTexts6Data( aFile, fileHeader );
           } },
         { true, ALTIUM_PCB_DIR::FILLS6,
-          [this]( const ALTIUM_COMPOUND_FILE& aFile, auto fileHeader )
+          [this]( const ALTIUM_PCB_COMPOUND_FILE& aFile, auto fileHeader )
           {
               this->ParseFills6Data( aFile, fileHeader );
           } },
         { false, ALTIUM_PCB_DIR::BOARDREGIONS,
-          [this]( const ALTIUM_COMPOUND_FILE& aFile, auto fileHeader )
+          [this]( const ALTIUM_PCB_COMPOUND_FILE& aFile, auto fileHeader )
           {
               this->ParseBoardRegionsData( aFile, fileHeader );
           } },
         { true, ALTIUM_PCB_DIR::SHAPEBASEDREGIONS6,
-          [this]( const ALTIUM_COMPOUND_FILE& aFile, auto fileHeader )
+          [this]( const ALTIUM_PCB_COMPOUND_FILE& aFile, auto fileHeader )
           {
               this->ParseShapeBasedRegions6Data( aFile, fileHeader );
           } },
         { true, ALTIUM_PCB_DIR::REGIONS6,
-          [this]( const ALTIUM_COMPOUND_FILE& aFile, auto fileHeader )
+          [this]( const ALTIUM_PCB_COMPOUND_FILE& aFile, auto fileHeader )
           {
               this->ParseRegions6Data( aFile, fileHeader );
           } }
@@ -654,8 +655,8 @@ void ALTIUM_PCB::Parse( const ALTIUM_COMPOUND_FILE&                  altiumPcbFi
 }
 
 
-FOOTPRINT* ALTIUM_PCB::ParseFootprint( ALTIUM_COMPOUND_FILE& altiumLibFile,
-                                       const wxString&             aFootprintName )
+FOOTPRINT* ALTIUM_PCB::ParseFootprint( ALTIUM_PCB_COMPOUND_FILE& altiumLibFile,
+                                       const wxString&       aFootprintName )
 {
     std::unique_ptr<FOOTPRINT> footprint = std::make_unique<FOOTPRINT>( m_board );
 
@@ -675,7 +676,8 @@ FOOTPRINT* ALTIUM_PCB::ParseFootprint( ALTIUM_COMPOUND_FILE& altiumLibFile,
     //        ParseWideStrings6Data( altiumLibFile, unicodeStringsData );
     //    }
 
-    std::tuple<wxString, const CFB::COMPOUND_FILE_ENTRY*> ret = altiumLibFile.FindLibFootprintDirName(aFootprintName);
+    std::tuple<wxString, const CFB::COMPOUND_FILE_ENTRY*> ret =
+            altiumLibFile.FindLibFootprintDirName( aFootprintName );
 
     wxString fpDirName = std::get<0>( ret );
     const CFB::COMPOUND_FILE_ENTRY* footprintStream = std::get<1>( ret );
@@ -803,7 +805,7 @@ FOOTPRINT* ALTIUM_PCB::ParseFootprint( ALTIUM_COMPOUND_FILE& altiumLibFile,
         case ALTIUM_RECORD::MODEL:
         {
             ACOMPONENTBODY6 componentBody( parser );
-            // Won't be supported for now, as we would need to extract the model
+            ConvertComponentBody6ToFootprintItem( altiumLibFile, footprint.get(), componentBody );
             break;
         }
         default:
@@ -909,7 +911,7 @@ const ARULE6* ALTIUM_PCB::GetRuleDefault( ALTIUM_RULE_KIND aKind ) const
     return nullptr;
 }
 
-void ALTIUM_PCB::ParseFileHeader( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+void ALTIUM_PCB::ParseFileHeader( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                                   const CFB::COMPOUND_FILE_ENTRY* aEntry )
 {
     ALTIUM_BINARY_PARSER reader( aAltiumPcbFile, aEntry );
@@ -927,7 +929,7 @@ void ALTIUM_PCB::ParseFileHeader( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile
 }
 
 
-void ALTIUM_PCB::ParseExtendedPrimitiveInformationData( const ALTIUM_COMPOUND_FILE& aAltiumPcbFile,
+void ALTIUM_PCB::ParseExtendedPrimitiveInformationData( const ALTIUM_PCB_COMPOUND_FILE& aAltiumPcbFile,
                                                         const CFB::COMPOUND_FILE_ENTRY* aEntry )
 {
     if( m_progressReporter )
@@ -949,7 +951,7 @@ void ALTIUM_PCB::ParseExtendedPrimitiveInformationData( const ALTIUM_COMPOUND_FI
 }
 
 
-void ALTIUM_PCB::ParseBoard6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+void ALTIUM_PCB::ParseBoard6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                                   const CFB::COMPOUND_FILE_ENTRY* aEntry )
 {
     if( m_progressReporter )
@@ -1201,7 +1203,7 @@ void ALTIUM_PCB::HelperCreateBoardOutline( const std::vector<ALTIUM_VERTICE>& aV
 }
 
 
-void ALTIUM_PCB::ParseClasses6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+void ALTIUM_PCB::ParseClasses6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                                     const CFB::COMPOUND_FILE_ENTRY* aEntry )
 {
     if( m_progressReporter )
@@ -1253,7 +1255,7 @@ void ALTIUM_PCB::ParseClasses6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFi
 }
 
 
-void ALTIUM_PCB::ParseComponents6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+void ALTIUM_PCB::ParseComponents6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                                        const CFB::COMPOUND_FILE_ENTRY* aEntry )
 {
     if( m_progressReporter )
@@ -1315,7 +1317,85 @@ double normalizeAngleDegrees( double Angle, double aMin, double aMax )
 }
 
 
-void ALTIUM_PCB::ParseComponentsBodies6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+void ALTIUM_PCB::ConvertComponentBody6ToFootprintItem( const ALTIUM_PCB_COMPOUND_FILE& aAltiumPcbFile,
+                                                       FOOTPRINT* aFootprint,
+                                                       const ACOMPONENTBODY6& aElem )
+{
+    if( m_progressReporter )
+        m_progressReporter->Report( _( "Loading component 3D models..." ) );
+
+    if( !aElem.modelIsEmbedded )
+        return;
+
+    auto model = aAltiumPcbFile.GetLibModel( aElem.modelId );
+
+    if( !model )
+    {
+        if( m_reporter )
+        {
+            m_reporter->Report( wxString::Format( wxT( "Model %s not found for footprint %s" ),
+                                                  aElem.modelId, aFootprint->GetReference() ),
+                                RPT_SEVERITY_ERROR );
+        }
+
+        return;
+    }
+
+    const VECTOR2I& fpPosition = aFootprint->GetPosition();
+
+    EMBEDDED_FILES::EMBEDDED_FILE* file = new EMBEDDED_FILES::EMBEDDED_FILE();
+    file->name = aElem.modelName;
+    // Decompress the model data before assigning
+    std::vector<char> decompressedData;
+    wxMemoryInputStream compressedStream(model->second.data(), model->second.size());
+    wxZlibInputStream zlibStream(compressedStream);
+
+    decompressedData.reserve(model->second.size() * 2); // Reserve some space, assuming decompressed data is larger
+    while (!zlibStream.Eof()) {
+        size_t currentSize = decompressedData.size();
+        decompressedData.resize(currentSize + 4096); // Increase buffer size
+        zlibStream.Read(decompressedData.data() + currentSize, 4096);
+        size_t bytesRead = zlibStream.LastRead();
+        decompressedData.resize(currentSize + bytesRead); // Resize to actual read size
+    }
+
+    file->decompressedData = std::move( decompressedData );
+    file->type = EMBEDDED_FILES::EMBEDDED_FILE::FILE_TYPE::MODEL;
+
+    EMBEDDED_FILES::CompressAndEncode( *file );
+    aFootprint->GetEmbeddedFiles()->AddFile( file );
+
+    FP_3DMODEL modelSettings;
+
+    modelSettings.m_Filename = aFootprint->GetEmbeddedFiles()->GetEmbeddedFileLink( *file );
+
+    modelSettings.m_Offset.x = pcbIUScale.IUTomm((int) aElem.modelPosition.x - fpPosition.x );
+    modelSettings.m_Offset.y = -pcbIUScale.IUTomm((int) aElem.modelPosition.y - fpPosition.y );
+    modelSettings.m_Offset.z = pcbIUScale.IUTomm( (int) aElem.modelPosition.z + model->first.z_offset );
+
+    EDA_ANGLE orientation = aFootprint->GetOrientation();
+
+    if( aFootprint->IsFlipped() )
+    {
+        modelSettings.m_Offset.y = -modelSettings.m_Offset.y;
+        orientation              = -orientation;
+    }
+
+    RotatePoint( &modelSettings.m_Offset.x, &modelSettings.m_Offset.y, orientation );
+
+    modelSettings.m_Rotation.x = normalizeAngleDegrees( -aElem.modelRotation.x + model->first.rotation.x, -180, 180 );
+    modelSettings.m_Rotation.y = normalizeAngleDegrees( -aElem.modelRotation.y + model->first.rotation.y, -180, 180 );
+    modelSettings.m_Rotation.z = normalizeAngleDegrees( -aElem.modelRotation.z + aElem.rotation
+                                                                               + orientation.AsDegrees()
+                                                                               + model->first.rotation.z,
+                                                                                 -180, 180 );
+    modelSettings.m_Opacity = aElem.body_opacity_3d;
+
+    aFootprint->Models().push_back( modelSettings );
+}
+
+
+void ALTIUM_PCB::ParseComponentsBodies6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                                              const CFB::COMPOUND_FILE_ENTRY* aEntry )
 {
     if( m_progressReporter )
@@ -1326,7 +1406,7 @@ void ALTIUM_PCB::ParseComponentsBodies6Data( const ALTIUM_COMPOUND_FILE&     aAl
     while( reader.GetRemainingBytes() >= 4 /* TODO: use Header section of file */ )
     {
         checkpoint();
-        ACOMPONENTBODY6 elem( reader ); // TODO: implement
+        ACOMPONENTBODY6 elem( reader );
 
         if( elem.component == ALTIUM_COMPONENT_NONE )
             continue; // TODO: we do not support components for the board yet
@@ -1334,7 +1414,7 @@ void ALTIUM_PCB::ParseComponentsBodies6Data( const ALTIUM_COMPOUND_FILE&     aAl
         if( m_components.size() <= elem.component )
         {
             THROW_IO_ERROR( wxString::Format( wxT( "ComponentsBodies6 stream tries to access "
-                                                   "component id %d of %d existing components" ),
+                                                   "component id %d of %zu existing components" ),
                                               elem.component,
                                               m_components.size() ) );
         }
@@ -1342,9 +1422,9 @@ void ALTIUM_PCB::ParseComponentsBodies6Data( const ALTIUM_COMPOUND_FILE&     aAl
         if( !elem.modelIsEmbedded )
             continue;
 
-        auto modelTuple = m_models.find( elem.modelId );
+        auto modelTuple = m_EmbeddedModels.find( elem.modelId );
 
-        if( modelTuple == m_models.end() )
+        if( modelTuple == m_EmbeddedModels.end() )
         {
             if( m_reporter )
             {
@@ -1357,34 +1437,36 @@ void ALTIUM_PCB::ParseComponentsBodies6Data( const ALTIUM_COMPOUND_FILE&     aAl
             continue;
         }
 
-        FOOTPRINT*     footprint  = m_components.at( elem.component );
-        const VECTOR2I& fpPosition = footprint->GetPosition();
+        FOOTPRINT*      footprint  = m_components.at( elem.component );
+
+        EMBEDDED_FILES::EMBEDDED_FILE* file = new EMBEDDED_FILES::EMBEDDED_FILE();
+        file->name = modelTuple->second.m_modelname;
+
+        wxMemoryInputStream compressedStream(modelTuple->second.m_data.data(), modelTuple->second.m_data.size());
+        wxZlibInputStream zlibStream(compressedStream);
+        wxMemoryOutputStream decompressedStream;
+        zlibStream.Read(decompressedStream);
+        file->decompressedData.resize(decompressedStream.GetSize());
+        decompressedStream.CopyTo(file->decompressedData.data(), file->decompressedData.size());
+
+        EMBEDDED_FILES::CompressAndEncode( *file );
+        m_board->GetEmbeddedFiles()->AddFile( file );
 
         FP_3DMODEL modelSettings;
 
-        modelSettings.m_Filename = modelTuple->second;
+        modelSettings.m_Filename = footprint->GetEmbeddedFiles()->GetEmbeddedFileLink( *file );
 
-        modelSettings.m_Offset.x = pcbIUScale.IUTomm((int) elem.modelPosition.x - fpPosition.x );
-        modelSettings.m_Offset.y = -pcbIUScale.IUTomm((int) elem.modelPosition.y - fpPosition.y );
-        modelSettings.m_Offset.z = pcbIUScale.IUTomm( (int) elem.modelPosition.z );
+        modelSettings.m_Offset.x = pcbIUScale.IUTomm((int) elem.modelPosition.x );
+        modelSettings.m_Offset.y = -pcbIUScale.IUTomm((int) elem.modelPosition.y );
+        modelSettings.m_Offset.z = pcbIUScale.IUTomm( (int) elem.modelPosition.z + modelTuple->second.m_z_offset );
 
-        EDA_ANGLE orientation = footprint->GetOrientation();
-
-        if( footprint->IsFlipped() )
-        {
-            modelSettings.m_Offset.y = -modelSettings.m_Offset.y;
-            orientation              = -orientation;
-        }
-
-        RotatePoint( &modelSettings.m_Offset.x, &modelSettings.m_Offset.y, orientation );
-
-        modelSettings.m_Rotation.x = normalizeAngleDegrees( -elem.modelRotation.x, -180, 180 );
-        modelSettings.m_Rotation.y = normalizeAngleDegrees( -elem.modelRotation.y, -180, 180 );
+        modelSettings.m_Rotation.x = normalizeAngleDegrees( -elem.modelRotation.x + modelTuple->second.m_rotation.x, -180, 180 );
+        modelSettings.m_Rotation.y = normalizeAngleDegrees( -elem.modelRotation.y + modelTuple->second.m_rotation.y, -180, 180 );
         modelSettings.m_Rotation.z = normalizeAngleDegrees( -elem.modelRotation.z
                                                                         + elem.rotation
-                                                                        + orientation.AsDegrees(),
+                                                                        + modelTuple->second.m_rotation.z,
                                                             -180, 180 );
-        modelSettings.m_Opacity = elem.bodyOpacity;
+        modelSettings.m_Opacity = elem.body_opacity_3d;
 
         footprint->Models().push_back( modelSettings );
     }
@@ -1756,7 +1838,7 @@ void ALTIUM_PCB::HelperParseDimensions6Center( const ADIMENSION6& aElem )
 }
 
 
-void ALTIUM_PCB::ParseDimensions6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+void ALTIUM_PCB::ParseDimensions6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                                        const CFB::COMPOUND_FILE_ENTRY* aEntry )
 {
     if( m_progressReporter )
@@ -1840,7 +1922,7 @@ void ALTIUM_PCB::ParseDimensions6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPc
 }
 
 
-void ALTIUM_PCB::ParseModelsData( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+void ALTIUM_PCB::ParseModelsData( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                                   const CFB::COMPOUND_FILE_ENTRY* aEntry,
                                   const std::vector<std::string>& aRootDir )
 {
@@ -1852,37 +1934,6 @@ void ALTIUM_PCB::ParseModelsData( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile
     if( reader.GetRemainingBytes() == 0 )
         return;
 
-    wxString projectPath = wxPathOnly( m_board->GetFileName() );
-    // TODO: set KIPRJMOD always after import (not only when loading project)?
-    wxSetEnv( PROJECT_VAR_NAME, projectPath );
-
-    // TODO: make this path configurable?
-    const wxString altiumModelDir = wxT( "ALTIUM_EMBEDDED_MODELS" );
-
-    wxFileName altiumModelsPath = wxFileName::DirName( projectPath );
-    wxString   kicadModelPrefix = wxT( "${KIPRJMOD}/" ) + altiumModelDir + wxT( "/" );
-
-    if( !altiumModelsPath.AppendDir( altiumModelDir ) )
-        THROW_IO_ERROR( wxT( "Cannot construct directory path for step models" ) );
-
-    // Create dir if it does not exist
-    if( !altiumModelsPath.DirExists() )
-    {
-        if( !altiumModelsPath.Mkdir() )
-        {
-            if( m_reporter )
-            {
-                wxString msg;
-                msg.Printf( _( "Failed to create folder '%s'." ) + wxS( " " )
-                          + _( "No 3D-models will be imported." ),
-                            altiumModelsPath.GetFullPath() );
-                m_reporter->Report( msg, RPT_SEVERITY_ERROR );
-            }
-
-            return;
-        }
-    }
-
     int      idx = 0;
     wxString invalidChars = wxFileName::GetForbiddenChars();
 
@@ -1898,7 +1949,6 @@ void ALTIUM_PCB::ParseModelsData( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile
                                    wxString::npos == elem.name.find_first_of( invalidChars );
         wxString       storageName = !validName ? wxString::Format( wxT( "model_%d" ), idx )
                                                 : elem.name;
-        wxFileName     storagePath( altiumModelsPath.GetPath(), storageName );
 
         idx++;
 
@@ -1924,40 +1974,9 @@ void ALTIUM_PCB::ParseModelsData( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile
         aAltiumPcbFile.GetCompoundFileReader().ReadFile( stepEntry, 0, stepContent.data(),
                                                          stepSize );
 
-        if( !storagePath.IsDirWritable() )
-        {
-            if( m_reporter )
-            {
-                wxString msg;
-                msg.Printf( _( "Insufficient permissions to save file '%s'." ),
-                            storagePath.GetFullPath() );
-                m_reporter->Report( msg, RPT_SEVERITY_ERROR );
-            }
-
-            continue;
-        }
-
-        wxMemoryInputStream stepStream( stepContent.data(), stepSize );
-        wxZlibInputStream   zlibInputStream( stepStream );
-
-        wxFFileOutputStream outputStream( storagePath.GetFullPath() );
-
-        if( !outputStream.IsOk() )
-        {
-            if( m_reporter )
-            {
-                wxString msg;
-                msg.Printf( _( "Unable to write file '%s'." ), storagePath.GetFullPath() );
-                m_reporter->Report( msg, RPT_SEVERITY_ERROR );
-            }
-
-            continue;
-        }
-
-        outputStream.Write( zlibInputStream );
-        outputStream.Close();
-
-        m_models.insert( { elem.id, kicadModelPrefix + storageName } );
+        m_EmbeddedModels.insert( std::make_pair(
+                elem.id, ALTIUM_EMBEDDED_MODEL_DATA( storageName, elem.rotation, elem.z_offset,
+                                                     std::move( stepContent ) ) ) );
     }
 
     if( reader.GetRemainingBytes() != 0 )
@@ -1965,7 +1984,7 @@ void ALTIUM_PCB::ParseModelsData( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile
 }
 
 
-void ALTIUM_PCB::ParseNets6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+void ALTIUM_PCB::ParseNets6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                                  const CFB::COMPOUND_FILE_ENTRY* aEntry )
 {
     if( m_progressReporter )
@@ -1991,7 +2010,7 @@ void ALTIUM_PCB::ParseNets6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
         THROW_IO_ERROR( wxT( "Nets6 stream is not fully parsed" ) );
 }
 
-void ALTIUM_PCB::ParsePolygons6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+void ALTIUM_PCB::ParsePolygons6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                                      const CFB::COMPOUND_FILE_ENTRY* aEntry )
 {
     if( m_progressReporter )
@@ -2158,7 +2177,7 @@ void ALTIUM_PCB::ParsePolygons6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbF
         THROW_IO_ERROR( wxT( "Polygons6 stream is not fully parsed" ) );
 }
 
-void ALTIUM_PCB::ParseRules6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+void ALTIUM_PCB::ParseRules6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                                   const CFB::COMPOUND_FILE_ENTRY* aEntry )
 {
     if( m_progressReporter )
@@ -2226,7 +2245,7 @@ void ALTIUM_PCB::ParseRules6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile
         THROW_IO_ERROR( wxT( "Rules6 stream is not fully parsed" ) );
 }
 
-void ALTIUM_PCB::ParseBoardRegionsData( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+void ALTIUM_PCB::ParseBoardRegionsData( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                                         const CFB::COMPOUND_FILE_ENTRY* aEntry )
 {
     if( m_progressReporter )
@@ -2246,7 +2265,7 @@ void ALTIUM_PCB::ParseBoardRegionsData( const ALTIUM_COMPOUND_FILE&     aAltiumP
         THROW_IO_ERROR( wxT( "BoardRegions stream is not fully parsed" ) );
 }
 
-void ALTIUM_PCB::ParseShapeBasedRegions6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+void ALTIUM_PCB::ParseShapeBasedRegions6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                                               const CFB::COMPOUND_FILE_ENTRY* aEntry )
 {
     if( m_progressReporter )
@@ -2680,7 +2699,7 @@ void ALTIUM_PCB::ConvertShapeBasedRegions6ToFootprintItemOnLayer( FOOTPRINT*
 }
 
 
-void ALTIUM_PCB::ParseRegions6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+void ALTIUM_PCB::ParseRegions6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                                     const CFB::COMPOUND_FILE_ENTRY* aEntry )
 {
     if( m_progressReporter )
@@ -2755,7 +2774,7 @@ void ALTIUM_PCB::ParseRegions6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFi
 }
 
 
-void ALTIUM_PCB::ParseArcs6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+void ALTIUM_PCB::ParseArcs6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                                  const CFB::COMPOUND_FILE_ENTRY* aEntry )
 {
     if( m_progressReporter )
@@ -3001,7 +3020,7 @@ void ALTIUM_PCB::ConvertArcs6ToFootprintItemOnLayer( FOOTPRINT* aFootprint, cons
 }
 
 
-void ALTIUM_PCB::ParsePads6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+void ALTIUM_PCB::ParsePads6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                                  const CFB::COMPOUND_FILE_ENTRY* aEntry )
 {
     if( m_progressReporter )
@@ -3710,7 +3729,7 @@ void ALTIUM_PCB::HelperParsePad6NonCopper( const APAD6& aElem, PCB_LAYER_ID aLay
 }
 
 
-void ALTIUM_PCB::ParseVias6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+void ALTIUM_PCB::ParseVias6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                                  const CFB::COMPOUND_FILE_ENTRY* aEntry )
 {
     if( m_progressReporter )
@@ -3785,7 +3804,7 @@ void ALTIUM_PCB::ParseVias6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
         THROW_IO_ERROR( wxT( "Vias6 stream is not fully parsed" ) );
 }
 
-void ALTIUM_PCB::ParseTracks6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+void ALTIUM_PCB::ParseTracks6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                                    const CFB::COMPOUND_FILE_ENTRY* aEntry )
 {
     if( m_progressReporter )
@@ -4002,7 +4021,7 @@ void ALTIUM_PCB::ConvertTracks6ToFootprintItemOnLayer( FOOTPRINT* aFootprint, co
 }
 
 
-void ALTIUM_PCB::ParseWideStrings6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+void ALTIUM_PCB::ParseWideStrings6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                                         const CFB::COMPOUND_FILE_ENTRY* aEntry )
 {
     if( m_progressReporter )
@@ -4016,7 +4035,7 @@ void ALTIUM_PCB::ParseWideStrings6Data( const ALTIUM_COMPOUND_FILE&     aAltiumP
         THROW_IO_ERROR( wxT( "WideStrings6 stream is not fully parsed" ) );
 }
 
-void ALTIUM_PCB::ParseTexts6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+void ALTIUM_PCB::ParseTexts6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                                   const CFB::COMPOUND_FILE_ENTRY* aEntry )
 {
     if( m_progressReporter )
@@ -4324,7 +4343,7 @@ void ALTIUM_PCB::ConvertTexts6ToEdaTextSettings( const ATEXT6& aElem, EDA_TEXT&
 }
 
 
-void ALTIUM_PCB::ParseFills6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+void ALTIUM_PCB::ParseFills6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                                   const CFB::COMPOUND_FILE_ENTRY* aEntry )
 {
     if( m_progressReporter )
diff --git a/pcbnew/pcb_io/altium/altium_pcb.h b/pcbnew/pcb_io/altium/altium_pcb.h
index 01077c75b9..69489fa452 100644
--- a/pcbnew/pcb_io/altium/altium_pcb.h
+++ b/pcbnew/pcb_io/altium/altium_pcb.h
@@ -97,11 +97,24 @@ namespace CFB
 struct COMPOUND_FILE_ENTRY;
 } // namespace CFB
 
-class ALTIUM_COMPOUND_FILE;
+class ALTIUM_PCB_COMPOUND_FILE;
+
+// Structure for storing embedded model data
+struct ALTIUM_EMBEDDED_MODEL_DATA
+{
+    wxString m_modelname;
+    VECTOR3D m_rotation;
+    double   m_z_offset;
+    std::vector<char> m_data;
+
+    // Constructor
+    ALTIUM_EMBEDDED_MODEL_DATA(const wxString& name, const VECTOR3D& rotation, double z_offset, std::vector<char>&& data)
+        : m_modelname(name), m_rotation(rotation), m_z_offset(z_offset), m_data(std::move(data)) {}
+};
 
 // type declaration required for a helper method
 class ALTIUM_PCB;
-typedef std::function<void( const ALTIUM_COMPOUND_FILE&, const CFB::COMPOUND_FILE_ENTRY* )>
+typedef std::function<void( const ALTIUM_PCB_COMPOUND_FILE&, const CFB::COMPOUND_FILE_ENTRY* )>
         PARSE_FUNCTION_POINTER_fp;
 
 class ALTIUM_PCB
@@ -114,11 +127,11 @@ public:
                          const wxString& aFootprintName = wxEmptyString);
     ~ALTIUM_PCB();
 
-    void Parse( const ALTIUM_COMPOUND_FILE&                  aAltiumPcbFile,
+    void Parse( const ALTIUM_PCB_COMPOUND_FILE&                  aAltiumPcbFile,
                 const std::map<ALTIUM_PCB_DIR, std::string>& aFileMapping );
 
-    FOOTPRINT* ParseFootprint( ALTIUM_COMPOUND_FILE& altiumLibFile,
-                               const wxString&             aFootprintName );
+    FOOTPRINT* ParseFootprint( ALTIUM_PCB_COMPOUND_FILE& altiumLibFile,
+                               const wxString&       aFootprintName );
 
 private:
     void checkpoint();
@@ -129,30 +142,30 @@ private:
     const ARULE6* GetRule( ALTIUM_RULE_KIND aKind, const wxString& aName ) const;
     const ARULE6* GetRuleDefault( ALTIUM_RULE_KIND aKind ) const;
 
-    void ParseFileHeader( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+    void ParseFileHeader( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                           const CFB::COMPOUND_FILE_ENTRY* aEntry );
 
     // Text Format
-    void ParseBoard6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+    void ParseBoard6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                           const CFB::COMPOUND_FILE_ENTRY* aEntry );
-    void ParseClasses6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+    void ParseClasses6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                             const CFB::COMPOUND_FILE_ENTRY* aEntry );
-    void ParseComponents6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+    void ParseComponents6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                                const CFB::COMPOUND_FILE_ENTRY* aEntry );
-    void ParseDimensions6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+    void ParseDimensions6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                                const CFB::COMPOUND_FILE_ENTRY* aEntry );
-    void ParseModelsData( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+    void ParseModelsData( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                           const CFB::COMPOUND_FILE_ENTRY* aEntry,
                           const std::vector<std::string>& aRootDir );
-    void ParseNets6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+    void ParseNets6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                          const CFB::COMPOUND_FILE_ENTRY* aEntry );
-    void ParsePolygons6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+    void ParsePolygons6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                              const CFB::COMPOUND_FILE_ENTRY* aEntry );
-    void ParseRules6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+    void ParseRules6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                           const CFB::COMPOUND_FILE_ENTRY* aEntry );
 
     // Binary Format
-    void ParseArcs6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+    void ParseArcs6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                          const CFB::COMPOUND_FILE_ENTRY* aEntry );
     void ConvertArcs6ToPcbShape( const AARC6& aElem, PCB_SHAPE* aShape );
     void ConvertArcs6ToBoardItem( const AARC6& aElem, const int aPrimitiveIndex );
@@ -161,19 +174,22 @@ private:
     void ConvertArcs6ToBoardItemOnLayer( const AARC6& aElem, PCB_LAYER_ID aLayer );
     void ConvertArcs6ToFootprintItemOnLayer( FOOTPRINT* aFootprint, const AARC6& aElem,
                                              PCB_LAYER_ID aLayer );
-    void ParseComponentsBodies6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+    void ParseComponentsBodies6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                                      const CFB::COMPOUND_FILE_ENTRY* aEntry );
-    void ParsePads6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+    void ConvertComponentBody6ToFootprintItem( const ALTIUM_PCB_COMPOUND_FILE& aAltiumPcbFile,
+                                              FOOTPRINT* aFootprint,
+                                              const ACOMPONENTBODY6& aElem );
+    void ParsePads6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                          const CFB::COMPOUND_FILE_ENTRY* aEntry );
     void ConvertPads6ToBoardItem( const APAD6& aElem );
     void ConvertPads6ToFootprintItem( FOOTPRINT* aFootprint, const APAD6& aElem );
     void ConvertPads6ToBoardItemOnNonCopper( const APAD6& aElem );
     void ConvertPads6ToFootprintItemOnCopper( FOOTPRINT* aFootprint, const APAD6& aElem );
     void ConvertPads6ToFootprintItemOnNonCopper( FOOTPRINT* aFootprint, const APAD6& aElem );
-    void ParseVias6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+    void ParseVias6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                          const CFB::COMPOUND_FILE_ENTRY* aEntry );
     void ConvertVias6ToFootprintItem( FOOTPRINT* aFootprint, const AVIA6& aElem );
-    void ParseTracks6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+    void ParseTracks6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                            const CFB::COMPOUND_FILE_ENTRY* aEntry );
     void ConvertTracks6ToBoardItem( const ATRACK6& aElem, const int aPrimitiveIndex );
     void ConvertTracks6ToFootprintItem( FOOTPRINT* aFootprint, const ATRACK6& aElem,
@@ -181,7 +197,7 @@ private:
     void ConvertTracks6ToBoardItemOnLayer( const ATRACK6& aElem, PCB_LAYER_ID aLayer );
     void ConvertTracks6ToFootprintItemOnLayer( FOOTPRINT* aFootprint, const ATRACK6& aElem,
                                                PCB_LAYER_ID aLayer );
-    void ParseTexts6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+    void ParseTexts6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                           const CFB::COMPOUND_FILE_ENTRY* aEntry );
     void ConvertTexts6ToBoardItem( const ATEXT6& aElem );
     void ConvertTexts6ToFootprintItem( FOOTPRINT* aFootprint, const ATEXT6& aElem );
@@ -189,7 +205,7 @@ private:
     void ConvertTexts6ToFootprintItemOnLayer( FOOTPRINT* aFootprint, const ATEXT6& aElem,
                                               PCB_LAYER_ID aLayer );
     void ConvertTexts6ToEdaTextSettings( const ATEXT6& aElem, EDA_TEXT& aEdaText );
-    void ParseFills6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+    void ParseFills6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                           const CFB::COMPOUND_FILE_ENTRY* aEntry );
     void ConvertFills6ToBoardItem( const AFILL6& aElem );
     void ConvertFills6ToFootprintItem( FOOTPRINT* aFootprint, const AFILL6& aElem,
@@ -197,9 +213,9 @@ private:
     void ConvertFills6ToBoardItemOnLayer( const AFILL6& aElem, PCB_LAYER_ID aLayer );
     void ConvertFills6ToFootprintItemOnLayer( FOOTPRINT* aFootprint, const AFILL6& aElem,
                                               PCB_LAYER_ID aLayer );
-    void ParseBoardRegionsData( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+    void ParseBoardRegionsData( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                                 const CFB::COMPOUND_FILE_ENTRY* aEntry );
-    void ParseShapeBasedRegions6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+    void ParseShapeBasedRegions6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                                       const CFB::COMPOUND_FILE_ENTRY* aEntry );
     void ConvertShapeBasedRegions6ToBoardItem( const AREGION6& aElem );
     void ConvertShapeBasedRegions6ToFootprintItem( FOOTPRINT* aFootprint, const AREGION6& aElem,
@@ -209,11 +225,11 @@ private:
                                                           const AREGION6& aElem,
                                                           PCB_LAYER_ID    aLayer,
                                                           const int       aPrimitiveIndex );
-    void ParseExtendedPrimitiveInformationData( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+    void ParseExtendedPrimitiveInformationData( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                                                 const CFB::COMPOUND_FILE_ENTRY* aEntry );
-    void ParseRegions6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+    void ParseRegions6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                             const CFB::COMPOUND_FILE_ENTRY* aEntry );
-    void ParseWideStrings6Data( const ALTIUM_COMPOUND_FILE&     aAltiumPcbFile,
+    void ParseWideStrings6Data( const ALTIUM_PCB_COMPOUND_FILE&     aAltiumPcbFile,
                                 const CFB::COMPOUND_FILE_ENTRY* aEntry );
 
     // Helper Functions
@@ -248,11 +264,12 @@ private:
     std::vector<FOOTPRINT*>              m_components;
     std::vector<ZONE*>                   m_polygons;
     std::vector<PCB_DIM_RADIAL*>         m_radialDimensions;
-    std::map<wxString, wxString>         m_models;
     std::map<uint32_t, wxString>         m_unicodeStrings;
     std::vector<int>                     m_altiumToKicadNetcodes;
     std::map<ALTIUM_LAYER, PCB_LAYER_ID> m_layermap; // used to correctly map copper layers
     std::map<ALTIUM_LAYER, wxString>     m_layerNames;
+
+    std::map<wxString, ALTIUM_EMBEDDED_MODEL_DATA>  m_EmbeddedModels;
     std::map<ALTIUM_RULE_KIND, std::vector<ARULE6>> m_rules;
     std::map<ALTIUM_RECORD, std::multimap<int, const AEXTENDED_PRIMITIVE_INFORMATION>>
             m_extendedPrimitiveInformationMaps;
diff --git a/pcbnew/pcb_io/altium/altium_pcb_compound_file.cpp b/pcbnew/pcb_io/altium/altium_pcb_compound_file.cpp
new file mode 100644
index 0000000000..9d9ad08323
--- /dev/null
+++ b/pcbnew/pcb_io/altium/altium_pcb_compound_file.cpp
@@ -0,0 +1,187 @@
+#include <altium_pcb_compound_file.h>
+#include <utf.h>
+
+#include <wx/string.h>
+
+#include <compoundfilereader.h>
+#include <map>
+
+ALTIUM_PCB_COMPOUND_FILE::ALTIUM_PCB_COMPOUND_FILE( const wxString& aFilePath )
+    : ALTIUM_COMPOUND_FILE( aFilePath )
+{
+}
+
+ALTIUM_PCB_COMPOUND_FILE::ALTIUM_PCB_COMPOUND_FILE( const void* aBuffer, size_t aLen )
+    : ALTIUM_COMPOUND_FILE( aBuffer, aLen )
+{
+}
+
+
+ALTIUM_PCB_COMPOUND_FILE::~ALTIUM_PCB_COMPOUND_FILE()
+{
+}
+
+std::map<wxString, wxString> ALTIUM_PCB_COMPOUND_FILE::ListLibFootprints()
+{
+    if( m_libFootprintDirNameCache.empty() )
+        cacheLibFootprintNames();
+
+    return m_libFootprintDirNameCache;
+}
+
+
+std::tuple<wxString, const CFB::COMPOUND_FILE_ENTRY*>
+ALTIUM_PCB_COMPOUND_FILE::FindLibFootprintDirName( const wxString& aFpUnicodeName )
+{
+    if( m_libFootprintNameCache.empty() )
+        cacheLibFootprintNames();
+
+    auto it = m_libFootprintNameCache.find( aFpUnicodeName );
+
+    if( it == m_libFootprintNameCache.end() )
+        return { wxEmptyString, nullptr };
+
+    return { it->first, it->second };
+}
+
+
+const std::pair<AMODEL, std::vector<char>>* ALTIUM_PCB_COMPOUND_FILE::GetLibModel( const wxString& aModelName ) const
+{
+    auto it = m_libModelsCache.find( aModelName );
+
+    if( it == m_libModelsCache.end() )
+        return nullptr;
+
+    return &it->second;
+}
+
+
+void ALTIUM_PCB_COMPOUND_FILE::cacheLibFootprintNames()
+{
+    m_libFootprintDirNameCache.clear();
+    m_libFootprintNameCache.clear();
+
+    if( !m_reader )
+        return;
+
+    const CFB::COMPOUND_FILE_ENTRY* root = m_reader->GetRootEntry();
+
+    if( !root )
+        return;
+
+    m_reader->EnumFiles( root, 1,
+        [this]( const CFB::COMPOUND_FILE_ENTRY* tentry, const CFB::utf16string& dir, int level ) -> int
+        {
+            if( m_reader->IsStream( tentry ) )
+                return 0;
+
+            m_reader->EnumFiles( tentry, 1,
+                        [&]( const CFB::COMPOUND_FILE_ENTRY* entry, const CFB::utf16string&, int ) -> int
+                        {
+                            std::wstring fileName = UTF16ToWstring( entry->name, entry->nameLen );
+
+                            if( m_reader->IsStream( entry ) && fileName == L"Parameters" )
+                            {
+                                ALTIUM_BINARY_PARSER         parametersReader( *this, entry );
+                                std::map<wxString, wxString> parameterProperties =
+                                        parametersReader.ReadProperties();
+
+                                wxString key = ALTIUM_PROPS_UTILS::ReadString(
+                                        parameterProperties, wxT( "PATTERN" ), wxT( "" ) );
+                                wxString fpName = ALTIUM_PROPS_UTILS::ReadUnicodeString(
+                                        parameterProperties, wxT( "PATTERN" ), wxT( "" ) );
+
+                                m_libFootprintDirNameCache[key] = fpName;
+                                m_libFootprintNameCache[fpName] = tentry;
+                            }
+
+                            return 0;
+                        } );
+            return 0;
+            } );
+}
+
+
+bool ALTIUM_PCB_COMPOUND_FILE::CacheLibModels()
+{
+    const CFB::COMPOUND_FILE_ENTRY* models_root = nullptr;
+    const CFB::COMPOUND_FILE_ENTRY* models_data = nullptr;
+    bool found = false;
+
+    if( !m_reader || !m_libModelsCache.empty() )
+        return false;
+
+    models_data = FindStream( { "Library", "Models", "Data" } );
+
+    if( !models_data )
+        return false;
+
+    ALTIUM_BINARY_PARSER parser( *this, models_data );
+
+    if( parser.GetRemainingBytes() == 0 )
+        return false;
+
+    std::vector<AMODEL> models;
+
+    // First, we parse and extract the model data from the Data stream
+    while( parser.GetRemainingBytes() >= 4 )
+    {
+        AMODEL elem( parser );
+        models.push_back( elem );
+    }
+
+    // Next, we need the model directory entry to read the compressed model streams
+    m_reader->EnumFiles( m_reader->GetRootEntry(), 2, [&]( const CFB::COMPOUND_FILE_ENTRY* entry, const CFB::utf16string& dir, int ) -> int
+    {
+        if( found )
+            return 1;
+
+        if( m_reader->IsStream( entry ) )
+            return 0;
+
+        std::string dir_str = UTF16ToUTF8( dir.c_str() );
+        std::string entry_str = UTF16ToUTF8( entry->name );
+
+        if( dir_str.compare( "Library" ) == 0 && entry_str.compare( "Models" ) == 0 )
+        {
+            models_root = entry;
+            found = true;
+            return 1;
+        }
+
+        return 0;
+    });
+
+    if( !models_root )
+        return false;
+
+    int      idx = 0;
+
+    m_reader->EnumFiles( models_root, 1, [&]( const CFB::COMPOUND_FILE_ENTRY* stepEntry, const CFB::utf16string&, int ) -> int
+    {
+        wxString fileName = UTF16ToUTF8( stepEntry->name, stepEntry->nameLen );
+
+        if( !m_reader->IsStream( stepEntry ) || idx >= models.size() )
+            return 0;
+
+        size_t            stepSize = static_cast<size_t>( stepEntry->size );
+        std::vector<char> stepContent( stepSize );
+
+        // read file into buffer
+        m_reader->ReadFile( stepEntry, 0, stepContent.data(), stepSize );
+
+        if( stepContent.empty() )
+            return 0;
+
+        // We store the models in their original compressed form so as to speed the caching process
+        // When we parse an individual footprint, we decompress and recompress the model data into
+        // our format
+        m_libModelsCache.emplace( models[idx].id, std::make_pair( std::move( models[idx] ),
+                                                                  std::move( stepContent ) ) );
+        idx++;
+
+        return 0;
+    } );
+
+    return true;
+}
\ No newline at end of file
diff --git a/pcbnew/pcb_io/altium/altium_pcb_compound_file.h b/pcbnew/pcb_io/altium/altium_pcb_compound_file.h
new file mode 100644
index 0000000000..e41cfa14fb
--- /dev/null
+++ b/pcbnew/pcb_io/altium/altium_pcb_compound_file.h
@@ -0,0 +1,55 @@
+/*
+ * This program source code file is part of KiCad, a free EDA CAD application.
+ *
+ * Copyright (C) 2024 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 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:
+ * http://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
+ */
+
+#ifndef ALTIUM_PCB_COMPOUND_FILE_H
+#define ALTIUM_PCB_COMPOUND_FILE_H
+
+#include <memory>
+#include <altium_parser_pcb.h>
+#include <io/altium/altium_binary_parser.h>
+
+
+class ALTIUM_PCB_COMPOUND_FILE : public ALTIUM_COMPOUND_FILE
+{
+public:
+    ALTIUM_PCB_COMPOUND_FILE( const wxString& aFilePath );
+    ALTIUM_PCB_COMPOUND_FILE( const void* aBuffer, size_t aLen );
+    ~ALTIUM_PCB_COMPOUND_FILE();
+
+    std::map<wxString, wxString> ListLibFootprints();
+
+    std::tuple<wxString, const CFB::COMPOUND_FILE_ENTRY*> FindLibFootprintDirName( const wxString& aFpUnicodeName );
+
+    const std::pair<AMODEL, std::vector<char>>* GetLibModel( const wxString& aModelID ) const;
+
+    bool CacheLibModels();
+private:
+
+    void cacheLibFootprintNames();
+
+    std::map<wxString, const CFB::COMPOUND_FILE_ENTRY*>      m_libFootprintNameCache;
+    std::map<wxString, wxString>                             m_libFootprintDirNameCache;
+    std::map<wxString, std::pair<AMODEL, std::vector<char>>> m_libModelsCache;
+};
+
+#endif // ALTIUM_PCB_COMPOUND_FILE_H
diff --git a/pcbnew/pcb_io/altium/pcb_io_altium_circuit_maker.cpp b/pcbnew/pcb_io/altium/pcb_io_altium_circuit_maker.cpp
index 5be1bb7b6a..dc9cd55c17 100644
--- a/pcbnew/pcb_io/altium/pcb_io_altium_circuit_maker.cpp
+++ b/pcbnew/pcb_io/altium/pcb_io_altium_circuit_maker.cpp
@@ -32,6 +32,7 @@
 #include <pcb_io_altium_circuit_maker.h>
 #include <pcb_io_altium_designer.h>
 #include <altium_pcb.h>
+#include <altium_pcb_compound_file.h>
 #include <io/altium/altium_binary_parser.h>
 #include <pcb_io/pcb_io.h>
 #include <reporter.h>
@@ -101,7 +102,7 @@ BOARD* PCB_IO_ALTIUM_CIRCUIT_MAKER::LoadBoard( const wxString& aFileName, BOARD*
     };
     // clang-format on
 
-    ALTIUM_COMPOUND_FILE altiumPcbFile( aFileName );
+    ALTIUM_PCB_COMPOUND_FILE altiumPcbFile( aFileName );
 
     try
     {
diff --git a/pcbnew/pcb_io/altium/pcb_io_altium_circuit_studio.cpp b/pcbnew/pcb_io/altium/pcb_io_altium_circuit_studio.cpp
index 45f2855480..80a836e404 100644
--- a/pcbnew/pcb_io/altium/pcb_io_altium_circuit_studio.cpp
+++ b/pcbnew/pcb_io/altium/pcb_io_altium_circuit_studio.cpp
@@ -33,6 +33,7 @@
 #include <pcb_io_altium_circuit_studio.h>
 #include <pcb_io_altium_designer.h>
 #include <altium_pcb.h>
+#include <altium_pcb_compound_file.h>
 #include <io/altium/altium_binary_parser.h>
 #include <pcb_io/pcb_io.h>
 #include <reporter.h>
@@ -102,7 +103,7 @@ BOARD* PCB_IO_ALTIUM_CIRCUIT_STUDIO::LoadBoard( const wxString& aFileName, BOARD
     };
     // clang-format on
 
-    ALTIUM_COMPOUND_FILE altiumPcbFile( aFileName );
+    ALTIUM_PCB_COMPOUND_FILE altiumPcbFile( aFileName );
 
     try
     {
diff --git a/pcbnew/pcb_io/altium/pcb_io_altium_designer.cpp b/pcbnew/pcb_io/altium/pcb_io_altium_designer.cpp
index f387c24942..d1ef5fdea4 100644
--- a/pcbnew/pcb_io/altium/pcb_io_altium_designer.cpp
+++ b/pcbnew/pcb_io/altium/pcb_io_altium_designer.cpp
@@ -31,6 +31,7 @@
 #include <font/fontconfig.h>
 #include <pcb_io_altium_designer.h>
 #include <altium_pcb.h>
+#include <altium_pcb_compound_file.h>
 #include <io/io_utils.h>
 #include <io/altium/altium_binary_parser.h>
 #include <pcb_io/pcb_io.h>
@@ -135,7 +136,7 @@ BOARD* PCB_IO_ALTIUM_DESIGNER::LoadBoard( const wxString& aFileName, BOARD* aApp
     };
     // clang-format on
 
-    ALTIUM_COMPOUND_FILE altiumPcbFile( aFileName );
+    ALTIUM_PCB_COMPOUND_FILE altiumPcbFile( aFileName );
 
     try
     {
@@ -187,18 +188,24 @@ void PCB_IO_ALTIUM_DESIGNER::loadAltiumLibrary( const wxString& aLibraryPath )
         if( aLibraryPath.Lower().EndsWith( wxS( ".pcblib" ) ) )
         {
             m_fplibFiles[aLibraryPath].emplace_back(
-                    std::make_unique<ALTIUM_COMPOUND_FILE>( aLibraryPath ) );
+                    std::make_unique<ALTIUM_PCB_COMPOUND_FILE>( aLibraryPath ) );
         }
         else if( aLibraryPath.Lower().EndsWith( wxS( ".intlib" ) ) )
         {
-            std::unique_ptr<ALTIUM_COMPOUND_FILE> intCom =
-                    std::make_unique<ALTIUM_COMPOUND_FILE>( aLibraryPath );
+            std::unique_ptr<ALTIUM_PCB_COMPOUND_FILE> intCom =
+                    std::make_unique<ALTIUM_PCB_COMPOUND_FILE>( aLibraryPath );
 
             std::map<wxString, const CFB::COMPOUND_FILE_ENTRY*> pcbLibFiles =
                     intCom->EnumDir( L"PCBLib" );
-
             for( const auto& [pcbLibName, pcbCfe] : pcbLibFiles )
-                m_fplibFiles[aLibraryPath].push_back( intCom->DecodeIntLibStream( *pcbCfe ) );
+            {
+                auto decodedStream = intCom->DecodeIntLibStream( *pcbCfe );
+                m_fplibFiles[aLibraryPath].push_back(
+                    std::unique_ptr<ALTIUM_PCB_COMPOUND_FILE>(
+                        static_cast<ALTIUM_PCB_COMPOUND_FILE*>(decodedStream.release())
+                    )
+                );
+            }
         }
     }
     catch( CFB::CFBException& exception )
@@ -296,8 +303,9 @@ FOOTPRINT* PCB_IO_ALTIUM_DESIGNER::FootprintLoad( const wxString& aLibraryPath,
 
     try
     {
-        for( std::unique_ptr<ALTIUM_COMPOUND_FILE>& altiumLibFile : it->second )
+        for( std::unique_ptr<ALTIUM_PCB_COMPOUND_FILE>& altiumLibFile : it->second )
         {
+            altiumLibFile->CacheLibModels();
             auto [dirName, fpCfe] = altiumLibFile->FindLibFootprintDirName( aFootprintName );
 
             if( dirName.IsEmpty() )
diff --git a/pcbnew/pcb_io/altium/pcb_io_altium_designer.h b/pcbnew/pcb_io/altium/pcb_io_altium_designer.h
index e9f6a049bc..c51897750a 100644
--- a/pcbnew/pcb_io/altium/pcb_io_altium_designer.h
+++ b/pcbnew/pcb_io/altium/pcb_io_altium_designer.h
@@ -32,7 +32,7 @@
 #include <map>
 #include <memory>
 
-class ALTIUM_COMPOUND_FILE;
+class ALTIUM_PCB_COMPOUND_FILE;
 
 class PCB_IO_ALTIUM_DESIGNER : public PCB_IO, public LAYER_MAPPABLE_PLUGIN
 {
@@ -85,7 +85,8 @@ public:
             const std::vector<INPUT_LAYER_DESC>& aInputLayerDescriptionVector );
 
 private:
-    std::map<wxString, std::vector<std::unique_ptr<ALTIUM_COMPOUND_FILE>>> m_fplibFiles;
+    std::map<wxString, std::vector<std::unique_ptr<ALTIUM_PCB_COMPOUND_FILE>>> m_fplibFiles;
+
 
     void loadAltiumLibrary( const wxString& aLibraryPath );
 };
diff --git a/pcbnew/pcb_io/altium/pcb_io_solidworks.cpp b/pcbnew/pcb_io/altium/pcb_io_solidworks.cpp
index 129521ed95..d473ceab21 100644
--- a/pcbnew/pcb_io/altium/pcb_io_solidworks.cpp
+++ b/pcbnew/pcb_io/altium/pcb_io_solidworks.cpp
@@ -23,6 +23,7 @@
 #include <pcb_io_solidworks.h>
 #include <pcb_io_altium_designer.h>
 #include <altium_pcb.h>
+#include <altium_pcb_compound_file.h>
 #include <io/altium/altium_binary_parser.h>
 #include <pcb_io/pcb_io.h>
 #include <reporter.h>
@@ -121,7 +122,7 @@ BOARD* PCB_IO_SOLIDWORKS::LoadBoard( const wxString& aFileName, BOARD* aAppendTo
     };
     // clang-format on
 
-    ALTIUM_COMPOUND_FILE altiumPcbFile( aFileName );
+    ALTIUM_PCB_COMPOUND_FILE altiumPcbFile( aFileName );
 
     try
     {
diff --git a/pcbnew/pcb_io/kicad_sexpr/pcb_io_kicad_sexpr.cpp b/pcbnew/pcb_io/kicad_sexpr/pcb_io_kicad_sexpr.cpp
index ed13c61302..0dc58e8cd4 100644
--- a/pcbnew/pcb_io/kicad_sexpr/pcb_io_kicad_sexpr.cpp
+++ b/pcbnew/pcb_io/kicad_sexpr/pcb_io_kicad_sexpr.cpp
@@ -112,6 +112,14 @@ void FP_CACHE::Save( FOOTPRINT* aFootprint )
         if( aFootprint && aFootprint != it->second->GetFootprint() )
             continue;
 
+        // If we've requested to embed the fonts in the footprint, do so.
+        // Otherwise, clear the embedded fonts from the footprint.  Embedded
+        // fonts will be used if available
+        if( aFootprint->GetAreFontsEmbedded() )
+            aFootprint->EmbedFonts();
+        else
+            aFootprint->GetEmbeddedFiles()->ClearEmbeddedFonts();
+
         WX_FILENAME fn = it->second->GetFileName();
 
         wxString tempFileName =
@@ -321,6 +329,13 @@ void PCB_IO_KICAD_SEXPR::SaveBoard( const wxString& aFileName, BOARD* aBoard,
 
     m_board = aBoard;       // after init()
 
+    // If the user wants fonts embedded, make sure that they are added to the board.  Otherwise,
+    // remove any fonts that were previously embedded.
+    if( m_board->GetAreFontsEmbedded() )
+        m_board->EmbedFonts();
+    else
+        m_board->GetEmbeddedFiles()->ClearEmbeddedFonts();
+
     // Prepare net mapping that assures that net codes saved in a file are consecutive integers
     m_mapping->SetBoard( aBoard );
 
@@ -860,6 +875,31 @@ void PCB_IO_KICAD_SEXPR::format( const BOARD* aBoard, int aNestLevel ) const
     // Save the generators
     for( BOARD_ITEM* gen : sorted_generators )
         Format( gen, aNestLevel );
+
+    // Save any embedded files
+    // Consolidate the embedded models in footprints into a single map
+    // to avoid duplicating the same model in the board file.
+    EMBEDDED_FILES files_to_write;
+
+    for( auto& file : aBoard->GetEmbeddedFiles()->EmbeddedFileMap() )
+        files_to_write.AddFile( file.second );
+
+    for( BOARD_ITEM* item : sorted_footprints )
+    {
+        FOOTPRINT* fp = static_cast<FOOTPRINT*>( item );
+
+        for( auto& file : fp->GetEmbeddedFiles()->EmbeddedFileMap() )
+            files_to_write.AddFile( file.second );
+    }
+
+    m_out->Print( aNestLevel + 1, "(embedded_fonts %s)\n",
+                  aBoard->GetEmbeddedFiles()->GetAreFontsEmbedded() ? "yes" : "no" );
+
+    if( !files_to_write.IsEmpty() )
+        files_to_write.WriteEmbeddedFiles( *m_out, aNestLevel + 1, ( m_ctl & CTL_FOR_BOARD ) );
+
+    // Remove the files so that they are not freed in the DTOR
+    files_to_write.ClearEmbeddedFiles( false );
 }
 
 
@@ -1341,6 +1381,14 @@ void PCB_IO_KICAD_SEXPR::format( const FOOTPRINT* aFootprint, int aNestLevel ) c
     for( BOARD_ITEM* group : sorted_groups )
         Format( group, aNestLevel + 1 );
 
+    m_out->Print( aNestLevel + 1, "(embedded_fonts %s)\n",
+                  aFootprint->GetEmbeddedFiles()->GetAreFontsEmbedded() ? "yes" : "no" );
+
+    if( !aFootprint->GetEmbeddedFiles()->IsEmpty() )
+    {
+        aFootprint->WriteEmbeddedFiles( *m_out, aNestLevel + 1, !( m_ctl & CTL_FOR_BOARD ) );
+    }
+
     // Save 3D info.
     auto bs3D = aFootprint->Models().begin();
     auto es3D = aFootprint->Models().end();
diff --git a/pcbnew/pcb_io/kicad_sexpr/pcb_io_kicad_sexpr.h b/pcbnew/pcb_io/kicad_sexpr/pcb_io_kicad_sexpr.h
index dbde711094..089b306de4 100644
--- a/pcbnew/pcb_io/kicad_sexpr/pcb_io_kicad_sexpr.h
+++ b/pcbnew/pcb_io/kicad_sexpr/pcb_io_kicad_sexpr.h
@@ -158,7 +158,8 @@ class PCB_IO_KICAD_SEXPR;   // forward decl
 //#define SEXPR_BOARD_FILE_VERSION    20240225  // Rationalization of solder_paste_margin
 //#define SEXPR_BOARD_FILE_VERSION    20240609  // Add 'tenting' keyword
 //#define SEXPR_BOARD_FILE_VERSION    20240617  // Table angles
-#define SEXPR_BOARD_FILE_VERSION      20240703  // User layer types
+//#define SEXPR_BOARD_FILE_VERSION    20240703  // User layer types
+#define SEXPR_BOARD_FILE_VERSION      20240706  // Embedded Files
 
 #define BOARD_FILE_HOST_VERSION       20200825  ///< Earlier files than this include the host tag
 #define LEGACY_ARC_FORMATTING         20210925  ///< These were the last to use old arc formatting
diff --git a/pcbnew/pcb_io/kicad_sexpr/pcb_io_kicad_sexpr_parser.cpp b/pcbnew/pcb_io/kicad_sexpr/pcb_io_kicad_sexpr_parser.cpp
index 8835147b1f..8b2614eae3 100644
--- a/pcbnew/pcb_io/kicad_sexpr/pcb_io_kicad_sexpr_parser.cpp
+++ b/pcbnew/pcb_io/kicad_sexpr/pcb_io_kicad_sexpr_parser.cpp
@@ -37,6 +37,8 @@
 
 #include <board.h>
 #include <board_design_settings.h>
+#include <embedded_files.h>
+#include <font/fontconfig.h>
 #include <pcb_dimension.h>
 #include <pcb_shape.h>
 #include <pcb_reference_image.h>
@@ -72,6 +74,7 @@
 // base64 code. Needed for PCB_REFERENCE_IMAGE
 #define wxUSE_BASE64 1
 #include <wx/base64.h>
+#include <wx/log.h>
 #include <wx/mstream.h>
 
 // We currently represent board units as integers.  Any values that are
@@ -580,8 +583,7 @@ void PCB_IO_KICAD_SEXPR_PARSER::parseEDA_TEXT( EDA_TEXT* aText )
 
             if( !faceName.IsEmpty() )
             {
-                aText->SetFont( KIFONT::FONT::GetFont( faceName, aText->IsBold(),
-                                                       aText->IsItalic() ) );
+                m_fontTextMap[aText] = { faceName, aText->IsBold(), aText->IsItalic() };
             }
 
             break;
@@ -865,6 +867,17 @@ BOARD_ITEM* PCB_IO_KICAD_SEXPR_PARSER::Parse()
         THROW_PARSE_ERROR( err, CurSource(), CurLine(), CurLineNumber(), CurOffset() );
     }
 
+    // Assign the fonts after we have parsed any potential embedded fonts from the board
+    // or footprint.  This also ensures that the embedded fonts are cached
+    for( auto& [text, params] : m_fontTextMap )
+    {
+        const std::vector<wxString>* embeddedFonts = item->GetEmbeddedFiles()->UpdateFontFiles();
+
+        text->SetFont( KIFONT::FONT::GetFont( std::get<0>( params ), std::get<1>( params ),
+                                              std::get<2>( params ),
+                                              embeddedFonts ) );
+    }
+
     resolveGroups( item );
 
     return item;
@@ -1073,6 +1086,31 @@ BOARD* PCB_IO_KICAD_SEXPR_PARSER::parseBOARD_unchecked()
             bulkAddedItems.push_back( item );
             break;
 
+        case T_embedded_fonts:
+        {
+            m_board->GetEmbeddedFiles()->SetAreFontsEmbedded( parseBool() );
+            NeedRIGHT();
+            break;
+        }
+
+        case T_embedded_files:
+        {
+            EMBEDDED_FILES_PARSER embeddedFilesParser( reader );
+            embeddedFilesParser.SyncLineReaderWith( *this );
+
+            try
+            {
+                embeddedFilesParser.ParseEmbedded( m_board->GetEmbeddedFiles() );
+            }
+            catch( const PARSE_ERROR& e )
+            {
+                wxLogError( e.What() );
+            }
+
+            SyncLineReaderWith( embeddedFilesParser );
+            break;
+        }
+
         default:
             wxString err;
             err.Printf( _( "Unknown token '%s'" ), FromUTF8() );
@@ -1184,6 +1222,9 @@ BOARD* PCB_IO_KICAD_SEXPR_PARSER::parseBOARD_unchecked()
         }
     }
 
+    // Ensure all footprints have their embedded data from the board
+    m_board->FixupEmbeddedData();
+
     return m_board;
 }
 
@@ -4675,13 +4716,38 @@ FOOTPRINT* PCB_IO_KICAD_SEXPR_PARSER::parseFOOTPRINT_unchecked( wxArrayString* a
             parseGROUP( footprint.get() );
             break;
 
+        case T_embedded_fonts:
+        {
+            footprint->GetEmbeddedFiles()->SetAreFontsEmbedded( parseBool() );
+            NeedRIGHT();
+            break;
+        }
+
+        case T_embedded_files:
+        {
+            EMBEDDED_FILES_PARSER embeddedFilesParser( reader );
+            embeddedFilesParser.SyncLineReaderWith( *this );
+
+            try
+            {
+                embeddedFilesParser.ParseEmbedded( footprint->GetEmbeddedFiles() );
+            }
+            catch( const PARSE_ERROR& e )
+            {
+                wxLogError( e.What() );
+            }
+
+            SyncLineReaderWith( embeddedFilesParser );
+            break;
+        }
+
         default:
-            Expecting( "locked, placed, tedit, tstamp, uuid, at, descr, tags, path, "
-                       "autoplace_cost90, autoplace_cost180, solder_mask_margin, "
-                       "solder_paste_margin, solder_paste_margin_ratio, clearance, "
-                       "zone_connect, thermal_gap, attr, fp_text, "
-                       "fp_arc, fp_circle, fp_curve, fp_line, fp_poly, fp_rect, pad, "
-                       "zone, group, generator, version or model" );
+            Expecting( "at, descr, locked, placed, tedit, tstamp, uuid, "
+                       "autoplace_cost90, autoplace_cost180, attr, clearance, "
+                       "embedded_files, fp_arc, fp_circle, fp_curve, fp_line, fp_poly, "
+                       "fp_rect, fp_text, pad, group, generator, model, path, solder_mask_margin, "
+                       "solder_paste_margin, solder_paste_margin_ratio, tags, thermal_gap, "
+                       "version, zone, or zone_connect" );
         }
     }
 
diff --git a/pcbnew/pcb_io/kicad_sexpr/pcb_io_kicad_sexpr_parser.h b/pcbnew/pcb_io/kicad_sexpr/pcb_io_kicad_sexpr_parser.h
index 63ea7984bc..b4ad365397 100644
--- a/pcbnew/pcb_io/kicad_sexpr/pcb_io_kicad_sexpr_parser.h
+++ b/pcbnew/pcb_io/kicad_sexpr/pcb_io_kicad_sexpr_parser.h
@@ -406,6 +406,8 @@ private:
     TIME_PT             m_lastProgressTime;  ///< for progress reporting
     unsigned            m_lineCount;         ///< for progress reporting
 
+    std::map<EDA_TEXT*, std::tuple<wxString, bool, bool>> m_fontTextMap;
+
     std::vector<GROUP_INFO>     m_groupInfos;
     std::vector<GENERATOR_INFO> m_generatorInfos;
 
diff --git a/pcbnew/pcbnew_config.cpp b/pcbnew/pcbnew_config.cpp
index b2b3e0af70..8507d7b937 100644
--- a/pcbnew/pcbnew_config.cpp
+++ b/pcbnew/pcbnew_config.cpp
@@ -29,8 +29,10 @@
 #include <tools/pcb_selection_tool.h>
 #include <board_design_settings.h>
 #include <drawing_sheet/ds_data_model.h>
+#include <filename_resolver.h>
 #include <pcbplot.h>
 #include <pcb_painter.h>
+#include <pgm_base.h>
 #include <project.h>
 #include <widgets/appearance_controls.h>
 #include <widgets/panel_selection_filter.h>
@@ -39,6 +41,27 @@
 #include <project/project_local_settings.h>
 
 
+void PCB_EDIT_FRAME::LoadDrawingSheet()
+{
+    PROJECT_FILE&           project       = Prj().GetProjectFile();
+
+    // Load the drawing sheet from the filename stored in the project
+    // If empty, or not existing, the default drawing sheet is loaded.
+    FILENAME_RESOLVER resolver;
+    resolver.SetProject( &Prj() );
+    resolver.SetProgramBase( &Pgm() );
+
+    wxString filename = resolver.ResolvePath( project.m_BoardDrawingSheetFile,
+                                              Prj().GetProjectPath(),
+                                              GetBoard()->GetEmbeddedFiles() );
+
+    wxString msg;
+
+    if( !DS_DATA_MODEL::GetTheInstance().LoadDrawingSheet( filename, &msg ) )
+        ShowInfoBarError( msg, true );
+
+}
+
 bool PCB_EDIT_FRAME::LoadProjectSettings()
 {
     PROJECT_FILE&           project       = Prj().GetProjectFile();
@@ -46,14 +69,6 @@ bool PCB_EDIT_FRAME::LoadProjectSettings()
 
     BASE_SCREEN::m_DrawingSheetFileName = project.m_BoardDrawingSheetFile;
 
-    // Load the drawing sheet from the filename stored in BASE_SCREEN::m_DrawingSheetFileName.
-    // If empty, or not existing, the default drawing sheet is loaded.
-    wxString filename = DS_DATA_MODEL::ResolvePath( BASE_SCREEN::m_DrawingSheetFileName,
-                                                    Prj().GetProjectPath());
-
-    if( !DS_DATA_MODEL::GetTheInstance().LoadDrawingSheet( filename, nullptr ) )
-        ShowInfoBarError( _( "Error loading drawing sheet." ), true );
-
     // Load render settings that aren't stored in PCB_DISPLAY_OPTIONS
 
     std::shared_ptr<NET_SETTINGS>& netSettings = project.NetSettings();
diff --git a/pcbnew/pcbnew_jobs_handler.cpp b/pcbnew/pcbnew_jobs_handler.cpp
index 39861f77f2..c6fd922b04 100644
--- a/pcbnew/pcbnew_jobs_handler.cpp
+++ b/pcbnew/pcbnew_jobs_handler.cpp
@@ -48,6 +48,7 @@
 #include <plotters/plotters_pslike.h>
 #include <tool/tool_manager.h>
 #include <tools/drc_tool.h>
+#include <filename_resolver.h>
 #include <gerber_jobfile_writer.h>
 #include "gerber_placefile_writer.h"
 #include <gendrill_Excellon_writer.h>
@@ -1552,8 +1553,13 @@ void PCBNEW_JOBS_HANDLER::loadOverrideDrawingSheet( BOARD* aBrd, const wxString&
             [&]( const wxString& path ) -> bool
             {
                 BASE_SCREEN::m_DrawingSheetFileName = path;
-                wxString filename = DS_DATA_MODEL::ResolvePath( BASE_SCREEN::m_DrawingSheetFileName,
-                                                                aBrd->GetProject()->GetProjectPath() );
+                FILENAME_RESOLVER resolver;
+                resolver.SetProject( aBrd->GetProject() );
+                resolver.SetProgramBase( &Pgm() );
+
+                wxString filename = resolver.ResolvePath( BASE_SCREEN::m_DrawingSheetFileName,
+                                                          aBrd->GetProject()->GetProjectPath(),
+                                                          aBrd->GetEmbeddedFiles() );
                 wxString msg;
 
                 if( !DS_DATA_MODEL::GetTheInstance().LoadDrawingSheet( filename, &msg ) )
diff --git a/pcbnew/python/scripting/pcbnew_scripting_helpers.cpp b/pcbnew/python/scripting/pcbnew_scripting_helpers.cpp
index ab337e5694..2485706660 100644
--- a/pcbnew/python/scripting/pcbnew_scripting_helpers.cpp
+++ b/pcbnew/python/scripting/pcbnew_scripting_helpers.cpp
@@ -44,8 +44,10 @@
 #include <core/ignore.h>
 #include <pcb_io/pcb_io_mgr.h>
 #include <string_utils.h>
+#include <filename_resolver.h>
 #include <macros.h>
 #include <pcbnew_scripting_helpers.h>
+#include <pgm_base.h>
 #include <project.h>
 #include <project_pcb.h>
 #include <project/net_settings.h>
@@ -179,18 +181,28 @@ BOARD* LoadBoard( wxString& aFileName, PCB_IO_MGR::PCB_FILE_T aFormat, bool aSet
 
     BASE_SCREEN::m_DrawingSheetFileName = project->GetProjectFile().m_BoardDrawingSheetFile;
 
-    // Load the drawing sheet from the filename stored in BASE_SCREEN::m_DrawingSheetFileName.
-    // If empty, or not existing, the default drawing sheet is loaded.
-    wxString filename = DS_DATA_MODEL::ResolvePath( BASE_SCREEN::m_DrawingSheetFileName,
-                                                    project->GetProjectPath() );
-
-    if( !DS_DATA_MODEL::GetTheInstance().LoadDrawingSheet( filename, nullptr ) )
-        wxFprintf( stderr, _( "Error loading drawing sheet." ) );
-
     BOARD* brd = PCB_IO_MGR::Load( aFormat, aFileName );
 
     if( brd )
     {
+        // Load the drawing sheet from the filename stored in BASE_SCREEN::m_DrawingSheetFileName.
+        // If empty, or not existing, the default drawing sheet is loaded.
+        FILENAME_RESOLVER resolver;
+        resolver.SetProject( project );
+        resolver.SetProgramBase( &Pgm() );
+
+        wxString filename = resolver.ResolvePath( BASE_SCREEN::m_DrawingSheetFileName,
+                                              project->GetProjectPath(),
+                                              brd->GetEmbeddedFiles() );
+
+        wxString msg;
+
+        if( !DS_DATA_MODEL::GetTheInstance().LoadDrawingSheet( filename, &msg ) )
+        {
+            wxFprintf( stderr, _( "Error loading drawing sheet '%s': %s" ),
+                       BASE_SCREEN::m_DrawingSheetFileName, msg );
+        }
+
         // JEY TODO: move this global to the board
         ENUM_MAP<PCB_LAYER_ID>& layerEnum = ENUM_MAP<PCB_LAYER_ID>::Instance();
 
diff --git a/pcbnew/tools/board_editor_control.cpp b/pcbnew/tools/board_editor_control.cpp
index 8d9aec7489..7da38de694 100644
--- a/pcbnew/tools/board_editor_control.cpp
+++ b/pcbnew/tools/board_editor_control.cpp
@@ -298,8 +298,8 @@ int BOARD_EDITOR_CONTROL::PageSettings( const TOOL_EVENT& aEvent )
     undoCmd.SetDescription( _( "Page Settings" ) );
     m_frame->SaveCopyInUndoList( undoCmd, UNDO_REDO::PAGESETTINGS );
 
-    DIALOG_PAGES_SETTINGS dlg( m_frame, pcbIUScale.IU_PER_MILS, VECTOR2I( MAX_PAGE_SIZE_PCBNEW_MILS,
-                                                                          MAX_PAGE_SIZE_PCBNEW_MILS ) );
+    DIALOG_PAGES_SETTINGS dlg( m_frame, m_frame->GetBoard()->GetEmbeddedFiles(), pcbIUScale.IU_PER_MILS,
+                               VECTOR2I( MAX_PAGE_SIZE_PCBNEW_MILS, MAX_PAGE_SIZE_PCBNEW_MILS ) );
     dlg.SetWksFileName( BASE_SCREEN::m_DrawingSheetFileName );
 
     if( dlg.ShowModal() == wxID_OK )
diff --git a/pcbnew/widgets/pcb_properties_panel.cpp b/pcbnew/widgets/pcb_properties_panel.cpp
index 2c374c159b..c716cef98c 100644
--- a/pcbnew/widgets/pcb_properties_panel.cpp
+++ b/pcbnew/widgets/pcb_properties_panel.cpp
@@ -275,7 +275,8 @@ void PCB_PROPERTIES_PANEL::updateLists( const BOARD* aBoard )
 
     // Regnerate font names
     std::vector<std::string> fontNames;
-    Fontconfig()->ListFonts( fontNames, std::string( Pgm().GetLanguageTag().utf8_str() ) );
+    Fontconfig()->ListFonts( fontNames, std::string( Pgm().GetLanguageTag().utf8_str() ),
+                             aBoard->GetFontFiles() );
 
     fonts.Add( KICAD_FONT_NAME, -1 );
 
diff --git a/qa/tests/common/CMakeLists.txt b/qa/tests/common/CMakeLists.txt
index b7a6da566c..e41699e68c 100644
--- a/qa/tests/common/CMakeLists.txt
+++ b/qa/tests/common/CMakeLists.txt
@@ -36,6 +36,7 @@ set( QA_COMMON_SRCS
     test_coroutine.cpp
     test_eda_shape.cpp
     test_eda_text.cpp
+    test_embedded_file_compress.cpp
     test_lib_table.cpp
     test_markup_parser.cpp
     test_kicad_string.cpp
diff --git a/qa/tests/common/test_embedded_file_compress.cpp b/qa/tests/common/test_embedded_file_compress.cpp
new file mode 100644
index 0000000000..56a1731bb1
--- /dev/null
+++ b/qa/tests/common/test_embedded_file_compress.cpp
@@ -0,0 +1,129 @@
+/*
+ * This program source code file is part of KiCad, a free EDA CAD application.
+ *
+ * Copyright (C) 1992-2021 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 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, see <http://www.gnu.org/licenses/>.
+ */
+
+#include <magic_enum.hpp>
+#include <boost/test/unit_test.hpp>
+#include <picosha2.h>
+#include <embedded_files.h>
+
+#include <random>
+using magic_enum::iostream_operators::operator<<;
+
+BOOST_AUTO_TEST_SUITE( EmbeddedFiles )
+
+BOOST_AUTO_TEST_CASE( CompressAndEncode_OK )
+{
+    EMBEDDED_FILES::EMBEDDED_FILE file;
+    file.name = "test_file";
+    std::string data = "Hello, World!";
+    file.decompressedData.assign(data.begin(), data.end());
+
+    picosha2::hash256_hex_string(file.decompressedData, file.data_sha);
+
+    EMBEDDED_FILES::RETURN_CODE result = EMBEDDED_FILES::CompressAndEncode(file);
+    BOOST_CHECK_EQUAL(result, EMBEDDED_FILES::RETURN_CODE::OK);
+}
+
+BOOST_AUTO_TEST_CASE( DecompressAndDecode_OK )
+{
+    EMBEDDED_FILES::EMBEDDED_FILE file;
+    file.name = "test_file";
+    std::string data = "Hello, World!";
+    file.decompressedData.assign( data.begin(), data.end() );
+
+    picosha2::hash256_hex_string( file.decompressedData, file.data_sha );
+
+    EMBEDDED_FILES::RETURN_CODE result = EMBEDDED_FILES::CompressAndEncode( file );
+    BOOST_CHECK_EQUAL( result, EMBEDDED_FILES::RETURN_CODE::OK );
+
+    result = EMBEDDED_FILES::DecompressAndDecode( file );
+    BOOST_CHECK_EQUAL( result, EMBEDDED_FILES::RETURN_CODE::OK );
+
+    // Create a large test data
+    data.clear();
+    data.reserve( 13 * 100000 + 1 );
+
+    for( int i = 0; i < 100000; ++i )
+        data += "Hello, World!";
+
+    file.decompressedData.assign( data.begin(), data.end() );
+
+    picosha2::hash256_hex_string( file.decompressedData, file.data_sha );
+
+    result = EMBEDDED_FILES::CompressAndEncode( file );
+    BOOST_CHECK_EQUAL( result, EMBEDDED_FILES::RETURN_CODE::OK );
+
+    result = EMBEDDED_FILES::DecompressAndDecode( file );
+    BOOST_CHECK_EQUAL( result, EMBEDDED_FILES::RETURN_CODE::OK );
+
+    // Create a sequential test dataset
+    data.clear();
+    data.reserve( 100000 );
+
+    for( int i = 0; i < 100000; ++i )
+        data += static_cast<char>( i % 256 );
+
+    file.decompressedData.assign( data.begin(), data.end() );
+    picosha2::hash256_hex_string( file.decompressedData, file.data_sha );
+
+    result = EMBEDDED_FILES::CompressAndEncode( file );
+    BOOST_CHECK_EQUAL( result, EMBEDDED_FILES::RETURN_CODE::OK );
+
+    result = EMBEDDED_FILES::DecompressAndDecode( file );
+    BOOST_CHECK_EQUAL( result, EMBEDDED_FILES::RETURN_CODE::OK );
+
+    // Create a random test dataset with a known seed
+    data.clear();
+    data.reserve( 100000 );
+
+    std::mt19937 rng;
+    rng.seed( 0 );
+
+    for( int i = 0; i < 100000; ++i )
+        data += static_cast<char>( rng() % 256 );
+
+    file.decompressedData.assign( data.begin(), data.end() );
+    picosha2::hash256_hex_string( file.decompressedData, file.data_sha );
+
+    result = EMBEDDED_FILES::CompressAndEncode( file );
+    BOOST_CHECK_EQUAL( result, EMBEDDED_FILES::RETURN_CODE::OK );
+
+    result = EMBEDDED_FILES::DecompressAndDecode( file );
+    BOOST_CHECK_EQUAL( result, EMBEDDED_FILES::RETURN_CODE::OK );
+
+}
+
+BOOST_AUTO_TEST_CASE( DecompressAndDecode_ChecksumError )
+{
+    EMBEDDED_FILES::EMBEDDED_FILE file;
+    file.name = "test_file";
+    std::string data = "Hello, World!";
+    file.decompressedData.assign(data.begin(), data.end());
+
+    EMBEDDED_FILES::RETURN_CODE result = EMBEDDED_FILES::CompressAndEncode(file);
+    BOOST_CHECK_EQUAL(result, EMBEDDED_FILES::RETURN_CODE::OK);
+
+    // Modify the checksum
+    file.data_sha[0] = 'x';
+
+    result = EMBEDDED_FILES::DecompressAndDecode(file);
+    BOOST_CHECK_EQUAL(result, EMBEDDED_FILES::RETURN_CODE::CHECKSUM_ERROR);
+}
+
+BOOST_AUTO_TEST_SUITE_END()