mirror of
https://gitlab.com/kicad/code/kicad.git
synced 2025-09-13 17:53:11 +02:00
476 lines
14 KiB
C++
476 lines
14 KiB
C++
/*
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
*
|
|
* Copyright (C) 2012 Brian Sidebotham <brian.sidebotham@gmail.com>
|
|
* Copyright The 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 "dialog_template_selector.h"
|
|
#include <bitmaps.h>
|
|
#include <widgets/std_bitmap_button.h>
|
|
#include <widgets/ui_common.h>
|
|
#include <algorithm>
|
|
#include <wx_filename.h>
|
|
#include <wx/dir.h>
|
|
#include <wx/dirdlg.h>
|
|
#include <wx/settings.h>
|
|
#include <wx/bitmap.h>
|
|
#include <wx/image.h>
|
|
#include <wx/math.h>
|
|
#include "template_default_html.h"
|
|
|
|
// Welcome / fallback HTML now provided by template_default_html.h
|
|
|
|
TEMPLATE_SELECTION_PANEL::TEMPLATE_SELECTION_PANEL( wxNotebookPage* aParent,
|
|
const wxString& aPath ) :
|
|
TEMPLATE_SELECTION_PANEL_BASE( aParent )
|
|
{
|
|
m_parent = aParent;
|
|
m_templatesPath = aPath;
|
|
}
|
|
|
|
|
|
void TEMPLATE_SELECTION_PANEL::AddTemplateWidget( TEMPLATE_WIDGET* aTemplateWidget )
|
|
{
|
|
m_SizerChoice->Add( aTemplateWidget );
|
|
Layout();
|
|
}
|
|
|
|
|
|
// Sort the widgets alphabetically, leaving Default at the top
|
|
void TEMPLATE_SELECTION_PANEL::SortAlphabetically()
|
|
{
|
|
std::vector<TEMPLATE_WIDGET*> sortedList;
|
|
TEMPLATE_WIDGET* default_temp = nullptr;
|
|
size_t count = m_SizerChoice->GetItemCount();
|
|
|
|
if( count <= 1 )
|
|
return;
|
|
|
|
for( size_t idx = 0; idx < count; idx++ )
|
|
{
|
|
wxSizerItem* item = m_SizerChoice->GetItem( idx );
|
|
if( item && item->IsWindow() )
|
|
{
|
|
TEMPLATE_WIDGET* temp = static_cast<TEMPLATE_WIDGET*>( item->GetWindow() );
|
|
|
|
const wxString title = *temp->GetTemplate()->GetTitle();
|
|
|
|
if( default_temp == nullptr && title.CmpNoCase( "default" ) == 0 )
|
|
default_temp = temp;
|
|
else
|
|
sortedList.push_back( temp );
|
|
}
|
|
}
|
|
|
|
std::sort(
|
|
sortedList.begin(), sortedList.end(),
|
|
[]( TEMPLATE_WIDGET* aWidgetA, TEMPLATE_WIDGET* aWidgetB ) -> bool
|
|
{
|
|
const wxString* a = aWidgetA->GetTemplate()->GetTitle();
|
|
const wxString* b = aWidgetB->GetTemplate()->GetTitle();
|
|
|
|
return ( *a ).CmpNoCase( *b ) < 0;
|
|
});
|
|
|
|
m_SizerChoice->Clear( false );
|
|
|
|
if( default_temp != nullptr )
|
|
m_SizerChoice->Add( default_temp );
|
|
|
|
for (TEMPLATE_WIDGET* temp : sortedList)
|
|
{
|
|
m_SizerChoice->Add( temp );
|
|
}
|
|
|
|
Layout();
|
|
}
|
|
|
|
|
|
TEMPLATE_WIDGET::TEMPLATE_WIDGET( wxWindow* aParent, DIALOG_TEMPLATE_SELECTOR* aDialog ) :
|
|
TEMPLATE_WIDGET_BASE( aParent )
|
|
{
|
|
m_parent = aParent;
|
|
m_dialog = aDialog;
|
|
|
|
// wxWidgets_3.xx way of doing the same...
|
|
// Bind(wxEVT_LEFT_DOWN, &TEMPLATE_WIDGET::OnMouse, this );
|
|
|
|
m_bitmapIcon->Connect( wxEVT_LEFT_DOWN, wxMouseEventHandler( TEMPLATE_WIDGET::OnMouse ),
|
|
nullptr, this );
|
|
m_staticTitle->Connect( wxEVT_LEFT_DOWN, wxMouseEventHandler( TEMPLATE_WIDGET::OnMouse ),
|
|
nullptr, this );
|
|
|
|
// We're not selected until we're clicked
|
|
Unselect();
|
|
|
|
// Start with template being NULL
|
|
m_currTemplate = nullptr;
|
|
}
|
|
|
|
|
|
void TEMPLATE_WIDGET::Select()
|
|
{
|
|
m_dialog->SetWidget( this );
|
|
SetBackgroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_BTNHIGHLIGHT ) );
|
|
m_selected = true;
|
|
Refresh();
|
|
}
|
|
|
|
|
|
void TEMPLATE_WIDGET::Unselect()
|
|
{
|
|
SetBackgroundColour( wxSystemSettings::GetColour( wxSYS_COLOUR_BTNFACE ) );
|
|
m_selected = false;
|
|
Refresh();
|
|
}
|
|
|
|
|
|
void TEMPLATE_WIDGET::SetTemplate( PROJECT_TEMPLATE* aTemplate )
|
|
{
|
|
m_currTemplate = aTemplate;
|
|
m_staticTitle->SetFont( KIUI::GetInfoFont( this ) );
|
|
m_staticTitle->SetLabel( *aTemplate->GetTitle() );
|
|
m_staticTitle->Wrap( 100 );
|
|
wxBitmap* icon = aTemplate->GetIcon();
|
|
|
|
if( icon && icon->IsOk() )
|
|
{
|
|
wxSize maxSize = m_bitmapIcon->GetSize();
|
|
|
|
if( icon->GetWidth() > maxSize.x || icon->GetHeight() > maxSize.y )
|
|
{
|
|
double scale = std::min( (double) maxSize.x / icon->GetWidth(),
|
|
(double) maxSize.y / icon->GetHeight() );
|
|
wxImage image = icon->ConvertToImage();
|
|
int w = wxRound( icon->GetWidth() * scale );
|
|
int h = wxRound( icon->GetHeight() * scale );
|
|
image.Rescale( w, h, wxIMAGE_QUALITY_HIGH );
|
|
m_bitmapIcon->SetBitmap( wxBitmap( image ) );
|
|
}
|
|
else
|
|
{
|
|
m_bitmapIcon->SetBitmap( *icon );
|
|
}
|
|
}
|
|
else
|
|
m_bitmapIcon->SetBitmap( KiBitmap( BITMAPS::icon_kicad ) );
|
|
}
|
|
|
|
|
|
void TEMPLATE_WIDGET::OnMouse( wxMouseEvent& event )
|
|
{
|
|
// Toggle selection here
|
|
Select();
|
|
event.Skip();
|
|
}
|
|
|
|
|
|
void DIALOG_TEMPLATE_SELECTOR::OnPageChange( wxNotebookEvent& event )
|
|
{
|
|
int newPage = event.GetSelection();
|
|
int oldPage = event.GetOldSelection();
|
|
|
|
// Move webview panel from old page to new page
|
|
if( oldPage != wxNOT_FOUND && (unsigned)oldPage < m_panels.size() )
|
|
{
|
|
// Detach webview from old panel
|
|
m_panels[oldPage]->m_SizerBase->Detach( m_webviewPanel );
|
|
m_panels[oldPage]->Layout();
|
|
}
|
|
|
|
if( newPage != wxNOT_FOUND && (unsigned)newPage < m_panels.size() )
|
|
{
|
|
// Reparent the webview to the new panel
|
|
m_webviewPanel->Reparent( m_panels[newPage] );
|
|
|
|
// Attach webview to new panel
|
|
m_panels[newPage]->m_SizerBase->Add( m_webviewPanel, 1, wxEXPAND | wxALL, 5 );
|
|
m_panels[newPage]->Layout();
|
|
|
|
// Update template path
|
|
m_tcTemplatePath->SetValue( m_panels[newPage]->GetPath() );
|
|
|
|
// Reset to welcome page when switching tabs if no template selected
|
|
m_webviewPanel->SetPage( GetWelcomeHtml() );
|
|
}
|
|
|
|
event.Skip();
|
|
}
|
|
|
|
|
|
DIALOG_TEMPLATE_SELECTOR::DIALOG_TEMPLATE_SELECTOR( wxWindow* aParent, const wxPoint& aPos,
|
|
const wxSize& aSize,
|
|
std::vector<std::pair<wxString, wxFileName>> aTitleDirList,
|
|
const wxFileName& aDefaultTemplate ) :
|
|
DIALOG_TEMPLATE_SELECTOR_BASE( aParent, wxID_ANY, _( "Project Template Selector" ), aPos,
|
|
aSize )
|
|
{
|
|
m_browseButton->SetBitmap( KiBitmapBundle( BITMAPS::small_folder ) );
|
|
m_reloadButton->SetBitmap( KiBitmapBundle( BITMAPS::small_refresh ) );
|
|
|
|
m_selectedWidget = nullptr;
|
|
m_defaultTemplatePath = aDefaultTemplate;
|
|
m_defaultWidget = nullptr;
|
|
|
|
for( auto& [title, pathFname] : aTitleDirList )
|
|
{
|
|
pathFname.Normalize( FN_NORMALIZE_FLAGS | wxPATH_NORM_ENV_VARS );
|
|
wxString path = pathFname.GetFullPath(); // caller ensures this ends with file separator.
|
|
|
|
TEMPLATE_SELECTION_PANEL* tpanel = new TEMPLATE_SELECTION_PANEL( m_notebook, path );
|
|
m_panels.push_back( tpanel );
|
|
|
|
m_notebook->AddPage( tpanel, title );
|
|
|
|
if( m_notebook->GetPageCount() == 1 )
|
|
m_tcTemplatePath->SetValue( path );
|
|
|
|
buildPageContent( path, m_notebook->GetPageCount() - 1 );
|
|
}
|
|
|
|
// Move webview panel from dialog to first template selection panel
|
|
if( !m_panels.empty() )
|
|
{
|
|
// Find the sizer containing the webview and detach it
|
|
wxSizer* parentSizer = m_webviewPanel->GetContainingSizer();
|
|
|
|
if( parentSizer )
|
|
{
|
|
parentSizer->Detach( m_webviewPanel );
|
|
}
|
|
|
|
// Reparent the webview to the first panel
|
|
m_webviewPanel->Reparent( m_panels[0] );
|
|
|
|
// Add webview to first panel
|
|
m_panels[0]->m_SizerBase->Add( m_webviewPanel, 1, wxEXPAND | wxALL, 5 );
|
|
m_panels[0]->Layout();
|
|
}
|
|
|
|
if( m_defaultWidget )
|
|
m_defaultWidget->Select();
|
|
|
|
// Set welcome HTML after dialog is fully constructed
|
|
CallAfter( [this]()
|
|
{
|
|
#if defined (_WIN32)
|
|
wxSafeYield();
|
|
m_tcTemplatePath->SelectNone();
|
|
#endif
|
|
|
|
if( m_selectedWidget )
|
|
{
|
|
wxFileName htmlFile = m_selectedWidget->GetTemplate()->GetHtmlFile();
|
|
|
|
if( htmlFile.FileExists() && htmlFile.IsFileReadable() && htmlFile.GetSize() > 100 /* Basic HTML */ )
|
|
m_webviewPanel->LoadURL( wxFileName::FileNameToURL( htmlFile ) );
|
|
else
|
|
m_webviewPanel->SetPage( GetWelcomeHtml() );
|
|
}
|
|
else
|
|
{
|
|
m_webviewPanel->SetPage( GetWelcomeHtml() );
|
|
}
|
|
});
|
|
|
|
// When all widgets have the size fixed, call finishDialogSettings to update sizers
|
|
finishDialogSettings();
|
|
}
|
|
|
|
|
|
void DIALOG_TEMPLATE_SELECTOR::SetWidget( TEMPLATE_WIDGET* aWidget )
|
|
{
|
|
if( m_selectedWidget != nullptr )
|
|
m_selectedWidget->Unselect();
|
|
|
|
m_selectedWidget = aWidget;
|
|
wxFileName htmlFile = aWidget->GetTemplate()->GetHtmlFile();
|
|
|
|
if( htmlFile.FileExists() && htmlFile.IsFileReadable() )
|
|
{
|
|
m_webviewPanel->LoadURL( wxFileName::FileNameToURL( htmlFile ) );
|
|
}
|
|
else
|
|
{
|
|
m_webviewPanel->SetPage( GetTemplateInfoHtml( *aWidget->GetTemplate()->GetTitle() ) );
|
|
}
|
|
}
|
|
|
|
|
|
void DIALOG_TEMPLATE_SELECTOR::AddTemplate( int aPage, PROJECT_TEMPLATE* aTemplate )
|
|
{
|
|
TEMPLATE_WIDGET* w = new TEMPLATE_WIDGET( m_panels[aPage]->m_scrolledWindow, this );
|
|
w->SetTemplate( aTemplate );
|
|
m_panels[aPage]->AddTemplateWidget( w );
|
|
m_allWidgets.push_back( w );
|
|
|
|
wxFileName base = aTemplate->GetHtmlFile();
|
|
base.RemoveLastDir();
|
|
|
|
if( !m_defaultWidget || (m_defaultTemplatePath.IsOk() && base == m_defaultTemplatePath) )
|
|
m_defaultWidget = w;
|
|
|
|
wxString dirName = base.GetDirs().IsEmpty() ? wxString() : base.GetDirs().back();
|
|
|
|
if( dirName.CmpNoCase( "default" ) == 0 )
|
|
{
|
|
// Prefer a directory literally named 'default'
|
|
m_defaultWidget = w;
|
|
}
|
|
}
|
|
|
|
|
|
PROJECT_TEMPLATE* DIALOG_TEMPLATE_SELECTOR::GetSelectedTemplate()
|
|
{
|
|
return m_selectedWidget? m_selectedWidget->GetTemplate() : nullptr;
|
|
}
|
|
|
|
PROJECT_TEMPLATE* DIALOG_TEMPLATE_SELECTOR::GetDefaultTemplate()
|
|
{
|
|
return m_defaultWidget? m_defaultWidget->GetTemplate() : nullptr;
|
|
}
|
|
|
|
|
|
void DIALOG_TEMPLATE_SELECTOR::buildPageContent( const wxString& aPath, int aPage )
|
|
{
|
|
// Get a list of files under the template path to include as choices...
|
|
wxDir dir;
|
|
|
|
if( dir.Open( aPath ) )
|
|
{
|
|
if( dir.HasSubDirs( "meta" ) )
|
|
{
|
|
AddTemplate( aPage, new PROJECT_TEMPLATE( aPath ) );
|
|
}
|
|
else
|
|
{
|
|
wxString sub_name;
|
|
wxArrayString subdirs;
|
|
|
|
bool cont = dir.GetFirst( &sub_name, wxEmptyString, wxDIR_DIRS );
|
|
|
|
while( cont )
|
|
{
|
|
subdirs.Add( wxString( sub_name ) );
|
|
cont = dir.GetNext( &sub_name );
|
|
}
|
|
|
|
if( !subdirs.IsEmpty() )
|
|
subdirs.Sort();
|
|
|
|
for( const wxString& dir_name : subdirs )
|
|
{
|
|
wxDir sub_dir;
|
|
wxString sub_full = aPath + dir_name;
|
|
|
|
if( sub_dir.Open( sub_full ) )
|
|
AddTemplate( aPage, new PROJECT_TEMPLATE( sub_full ) );
|
|
}
|
|
}
|
|
}
|
|
|
|
m_panels[aPage]->SortAlphabetically();
|
|
Layout();
|
|
}
|
|
|
|
|
|
void DIALOG_TEMPLATE_SELECTOR::onDirectoryBrowseClicked( wxCommandEvent& event )
|
|
{
|
|
wxFileName fn;
|
|
fn.AssignDir( m_tcTemplatePath->GetValue() );
|
|
fn.Normalize( FN_NORMALIZE_FLAGS | wxPATH_NORM_ENV_VARS );
|
|
wxString currPath = fn.GetFullPath();
|
|
|
|
wxDirDialog dirDialog( this, _( "Select Templates Directory" ), currPath,
|
|
wxDD_DEFAULT_STYLE | wxDD_DIR_MUST_EXIST );
|
|
|
|
if( dirDialog.ShowModal() != wxID_OK )
|
|
return;
|
|
|
|
wxFileName dirName = wxFileName::DirName( dirDialog.GetPath() );
|
|
|
|
m_tcTemplatePath->SetValue( dirName.GetFullPath() );
|
|
|
|
// Rebuild the page from the new templates path:
|
|
replaceCurrentPage();
|
|
}
|
|
|
|
|
|
void DIALOG_TEMPLATE_SELECTOR::onReload( wxCommandEvent& event )
|
|
{
|
|
int page = m_notebook->GetSelection();
|
|
|
|
if( page < 0 )
|
|
return; // Should not happen
|
|
|
|
wxString currPath = m_tcTemplatePath->GetValue();
|
|
|
|
wxFileName fn;
|
|
fn.AssignDir( m_tcTemplatePath->GetValue() );
|
|
fn.Normalize( FN_NORMALIZE_FLAGS | wxPATH_NORM_ENV_VARS );
|
|
currPath = fn.GetFullPath();
|
|
m_tcTemplatePath->SetValue( currPath );
|
|
|
|
replaceCurrentPage();
|
|
}
|
|
|
|
|
|
void DIALOG_TEMPLATE_SELECTOR::replaceCurrentPage()
|
|
{
|
|
// Rebuild the page from the new templates path:
|
|
int page = m_notebook->GetSelection();
|
|
|
|
if( page < 0 )
|
|
return; // Should not happen
|
|
|
|
wxString title = m_notebook->GetPageText( page );
|
|
wxString currPath = m_tcTemplatePath->GetValue();
|
|
|
|
// Block all events to the notebook and its children
|
|
wxEventBlocker blocker( m_notebook );
|
|
|
|
// Detach webview from current panel before deleting it
|
|
if( (unsigned)page < m_panels.size() )
|
|
{
|
|
m_panels[page]->m_SizerBase->Detach( m_webviewPanel );
|
|
m_webviewPanel->Reparent( this ); // Reparent to dialog
|
|
m_webviewPanel->Hide(); // Hide webview panel temporarily
|
|
}
|
|
|
|
m_notebook->DeletePage( page );
|
|
|
|
TEMPLATE_SELECTION_PANEL* tpanel = new TEMPLATE_SELECTION_PANEL( m_notebook, currPath );
|
|
m_panels[page] = tpanel;
|
|
m_notebook->InsertPage( page, tpanel, title, true );
|
|
|
|
// Reparent and add webview back to the new panel
|
|
m_webviewPanel->Reparent( tpanel );
|
|
m_webviewPanel->Show();
|
|
tpanel->m_SizerBase->Add( m_webviewPanel, 1, wxEXPAND | wxALL, 5 );
|
|
|
|
buildPageContent( m_tcTemplatePath->GetValue(), page );
|
|
|
|
m_selectedWidget = nullptr;
|
|
// Reset to welcome page after rebuilding
|
|
m_webviewPanel->SetPage( GetWelcomeHtml() );
|
|
|
|
Layout();
|
|
} |