2011-11-10 15:55:05 +00:00
|
|
|
/*
|
|
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
|
|
*
|
2015-01-07 15:04:57 +00:00
|
|
|
* Copyright (C) 2014-2015 Jean-Pierre Charras, jp.charras at wanadoo.fr
|
2018-05-07 15:47:16 +00:00
|
|
|
* Copyright (C) 2008 Wayne Stambaugh <stambaughw@gmail.com>
|
|
|
|
* Copyright (C) 1992-2018 KiCad Developers, see AUTHORS.txt for contributors.
|
2011-11-10 15:55:05 +00:00
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, you may find one here:
|
|
|
|
* http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
|
|
|
|
* or you may search the http://www.gnu.org website for the version 2 license,
|
|
|
|
* or you may write to the Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @file common.cpp
|
|
|
|
*/
|
2009-11-23 15:16:50 +00:00
|
|
|
|
2012-01-23 04:33:36 +00:00
|
|
|
#include <fctsys.h>
|
2018-01-29 15:39:40 +00:00
|
|
|
#include <eda_base_frame.h>
|
2012-01-23 04:33:36 +00:00
|
|
|
#include <base_struct.h>
|
|
|
|
#include <common.h>
|
|
|
|
#include <macros.h>
|
2012-04-13 18:51:24 +00:00
|
|
|
#include <base_units.h>
|
2014-10-26 13:59:01 +00:00
|
|
|
#include <reporter.h>
|
2012-04-13 18:51:24 +00:00
|
|
|
|
2008-04-24 16:55:35 +00:00
|
|
|
#include <wx/process.h>
|
2014-09-05 21:12:38 +00:00
|
|
|
#include <wx/config.h>
|
|
|
|
#include <wx/utils.h>
|
|
|
|
#include <wx/stdpaths.h>
|
2018-03-08 22:57:31 +00:00
|
|
|
#include <wx/url.h>
|
2008-01-05 17:20:46 +00:00
|
|
|
|
2016-05-10 07:11:09 +00:00
|
|
|
#include <pgm_base.h>
|
|
|
|
|
2017-02-20 16:57:41 +00:00
|
|
|
using KIGFX::COLOR4D;
|
|
|
|
|
2013-06-28 16:29:39 +00:00
|
|
|
|
2009-04-05 20:49:15 +00:00
|
|
|
/**
|
|
|
|
* Global variables definitions.
|
|
|
|
*
|
2013-03-19 01:25:19 +00:00
|
|
|
* TODO: All of these variables should be moved into the class were they
|
2009-04-05 20:49:15 +00:00
|
|
|
* are defined and used. Most of them probably belong in the
|
|
|
|
* application class.
|
|
|
|
*/
|
|
|
|
|
2017-02-20 16:57:41 +00:00
|
|
|
COLOR4D g_GhostColor;
|
2009-04-05 20:49:15 +00:00
|
|
|
|
2012-03-26 21:45:05 +00:00
|
|
|
|
2016-05-28 16:46:22 +00:00
|
|
|
std::atomic<unsigned int> LOCALE_IO::m_c_count(0);
|
2012-04-17 06:13:22 +00:00
|
|
|
|
2016-05-10 07:11:09 +00:00
|
|
|
LOCALE_IO::LOCALE_IO()
|
2008-06-06 16:39:45 +00:00
|
|
|
{
|
2016-05-10 07:11:09 +00:00
|
|
|
// use thread safe, atomic operation
|
2016-05-28 16:46:22 +00:00
|
|
|
if( m_c_count++ == 0 )
|
2016-05-10 07:11:09 +00:00
|
|
|
{
|
|
|
|
// Store the user locale name, to restore this locale later, in dtor
|
|
|
|
m_user_locale = setlocale( LC_ALL, 0 );
|
|
|
|
// Switch the locale to C locale, to read/write files with fp numbers
|
|
|
|
setlocale( LC_ALL, "C" );
|
|
|
|
}
|
2008-06-06 16:39:45 +00:00
|
|
|
}
|
|
|
|
|
2018-05-07 15:47:16 +00:00
|
|
|
|
2016-05-10 07:11:09 +00:00
|
|
|
LOCALE_IO::~LOCALE_IO()
|
2008-06-06 16:39:45 +00:00
|
|
|
{
|
2016-05-10 07:11:09 +00:00
|
|
|
// use thread safe, atomic operation
|
2016-05-28 16:46:22 +00:00
|
|
|
if( --m_c_count == 0 )
|
2016-05-10 07:11:09 +00:00
|
|
|
{
|
|
|
|
// revert to the user locale
|
|
|
|
setlocale( LC_ALL, m_user_locale.c_str() );
|
|
|
|
}
|
2008-06-06 16:39:45 +00:00
|
|
|
}
|
|
|
|
|
2008-09-09 11:32:21 +00:00
|
|
|
|
2012-05-06 23:32:01 +00:00
|
|
|
wxSize GetTextSize( const wxString& aSingleLine, wxWindow* aWindow )
|
|
|
|
{
|
|
|
|
wxCoord width;
|
|
|
|
wxCoord height;
|
|
|
|
|
|
|
|
{
|
|
|
|
wxClientDC dc( aWindow );
|
|
|
|
dc.SetFont( aWindow->GetFont() );
|
|
|
|
dc.GetTextExtent( aSingleLine, &width, &height );
|
|
|
|
}
|
|
|
|
|
|
|
|
return wxSize( width, height );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-11-10 15:55:05 +00:00
|
|
|
bool EnsureTextCtrlWidth( wxTextCtrl* aCtrl, const wxString* aString )
|
2009-01-29 17:30:38 +00:00
|
|
|
{
|
2009-06-13 17:06:07 +00:00
|
|
|
wxWindow* window = aCtrl->GetParent();
|
|
|
|
|
2009-01-29 17:30:38 +00:00
|
|
|
if( !window )
|
|
|
|
window = aCtrl;
|
|
|
|
|
2009-06-13 17:06:07 +00:00
|
|
|
wxString ctrlText;
|
2009-01-29 17:30:38 +00:00
|
|
|
|
|
|
|
if( !aString )
|
|
|
|
{
|
|
|
|
ctrlText = aCtrl->GetValue();
|
2009-06-13 17:06:07 +00:00
|
|
|
aString = &ctrlText;
|
2009-01-29 17:30:38 +00:00
|
|
|
}
|
|
|
|
|
2012-05-06 23:32:01 +00:00
|
|
|
wxSize textz = GetTextSize( *aString, window );
|
|
|
|
wxSize ctrlz = aCtrl->GetSize();
|
2011-11-10 15:55:05 +00:00
|
|
|
|
2012-05-06 23:32:01 +00:00
|
|
|
if( ctrlz.GetWidth() < textz.GetWidth() + 10 )
|
2009-01-29 17:30:38 +00:00
|
|
|
{
|
2012-05-06 23:32:01 +00:00
|
|
|
ctrlz.SetWidth( textz.GetWidth() + 10 );
|
|
|
|
aCtrl->SetSizeHints( ctrlz );
|
2009-01-29 17:30:38 +00:00
|
|
|
return true;
|
|
|
|
}
|
2011-11-10 15:55:05 +00:00
|
|
|
|
2009-01-29 17:30:38 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-01-15 20:01:53 +00:00
|
|
|
void wxStringSplit( const wxString& aText, wxArrayString& aStrings, wxChar aSplitter )
|
2009-04-28 19:34:42 +00:00
|
|
|
{
|
2015-01-15 20:01:53 +00:00
|
|
|
wxString tmp;
|
2009-06-13 17:06:07 +00:00
|
|
|
|
2015-01-15 20:01:53 +00:00
|
|
|
for( unsigned ii = 0; ii < aText.Length(); ii++ )
|
2009-04-28 19:34:42 +00:00
|
|
|
{
|
2015-01-15 20:01:53 +00:00
|
|
|
if( aText[ii] == aSplitter )
|
|
|
|
{
|
|
|
|
aStrings.Add( tmp );
|
|
|
|
tmp.Clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
else
|
|
|
|
tmp << aText[ii];
|
2009-04-28 19:34:42 +00:00
|
|
|
}
|
2009-05-02 07:35:04 +00:00
|
|
|
|
2015-01-15 20:01:53 +00:00
|
|
|
if( !tmp.IsEmpty() )
|
2009-04-28 19:34:42 +00:00
|
|
|
{
|
2015-01-15 20:01:53 +00:00
|
|
|
aStrings.Add( tmp );
|
2009-04-28 19:34:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-08-22 05:11:01 +00:00
|
|
|
|
2013-01-18 10:42:23 +00:00
|
|
|
int ProcessExecute( const wxString& aCommandLine, int aFlags, wxProcess *callback )
|
2008-04-24 16:55:35 +00:00
|
|
|
{
|
2013-01-18 10:42:23 +00:00
|
|
|
return wxExecute( aCommandLine, aFlags, callback );
|
2008-04-24 16:55:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-12-07 08:29:10 +00:00
|
|
|
timestamp_t GetNewTimeStamp()
|
2007-06-05 12:10:51 +00:00
|
|
|
{
|
2017-12-07 08:29:10 +00:00
|
|
|
static timestamp_t oldTimeStamp;
|
|
|
|
timestamp_t newTimeStamp;
|
2007-06-05 12:10:51 +00:00
|
|
|
|
2011-12-07 15:49:32 +00:00
|
|
|
newTimeStamp = time( NULL );
|
2011-04-19 14:28:34 +00:00
|
|
|
|
2011-12-07 15:49:32 +00:00
|
|
|
if( newTimeStamp <= oldTimeStamp )
|
|
|
|
newTimeStamp = oldTimeStamp + 1;
|
2011-04-19 14:28:34 +00:00
|
|
|
|
2011-12-07 15:49:32 +00:00
|
|
|
oldTimeStamp = newTimeStamp;
|
2011-04-19 14:28:34 +00:00
|
|
|
|
2011-12-07 15:49:32 +00:00
|
|
|
return newTimeStamp;
|
2007-06-05 12:10:51 +00:00
|
|
|
}
|
|
|
|
|
2007-08-22 05:11:01 +00:00
|
|
|
|
2012-04-13 18:51:24 +00:00
|
|
|
double RoundTo0( double x, double precision )
|
|
|
|
{
|
|
|
|
assert( precision != 0 );
|
2011-03-25 19:16:05 +00:00
|
|
|
|
// Dick Hollenbeck's KiROUND R&D
// This provides better project control over rounding to int from double
// than wxRound() did. This scheme provides better logging in Debug builds
// and it provides for compile time calculation of constants.
#include <stdio.h>
#include <assert.h>
#include <limits.h>
//-----<KiROUND KIT>------------------------------------------------------------
/**
* KiROUND
* rounds a floating point number to an int using
* "round halfway cases away from zero".
* In Debug build an assert fires if will not fit into an int.
*/
#if defined( DEBUG )
// DEBUG: a macro to capture line and file, then calls this inline
static inline int KiRound( double v, int line, const char* filename )
{
v = v < 0 ? v - 0.5 : v + 0.5;
if( v > INT_MAX + 0.5 )
{
printf( "%s: in file %s on line %d, val: %.16g too ' > 0 ' for int\n", __FUNCTION__, filename, line, v );
}
else if( v < INT_MIN - 0.5 )
{
printf( "%s: in file %s on line %d, val: %.16g too ' < 0 ' for int\n", __FUNCTION__, filename, line, v );
}
return int( v );
}
#define KiROUND( v ) KiRound( v, __LINE__, __FILE__ )
#else
// RELEASE: a macro so compile can pre-compute constants.
#define KiROUND( v ) int( (v) < 0 ? (v) - 0.5 : (v) + 0.5 )
#endif
//-----</KiROUND KIT>-----------------------------------------------------------
// Only a macro is compile time calculated, an inline function causes a static constructor
// in a situation like this.
// Therefore the Release build is best done with a MACRO not an inline function.
int Computed = KiROUND( 14.3 * 8 );
int main( int argc, char** argv )
{
for( double d = double(INT_MAX)-1; d < double(INT_MAX)+8; d += 2.0 )
{
int i = KiROUND( d );
printf( "t: %d %.16g\n", i, d );
}
return 0;
}
2012-04-19 06:55:45 +00:00
|
|
|
long long ix = KiROUND( x * precision );
|
2011-03-25 19:16:05 +00:00
|
|
|
|
2012-04-13 18:51:24 +00:00
|
|
|
if ( x < 0.0 )
|
2015-06-26 13:41:56 +00:00
|
|
|
ix = -ix;
|
2011-03-25 19:16:05 +00:00
|
|
|
|
2012-04-13 18:51:24 +00:00
|
|
|
int remainder = ix % 10; // remainder is in precision mm
|
2008-05-05 19:43:56 +00:00
|
|
|
|
2015-01-07 15:04:57 +00:00
|
|
|
if( remainder <= 2 )
|
2012-04-13 18:51:24 +00:00
|
|
|
ix -= remainder; // truncate to the near number
|
2015-01-07 15:04:57 +00:00
|
|
|
else if( remainder >= 8 )
|
2012-04-13 18:51:24 +00:00
|
|
|
ix += 10 - remainder; // round to near number
|
2008-05-05 19:43:56 +00:00
|
|
|
|
2012-04-13 18:51:24 +00:00
|
|
|
if ( x < 0 )
|
2015-06-26 13:41:56 +00:00
|
|
|
ix = -ix;
|
2012-04-13 18:51:24 +00:00
|
|
|
|
|
|
|
return (double) ix / precision;
|
2008-05-05 19:43:56 +00:00
|
|
|
}
|
2013-05-23 18:45:23 +00:00
|
|
|
|
2014-09-05 21:12:38 +00:00
|
|
|
|
|
|
|
wxConfigBase* GetNewConfig( const wxString& aProgName )
|
|
|
|
{
|
|
|
|
wxConfigBase* cfg = 0;
|
|
|
|
wxFileName configname;
|
|
|
|
configname.AssignDir( GetKicadConfigPath() );
|
|
|
|
configname.SetFullName( aProgName );
|
|
|
|
|
|
|
|
cfg = new wxFileConfig( wxT( "" ), wxT( "" ), configname.GetFullPath() );
|
|
|
|
return cfg;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
wxString GetKicadConfigPath()
|
|
|
|
{
|
|
|
|
wxFileName cfgpath;
|
|
|
|
|
2018-04-24 15:53:59 +00:00
|
|
|
// http://docs.wxwidgets.org/3.0/classwx_standard_paths.html#a7c7cf595d94d29147360d031647476b0
|
2014-09-05 21:12:38 +00:00
|
|
|
cfgpath.AssignDir( wxStandardPaths::Get().GetUserConfigDir() );
|
|
|
|
|
2018-05-07 15:47:16 +00:00
|
|
|
// GetUserConfigDir() does not default to ~/.config which is the current standard
|
|
|
|
// configuration file location on Linux. This has been fixed in later versions of wxWidgets.
|
|
|
|
#if !defined( __WXMSW__ ) && !defined( __WXMAC__ )
|
|
|
|
wxArrayString dirs = cfgpath.GetDirs();
|
|
|
|
|
|
|
|
if( dirs.Last() != ".config" )
|
|
|
|
cfgpath.AppendDir( ".config" );
|
|
|
|
#endif
|
|
|
|
|
2014-09-05 21:12:38 +00:00
|
|
|
wxString envstr;
|
|
|
|
|
2018-05-07 15:47:16 +00:00
|
|
|
// This shouldn't cause any issues on Windows or MacOS.
|
2018-04-24 15:53:59 +00:00
|
|
|
if( wxGetEnv( wxT( "XDG_CONFIG_HOME" ), &envstr ) && !envstr.IsEmpty() )
|
2014-09-05 21:12:38 +00:00
|
|
|
{
|
|
|
|
// Override the assignment above with XDG_CONFIG_HOME
|
|
|
|
cfgpath.AssignDir( envstr );
|
|
|
|
}
|
2018-04-24 15:53:59 +00:00
|
|
|
|
2014-09-05 21:12:38 +00:00
|
|
|
cfgpath.AppendDir( wxT( "kicad" ) );
|
|
|
|
|
2018-04-24 15:53:59 +00:00
|
|
|
// Use KICAD_CONFIG_HOME to allow the user to force a specific configuration path.
|
|
|
|
if( wxGetEnv( wxT( "KICAD_CONFIG_HOME" ), &envstr ) && !envstr.IsEmpty() )
|
|
|
|
{
|
2018-04-27 13:29:53 +00:00
|
|
|
// Override the assignment above with KICAD_CONFIG_HOME
|
2018-04-24 15:53:59 +00:00
|
|
|
cfgpath.AssignDir( envstr );
|
|
|
|
}
|
|
|
|
|
2014-09-05 21:12:38 +00:00
|
|
|
if( !cfgpath.DirExists() )
|
|
|
|
{
|
|
|
|
cfgpath.Mkdir( wxS_DIR_DEFAULT, wxPATH_MKDIR_FULL );
|
|
|
|
}
|
|
|
|
|
|
|
|
return cfgpath.GetPath();
|
|
|
|
}
|
2014-10-16 22:13:01 +00:00
|
|
|
|
2014-10-26 13:59:01 +00:00
|
|
|
|
2018-07-30 13:18:37 +00:00
|
|
|
enum Bracket
|
|
|
|
{
|
|
|
|
Bracket_None,
|
|
|
|
Bracket_Normal = ')',
|
|
|
|
Bracket_Curly = '}',
|
|
|
|
#ifdef __WINDOWS__
|
|
|
|
Bracket_Windows = '%', // yeah, Windows people are a bit strange ;-)
|
|
|
|
#endif
|
|
|
|
Bracket_Max
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Stolen from wxExpandEnvVars and then heavily optimized
|
|
|
|
//
|
|
|
|
wxString KIwxExpandEnvVars(const wxString& str)
|
|
|
|
{
|
|
|
|
size_t strlen = str.length();
|
|
|
|
|
|
|
|
wxString strResult;
|
|
|
|
strResult.Alloc(strlen);
|
|
|
|
|
|
|
|
for ( size_t n = 0; n < strlen; n++ ) {
|
|
|
|
wxUniChar str_n = str[n];
|
|
|
|
|
|
|
|
switch ( str_n.GetValue() ) {
|
|
|
|
#ifdef __WINDOWS__
|
2018-08-04 10:28:46 +00:00
|
|
|
case wxT('%'):
|
2018-07-30 13:18:37 +00:00
|
|
|
#endif // __WINDOWS__
|
|
|
|
case wxT('$'):
|
|
|
|
{
|
|
|
|
Bracket bracket;
|
|
|
|
#ifdef __WINDOWS__
|
|
|
|
if ( str_n == wxT('%') )
|
|
|
|
bracket = Bracket_Windows;
|
|
|
|
else
|
|
|
|
#endif // __WINDOWS__
|
|
|
|
if ( n == strlen - 1 ) {
|
|
|
|
bracket = Bracket_None;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
switch ( str[n + 1].GetValue() ) {
|
|
|
|
case wxT('('):
|
|
|
|
bracket = Bracket_Normal;
|
2018-08-04 10:28:46 +00:00
|
|
|
str_n = str[++n]; // skip the bracket
|
2018-07-30 13:18:37 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case wxT('{'):
|
|
|
|
bracket = Bracket_Curly;
|
2018-08-04 10:28:46 +00:00
|
|
|
str_n = str[++n]; // skip the bracket
|
2018-07-30 13:18:37 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
bracket = Bracket_None;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t m = n + 1;
|
|
|
|
wxUniChar str_m = str[m];
|
|
|
|
|
|
|
|
while ( m < strlen && (wxIsalnum(str_m) || str_m == wxT('_')) )
|
|
|
|
str_m = str[++m];
|
|
|
|
|
|
|
|
wxString strVarName(str.c_str() + n + 1, m - n - 1);
|
|
|
|
|
|
|
|
#ifdef __WXWINCE__
|
|
|
|
const bool expanded = false;
|
|
|
|
#else
|
|
|
|
// NB: use wxGetEnv instead of wxGetenv as otherwise variables
|
|
|
|
// set through wxSetEnv may not be read correctly!
|
|
|
|
bool expanded = false;
|
|
|
|
wxString tmp;
|
|
|
|
if (wxGetEnv(strVarName, &tmp))
|
|
|
|
{
|
|
|
|
strResult += tmp;
|
|
|
|
expanded = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
// variable doesn't exist => don't change anything
|
|
|
|
#ifdef __WINDOWS__
|
|
|
|
if ( bracket != Bracket_Windows )
|
|
|
|
#endif
|
|
|
|
if ( bracket != Bracket_None )
|
|
|
|
strResult << str[n - 1];
|
|
|
|
strResult << str_n << strVarName;
|
|
|
|
}
|
|
|
|
|
|
|
|
// check the closing bracket
|
|
|
|
if ( bracket != Bracket_None ) {
|
|
|
|
if ( m == strlen || str_m != (wxChar)bracket ) {
|
|
|
|
// under MSW it's common to have '%' characters in the registry
|
|
|
|
// and it's annoying to have warnings about them each time, so
|
|
|
|
// ignroe them silently if they are not used for env vars
|
|
|
|
//
|
|
|
|
// under Unix, OTOH, this warning could be useful for the user to
|
|
|
|
// understand why isn't the variable expanded as intended
|
|
|
|
#ifndef __WINDOWS__
|
|
|
|
wxLogWarning(_("Environment variables expansion failed: missing '%c' at position %u in '%s'."),
|
|
|
|
(char)bracket, (unsigned int) (m + 1), str.c_str());
|
|
|
|
#endif // __WINDOWS__
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// skip closing bracket unless the variables wasn't expanded
|
|
|
|
if ( !expanded )
|
|
|
|
strResult << (wxChar)bracket;
|
|
|
|
str_m = str[++m];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
n = m - 1; // skip variable name
|
2018-08-04 10:28:46 +00:00
|
|
|
str_n = str[n];
|
2018-07-30 13:18:37 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case wxT('\\'):
|
|
|
|
// backslash can be used to suppress special meaning of % and $
|
|
|
|
if ( n != strlen - 1 && (str[n + 1] == wxT('%') || str[n + 1] == wxT('$')) ) {
|
2018-08-04 10:28:46 +00:00
|
|
|
str_n = str[++n];
|
|
|
|
strResult += str_n;
|
2018-07-30 13:18:37 +00:00
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
//else: fall through
|
|
|
|
|
|
|
|
default:
|
|
|
|
strResult += str_n;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return strResult;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-12-23 13:01:59 +00:00
|
|
|
#include <ki_mutex.h>
|
|
|
|
const wxString ExpandEnvVarSubstitutions( const wxString& aString )
|
|
|
|
{
|
|
|
|
// wxGetenv( wchar_t* ) is not re-entrant on linux.
|
|
|
|
// Put a lock on multithreaded use of wxGetenv( wchar_t* ), called from wxEpandEnvVars(),
|
|
|
|
static MUTEX getenv_mutex;
|
|
|
|
|
|
|
|
MUTLOCK lock( getenv_mutex );
|
|
|
|
|
|
|
|
// We reserve the right to do this another way, by providing our own member
|
|
|
|
// function.
|
2018-07-30 13:18:37 +00:00
|
|
|
return KIwxExpandEnvVars( aString );
|
2014-12-23 13:01:59 +00:00
|
|
|
}
|
2014-10-26 13:59:01 +00:00
|
|
|
|
2018-03-08 22:57:31 +00:00
|
|
|
|
|
|
|
const wxString ResolveUriByEnvVars( const wxString& aUri )
|
|
|
|
{
|
|
|
|
// URL-like URI: return as is.
|
|
|
|
wxURL url( aUri );
|
|
|
|
if( url.GetError() == wxURL_NOERR )
|
|
|
|
return aUri;
|
|
|
|
|
|
|
|
// Otherwise, the path points to a local file. Resolve environment
|
|
|
|
// variables if any.
|
|
|
|
return ExpandEnvVarSubstitutions( aUri );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-10-26 13:59:01 +00:00
|
|
|
bool EnsureFileDirectoryExists( wxFileName* aTargetFullFileName,
|
|
|
|
const wxString& aBaseFilename,
|
|
|
|
REPORTER* aReporter )
|
|
|
|
{
|
|
|
|
wxString msg;
|
|
|
|
wxString baseFilePath = wxFileName( aBaseFilename ).GetPath();
|
|
|
|
|
|
|
|
// make aTargetFullFileName path, which is relative to aBaseFilename path (if it is not
|
|
|
|
// already an absolute path) absolute:
|
|
|
|
if( !aTargetFullFileName->MakeAbsolute( baseFilePath ) )
|
|
|
|
{
|
|
|
|
if( aReporter )
|
|
|
|
{
|
2017-12-15 11:37:46 +00:00
|
|
|
msg.Printf( _( "Cannot make path \"%s\" absolute with respect to \"%s\"." ),
|
2014-10-26 13:59:01 +00:00
|
|
|
GetChars( aTargetFullFileName->GetPath() ),
|
|
|
|
GetChars( baseFilePath ) );
|
2015-06-16 12:20:42 +00:00
|
|
|
aReporter->Report( msg, REPORTER::RPT_ERROR );
|
2014-10-26 13:59:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure the path of aTargetFullFileName exists, and create it if needed:
|
|
|
|
wxString outputPath( aTargetFullFileName->GetPath() );
|
|
|
|
|
|
|
|
if( !wxFileName::DirExists( outputPath ) )
|
|
|
|
{
|
|
|
|
if( wxMkdir( outputPath ) )
|
|
|
|
{
|
|
|
|
if( aReporter )
|
|
|
|
{
|
2017-12-15 11:37:46 +00:00
|
|
|
msg.Printf( _( "Output directory \"%s\" created.\n" ), GetChars( outputPath ) );
|
2015-06-16 12:20:42 +00:00
|
|
|
aReporter->Report( msg, REPORTER::RPT_INFO );
|
2014-10-26 13:59:01 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if( aReporter )
|
|
|
|
{
|
2017-12-15 11:37:46 +00:00
|
|
|
msg.Printf( _( "Cannot create output directory \"%s\".\n" ),
|
2014-10-26 13:59:01 +00:00
|
|
|
GetChars( outputPath ) );
|
2015-06-16 12:20:42 +00:00
|
|
|
aReporter->Report( msg, REPORTER::RPT_ERROR );
|
2014-10-26 13:59:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-10-16 22:13:01 +00:00
|
|
|
#ifdef __WXMAC__
|
|
|
|
wxString GetOSXKicadUserDataDir()
|
|
|
|
{
|
|
|
|
// According to wxWidgets documentation for GetUserDataDir:
|
|
|
|
// Mac: ~/Library/Application Support/appname
|
|
|
|
wxFileName udir( wxStandardPaths::Get().GetUserDataDir(), wxEmptyString );
|
|
|
|
|
|
|
|
// Since appname is different if started via launcher or standalone binary
|
|
|
|
// map all to "kicad" here
|
|
|
|
udir.RemoveLastDir();
|
|
|
|
udir.AppendDir( wxT( "kicad" ) );
|
|
|
|
|
|
|
|
return udir.GetPath();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
wxString GetOSXKicadMachineDataDir()
|
|
|
|
{
|
|
|
|
return wxT( "/Library/Application Support/kicad" );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
wxString GetOSXKicadDataDir()
|
|
|
|
{
|
|
|
|
// According to wxWidgets documentation for GetDataDir:
|
|
|
|
// Mac: appname.app/Contents/SharedSupport bundle subdirectory
|
|
|
|
wxFileName ddir( wxStandardPaths::Get().GetDataDir(), wxEmptyString );
|
|
|
|
|
|
|
|
// This must be mapped to main bundle for everything but kicad.app
|
|
|
|
const wxArrayString dirs = ddir.GetDirs();
|
|
|
|
if( dirs[dirs.GetCount() - 3] != wxT( "kicad.app" ) )
|
|
|
|
{
|
|
|
|
// Bundle structure resp. current path is
|
|
|
|
// kicad.app/Contents/Applications/<standalone>.app/Contents/SharedSupport
|
|
|
|
// and will be mapped to
|
|
|
|
// kicad.app/Contents/SharedSupprt
|
|
|
|
ddir.RemoveLastDir();
|
|
|
|
ddir.RemoveLastDir();
|
|
|
|
ddir.RemoveLastDir();
|
|
|
|
ddir.RemoveLastDir();
|
|
|
|
ddir.AppendDir( wxT( "SharedSupport" ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
return ddir.GetPath();
|
|
|
|
}
|
|
|
|
#endif
|
2017-12-12 16:45:45 +00:00
|
|
|
|
2017-12-18 08:53:14 +00:00
|
|
|
// add this only if it is not in wxWidgets (for instance before 3.1.0)
|
|
|
|
#ifdef USE_KICAD_WXSTRING_HASH
|
2017-12-12 16:45:45 +00:00
|
|
|
size_t std::hash<wxString>::operator()( const wxString& s ) const
|
|
|
|
{
|
|
|
|
return std::hash<std::wstring>{}( s.ToStdWstring() );
|
|
|
|
}
|
2017-12-18 08:53:14 +00:00
|
|
|
#endif
|
2018-04-12 16:09:18 +00:00
|
|
|
|
|
|
|
|
2018-04-13 09:06:46 +00:00
|
|
|
#ifdef USE_KICAD_WXPOINT_LESS
|
2018-04-12 16:09:18 +00:00
|
|
|
bool std::less<wxPoint>::operator()( const wxPoint& aA, const wxPoint& aB ) const
|
|
|
|
{
|
|
|
|
if( aA.x == aB.x )
|
|
|
|
return aA.y < aB.y;
|
|
|
|
|
|
|
|
return aA.x < aB.x;
|
|
|
|
}
|
2018-04-13 09:06:46 +00:00
|
|
|
#endif
|
2018-08-03 16:53:38 +00:00
|
|
|
|
|
|
|
|
2018-08-08 18:07:26 +00:00
|
|
|
/**
|
|
|
|
* Performance enhancements to file and directory operations.
|
|
|
|
*
|
|
|
|
* Note: while it's annoying to have to make copies of wxWidgets stuff and then
|
|
|
|
* add platform-specific performance optimizations, the following routies offer
|
|
|
|
* SIGNIFICANT performance benefits.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* WX_FILENAME
|
|
|
|
*
|
|
|
|
* A wrapper around a wxFileName which avoids expensive calls to wxFileName::SplitPath()
|
|
|
|
* and string concatenations by caching the path and filename locally and only resolving
|
|
|
|
* the wxFileName when it has to.
|
|
|
|
*/
|
2018-08-05 11:56:02 +00:00
|
|
|
WX_FILENAME::WX_FILENAME( const wxString& aPath, const wxString& aFilename ) :
|
|
|
|
m_fn( aPath, aFilename ),
|
|
|
|
m_path( aPath ),
|
|
|
|
m_fullName( aFilename )
|
|
|
|
{ }
|
|
|
|
|
|
|
|
|
2018-08-03 16:53:38 +00:00
|
|
|
void WX_FILENAME::SetFullName( const wxString& aFileNameAndExtension )
|
|
|
|
{
|
|
|
|
m_fullName = aFileNameAndExtension;
|
2018-08-05 11:56:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
wxString WX_FILENAME::GetName() const
|
|
|
|
{
|
|
|
|
size_t dot = m_fullName.find_last_of( wxT( '.' ) );
|
|
|
|
return m_fullName.substr( 0, dot );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
wxString WX_FILENAME::GetFullName() const
|
|
|
|
{
|
|
|
|
return m_fullName;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
wxString WX_FILENAME::GetPath() const
|
|
|
|
{
|
|
|
|
return m_path;
|
|
|
|
}
|
|
|
|
|
2018-08-03 16:53:38 +00:00
|
|
|
|
2018-08-05 11:56:02 +00:00
|
|
|
wxString WX_FILENAME::GetFullPath() const
|
|
|
|
{
|
|
|
|
return m_path + wxT( '/' ) + m_fullName;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Write locally-cached values to the wxFileName. MUST be called before using m_fn.
|
|
|
|
void WX_FILENAME::resolve()
|
|
|
|
{
|
2018-08-03 16:53:38 +00:00
|
|
|
size_t dot = m_fullName.find_last_of( wxT( '.' ) );
|
|
|
|
m_fn.SetName( m_fullName.substr( 0, dot ) );
|
|
|
|
m_fn.SetExt( m_fullName.substr( dot + 1 ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
long long WX_FILENAME::GetTimestamp()
|
|
|
|
{
|
2018-08-05 11:56:02 +00:00
|
|
|
resolve();
|
|
|
|
|
2018-08-03 16:53:38 +00:00
|
|
|
if( m_fn.FileExists() )
|
|
|
|
return m_fn.GetModificationTime().GetValue().GetValue();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-08-08 18:07:26 +00:00
|
|
|
/**
|
|
|
|
* A copy of wxMatchWild(), which wxWidgets attributes to Douglas A. Lewis
|
|
|
|
* <dalewis@cs.Buffalo.EDU> and ircII's reg.c.
|
|
|
|
*
|
|
|
|
* This version is modified to skip any encoding conversions (for performance).
|
|
|
|
*/
|
2018-08-05 11:56:02 +00:00
|
|
|
bool matchWild( const char* pat, const char* text, bool dot_special )
|
2018-08-03 16:53:38 +00:00
|
|
|
{
|
2018-08-05 11:56:02 +00:00
|
|
|
if ( !*text )
|
|
|
|
{
|
|
|
|
/* Match if both are empty. */
|
|
|
|
return !*pat;
|
|
|
|
}
|
2018-08-03 16:53:38 +00:00
|
|
|
|
2018-08-05 11:56:02 +00:00
|
|
|
const char *m = pat,
|
|
|
|
*n = text,
|
|
|
|
*ma = NULL,
|
|
|
|
*na = NULL;
|
|
|
|
int just = 0,
|
|
|
|
acount = 0,
|
|
|
|
count = 0;
|
2018-08-03 16:53:38 +00:00
|
|
|
|
2018-08-05 11:56:02 +00:00
|
|
|
if (dot_special && (*n == '.'))
|
|
|
|
{
|
|
|
|
/* Never match so that hidden Unix files
|
|
|
|
* are never found. */
|
|
|
|
return false;
|
|
|
|
}
|
2018-08-03 16:53:38 +00:00
|
|
|
|
2018-08-05 11:56:02 +00:00
|
|
|
for (;;)
|
|
|
|
{
|
|
|
|
if (*m == '*')
|
|
|
|
{
|
|
|
|
ma = ++m;
|
|
|
|
na = n;
|
|
|
|
just = 1;
|
|
|
|
acount = count;
|
|
|
|
}
|
|
|
|
else if (*m == '?')
|
|
|
|
{
|
|
|
|
m++;
|
|
|
|
if (!*n++)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (*m == '\\')
|
|
|
|
{
|
|
|
|
m++;
|
|
|
|
/* Quoting "nothing" is a bad thing */
|
|
|
|
if (!*m)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (!*m)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* If we are out of both strings or we just
|
|
|
|
* saw a wildcard, then we can say we have a
|
|
|
|
* match
|
|
|
|
*/
|
|
|
|
if (!*n)
|
|
|
|
return true;
|
|
|
|
if (just)
|
|
|
|
return true;
|
|
|
|
just = 0;
|
|
|
|
goto not_matched;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* We could check for *n == NULL at this point, but
|
|
|
|
* since it's more common to have a character there,
|
|
|
|
* check to see if they match first (m and n) and
|
|
|
|
* then if they don't match, THEN we can check for
|
|
|
|
* the NULL of n
|
|
|
|
*/
|
|
|
|
just = 0;
|
|
|
|
if (*m == *n)
|
|
|
|
{
|
|
|
|
m++;
|
|
|
|
count++;
|
|
|
|
n++;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-08-03 16:53:38 +00:00
|
|
|
|
2018-08-05 11:56:02 +00:00
|
|
|
not_matched:
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If there are no more characters in the
|
|
|
|
* string, but we still need to find another
|
|
|
|
* character (*m != NULL), then it will be
|
|
|
|
* impossible to match it
|
|
|
|
*/
|
|
|
|
if (!*n)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (ma)
|
|
|
|
{
|
|
|
|
m = ma;
|
|
|
|
n = ++na;
|
|
|
|
count = acount;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-08-03 16:53:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-08-08 18:07:26 +00:00
|
|
|
/**
|
|
|
|
* A copy of ConvertFileTimeToWx() because wxWidgets left it as a static function
|
|
|
|
* private to src/common/filename.cpp.
|
|
|
|
*/
|
|
|
|
#if wxUSE_DATETIME && defined(__WIN32__) && !defined(__WXMICROWIN__)
|
|
|
|
|
|
|
|
// Convert between wxDateTime and FILETIME which is a 64-bit value representing
|
|
|
|
// the number of 100-nanosecond intervals since January 1, 1601 UTC.
|
|
|
|
//
|
|
|
|
// This is the offset between FILETIME epoch and the Unix/wxDateTime Epoch.
|
|
|
|
static wxInt64 EPOCH_OFFSET_IN_MSEC = wxLL(11644473600000);
|
|
|
|
|
|
|
|
static void ConvertFileTimeToWx(wxDateTime *dt, const FILETIME &ft)
|
|
|
|
{
|
|
|
|
wxLongLong t(ft.dwHighDateTime, ft.dwLowDateTime);
|
|
|
|
t /= 10000; // Convert hundreds of nanoseconds to milliseconds.
|
|
|
|
t -= EPOCH_OFFSET_IN_MSEC;
|
|
|
|
|
|
|
|
*dt = wxDateTime(t);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // wxUSE_DATETIME && __WIN32__
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* TimestampDir
|
|
|
|
*
|
|
|
|
* This routine offers SIGNIFICANT performance benefits over using wxWidgets to gather
|
|
|
|
* timestamps from matching files in a directory.
|
|
|
|
* @param aDirPath the directory to search
|
|
|
|
* @param aFilespec a (wildcarded) file spec to match against
|
|
|
|
* @return a hash of the last-mod-dates of all matching files in the directory
|
|
|
|
*/
|
2018-08-05 11:56:02 +00:00
|
|
|
long long TimestampDir( const wxString& aDirPath, const wxString& aFilespec )
|
2018-08-03 16:53:38 +00:00
|
|
|
{
|
2018-08-05 11:56:02 +00:00
|
|
|
long long timestamp = 0;
|
2018-08-03 16:53:38 +00:00
|
|
|
|
2018-08-08 18:07:26 +00:00
|
|
|
#if defined(__WIN32__)
|
|
|
|
// Win32 version.
|
|
|
|
// Save time by not searching for each path twice: once in wxDir.GetNext() and once in
|
|
|
|
// wxFileName.GetModificationTime(). Also cuts out wxWidgets' string-matching and case
|
|
|
|
// conversion by staying on the MSW side of things.
|
|
|
|
std::wstring filespec( aDirPath.t_str() );
|
|
|
|
filespec += '\\';
|
|
|
|
filespec += aFilespec.t_str();
|
|
|
|
|
|
|
|
WIN32_FIND_DATA findData;
|
|
|
|
wxDateTime lastModDate;
|
2018-08-03 16:53:38 +00:00
|
|
|
|
2018-08-08 18:07:26 +00:00
|
|
|
HANDLE fileHandle = ::FindFirstFile( filespec.data(), &findData );
|
|
|
|
|
|
|
|
if( fileHandle != INVALID_HANDLE_VALUE )
|
2018-08-05 11:56:02 +00:00
|
|
|
{
|
2018-08-08 18:07:26 +00:00
|
|
|
do
|
2018-08-05 11:56:02 +00:00
|
|
|
{
|
2018-08-08 18:07:26 +00:00
|
|
|
ConvertFileTimeToWx( &lastModDate, findData.ftLastWriteTime );
|
|
|
|
timestamp += lastModDate.GetValue().GetValue();
|
2018-08-05 11:56:02 +00:00
|
|
|
}
|
2018-08-08 18:07:26 +00:00
|
|
|
while ( FindNextFile( fileHandle, &findData ) != 0);
|
2018-08-05 11:56:02 +00:00
|
|
|
}
|
2018-08-08 18:07:26 +00:00
|
|
|
|
|
|
|
FindClose( fileHandle );
|
2018-08-05 11:56:02 +00:00
|
|
|
#else
|
2018-08-08 18:07:26 +00:00
|
|
|
// POSIX version.
|
|
|
|
// Save time by not converting between encodings -- do everything on the file-system side.
|
2018-08-05 11:56:02 +00:00
|
|
|
std::string filespec( aFilespec.fn_str() );
|
|
|
|
std::string dir_path( aDirPath.fn_str() );
|
2018-08-03 16:53:38 +00:00
|
|
|
|
2018-08-05 11:56:02 +00:00
|
|
|
DIR* dir = opendir( dir_path.c_str() );
|
2018-08-03 16:53:38 +00:00
|
|
|
|
2018-08-05 11:56:02 +00:00
|
|
|
if( dir )
|
|
|
|
{
|
|
|
|
for( dirent* dir_entry = readdir( dir ); dir_entry; dir_entry = readdir( dir ) )
|
|
|
|
{
|
|
|
|
if( !matchWild( filespec.c_str(), dir_entry->d_name, true ) )
|
|
|
|
continue;
|
2018-08-03 16:53:38 +00:00
|
|
|
|
2018-08-05 11:56:02 +00:00
|
|
|
std::string entry_path = dir_path + '/' + dir_entry->d_name;
|
|
|
|
struct stat entry_stat;
|
2018-08-03 16:53:38 +00:00
|
|
|
|
2018-08-05 11:56:02 +00:00
|
|
|
wxCRT_Lstat( entry_path.c_str(), &entry_stat );
|
2018-08-03 16:53:38 +00:00
|
|
|
|
2018-08-05 11:56:02 +00:00
|
|
|
// Timestamp the source file, not the symlink
|
|
|
|
if( S_ISLNK( entry_stat.st_mode ) ) // wxFILE_EXISTS_SYMLINK
|
|
|
|
{
|
|
|
|
char buffer[ PATH_MAX + 1 ];
|
|
|
|
ssize_t pathLen = readlink( entry_path.c_str(), buffer, PATH_MAX );
|
2018-08-03 16:53:38 +00:00
|
|
|
|
2018-08-05 11:56:02 +00:00
|
|
|
if( pathLen > 0 )
|
|
|
|
{
|
|
|
|
buffer[ pathLen ] = '\0';
|
|
|
|
entry_path = dir_path + buffer;
|
2018-08-03 16:53:38 +00:00
|
|
|
|
2018-08-05 11:56:02 +00:00
|
|
|
wxCRT_Lstat( entry_path.c_str(), &entry_stat );
|
|
|
|
}
|
|
|
|
}
|
2018-08-03 16:53:38 +00:00
|
|
|
|
2018-08-05 11:56:02 +00:00
|
|
|
if( S_ISREG( entry_stat.st_mode ) ) // wxFileExists()
|
|
|
|
timestamp += entry_stat.st_mtime * 1000;
|
|
|
|
}
|
2018-08-21 17:50:55 +00:00
|
|
|
|
|
|
|
closedir( dir );
|
2018-08-05 11:56:02 +00:00
|
|
|
}
|
2018-08-03 16:53:38 +00:00
|
|
|
#endif
|
|
|
|
|
2018-08-05 11:56:02 +00:00
|
|
|
return timestamp;
|
|
|
|
}
|
2018-08-03 16:53:38 +00:00
|
|
|
|
|
|
|
|