2013-08-09 06:57:18 +02:00
# include "lc_global.h"
# include "lc_library.h"
# include "lc_zipfile.h"
# include "lc_file.h"
# include "pieceinf.h"
# include "lc_colors.h"
# include "lc_texture.h"
# include "lc_category.h"
# include "lc_application.h"
2015-04-26 20:14:33 +02:00
# include "lc_context.h"
2015-05-09 21:54:29 +02:00
# include "lc_glextensions.h"
2016-02-29 21:13:54 +01:00
# include "lc_synth.h"
2015-02-22 03:39:15 +01:00
# include "project.h"
2019-06-21 03:52:33 +02:00
# include "lc_profile.h"
2019-07-28 01:31:16 +02:00
# include "lc_meshloader.h"
2013-08-09 06:57:18 +02:00
# include <ctype.h>
# include <locale.h>
2015-07-22 06:00:47 +02:00
# include <zlib.h>
2017-01-23 06:22:46 +01:00
# if (QT_VERSION >= QT_VERSION_CHECK(5, 0, 0))
2017-01-23 05:50:43 +01:00
# include <QtConcurrent>
2017-01-23 06:22:46 +01:00
# endif
2015-07-22 06:00:47 +02:00
# if MAX_MEM_LEVEL >= 8
# define DEF_MEM_LEVEL 8
# else
# define DEF_MEM_LEVEL MAX_MEM_LEVEL
# endif
2013-08-09 06:57:18 +02:00
2019-09-24 02:06:16 +02:00
# define LC_LIBRARY_CACHE_VERSION 0x0108
2013-08-09 06:57:18 +02:00
# define LC_LIBRARY_CACHE_ARCHIVE 0x0001
# define LC_LIBRARY_CACHE_DIRECTORY 0x0002
lcPiecesLibrary : : lcPiecesLibrary ( )
2017-01-23 04:28:05 +01:00
: mLoadMutex ( QMutex : : Recursive )
2013-08-09 06:57:18 +02:00
{
2015-07-22 06:00:47 +02:00
# if (QT_VERSION >= QT_VERSION_CHECK(5, 0, 0))
QStringList cachePathList = QStandardPaths : : standardLocations ( QStandardPaths : : CacheLocation ) ;
mCachePath = cachePathList . first ( ) ;
# else
mCachePath = QDesktopServices : : storageLocation ( QDesktopServices : : CacheLocation ) ;
# endif
QDir Dir ;
Dir . mkpath ( mCachePath ) ;
2013-08-09 06:57:18 +02:00
mNumOfficialPieces = 0 ;
2015-04-26 20:14:33 +02:00
mBuffersDirty = false ;
2017-03-19 21:12:24 +01:00
mHasUnofficial = false ;
2017-11-25 21:57:41 +01:00
mCancelLoading = false ;
2019-09-24 01:55:24 +02:00
mStudLogo = lcGetProfileInt ( LC_PROFILE_STUD_LOGO ) ;
2013-08-09 06:57:18 +02:00
}
lcPiecesLibrary : : ~ lcPiecesLibrary ( )
{
2017-01-23 04:28:05 +01:00
mLoadMutex . lock ( ) ;
mLoadQueue . clear ( ) ;
mLoadMutex . unlock ( ) ;
2017-11-25 21:57:41 +01:00
mCancelLoading = true ;
2017-01-23 04:28:05 +01:00
WaitForLoadQueue ( ) ;
2013-08-09 06:57:18 +02:00
Unload ( ) ;
}
void lcPiecesLibrary : : Unload ( )
{
2018-02-22 02:27:24 +01:00
for ( const auto & PieceIt : mPieces )
2017-07-24 04:35:18 +02:00
delete PieceIt . second ;
mPieces . clear ( ) ;
2013-08-09 06:57:18 +02:00
2018-02-22 02:27:24 +01:00
for ( const auto & PrimitiveIt : mPrimitives )
2017-07-24 01:19:09 +02:00
delete PrimitiveIt . second ;
mPrimitives . clear ( ) ;
2013-08-09 06:57:18 +02:00
2019-07-05 02:06:26 +02:00
for ( lcTexture * Texture : mTextures )
delete Texture ;
mTextures . clear ( ) ;
2013-08-09 06:57:18 +02:00
mNumOfficialPieces = 0 ;
2017-04-14 02:26:40 +02:00
mZipFiles [ LC_ZIPFILE_OFFICIAL ] = nullptr ;
mZipFiles [ LC_ZIPFILE_UNOFFICIAL ] = nullptr ;
2013-08-09 06:57:18 +02:00
}
2015-01-07 17:52:42 +01:00
void lcPiecesLibrary : : RemoveTemporaryPieces ( )
{
2017-01-23 04:28:05 +01:00
QMutexLocker LoadLock ( & mLoadMutex ) ;
2017-07-24 04:35:18 +02:00
for ( auto PieceIt = mPieces . begin ( ) ; PieceIt ! = mPieces . end ( ) ; )
2015-01-07 17:52:42 +01:00
{
2017-07-24 04:35:18 +02:00
PieceInfo * Info = PieceIt - > second ;
2015-01-07 17:52:42 +01:00
2017-07-24 04:35:18 +02:00
if ( Info - > IsTemporary ( ) & & Info - > GetRefCount ( ) = = 0 )
2015-01-07 17:52:42 +01:00
{
2017-07-24 04:35:18 +02:00
PieceIt = mPieces . erase ( PieceIt ) ;
2015-01-07 17:52:42 +01:00
delete Info ;
}
2017-07-24 04:35:18 +02:00
else
PieceIt + + ;
2015-01-07 17:52:42 +01:00
}
}
2015-02-22 03:39:15 +01:00
void lcPiecesLibrary : : RemovePiece ( PieceInfo * Info )
{
2017-07-24 04:35:18 +02:00
for ( auto PieceIt = mPieces . begin ( ) ; PieceIt ! = mPieces . end ( ) ; PieceIt + + )
{
if ( PieceIt - > second = = Info )
{
mPieces . erase ( PieceIt ) ;
break ;
}
}
2015-02-22 03:39:15 +01:00
delete Info ;
}
2017-07-24 04:35:18 +02:00
void lcPiecesLibrary : : RenamePiece ( PieceInfo * Info , const char * NewName )
{
2017-07-27 18:21:55 +02:00
for ( auto PieceIt = mPieces . begin ( ) ; PieceIt ! = mPieces . end ( ) ; PieceIt + + )
{
if ( PieceIt - > second = = Info )
{
mPieces . erase ( PieceIt ) ;
break ;
}
}
strncpy ( Info - > mFileName , NewName , sizeof ( Info - > mFileName ) ) ;
Info - > mFileName [ sizeof ( Info - > mFileName ) - 1 ] = 0 ;
strncpy ( Info - > m_strDescription , NewName , sizeof ( Info - > m_strDescription ) ) ;
Info - > m_strDescription [ sizeof ( Info - > m_strDescription ) - 1 ] = 0 ;
char PieceName [ LC_PIECE_NAME_LEN ] ;
strcpy ( PieceName , Info - > mFileName ) ;
strupr ( PieceName ) ;
mPieces [ PieceName ] = Info ;
2017-07-24 04:35:18 +02:00
}
2016-11-26 02:12:19 +01:00
PieceInfo * lcPiecesLibrary : : FindPiece ( const char * PieceName , Project * CurrentProject , bool CreatePlaceholder , bool SearchProjectFolder )
2013-08-09 06:57:18 +02:00
{
2016-11-26 02:12:19 +01:00
QString ProjectPath ;
if ( SearchProjectFolder )
{
QString FileName = CurrentProject - > GetFileName ( ) ;
if ( ! FileName . isEmpty ( ) )
ProjectPath = QFileInfo ( FileName ) . absolutePath ( ) ;
}
2017-07-27 18:21:55 +02:00
char CleanName [ LC_PIECE_NAME_LEN ] ;
const char * Src = PieceName ;
char * Dst = CleanName ;
while ( * Src & & Dst - CleanName ! = sizeof ( CleanName ) )
{
if ( * Src = = ' \\ ' )
* Dst = ' / ' ;
else if ( * Src > = ' a ' & & * Src < = ' z ' )
* Dst = * Src + ' A ' - ' a ' ;
else
* Dst = * Src ;
Src + + ;
Dst + + ;
}
* Dst = 0 ;
const auto PieceIt = mPieces . find ( CleanName ) ;
2015-02-22 03:39:15 +01:00
2017-07-24 04:35:18 +02:00
if ( PieceIt ! = mPieces . end ( ) )
{
PieceInfo * Info = PieceIt - > second ;
2015-02-22 03:39:15 +01:00
2017-07-24 04:35:18 +02:00
if ( ( ! CurrentProject | | ! Info - > IsModel ( ) | | CurrentProject - > GetModels ( ) . FindIndex ( Info - > GetModel ( ) ) ! = - 1 ) & & ( ! ProjectPath . isEmpty ( ) | | ! Info - > IsProject ( ) ) )
return Info ;
2015-02-22 03:39:15 +01:00
}
2013-08-09 06:57:18 +02:00
2016-11-26 02:12:19 +01:00
if ( ! ProjectPath . isEmpty ( ) )
{
QFileInfo ProjectFile = QFileInfo ( ProjectPath + QDir : : separator ( ) + PieceName ) ;
if ( ProjectFile . isFile ( ) )
{
Project * NewProject = new Project ( ) ;
if ( NewProject - > Load ( ProjectFile . absoluteFilePath ( ) ) )
{
PieceInfo * Info = new PieceInfo ( ) ;
2017-07-27 18:21:55 +02:00
Info - > CreateProject ( NewProject , PieceName ) ;
mPieces [ CleanName ] = Info ;
2016-11-26 02:12:19 +01:00
return Info ;
}
else
delete NewProject ;
}
}
2014-12-24 16:52:52 +01:00
if ( CreatePlaceholder )
{
PieceInfo * Info = new PieceInfo ( ) ;
2013-08-09 06:57:18 +02:00
2014-12-24 16:52:52 +01:00
Info - > CreatePlaceholder ( PieceName ) ;
2017-07-27 18:21:55 +02:00
mPieces [ CleanName ] = Info ;
2013-08-09 06:57:18 +02:00
2014-12-24 16:52:52 +01:00
return Info ;
}
2013-08-09 06:57:18 +02:00
2017-04-14 02:26:40 +02:00
return nullptr ;
2013-08-09 06:57:18 +02:00
}
2017-07-02 02:12:09 +02:00
lcTexture * lcPiecesLibrary : : FindTexture ( const char * TextureName , Project * CurrentProject , bool SearchProjectFolder )
2013-08-09 06:57:18 +02:00
{
2019-07-05 02:06:26 +02:00
for ( lcTexture * Texture : mTextures )
if ( ! strcmp ( TextureName , Texture - > mName ) )
return Texture ;
2013-08-09 06:57:18 +02:00
2017-07-02 02:12:09 +02:00
QString ProjectPath ;
if ( SearchProjectFolder )
{
QString FileName = CurrentProject - > GetFileName ( ) ;
if ( ! FileName . isEmpty ( ) )
ProjectPath = QFileInfo ( FileName ) . absolutePath ( ) ;
}
if ( ! ProjectPath . isEmpty ( ) )
{
QFileInfo TextureFile = QFileInfo ( ProjectPath + QDir : : separator ( ) + TextureName + " .png " ) ;
if ( TextureFile . isFile ( ) )
{
lcTexture * Texture = lcLoadTexture ( TextureFile . absoluteFilePath ( ) , LC_TEXTURE_WRAPU | LC_TEXTURE_WRAPV ) ;
if ( Texture )
{
2019-07-05 02:06:26 +02:00
mTextures . push_back ( Texture ) ;
2017-07-02 02:12:09 +02:00
return Texture ;
}
}
}
2017-04-14 02:26:40 +02:00
return nullptr ;
2013-08-09 06:57:18 +02:00
}
2017-11-25 05:00:16 +01:00
bool lcPiecesLibrary : : Load ( const QString & LibraryPath , bool ShowProgress )
2013-08-09 06:57:18 +02:00
{
2014-05-08 00:58:59 +02:00
Unload ( ) ;
2019-06-21 03:52:33 +02:00
auto LoadCustomColors = [ ] ( )
{
QString CustomColorsPath = lcGetProfileString ( LC_PROFILE_COLOR_CONFIG ) ;
if ( CustomColorsPath . isEmpty ( ) )
return false ;
lcDiskFile ColorFile ( CustomColorsPath ) ;
return ColorFile . Open ( QIODevice : : ReadOnly ) & & lcLoadColorFile ( ColorFile ) ;
} ;
2014-05-08 00:58:59 +02:00
if ( OpenArchive ( LibraryPath , LC_ZIPFILE_OFFICIAL ) )
2013-08-09 06:57:18 +02:00
{
lcMemFile ColorFile ;
2019-06-21 03:52:33 +02:00
if ( ! LoadCustomColors ( ) )
if ( ! mZipFiles [ LC_ZIPFILE_OFFICIAL ] - > ExtractFile ( " ldraw/ldconfig.ldr " , ColorFile ) | | ! lcLoadColorFile ( ColorFile ) )
lcLoadDefaultColors ( ) ;
2013-08-09 06:57:18 +02:00
2017-05-29 22:31:24 +02:00
mLibraryDir = QFileInfo ( LibraryPath ) . absoluteDir ( ) ;
QString UnofficialFileName = mLibraryDir . absoluteFilePath ( QLatin1String ( " ldrawunf.zip " ) ) ;
2014-05-08 00:58:59 +02:00
2016-12-22 01:49:45 +01:00
if ( ! OpenArchive ( UnofficialFileName , LC_ZIPFILE_UNOFFICIAL ) )
2017-05-29 22:31:24 +02:00
UnofficialFileName . clear ( ) ;
2016-12-22 01:49:45 +01:00
2015-07-22 06:00:47 +02:00
ReadArchiveDescriptions ( LibraryPath , UnofficialFileName ) ;
2013-08-09 06:57:18 +02:00
}
else
{
2020-02-24 23:31:08 +01:00
mLibraryDir . setPath ( LibraryPath ) ;
2013-08-09 06:57:18 +02:00
2017-11-25 05:00:16 +01:00
if ( OpenDirectory ( mLibraryDir , ShowProgress ) )
2013-08-09 06:57:18 +02:00
{
2019-06-21 03:52:33 +02:00
if ( ! LoadCustomColors ( ) )
2019-05-18 19:49:48 +02:00
{
2019-06-21 03:52:33 +02:00
lcDiskFile ColorFile ( mLibraryDir . absoluteFilePath ( QLatin1String ( " ldconfig.ldr " ) ) ) ;
2019-05-18 19:49:48 +02:00
if ( ! ColorFile . Open ( QIODevice : : ReadOnly ) | | ! lcLoadColorFile ( ColorFile ) )
2019-06-21 03:52:33 +02:00
{
ColorFile . SetFileName ( mLibraryDir . absoluteFilePath ( QLatin1String ( " LDConfig.ldr " ) ) ) ;
if ( ! ColorFile . Open ( QIODevice : : ReadOnly ) | | ! lcLoadColorFile ( ColorFile ) )
lcLoadDefaultColors ( ) ;
}
2019-05-18 19:49:48 +02:00
}
2013-08-09 06:57:18 +02:00
}
else
return false ;
}
lcLoadDefaultCategories ( ) ;
2016-02-29 21:13:54 +01:00
lcSynthInit ( ) ;
2013-08-09 06:57:18 +02:00
return true ;
}
2017-05-29 22:31:24 +02:00
bool lcPiecesLibrary : : OpenArchive ( const QString & FileName , lcZipFileType ZipFileType )
2014-09-11 21:55:34 +02:00
{
2020-12-13 16:40:29 +01:00
std : : unique_ptr < lcDiskFile > File ( new lcDiskFile ( FileName ) ) ;
2014-09-11 21:55:34 +02:00
2020-12-13 16:40:29 +01:00
if ( ! File - > Open ( QIODevice : : ReadOnly ) )
2014-09-11 21:55:34 +02:00
return false ;
2020-12-19 20:12:31 +01:00
return OpenArchive ( std : : move ( File ) , ZipFileType ) ;
2014-09-11 21:55:34 +02:00
}
2020-12-19 20:12:31 +01:00
bool lcPiecesLibrary : : OpenArchive ( std : : unique_ptr < lcFile > File , lcZipFileType ZipFileType )
2013-08-09 06:57:18 +02:00
{
2020-12-13 17:23:05 +01:00
std : : unique_ptr < lcZipFile > ZipFile ( new lcZipFile ( ) ) ;
2013-08-09 06:57:18 +02:00
2020-12-13 16:40:29 +01:00
if ( ! ZipFile - > OpenRead ( std : : move ( File ) ) )
2013-08-09 06:57:18 +02:00
return false ;
2014-05-08 00:58:59 +02:00
for ( int FileIdx = 0 ; FileIdx < ZipFile - > mFiles . GetSize ( ) ; FileIdx + + )
2013-08-09 06:57:18 +02:00
{
2014-05-08 00:58:59 +02:00
lcZipFileInfo & FileInfo = ZipFile - > mFiles [ FileIdx ] ;
char NameBuffer [ LC_PIECE_NAME_LEN ] ;
char * Name = NameBuffer ;
2013-08-09 06:57:18 +02:00
const char * Src = FileInfo . file_name ;
char * Dst = Name ;
while ( * Src & & Dst - Name < LC_PIECE_NAME_LEN )
{
if ( * Src > = ' a ' & & * Src < = ' z ' )
* Dst = * Src + ' A ' - ' a ' ;
else if ( * Src = = ' \\ ' )
* Dst = ' / ' ;
else
* Dst = * Src ;
Src + + ;
Dst + + ;
}
if ( Dst - Name < = 4 )
continue ;
2017-07-27 18:21:55 +02:00
* Dst = 0 ;
2013-08-09 06:57:18 +02:00
Dst - = 4 ;
if ( memcmp ( Dst , " .DAT " , 4 ) )
{
2017-12-22 13:38:47 +01:00
if ( ! memcmp ( Dst , " .PNG " , 4 ) )
2013-08-09 06:57:18 +02:00
{
2017-12-22 13:38:47 +01:00
if ( ( ZipFileType = = LC_ZIPFILE_OFFICIAL & & ! memcmp ( Name , " LDRAW/PARTS/TEXTURES/ " , 21 ) ) | |
2019-09-21 18:47:33 +02:00
( ZipFileType = = LC_ZIPFILE_UNOFFICIAL & & ! memcmp ( Name , " PARTS/TEXTURES/ " , 15 ) ) )
2017-12-22 13:38:47 +01:00
{
lcTexture * Texture = new lcTexture ( ) ;
2019-07-05 02:06:26 +02:00
mTextures . push_back ( Texture ) ;
2013-08-09 06:57:18 +02:00
2017-12-22 13:38:47 +01:00
* Dst = 0 ;
strncpy ( Texture - > mName , Name + ( ZipFileType = = LC_ZIPFILE_OFFICIAL ? 21 : 15 ) , sizeof ( Texture - > mName ) ) ;
Texture - > mName [ sizeof ( Texture - > mName ) - 1 ] = 0 ;
}
2013-08-09 06:57:18 +02:00
}
continue ;
}
2014-05-08 00:58:59 +02:00
if ( ZipFileType = = LC_ZIPFILE_OFFICIAL )
{
if ( memcmp ( Name , " LDRAW/ " , 6 ) )
continue ;
Name + = 6 ;
}
2013-08-09 06:57:18 +02:00
2014-05-08 00:58:59 +02:00
if ( ! memcmp ( Name , " PARTS/ " , 6 ) )
2013-08-09 06:57:18 +02:00
{
2014-05-08 00:58:59 +02:00
Name + = 6 ;
if ( memcmp ( Name , " S/ " , 2 ) )
2013-08-09 06:57:18 +02:00
{
2017-04-14 02:26:40 +02:00
PieceInfo * Info = FindPiece ( Name , nullptr , false , false ) ;
2014-05-08 00:58:59 +02:00
if ( ! Info )
{
Info = new PieceInfo ( ) ;
2013-08-09 06:57:18 +02:00
2017-07-27 18:21:55 +02:00
strncpy ( Info - > mFileName , FileInfo . file_name + ( Name - NameBuffer ) , sizeof ( Info - > mFileName ) ) ;
Info - > mFileName [ sizeof ( Info - > mFileName ) - 1 ] = 0 ;
2017-07-24 04:35:18 +02:00
2017-07-27 18:21:55 +02:00
mPieces [ Name ] = Info ;
2014-05-08 00:58:59 +02:00
}
Info - > SetZipFile ( ZipFileType , FileIdx ) ;
2013-08-09 06:57:18 +02:00
}
else
{
2017-07-24 01:19:09 +02:00
lcLibraryPrimitive * Primitive = FindPrimitive ( Name ) ;
2014-05-08 00:58:59 +02:00
2017-07-24 01:19:09 +02:00
if ( ! Primitive )
2019-06-02 04:54:09 +02:00
mPrimitives [ Name ] = new lcLibraryPrimitive ( QString ( ) , FileInfo . file_name + ( Name - NameBuffer ) , ZipFileType , FileIdx , false , true ) ;
2014-05-08 00:58:59 +02:00
else
2017-07-24 01:19:09 +02:00
Primitive - > SetZipFile ( ZipFileType , FileIdx ) ;
2013-08-09 06:57:18 +02:00
}
}
2014-05-08 00:58:59 +02:00
else if ( ! memcmp ( Name , " P/ " , 2 ) )
2013-08-09 06:57:18 +02:00
{
2014-05-08 00:58:59 +02:00
Name + = 2 ;
2017-07-24 01:19:09 +02:00
lcLibraryPrimitive * Primitive = FindPrimitive ( Name ) ;
2014-05-08 00:58:59 +02:00
2017-07-24 01:19:09 +02:00
if ( ! Primitive )
2019-06-02 04:54:09 +02:00
mPrimitives [ Name ] = new lcLibraryPrimitive ( QString ( ) , FileInfo . file_name + ( Name - NameBuffer ) , ZipFileType , FileIdx , ( memcmp ( Name , " STU " , 3 ) = = 0 ) , false ) ;
2014-05-08 00:58:59 +02:00
else
2017-07-24 01:19:09 +02:00
Primitive - > SetZipFile ( ZipFileType , FileIdx ) ;
2013-08-09 06:57:18 +02:00
}
}
2020-12-13 17:23:05 +01:00
mZipFiles [ ZipFileType ] = std : : move ( ZipFile ) ;
2014-05-08 00:58:59 +02:00
return true ;
}
2015-07-22 06:00:47 +02:00
void lcPiecesLibrary : : ReadArchiveDescriptions ( const QString & OfficialFileName , const QString & UnofficialFileName )
2014-05-08 00:58:59 +02:00
{
2015-07-22 06:00:47 +02:00
QFileInfo OfficialInfo ( OfficialFileName ) ;
2016-02-19 18:53:54 +01:00
QFileInfo UnofficialInfo ( UnofficialFileName ) ;
2016-12-22 01:49:45 +01:00
2015-07-22 06:00:47 +02:00
mArchiveCheckSum [ 0 ] = OfficialInfo . size ( ) ;
2016-09-22 17:04:51 +02:00
# if (QT_VERSION >= QT_VERSION_CHECK(4, 7, 0))
2015-07-22 06:00:47 +02:00
mArchiveCheckSum [ 1 ] = OfficialInfo . lastModified ( ) . toMSecsSinceEpoch ( ) ;
2016-09-22 17:04:51 +02:00
# else
mArchiveCheckSum [ 1 ] = OfficialInfo . lastModified ( ) . toTime_t ( ) ;
# endif
2016-12-22 01:49:45 +01:00
if ( ! UnofficialFileName . isEmpty ( ) )
{
mArchiveCheckSum [ 2 ] = UnofficialInfo . size ( ) ;
2016-09-22 17:04:51 +02:00
# if (QT_VERSION >= QT_VERSION_CHECK(4, 7, 0))
2016-12-22 01:49:45 +01:00
mArchiveCheckSum [ 3 ] = UnofficialInfo . lastModified ( ) . toMSecsSinceEpoch ( ) ;
2016-09-22 17:04:51 +02:00
# else
2016-12-22 01:49:45 +01:00
mArchiveCheckSum [ 3 ] = UnofficialInfo . lastModified ( ) . toTime_t ( ) ;
2016-09-22 17:04:51 +02:00
# endif
2016-12-22 01:49:45 +01:00
}
else
{
mArchiveCheckSum [ 2 ] = 0 ;
mArchiveCheckSum [ 3 ] = 0 ;
}
2015-07-22 06:00:47 +02:00
QString IndexFileName = QFileInfo ( QDir ( mCachePath ) , QLatin1String ( " index " ) ) . absoluteFilePath ( ) ;
2013-08-09 06:57:18 +02:00
2015-07-22 06:00:47 +02:00
if ( ! LoadCacheIndex ( IndexFileName ) )
2013-08-09 06:57:18 +02:00
{
lcMemFile PieceFile ;
2018-02-22 02:27:24 +01:00
for ( const auto & PieceIt : mPieces )
2013-08-09 06:57:18 +02:00
{
2017-07-24 04:35:18 +02:00
PieceInfo * Info = PieceIt . second ;
2013-08-09 06:57:18 +02:00
2014-05-08 00:58:59 +02:00
mZipFiles [ Info - > mZipFileType ] - > ExtractFile ( Info - > mZipFileIndex , PieceFile , 256 ) ;
2013-08-09 06:57:18 +02:00
PieceFile . Seek ( 0 , SEEK_END ) ;
PieceFile . WriteU8 ( 0 ) ;
char * Src = ( char * ) PieceFile . mBuffer + 2 ;
char * Dst = Info - > m_strDescription ;
for ( ; ; )
{
if ( * Src ! = ' \r ' & & * Src ! = ' \n ' & & * Src & & Dst - Info - > m_strDescription < ( int ) sizeof ( Info - > m_strDescription ) - 1 )
{
* Dst + + = * Src + + ;
continue ;
}
* Dst = 0 ;
break ;
}
}
2015-07-22 06:00:47 +02:00
2017-11-27 04:21:54 +01:00
SaveArchiveCacheIndex ( IndexFileName ) ;
2013-08-09 06:57:18 +02:00
}
}
2017-11-25 05:00:16 +01:00
bool lcPiecesLibrary : : OpenDirectory ( const QDir & LibraryDir , bool ShowProgress )
2013-08-09 06:57:18 +02:00
{
2017-11-27 04:21:54 +01:00
const QLatin1String BaseFolders [ LC_NUM_FOLDERTYPES ] = { QLatin1String ( " unofficial/ " ) , QLatin1String ( " " ) } ;
2020-03-23 04:18:52 +01:00
constexpr int NumBaseFolders = LC_ARRAY_COUNT ( BaseFolders ) ;
2017-03-19 21:12:24 +01:00
2017-11-27 04:21:54 +01:00
QFileInfoList FileLists [ NumBaseFolders ] ;
2013-08-09 06:57:18 +02:00
2017-11-27 04:21:54 +01:00
for ( unsigned int BaseFolderIdx = 0 ; BaseFolderIdx < NumBaseFolders ; BaseFolderIdx + + )
{
QString ParstPath = QDir ( LibraryDir . absoluteFilePath ( BaseFolders [ BaseFolderIdx ] ) ) . absoluteFilePath ( QLatin1String ( " parts/ " ) ) ;
QDir Dir = QDir ( ParstPath , QLatin1String ( " *.dat " ) , QDir : : SortFlags ( QDir : : Name | QDir : : IgnoreCase ) , QDir : : Files | QDir : : Hidden | QDir : : Readable ) ;
FileLists [ BaseFolderIdx ] = Dir . entryInfoList ( ) ;
2013-08-09 06:57:18 +02:00
}
2017-11-27 04:21:54 +01:00
if ( FileLists [ LC_FOLDER_OFFICIAL ] . isEmpty ( ) )
2013-08-09 06:57:18 +02:00
return false ;
2017-11-27 04:21:54 +01:00
mHasUnofficial = ! FileLists [ LC_FOLDER_UNOFFICIAL ] . isEmpty ( ) ;
ReadDirectoryDescriptions ( FileLists , ShowProgress ) ;
2019-12-09 20:43:54 +01:00
for ( unsigned int BaseFolderIdx = 0 ; BaseFolderIdx < LC_ARRAY_COUNT ( BaseFolders ) ; BaseFolderIdx + + )
2013-08-09 06:57:18 +02:00
{
2019-06-02 04:54:09 +02:00
const char * PrimitiveDirectories [ ] = { " p/ " , " parts/s/ " } ;
2017-03-19 21:12:24 +01:00
bool SubFileDirectories [ ] = { false , false , true } ;
2017-05-29 22:31:24 +02:00
QDir BaseDir ( LibraryDir . absoluteFilePath ( QLatin1String ( BaseFolders [ BaseFolderIdx ] ) ) ) ;
2013-08-09 06:57:18 +02:00
2019-12-09 20:43:54 +01:00
for ( int DirectoryIdx = 0 ; DirectoryIdx < ( int ) ( LC_ARRAY_COUNT ( PrimitiveDirectories ) ) ; DirectoryIdx + + )
2013-08-09 06:57:18 +02:00
{
2019-06-02 04:54:09 +02:00
QString ChildPath = BaseDir . absoluteFilePath ( QLatin1String ( PrimitiveDirectories [ DirectoryIdx ] ) ) ;
QDirIterator DirIterator ( ChildPath , QStringList ( ) < < QLatin1String ( " *.dat " ) , QDir : : Files | QDir : : Hidden | QDir : : Readable , QDirIterator : : Subdirectories ) ;
2013-08-09 06:57:18 +02:00
2019-06-02 04:54:09 +02:00
while ( DirIterator . hasNext ( ) )
2013-08-09 06:57:18 +02:00
{
2017-03-19 21:12:24 +01:00
char Name [ LC_PIECE_NAME_LEN ] ;
2019-06-02 04:54:09 +02:00
QString FileName = DirIterator . next ( ) ;
QByteArray FileString = BaseDir . relativeFilePath ( FileName ) . toLatin1 ( ) ;
const char * Src = strchr ( FileString , ' / ' ) + 1 ;
char * Dst = Name ;
2013-08-09 06:57:18 +02:00
2017-03-19 21:12:24 +01:00
while ( * Src & & Dst - Name < ( int ) sizeof ( Name ) )
{
if ( * Src > = ' a ' & & * Src < = ' z ' )
* Dst = * Src + ' A ' - ' a ' ;
else if ( * Src = = ' \\ ' )
* Dst = ' / ' ;
else
* Dst = * Src ;
2013-08-09 06:57:18 +02:00
2017-03-19 21:12:24 +01:00
Src + + ;
Dst + + ;
}
2017-07-28 01:09:38 +02:00
* Dst = 0 ;
2013-08-09 06:57:18 +02:00
2017-03-19 21:12:24 +01:00
if ( Dst - Name < = 4 )
continue ;
Dst - = 4 ;
if ( memcmp ( Dst , " .DAT " , 4 ) )
continue ;
2017-07-24 01:19:09 +02:00
if ( mHasUnofficial & & IsPrimitive ( Name ) )
2017-03-19 21:12:24 +01:00
continue ;
if ( BaseFolderIdx = = 0 )
mHasUnofficial = true ;
2020-03-23 04:18:52 +01:00
const bool SubFile = SubFileDirectories [ DirectoryIdx ] ;
2019-06-02 04:54:09 +02:00
mPrimitives [ Name ] = new lcLibraryPrimitive ( std : : move ( FileName ) , strchr ( FileString , ' / ' ) + 1 , LC_NUM_ZIPFILES , 0 , ! SubFile & & ( memcmp ( Name , " STU " , 3 ) = = 0 ) , SubFile ) ;
2017-03-19 21:12:24 +01:00
}
2013-08-09 06:57:18 +02:00
}
}
2019-12-09 20:43:54 +01:00
for ( unsigned int BaseFolderIdx = 0 ; BaseFolderIdx < LC_ARRAY_COUNT ( BaseFolders ) ; BaseFolderIdx + + )
2013-08-09 06:57:18 +02:00
{
2019-10-26 20:41:49 +02:00
QDir BaseDir ( LibraryDir . absoluteFilePath ( QLatin1String ( BaseFolders [ BaseFolderIdx ] ) ) ) ;
QDir Dir ( BaseDir . absoluteFilePath ( QLatin1String ( " parts/textures/ " ) ) , QLatin1String ( " *.png " ) , QDir : : SortFlags ( QDir : : Name | QDir : : IgnoreCase ) , QDir : : Files | QDir : : Hidden | QDir : : Readable ) ;
QStringList FileList = Dir . entryList ( ) ;
mTextures . reserve ( mTextures . size ( ) + FileList . size ( ) ) ;
2013-08-09 06:57:18 +02:00
2019-10-26 20:41:49 +02:00
for ( int FileIdx = 0 ; FileIdx < FileList . size ( ) ; FileIdx + + )
2013-08-09 06:57:18 +02:00
{
2019-10-26 20:41:49 +02:00
char Name [ LC_MAXPATH ] ;
QByteArray FileString = FileList [ FileIdx ] . toLatin1 ( ) ;
const char * Src = FileString ;
char * Dst = Name ;
2013-08-09 06:57:18 +02:00
2019-10-26 20:41:49 +02:00
while ( * Src & & Dst - Name < ( int ) sizeof ( Name ) )
{
if ( * Src > = ' a ' & & * Src < = ' z ' )
* Dst = * Src + ' A ' - ' a ' ;
else if ( * Src = = ' \\ ' )
* Dst = ' / ' ;
else
* Dst = * Src ;
2013-08-09 06:57:18 +02:00
2019-10-26 20:41:49 +02:00
Src + + ;
Dst + + ;
}
2013-08-09 06:57:18 +02:00
2019-10-26 20:41:49 +02:00
if ( Dst - Name < = 4 )
continue ;
Dst - = 4 ;
if ( memcmp ( Dst , " .PNG " , 4 ) )
continue ;
* Dst = 0 ;
2013-08-09 06:57:18 +02:00
2019-10-26 20:41:49 +02:00
lcTexture * Texture = new lcTexture ( ) ;
mTextures . push_back ( Texture ) ;
2013-08-09 06:57:18 +02:00
2019-10-26 20:41:49 +02:00
strncpy ( Texture - > mName , Name , sizeof ( Texture - > mName ) ) ;
Texture - > mName [ sizeof ( Texture - > mName ) - 1 ] = 0 ;
Texture - > mFileName = Dir . absoluteFilePath ( FileList [ FileIdx ] ) ;
}
2013-08-09 06:57:18 +02:00
}
return true ;
}
2017-11-27 04:21:54 +01:00
void lcPiecesLibrary : : ReadDirectoryDescriptions ( const QFileInfoList ( & FileLists ) [ LC_NUM_FOLDERTYPES ] , bool ShowProgress )
{
QString IndexFileName = QFileInfo ( QDir ( mCachePath ) , QLatin1String ( " index " ) ) . absoluteFilePath ( ) ;
lcMemFile IndexFile ;
std : : vector < const char * > CachedDescriptions ;
if ( ReadDirectoryCacheFile ( IndexFileName , IndexFile ) )
{
QString LibraryPath = IndexFile . ReadQString ( ) ;
if ( LibraryPath = = mLibraryDir . absolutePath ( ) )
{
int NumDescriptions = IndexFile . ReadU32 ( ) ;
CachedDescriptions . reserve ( NumDescriptions ) ;
while ( NumDescriptions - - )
{
const char * FileName = ( const char * ) IndexFile . mBuffer + IndexFile . GetPosition ( ) ;
CachedDescriptions . push_back ( FileName ) ;
IndexFile . Seek ( strlen ( FileName ) + 1 , SEEK_CUR ) ;
2019-09-21 18:47:33 +02:00
const char * Description = ( const char * ) IndexFile . mBuffer + IndexFile . GetPosition ( ) ;
2017-11-27 04:21:54 +01:00
IndexFile . Seek ( strlen ( Description ) + 1 , SEEK_CUR ) ;
IndexFile . Seek ( 4 + 1 + 8 , SEEK_CUR ) ;
}
}
}
for ( int FolderIdx = 0 ; FolderIdx < LC_NUM_FOLDERTYPES ; FolderIdx + + )
{
const QFileInfoList & FileList = FileLists [ FolderIdx ] ;
for ( int FileIdx = 0 ; FileIdx < FileList . size ( ) ; FileIdx + + )
{
char Name [ LC_PIECE_NAME_LEN ] ;
QByteArray FileString = FileList [ FileIdx ] . fileName ( ) . toLatin1 ( ) ;
const char * Src = FileString ;
char * Dst = Name ;
while ( * Src & & Dst - Name < ( int ) sizeof ( Name ) )
{
if ( * Src > = ' a ' & & * Src < = ' z ' )
* Dst = * Src + ' A ' - ' a ' ;
else if ( * Src = = ' \\ ' )
* Dst = ' / ' ;
else
* Dst = * Src ;
Src + + ;
Dst + + ;
}
* Dst = 0 ;
if ( FolderIdx = = LC_FOLDER_OFFICIAL & & mHasUnofficial & & mPieces . find ( Name ) ! = mPieces . end ( ) )
continue ;
PieceInfo * Info = new PieceInfo ( ) ;
strncpy ( Info - > mFileName , FileString , sizeof ( Info - > mFileName ) ) ;
Info - > mFileName [ sizeof ( Info - > mFileName ) - 1 ] = 0 ;
Info - > mFolderType = FolderIdx ;
Info - > mFolderIndex = FileIdx ;
mPieces [ Name ] = Info ;
}
}
QAtomicInt FilesLoaded ;
bool Modified = false ;
auto ReadDescriptions = [ & FileLists , & CachedDescriptions , & FilesLoaded , & Modified ] ( const std : : pair < std : : string , PieceInfo * > & Entry )
{
PieceInfo * Info = Entry . second ;
FilesLoaded . ref ( ) ;
lcDiskFile PieceFile ( FileLists [ Info - > mFolderType ] [ Info - > mFolderIndex ] . absoluteFilePath ( ) ) ;
char Line [ 1024 ] ;
if ( ! CachedDescriptions . empty ( ) )
{
auto DescriptionCompare = [ ] ( const void * Key , const void * Element )
{
return strcmp ( ( const char * ) Key , * ( const char * * ) Element ) ;
} ;
void * CachedDescription = bsearch ( Info - > mFileName , & CachedDescriptions . front ( ) , CachedDescriptions . size ( ) , sizeof ( char * ) , DescriptionCompare ) ;
if ( CachedDescription )
{
const char * FileName = * ( const char * * ) CachedDescription ;
const char * Description = FileName + strlen ( FileName ) + 1 ;
2020-03-23 04:18:52 +01:00
const uint64_t CachedFileTime = * ( uint64_t * ) ( Description + strlen ( Description ) + 1 + 4 + 1 ) ;
2018-02-01 21:48:31 +01:00
# if (QT_VERSION >= QT_VERSION_CHECK(4, 7, 0))
quint64 FileTime = FileLists [ Info - > mFolderType ] [ Info - > mFolderIndex ] . lastModified ( ) . toMSecsSinceEpoch ( ) ;
# else
quint64 FileTime = FileLists [ Info - > mFolderType ] [ Info - > mFolderIndex ] . lastModified ( ) . toTime_t ( ) ;
# endif
if ( FileTime = = CachedFileTime )
{
strcpy ( Info - > m_strDescription , Description ) ;
return ;
}
2017-11-27 04:21:54 +01:00
}
}
if ( ! PieceFile . Open ( QIODevice : : ReadOnly ) | | ! PieceFile . ReadLine ( Line , sizeof ( Line ) ) )
{
strcpy ( Info - > m_strDescription , " Unknown " ) ;
return ;
}
const char * Src = Line + 2 ;
char * Dst = Info - > m_strDescription ;
for ( ; ; )
{
if ( * Src ! = ' \r ' & & * Src ! = ' \n ' & & * Src & & Dst - Info - > m_strDescription < ( int ) sizeof ( Info - > m_strDescription ) - 1 )
{
* Dst + + = * Src + + ;
continue ;
}
* Dst = 0 ;
break ;
}
Modified = true ;
} ;
QProgressDialog * ProgressDialog = new QProgressDialog ( nullptr ) ;
ProgressDialog - > setWindowFlags ( ProgressDialog - > windowFlags ( ) & ~ Qt : : WindowCloseButtonHint ) ;
ProgressDialog - > setWindowTitle ( tr ( " Initializing " ) ) ;
ProgressDialog - > setLabelText ( tr ( " Loading Parts Library " ) ) ;
2019-05-18 20:33:27 +02:00
ProgressDialog - > setMaximum ( ( int ) mPieces . size ( ) ) ;
2017-11-27 04:21:54 +01:00
ProgressDialog - > setMinimum ( 0 ) ;
ProgressDialog - > setValue ( 0 ) ;
ProgressDialog - > setCancelButton ( nullptr ) ;
ProgressDialog - > setAutoReset ( false ) ;
if ( ShowProgress )
ProgressDialog - > show ( ) ;
QFuture < void > LoadFuture = QtConcurrent : : map ( mPieces , ReadDescriptions ) ;
while ( ! LoadFuture . isFinished ( ) )
{
2017-12-10 19:13:58 +01:00
# if (QT_VERSION >= QT_VERSION_CHECK(5, 3, 0) || QT_VERSION < QT_VERSION_CHECK(5, 0, 0) )
2017-11-27 04:21:54 +01:00
ProgressDialog - > setValue ( FilesLoaded ) ;
2017-12-10 19:13:58 +01:00
# else
ProgressDialog - > setValue ( FilesLoaded . load ( ) ) ;
# endif
2017-11-27 04:21:54 +01:00
QApplication : : processEvents ( QEventLoop : : ExcludeUserInputEvents ) ;
}
2017-12-10 19:13:58 +01:00
# if (QT_VERSION >= QT_VERSION_CHECK(5, 3, 0) || QT_VERSION < QT_VERSION_CHECK(5, 0, 0) )
2017-11-27 04:21:54 +01:00
ProgressDialog - > setValue ( FilesLoaded ) ;
2017-12-10 19:13:58 +01:00
# else
ProgressDialog - > setValue ( FilesLoaded . load ( ) ) ;
# endif
2017-11-27 04:21:54 +01:00
QApplication : : processEvents ( QEventLoop : : ExcludeUserInputEvents ) ;
ProgressDialog - > deleteLater ( ) ;
if ( Modified )
{
2019-05-18 20:33:27 +02:00
lcMemFile NewIndexFile ;
2017-11-27 04:21:54 +01:00
2019-05-18 20:33:27 +02:00
NewIndexFile . WriteQString ( mLibraryDir . absolutePath ( ) ) ;
2017-11-27 04:21:54 +01:00
2019-05-18 20:33:27 +02:00
NewIndexFile . WriteU32 ( ( quint32 ) mPieces . size ( ) ) ;
2017-11-27 04:21:54 +01:00
std : : vector < PieceInfo * > SortedPieces ;
SortedPieces . reserve ( mPieces . size ( ) ) ;
2018-02-22 02:27:24 +01:00
for ( const auto & PieceIt : mPieces )
2017-11-27 04:21:54 +01:00
SortedPieces . push_back ( PieceIt . second ) ;
auto PieceInfoCompare = [ ] ( PieceInfo * Info1 , PieceInfo * Info2 )
{
return strcmp ( Info1 - > mFileName , Info2 - > mFileName ) < 0 ;
} ;
std : : sort ( SortedPieces . begin ( ) , SortedPieces . end ( ) , PieceInfoCompare ) ;
for ( const PieceInfo * Info : SortedPieces )
{
2019-05-18 20:33:27 +02:00
if ( NewIndexFile . WriteBuffer ( Info - > mFileName , strlen ( Info - > mFileName ) + 1 ) = = 0 )
2017-11-27 04:21:54 +01:00
return ;
2019-05-18 20:33:27 +02:00
if ( NewIndexFile . WriteBuffer ( Info - > m_strDescription , strlen ( Info - > m_strDescription ) + 1 ) = = 0 )
2017-11-27 04:21:54 +01:00
return ;
2019-05-18 20:33:27 +02:00
NewIndexFile . WriteU8 ( Info - > mFolderType ) ;
2017-11-27 04:21:54 +01:00
# if (QT_VERSION >= QT_VERSION_CHECK(4, 7, 0))
2017-12-02 21:22:04 +01:00
quint64 FileTime = FileLists [ Info - > mFolderType ] [ Info - > mFolderIndex ] . lastModified ( ) . toMSecsSinceEpoch ( ) ;
2017-11-27 04:21:54 +01:00
# else
2017-12-02 21:22:04 +01:00
quint64 FileTime = FileLists [ Info - > mFolderType ] [ Info - > mFolderIndex ] . lastModified ( ) . toTime_t ( ) ;
2017-11-27 04:21:54 +01:00
# endif
2018-02-01 21:48:31 +01:00
2019-05-18 20:33:27 +02:00
NewIndexFile . WriteU64 ( FileTime ) ;
2017-11-27 04:21:54 +01:00
}
2019-05-18 20:33:27 +02:00
WriteDirectoryCacheFile ( IndexFileName , NewIndexFile ) ;
2017-11-27 04:21:54 +01:00
}
}
bool lcPiecesLibrary : : ReadArchiveCacheFile ( const QString & FileName , lcMemFile & CacheFile )
2013-08-09 06:57:18 +02:00
{
2015-07-22 06:00:47 +02:00
QFile File ( FileName ) ;
2013-08-09 06:57:18 +02:00
2015-07-22 06:00:47 +02:00
if ( ! File . open ( QIODevice : : ReadOnly ) )
2013-08-09 06:57:18 +02:00
return false ;
2015-07-22 06:00:47 +02:00
quint32 CacheVersion , CacheFlags ;
2019-09-21 18:47:33 +02:00
2015-07-22 06:00:47 +02:00
if ( File . read ( ( char * ) & CacheVersion , sizeof ( CacheVersion ) ) = = - 1 | | CacheVersion ! = LC_LIBRARY_CACHE_VERSION )
2013-08-09 06:57:18 +02:00
return false ;
2015-07-22 06:00:47 +02:00
if ( File . read ( ( char * ) & CacheFlags , sizeof ( CacheFlags ) ) = = - 1 | | CacheFlags ! = LC_LIBRARY_CACHE_ARCHIVE )
return false ;
2013-08-09 06:57:18 +02:00
2015-07-22 06:00:47 +02:00
qint64 CacheCheckSum [ 4 ] ;
2013-08-09 06:57:18 +02:00
2015-07-22 06:00:47 +02:00
if ( File . read ( ( char * ) & CacheCheckSum , sizeof ( CacheCheckSum ) ) = = - 1 | | memcmp ( CacheCheckSum , mArchiveCheckSum , sizeof ( CacheCheckSum ) ) )
2013-08-09 06:57:18 +02:00
return false ;
2015-07-22 06:00:47 +02:00
quint32 UncompressedSize ;
2013-08-09 06:57:18 +02:00
2015-07-22 06:00:47 +02:00
if ( File . read ( ( char * ) & UncompressedSize , sizeof ( UncompressedSize ) ) = = - 1 )
return false ;
2013-08-09 06:57:18 +02:00
2015-07-22 06:00:47 +02:00
QByteArray CompressedData = File . readAll ( ) ;
2013-08-09 06:57:18 +02:00
2015-07-22 06:00:47 +02:00
CacheFile . SetLength ( UncompressedSize ) ;
CacheFile . Seek ( 0 , SEEK_SET ) ;
2013-08-09 06:57:18 +02:00
2020-03-23 04:18:52 +01:00
constexpr int CHUNK = 16384 ;
2015-07-22 06:00:47 +02:00
int ret ;
unsigned have ;
z_stream strm ;
unsigned char in [ CHUNK ] ;
unsigned char out [ CHUNK ] ;
int pos ;
2013-08-09 06:57:18 +02:00
2015-07-22 06:00:47 +02:00
strm . zalloc = Z_NULL ;
strm . zfree = Z_NULL ;
strm . opaque = Z_NULL ;
strm . avail_in = 0 ;
strm . next_in = Z_NULL ;
pos = 0 ;
2013-08-09 06:57:18 +02:00
2015-07-22 06:00:47 +02:00
ret = inflateInit2 ( & strm , - MAX_WBITS ) ;
if ( ret ! = Z_OK )
return ret ;
2013-08-09 06:57:18 +02:00
2015-07-22 06:00:47 +02:00
do
2013-08-09 06:57:18 +02:00
{
2015-07-22 06:00:47 +02:00
strm . avail_in = lcMin ( CompressedData . size ( ) - pos , CHUNK ) ;
strm . next_in = in ;
2013-08-09 06:57:18 +02:00
2015-07-22 06:00:47 +02:00
if ( strm . avail_in = = 0 )
break ;
2013-08-09 06:57:18 +02:00
2015-07-22 06:00:47 +02:00
memcpy ( in , CompressedData . constData ( ) + pos , strm . avail_in ) ;
pos + = strm . avail_in ;
2013-08-09 06:57:18 +02:00
2015-07-22 06:00:47 +02:00
do
{
strm . avail_out = CHUNK ;
strm . next_out = out ;
ret = inflate ( & strm , Z_NO_FLUSH ) ;
2013-08-09 06:57:18 +02:00
2015-07-22 06:00:47 +02:00
switch ( ret )
{
case Z_NEED_DICT :
ret = Z_DATA_ERROR ;
2018-09-24 20:29:05 +02:00
Q_FALLTHROUGH ( ) ;
2015-07-22 06:00:47 +02:00
case Z_DATA_ERROR :
2018-09-24 20:29:05 +02:00
Q_FALLTHROUGH ( ) ;
2015-07-22 06:00:47 +02:00
case Z_MEM_ERROR :
( void ) inflateEnd ( & strm ) ;
return ret ;
}
2013-08-09 06:57:18 +02:00
2015-07-22 06:00:47 +02:00
have = CHUNK - strm . avail_out ;
CacheFile . WriteBuffer ( out , have ) ;
} while ( strm . avail_out = = 0 ) ;
} while ( ret ! = Z_STREAM_END ) ;
( void ) inflateEnd ( & strm ) ;
2019-09-21 18:47:33 +02:00
2015-07-22 06:00:47 +02:00
CacheFile . Seek ( 0 , SEEK_SET ) ;
return ret = = Z_STREAM_END ;
2013-08-09 06:57:18 +02:00
}
2017-11-27 04:21:54 +01:00
bool lcPiecesLibrary : : WriteArchiveCacheFile ( const QString & FileName , lcMemFile & CacheFile )
2013-08-09 06:57:18 +02:00
{
2015-07-22 06:00:47 +02:00
QFile File ( FileName ) ;
if ( ! File . open ( QIODevice : : WriteOnly ) )
2013-08-09 06:57:18 +02:00
return false ;
2020-03-23 04:18:52 +01:00
constexpr quint32 CacheVersion = LC_LIBRARY_CACHE_VERSION ;
constexpr quint32 CacheFlags = LC_LIBRARY_CACHE_ARCHIVE ;
2019-09-21 18:47:33 +02:00
2015-07-22 06:00:47 +02:00
if ( File . write ( ( char * ) & CacheVersion , sizeof ( CacheVersion ) ) = = - 1 )
return false ;
2013-08-09 06:57:18 +02:00
2015-07-22 06:00:47 +02:00
if ( File . write ( ( char * ) & CacheFlags , sizeof ( CacheFlags ) ) = = - 1 )
return false ;
2013-08-09 06:57:18 +02:00
2015-07-22 06:00:47 +02:00
if ( File . write ( ( char * ) & mArchiveCheckSum , sizeof ( mArchiveCheckSum ) ) = = - 1 )
return false ;
2013-08-09 06:57:18 +02:00
2020-03-23 04:18:52 +01:00
const quint32 UncompressedSize = ( quint32 ) CacheFile . GetLength ( ) ;
2013-08-09 06:57:18 +02:00
2015-07-22 06:00:47 +02:00
if ( File . write ( ( char * ) & UncompressedSize , sizeof ( UncompressedSize ) ) = = - 1 )
return false ;
2013-08-09 06:57:18 +02:00
2020-03-23 04:18:52 +01:00
constexpr size_t BufferSize = 16384 ;
2015-07-22 06:00:47 +02:00
char WriteBuffer [ BufferSize ] ;
z_stream Stream ;
2017-12-02 21:22:04 +01:00
quint32 Crc32 = 0 ;
2013-08-09 06:57:18 +02:00
2015-07-22 06:00:47 +02:00
CacheFile . Seek ( 0 , SEEK_SET ) ;
2013-08-09 06:57:18 +02:00
2019-06-03 22:23:04 +02:00
Stream . zalloc = nullptr ;
Stream . zfree = nullptr ;
Stream . opaque = nullptr ;
2013-08-09 06:57:18 +02:00
2015-07-22 06:00:47 +02:00
if ( deflateInit2 ( & Stream , Z_DEFAULT_COMPRESSION , Z_DEFLATED , - MAX_WBITS , DEF_MEM_LEVEL , Z_DEFAULT_STRATEGY ) ! = Z_OK )
return false ;
2013-08-09 06:57:18 +02:00
2015-07-22 06:00:47 +02:00
Bytef * BufferIn = CacheFile . mBuffer ;
int FlushMode ;
2013-08-09 06:57:18 +02:00
2015-07-22 06:00:47 +02:00
do
{
2016-02-19 18:53:54 +01:00
uInt Read = ( uInt ) lcMin ( CacheFile . GetLength ( ) - ( BufferIn - CacheFile . mBuffer ) , BufferSize ) ;
2015-07-22 06:00:47 +02:00
Stream . avail_in = Read ;
Stream . next_in = BufferIn ;
Crc32 = crc32 ( Crc32 , BufferIn , Read ) ;
BufferIn + = Read ;
FlushMode = ( BufferIn > = CacheFile . mBuffer + CacheFile . GetLength ( ) ) ? Z_FINISH : Z_NO_FLUSH ;
do
{
Stream . avail_out = BufferSize ;
Stream . next_out = ( Bytef * ) WriteBuffer ;
deflate ( & Stream , FlushMode ) ;
File . write ( WriteBuffer , BufferSize - Stream . avail_out ) ;
} while ( Stream . avail_out = = 0 ) ;
} while ( FlushMode ! = Z_FINISH ) ;
2019-09-21 18:47:33 +02:00
deflateEnd ( & Stream ) ;
2015-07-22 06:00:47 +02:00
return true ;
2013-08-09 06:57:18 +02:00
}
2017-11-27 04:21:54 +01:00
bool lcPiecesLibrary : : ReadDirectoryCacheFile ( const QString & FileName , lcMemFile & CacheFile )
{
QFile File ( FileName ) ;
if ( ! File . open ( QIODevice : : ReadOnly ) )
return false ;
quint32 CacheVersion , CacheFlags ;
if ( File . read ( ( char * ) & CacheVersion , sizeof ( CacheVersion ) ) = = - 1 | | CacheVersion ! = LC_LIBRARY_CACHE_VERSION )
return false ;
if ( File . read ( ( char * ) & CacheFlags , sizeof ( CacheFlags ) ) = = - 1 | | CacheFlags ! = LC_LIBRARY_CACHE_DIRECTORY )
return false ;
quint32 UncompressedSize ;
if ( File . read ( ( char * ) & UncompressedSize , sizeof ( UncompressedSize ) ) = = - 1 )
return false ;
QByteArray Data = qUncompress ( File . readAll ( ) ) ;
if ( Data . isEmpty ( ) )
return false ;
CacheFile . SetLength ( Data . size ( ) ) ;
CacheFile . Seek ( 0 , SEEK_SET ) ;
CacheFile . WriteBuffer ( Data . constData ( ) , Data . size ( ) ) ;
CacheFile . Seek ( 0 , SEEK_SET ) ;
return true ;
}
bool lcPiecesLibrary : : WriteDirectoryCacheFile ( const QString & FileName , lcMemFile & CacheFile )
{
QFile File ( FileName ) ;
if ( ! File . open ( QIODevice : : WriteOnly ) )
return false ;
2020-03-23 04:18:52 +01:00
constexpr quint32 CacheVersion = LC_LIBRARY_CACHE_VERSION ;
2017-11-27 04:21:54 +01:00
if ( File . write ( ( char * ) & CacheVersion , sizeof ( CacheVersion ) ) = = - 1 )
return false ;
2020-03-23 04:18:52 +01:00
constexpr quint32 CacheFlags = LC_LIBRARY_CACHE_DIRECTORY ;
2017-11-27 04:21:54 +01:00
if ( File . write ( ( char * ) & CacheFlags , sizeof ( CacheFlags ) ) = = - 1 )
return false ;
2020-03-23 04:18:52 +01:00
const quint32 UncompressedSize = ( quint32 ) CacheFile . GetLength ( ) ;
2017-11-27 04:21:54 +01:00
if ( File . write ( ( char * ) & UncompressedSize , sizeof ( UncompressedSize ) ) = = - 1 )
return false ;
2019-05-18 20:33:27 +02:00
File . write ( qCompress ( CacheFile . mBuffer , ( int ) CacheFile . GetLength ( ) ) ) ;
2017-11-27 04:21:54 +01:00
return true ;
}
2015-07-22 06:00:47 +02:00
bool lcPiecesLibrary : : LoadCacheIndex ( const QString & FileName )
2013-08-09 06:57:18 +02:00
{
2015-07-22 06:00:47 +02:00
lcMemFile IndexFile ;
2013-08-09 06:57:18 +02:00
2017-11-27 04:21:54 +01:00
if ( ! ReadArchiveCacheFile ( FileName , IndexFile ) )
2015-07-22 06:00:47 +02:00
return false ;
2017-07-27 21:40:52 +02:00
quint32 NumFiles ;
2015-07-22 06:00:47 +02:00
2017-07-24 04:35:18 +02:00
if ( IndexFile . ReadBuffer ( ( char * ) & NumFiles , sizeof ( NumFiles ) ) = = 0 | | NumFiles ! = mPieces . size ( ) )
2015-07-22 06:00:47 +02:00
return false ;
2013-08-09 06:57:18 +02:00
2018-02-22 02:27:24 +01:00
for ( const auto & PieceIt : mPieces )
2013-08-09 06:57:18 +02:00
{
2017-07-24 04:35:18 +02:00
PieceInfo * Info = PieceIt . second ;
2015-07-22 06:00:47 +02:00
quint8 Length ;
2013-08-09 06:57:18 +02:00
2015-07-22 06:00:47 +02:00
if ( IndexFile . ReadBuffer ( ( char * ) & Length , sizeof ( Length ) ) = = 0 | | Length > = sizeof ( Info - > m_strDescription ) )
return false ;
2013-08-09 06:57:18 +02:00
2019-07-21 18:26:36 +02:00
if ( IndexFile . ReadBuffer ( ( char * ) Info - > m_strDescription , Length ) = = 0 )
2015-07-22 06:00:47 +02:00
return false ;
2013-08-09 06:57:18 +02:00
2015-07-22 06:00:47 +02:00
Info - > m_strDescription [ Length ] = 0 ;
}
2014-05-08 00:58:59 +02:00
2015-07-22 06:00:47 +02:00
return true ;
}
2014-05-08 00:58:59 +02:00
2017-11-27 04:21:54 +01:00
bool lcPiecesLibrary : : SaveArchiveCacheIndex ( const QString & FileName )
2015-07-22 06:00:47 +02:00
{
lcMemFile IndexFile ;
2013-08-09 06:57:18 +02:00
2020-03-23 04:18:52 +01:00
const quint32 NumFiles = ( quint32 ) mPieces . size ( ) ;
2013-08-09 06:57:18 +02:00
2015-07-22 06:00:47 +02:00
if ( IndexFile . WriteBuffer ( ( char * ) & NumFiles , sizeof ( NumFiles ) ) = = 0 )
return false ;
2018-02-22 02:27:24 +01:00
for ( const auto & PieceIt : mPieces )
2013-08-09 06:57:18 +02:00
{
2020-03-23 04:18:52 +01:00
const PieceInfo * Info = PieceIt . second ;
const quint8 Length = ( quint8 ) strlen ( Info - > m_strDescription ) ;
2013-08-09 06:57:18 +02:00
2015-07-22 06:00:47 +02:00
if ( IndexFile . WriteBuffer ( ( char * ) & Length , sizeof ( Length ) ) = = 0 )
return false ;
2019-07-21 18:26:36 +02:00
if ( IndexFile . WriteBuffer ( ( char * ) Info - > m_strDescription , Length ) = = 0 )
2015-07-22 06:00:47 +02:00
return false ;
2013-08-09 06:57:18 +02:00
}
2017-11-27 04:21:54 +01:00
return WriteArchiveCacheFile ( FileName , IndexFile ) ;
2015-07-22 06:00:47 +02:00
}
2013-08-09 06:57:18 +02:00
2015-07-22 06:00:47 +02:00
bool lcPiecesLibrary : : LoadCachePiece ( PieceInfo * Info )
{
2017-07-27 18:21:55 +02:00
QString FileName = QFileInfo ( QDir ( mCachePath ) , QString : : fromLatin1 ( Info - > mFileName ) ) . absoluteFilePath ( ) ;
2015-07-22 06:00:47 +02:00
lcMemFile MeshData ;
2013-08-09 06:57:18 +02:00
2017-11-27 04:21:54 +01:00
if ( ! ReadArchiveCacheFile ( FileName , MeshData ) )
2015-07-22 06:00:47 +02:00
return false ;
2013-08-09 06:57:18 +02:00
2019-09-27 23:52:50 +02:00
qint32 Flags ;
2019-09-24 02:06:16 +02:00
if ( MeshData . ReadBuffer ( ( char * ) & Flags , sizeof ( Flags ) ) = = 0 )
return false ;
if ( Flags ! = mStudLogo )
return false ;
2017-01-23 04:28:05 +01:00
lcMesh * Mesh = new lcMesh ;
if ( Mesh - > FileLoad ( MeshData ) )
{
Info - > SetMesh ( Mesh ) ;
return true ;
}
else
{
delete Mesh ;
return false ;
}
2015-07-22 06:00:47 +02:00
}
2013-08-09 06:57:18 +02:00
2015-07-22 06:00:47 +02:00
bool lcPiecesLibrary : : SaveCachePiece ( PieceInfo * Info )
{
lcMemFile MeshData ;
2013-08-09 06:57:18 +02:00
2020-03-23 04:18:52 +01:00
const qint32 Flags = mStudLogo ;
2019-09-24 02:06:16 +02:00
if ( MeshData . WriteBuffer ( ( char * ) & Flags , sizeof ( Flags ) ) = = 0 )
return false ;
2015-07-22 06:00:47 +02:00
if ( ! Info - > GetMesh ( ) - > FileSave ( MeshData ) )
return false ;
2013-08-09 06:57:18 +02:00
2017-07-27 18:21:55 +02:00
QString FileName = QFileInfo ( QDir ( mCachePath ) , QString : : fromLatin1 ( Info - > mFileName ) ) . absoluteFilePath ( ) ;
2013-08-09 06:57:18 +02:00
2017-11-27 04:21:54 +01:00
return WriteArchiveCacheFile ( FileName , MeshData ) ;
2013-08-09 06:57:18 +02:00
}
2017-01-23 07:44:54 +01:00
class lcSleeper : public QThread
{
public :
static void msleep ( unsigned long Msecs )
{
QThread : : msleep ( Msecs ) ;
}
} ;
2017-01-23 04:28:05 +01:00
void lcPiecesLibrary : : LoadPieceInfo ( PieceInfo * Info , bool Wait , bool Priority )
{
QMutexLocker LoadLock ( & mLoadMutex ) ;
if ( Wait )
{
if ( Info - > AddRef ( ) = = 1 )
Info - > Load ( ) ;
else
{
if ( Info - > mState = = LC_PIECEINFO_UNLOADED )
{
Info - > Load ( ) ;
emit PartLoaded ( Info ) ;
}
else
{
LoadLock . unlock ( ) ;
while ( Info - > mState ! = LC_PIECEINFO_LOADED )
2017-01-23 07:44:54 +01:00
lcSleeper : : msleep ( 10 ) ;
2017-01-23 04:28:05 +01:00
}
}
}
else
{
2019-09-24 01:55:24 +02:00
if ( Info - > AddRef ( ) = = 1 )
2017-01-23 04:28:05 +01:00
{
if ( Priority )
mLoadQueue . prepend ( Info ) ;
else
mLoadQueue . append ( Info ) ;
2017-12-22 13:38:47 +01:00
mLoadFutures . append ( QtConcurrent : : run ( [ this ] ( ) { LoadQueuedPiece ( ) ; } ) ) ;
2017-01-23 04:28:05 +01:00
}
}
}
void lcPiecesLibrary : : ReleasePieceInfo ( PieceInfo * Info )
{
QMutexLocker LoadLock ( & mLoadMutex ) ;
if ( Info - > GetRefCount ( ) = = 0 | | Info - > Release ( ) = = 0 )
Info - > Unload ( ) ;
}
void lcPiecesLibrary : : LoadQueuedPiece ( )
{
mLoadMutex . lock ( ) ;
2017-04-14 02:26:40 +02:00
PieceInfo * Info = nullptr ;
2017-01-23 04:28:05 +01:00
while ( ! mLoadQueue . isEmpty ( ) )
{
Info = mLoadQueue . takeFirst ( ) ;
if ( Info - > mState = = LC_PIECEINFO_UNLOADED & & Info - > GetRefCount ( ) > 0 )
{
Info - > mState = LC_PIECEINFO_LOADING ;
break ;
}
2017-01-24 05:26:57 +01:00
2017-04-14 02:26:40 +02:00
Info = nullptr ;
2017-01-23 04:28:05 +01:00
}
mLoadMutex . unlock ( ) ;
if ( Info )
Info - > Load ( ) ;
emit PartLoaded ( Info ) ;
}
void lcPiecesLibrary : : WaitForLoadQueue ( )
{
for ( QFuture < void > & Future : mLoadFutures )
Future . waitForFinished ( ) ;
mLoadFutures . clear ( ) ;
}
bool lcPiecesLibrary : : LoadPieceData ( PieceInfo * Info )
2013-08-09 06:57:18 +02:00
{
lcLibraryMeshData MeshData ;
2019-07-28 01:31:16 +02:00
lcMeshLoader MeshLoader ( MeshData , true , nullptr , false ) ;
2013-08-09 06:57:18 +02:00
2016-05-28 19:35:13 +02:00
bool Loaded = false ;
bool SaveCache = false ;
2014-05-18 01:03:05 +02:00
if ( Info - > mZipFileType ! = LC_NUM_ZIPFILES & & mZipFiles [ Info - > mZipFileType ] )
2013-08-09 06:57:18 +02:00
{
if ( LoadCachePiece ( Info ) )
return true ;
lcMemFile PieceFile ;
2016-05-28 19:35:13 +02:00
if ( mZipFiles [ Info - > mZipFileType ] - > ExtractFile ( Info - > mZipFileIndex , PieceFile ) )
2019-07-28 01:31:16 +02:00
Loaded = MeshLoader . LoadMesh ( PieceFile , LC_MESHDATA_SHARED ) ;
2013-08-09 06:57:18 +02:00
2016-05-28 19:35:13 +02:00
SaveCache = Loaded & & ( Info - > mZipFileType = = LC_ZIPFILE_OFFICIAL ) ;
2013-08-09 06:57:18 +02:00
}
else
{
char FileName [ LC_MAXPATH ] ;
lcDiskFile PieceFile ;
2017-03-19 21:12:24 +01:00
if ( mHasUnofficial )
{
2017-07-27 18:21:55 +02:00
sprintf ( FileName , " unofficial/parts/%s " , Info - > mFileName ) ;
2017-05-29 22:31:24 +02:00
PieceFile . SetFileName ( mLibraryDir . absoluteFilePath ( QLatin1String ( FileName ) ) ) ;
if ( PieceFile . Open ( QIODevice : : ReadOnly ) )
2019-07-28 01:31:16 +02:00
Loaded = MeshLoader . LoadMesh ( PieceFile , LC_MESHDATA_SHARED ) ;
2017-03-19 21:12:24 +01:00
}
if ( ! Loaded )
{
2017-07-27 18:21:55 +02:00
sprintf ( FileName , " parts/%s " , Info - > mFileName ) ;
2017-05-29 22:31:24 +02:00
PieceFile . SetFileName ( mLibraryDir . absoluteFilePath ( QLatin1String ( FileName ) ) ) ;
if ( PieceFile . Open ( QIODevice : : ReadOnly ) )
2019-07-28 01:31:16 +02:00
Loaded = MeshLoader . LoadMesh ( PieceFile , LC_MESHDATA_SHARED ) ;
2017-03-19 21:12:24 +01:00
}
2016-05-28 19:35:13 +02:00
}
2019-09-21 18:47:33 +02:00
2017-11-25 21:57:41 +01:00
if ( ! Loaded | | mCancelLoading )
2016-05-28 19:35:13 +02:00
return false ;
2019-09-24 01:22:11 +02:00
if ( Info )
2019-07-28 01:31:16 +02:00
Info - > SetMesh ( MeshData . CreateMesh ( ) ) ;
2015-02-23 01:50:37 +01:00
2016-05-28 19:35:13 +02:00
if ( SaveCache )
2015-07-22 06:00:47 +02:00
SaveCachePiece ( Info ) ;
2015-02-23 01:50:37 +01:00
return true ;
}
2019-07-28 01:31:16 +02:00
void lcPiecesLibrary : : GetPrimitiveFile ( lcLibraryPrimitive * Primitive , std : : function < void ( lcFile & File ) > Callback )
2015-02-23 01:50:37 +01:00
{
2019-07-28 01:31:16 +02:00
if ( mZipFiles [ LC_ZIPFILE_OFFICIAL ] )
2013-08-09 06:57:18 +02:00
{
2019-07-28 01:31:16 +02:00
lcMemFile IncludeFile ;
2019-05-28 01:07:20 +02:00
2019-07-28 01:31:16 +02:00
if ( mZipFiles [ Primitive - > mZipFileType ] - > ExtractFile ( Primitive - > mZipFileIndex , IncludeFile ) )
Callback ( IncludeFile ) ;
2019-05-28 01:07:20 +02:00
}
else
{
2019-07-28 01:31:16 +02:00
lcDiskFile IncludeFile ( Primitive - > mFileName ) ;
2019-05-28 01:07:20 +02:00
2019-07-28 01:31:16 +02:00
if ( IncludeFile . Open ( QIODevice : : ReadOnly ) )
Callback ( IncludeFile ) ;
2013-08-09 06:57:18 +02:00
}
2019-07-28 01:31:16 +02:00
}
2013-08-09 06:57:18 +02:00
2019-07-28 01:31:16 +02:00
void lcPiecesLibrary : : GetPieceFile ( const char * PieceName , std : : function < void ( lcFile & File ) > Callback )
{
const auto PieceIt = mPieces . find ( PieceName ) ;
2015-05-24 06:36:25 +02:00
2019-07-28 01:31:16 +02:00
if ( PieceIt ! = mPieces . end ( ) )
2015-05-24 06:36:25 +02:00
{
2019-07-28 01:31:16 +02:00
PieceInfo * Info = PieceIt - > second ;
2015-05-24 06:36:25 +02:00
2019-07-28 01:31:16 +02:00
if ( mZipFiles [ LC_ZIPFILE_OFFICIAL ] & & Info - > mZipFileType ! = LC_NUM_ZIPFILES )
2015-05-24 06:36:25 +02:00
{
2019-07-28 01:31:16 +02:00
lcMemFile IncludeFile ;
2015-05-24 06:36:25 +02:00
2019-07-28 01:31:16 +02:00
if ( mZipFiles [ Info - > mZipFileType ] - > ExtractFile ( Info - > mZipFileIndex , IncludeFile ) )
Callback ( IncludeFile ) ;
2015-05-24 06:36:25 +02:00
}
2019-07-28 01:31:16 +02:00
else
2015-05-24 06:36:25 +02:00
{
2019-07-28 01:31:16 +02:00
lcDiskFile IncludeFile ;
char FileName [ LC_MAXPATH ] ;
2015-05-24 06:36:25 +02:00
bool Found = false ;
2019-07-28 01:31:16 +02:00
if ( mHasUnofficial )
2015-05-24 06:36:25 +02:00
{
2019-07-28 01:31:16 +02:00
sprintf ( FileName , " unofficial/parts/%s " , Info - > mFileName ) ;
IncludeFile . SetFileName ( mLibraryDir . absoluteFilePath ( QLatin1String ( FileName ) ) ) ;
Found = IncludeFile . Open ( QIODevice : : ReadOnly ) ;
2015-05-24 06:36:25 +02:00
}
if ( ! Found )
{
2019-07-28 01:31:16 +02:00
sprintf ( FileName , " parts/%s " , Info - > mFileName ) ;
IncludeFile . SetFileName ( mLibraryDir . absoluteFilePath ( QLatin1String ( FileName ) ) ) ;
Found = IncludeFile . Open ( QIODevice : : ReadOnly ) ;
2019-05-28 01:07:20 +02:00
}
2013-08-09 06:57:18 +02:00
2019-07-28 01:31:16 +02:00
if ( Found )
Callback ( IncludeFile ) ;
2019-05-28 01:07:20 +02:00
}
}
else
{
2019-07-28 01:31:16 +02:00
bool Found = false ;
2019-05-28 01:07:20 +02:00
2019-07-28 01:31:16 +02:00
if ( mZipFiles [ LC_ZIPFILE_OFFICIAL ] )
2018-01-21 02:35:21 +01:00
{
2019-07-28 01:31:16 +02:00
lcMemFile IncludeFile ;
2013-08-09 06:57:18 +02:00
2020-01-17 23:44:35 +01:00
auto LoadIncludeFile = [ & IncludeFile , PieceName , this ] ( const char * Folder , int ZipFileIndex )
2019-07-28 01:31:16 +02:00
{
char IncludeFileName [ LC_MAXPATH ] ;
2020-01-17 23:44:35 +01:00
sprintf ( IncludeFileName , Folder , PieceName ) ;
2019-07-28 01:31:16 +02:00
return mZipFiles [ ZipFileIndex ] - > ExtractFile ( IncludeFileName , IncludeFile ) ;
} ;
2019-02-23 02:02:38 +01:00
2019-07-28 01:31:16 +02:00
if ( mHasUnofficial )
{
Found = LoadIncludeFile ( " parts/%s " , LC_ZIPFILE_UNOFFICIAL ) ;
2019-05-28 01:07:20 +02:00
2019-07-28 01:31:16 +02:00
if ( ! Found )
Found = LoadIncludeFile ( " p/%s " , LC_ZIPFILE_UNOFFICIAL ) ;
2018-01-21 02:35:21 +01:00
}
2013-08-09 06:57:18 +02:00
2019-07-28 01:31:16 +02:00
if ( ! Found )
2019-05-28 01:07:20 +02:00
{
2019-07-28 01:31:16 +02:00
Found = LoadIncludeFile ( " ldraw/parts/%s " , LC_ZIPFILE_OFFICIAL ) ;
2019-05-28 01:07:20 +02:00
2019-07-28 01:31:16 +02:00
if ( ! Found )
Found = LoadIncludeFile ( " ldraw/p/%s " , LC_ZIPFILE_OFFICIAL ) ;
2019-05-28 01:07:20 +02:00
}
2013-08-09 06:57:18 +02:00
2019-07-28 01:31:16 +02:00
if ( Found )
Callback ( IncludeFile ) ;
}
else
2015-05-24 06:36:25 +02:00
{
2019-07-28 01:31:16 +02:00
lcDiskFile IncludeFile ;
2015-05-24 06:36:25 +02:00
2020-01-17 23:44:35 +01:00
auto LoadIncludeFile = [ & IncludeFile , PieceName , this ] ( const char * Folder )
2015-05-24 06:36:25 +02:00
{
2019-07-28 01:31:16 +02:00
char IncludeFileName [ LC_MAXPATH ] ;
2020-01-17 23:44:35 +01:00
sprintf ( IncludeFileName , Folder , PieceName ) ;
2019-07-28 01:31:16 +02:00
IncludeFile . SetFileName ( mLibraryDir . absoluteFilePath ( QLatin1String ( IncludeFileName ) ) ) ;
if ( IncludeFile . Open ( QIODevice : : ReadOnly ) )
return true ;
2015-05-24 06:36:25 +02:00
2019-07-28 01:31:16 +02:00
# if defined(Q_OS_MACOS) || defined(Q_OS_LINUX)
// todo: instead of using strlwr, search the parts/primitive lists and get the file name from there
strlwr ( IncludeFileName ) ;
IncludeFile . SetFileName ( mLibraryDir . absoluteFilePath ( QLatin1String ( IncludeFileName ) ) ) ;
return IncludeFile . Open ( QIODevice : : ReadOnly ) ;
# else
return false ;
# endif
} ;
2015-05-24 06:36:25 +02:00
2019-07-28 01:31:16 +02:00
if ( mHasUnofficial )
2015-05-24 06:36:25 +02:00
{
2019-07-28 01:31:16 +02:00
Found = LoadIncludeFile ( " unofficial/parts/%s " ) ;
2015-05-24 06:36:25 +02:00
2019-07-28 01:31:16 +02:00
if ( ! Found )
Found = LoadIncludeFile ( " unofficial/p/%s " ) ;
2015-05-24 06:36:25 +02:00
}
2019-07-28 01:31:16 +02:00
if ( ! Found )
2015-05-24 06:36:25 +02:00
{
2019-07-28 01:31:16 +02:00
Found = LoadIncludeFile ( " parts/%s " ) ;
2013-08-09 06:57:18 +02:00
2019-07-28 01:31:16 +02:00
if ( ! Found )
Found = LoadIncludeFile ( " p/%s " ) ;
2013-08-09 06:57:18 +02:00
}
2019-07-28 01:31:16 +02:00
if ( Found )
Callback ( IncludeFile ) ;
}
2013-08-09 06:57:18 +02:00
}
}
2016-12-28 22:30:31 +01:00
void lcPiecesLibrary : : ReleaseBuffers ( lcContext * Context )
{
Context - > DestroyVertexBuffer ( mVertexBuffer ) ;
Context - > DestroyIndexBuffer ( mIndexBuffer ) ;
mBuffersDirty = true ;
}
2015-04-26 20:14:33 +02:00
void lcPiecesLibrary : : UpdateBuffers ( lcContext * Context )
{
2015-05-09 21:54:29 +02:00
if ( ! gSupportsVertexBufferObject | | ! mBuffersDirty )
2015-04-26 20:14:33 +02:00
return ;
int VertexDataSize = 0 ;
int IndexDataSize = 0 ;
2020-01-24 22:12:34 +01:00
std : : vector < lcMesh * > Meshes ;
2015-04-26 20:14:33 +02:00
2018-02-22 02:27:24 +01:00
for ( const auto & PieceIt : mPieces )
2015-04-26 20:14:33 +02:00
{
2020-03-23 04:18:52 +01:00
const PieceInfo * const Info = PieceIt . second ;
2015-04-26 20:41:16 +02:00
lcMesh * Mesh = Info - > IsPlaceholder ( ) ? gPlaceholderMesh : Info - > GetMesh ( ) ;
2015-04-26 20:14:33 +02:00
if ( ! Mesh )
continue ;
2016-05-29 04:46:34 +02:00
if ( Mesh - > mVertexDataSize > 16 * 1024 * 1024 | | Mesh - > mIndexDataSize > 16 * 1024 * 1024 )
continue ;
2015-04-26 20:14:33 +02:00
VertexDataSize + = Mesh - > mVertexDataSize ;
IndexDataSize + = Mesh - > mIndexDataSize ;
2020-01-24 22:12:34 +01:00
Meshes . push_back ( Mesh ) ;
2015-04-26 20:14:33 +02:00
}
Context - > DestroyVertexBuffer ( mVertexBuffer ) ;
Context - > DestroyIndexBuffer ( mIndexBuffer ) ;
if ( ! VertexDataSize | | ! IndexDataSize )
return ;
void * VertexData = malloc ( VertexDataSize ) ;
void * IndexData = malloc ( IndexDataSize ) ;
VertexDataSize = 0 ;
IndexDataSize = 0 ;
2020-01-24 22:12:34 +01:00
for ( lcMesh * Mesh : Meshes )
2015-04-26 20:14:33 +02:00
{
Mesh - > mVertexCacheOffset = VertexDataSize ;
Mesh - > mIndexCacheOffset = IndexDataSize ;
memcpy ( ( char * ) VertexData + VertexDataSize , Mesh - > mVertexData , Mesh - > mVertexDataSize ) ;
memcpy ( ( char * ) IndexData + IndexDataSize , Mesh - > mIndexData , Mesh - > mIndexDataSize ) ;
VertexDataSize + = Mesh - > mVertexDataSize ;
IndexDataSize + = Mesh - > mIndexDataSize ;
}
mVertexBuffer = Context - > CreateVertexBuffer ( VertexDataSize , VertexData ) ;
mIndexBuffer = Context - > CreateIndexBuffer ( IndexDataSize , IndexData ) ;
mBuffersDirty = false ;
free ( VertexData ) ;
free ( IndexData ) ;
}
2015-11-16 03:41:16 +01:00
void lcPiecesLibrary : : UnloadUnusedParts ( )
{
2017-01-23 04:28:05 +01:00
QMutexLocker LoadLock ( & mLoadMutex ) ;
2018-02-22 02:27:24 +01:00
for ( const auto & PieceIt : mPieces )
2017-01-23 04:28:05 +01:00
{
2017-07-24 04:35:18 +02:00
PieceInfo * Info = PieceIt . second ;
2017-01-23 04:28:05 +01:00
if ( Info - > GetRefCount ( ) = = 0 & & Info - > mState ! = LC_PIECEINFO_UNLOADED )
ReleasePieceInfo ( Info ) ;
}
2015-11-16 03:41:16 +01:00
}
2013-08-09 06:57:18 +02:00
bool lcPiecesLibrary : : LoadTexture ( lcTexture * Texture )
{
2019-06-03 22:23:04 +02:00
char FileName [ 2 * LC_MAXPATH ] ;
2013-08-09 06:57:18 +02:00
2014-05-08 00:58:59 +02:00
if ( mZipFiles [ LC_ZIPFILE_OFFICIAL ] )
2013-08-09 06:57:18 +02:00
{
lcMemFile TextureFile ;
2018-02-17 20:32:29 +01:00
sprintf ( FileName , " parts/textures/%s.png " , Texture - > mName ) ;
2013-08-09 06:57:18 +02:00
2014-05-08 00:58:59 +02:00
if ( ! mZipFiles [ LC_ZIPFILE_UNOFFICIAL ] | | ! mZipFiles [ LC_ZIPFILE_UNOFFICIAL ] - > ExtractFile ( FileName , TextureFile ) )
2017-12-22 13:38:47 +01:00
{
2018-02-17 20:32:29 +01:00
sprintf ( FileName , " ldraw/parts/textures/%s.png " , Texture - > mName ) ;
2017-12-22 13:38:47 +01:00
2014-05-08 00:58:59 +02:00
if ( ! mZipFiles [ LC_ZIPFILE_OFFICIAL ] - > ExtractFile ( FileName , TextureFile ) )
return false ;
2017-12-22 13:38:47 +01:00
}
2013-08-09 06:57:18 +02:00
2018-02-17 20:32:29 +01:00
return Texture - > Load ( TextureFile ) ;
2013-08-09 06:57:18 +02:00
}
else
2019-10-26 20:41:49 +02:00
return Texture - > Load ( Texture - > mFileName ) ;
2013-08-09 06:57:18 +02:00
}
2017-07-02 02:12:09 +02:00
void lcPiecesLibrary : : ReleaseTexture ( lcTexture * Texture )
{
QMutexLocker LoadLock ( & mLoadMutex ) ;
if ( Texture - > Release ( ) = = 0 & & Texture - > IsTemporary ( ) )
{
2019-07-05 02:06:26 +02:00
std : : vector < lcTexture * > : : iterator TextureIt = std : : find ( mTextures . begin ( ) , mTextures . end ( ) , Texture ) ;
if ( TextureIt ! = mTextures . end ( ) )
mTextures . erase ( TextureIt ) ;
2017-07-02 02:12:09 +02:00
delete Texture ;
}
}
2017-12-22 14:42:28 +01:00
void lcPiecesLibrary : : QueueTextureUpload ( lcTexture * Texture )
{
QMutexLocker Lock ( & mTextureMutex ) ;
mTextureUploads . push_back ( Texture ) ;
}
2018-09-30 04:45:21 +02:00
void lcPiecesLibrary : : UploadTextures ( lcContext * Context )
2017-12-22 14:42:28 +01:00
{
QMutexLocker Lock ( & mTextureMutex ) ;
for ( lcTexture * Texture : mTextureUploads )
2018-09-30 04:45:21 +02:00
Texture - > Upload ( Context ) ;
2017-12-22 14:42:28 +01:00
mTextureUploads . clear ( ) ;
}
2020-12-13 00:21:30 +01:00
bool lcPiecesLibrary : : SupportsStudLogo ( ) const
{
return mZipFiles [ LC_ZIPFILE_UNOFFICIAL ] | | mHasUnofficial ;
}
2019-09-24 01:55:24 +02:00
void lcPiecesLibrary : : SetStudLogo ( int StudLogo , bool Reload )
2019-09-24 01:22:11 +02:00
{
2019-09-24 01:55:24 +02:00
mStudLogo = StudLogo ;
mLoadMutex . lock ( ) ;
2019-09-27 23:52:50 +02:00
for ( const auto & PrimitiveIt : mPrimitives )
2019-09-24 01:55:24 +02:00
{
2019-09-27 23:52:50 +02:00
lcLibraryPrimitive * Primitive = PrimitiveIt . second ;
if ( Primitive - > mMeshData . mHasLogoStud )
Primitive - > Unload ( ) ;
2019-09-24 01:55:24 +02:00
}
mLoadMutex . unlock ( ) ;
if ( Reload )
2019-09-24 01:22:11 +02:00
{
2019-09-24 01:55:24 +02:00
mLoadMutex . lock ( ) ;
for ( const auto & PieceIt : mPieces )
{
PieceInfo * Info = PieceIt . second ;
2019-09-29 21:58:18 +02:00
if ( Info - > mState = = LC_PIECEINFO_LOADED & & Info - > GetMesh ( ) & & Info - > GetMesh ( ) - > mFlags & lcMeshFlag : : HasLogoStud )
2019-09-24 01:55:24 +02:00
{
Info - > Unload ( ) ;
mLoadQueue . append ( Info ) ;
mLoadFutures . append ( QtConcurrent : : run ( [ this ] ( ) { LoadQueuedPiece ( ) ; } ) ) ;
}
2019-09-24 01:22:11 +02:00
}
2019-09-24 01:55:24 +02:00
mLoadMutex . unlock ( ) ;
WaitForLoadQueue ( ) ;
2019-09-24 01:22:11 +02:00
}
}
2019-09-29 21:58:18 +02:00
bool lcPiecesLibrary : : GetStudLogoFile ( lcMemFile & PrimFile , int StudLogo , bool OpenStud )
2019-09-21 18:47:33 +02:00
{
if ( ! StudLogo | | ( ! mZipFiles [ LC_ZIPFILE_UNOFFICIAL ] & & ! mHasUnofficial ) )
return false ;
QString Logo = QString ( " %1 " ) . arg ( StudLogo ) ;
QString LogoRefLine = QString ( " 1 16 0 0 0 1 0 0 0 1 0 0 0 1 " ) ;
LogoRefLine + = ( OpenStud ? QString ( " stud2-logo%1.dat " ) . arg ( StudLogo > 1 ? Logo : " " ) :
QString ( " stud-logo%1.dat " ) . arg ( StudLogo > 1 ? Logo : " " ) ) ;
2020-12-15 00:08:43 +01:00
const QLatin1String LineEnding ( " \r \n " ) ;
2019-09-21 18:47:33 +02:00
QByteArray FileData ;
2020-12-15 00:08:43 +01:00
QTextStream TextStream ( & FileData ) ;
TextStream < < ( OpenStud ? " 0 Stud Open " : " 0 Stud " ) < < LineEnding ;
TextStream < < ( OpenStud ? " 0 Name: stud2.dat " : " 0 Name: stud.dat " ) < < LineEnding ;
TextStream < < " 0 Author: James Jessiman " < < LineEnding ;
TextStream < < " 0 !LDRAW_ORG Primitive " < < LineEnding ;
TextStream < < " 0 BFC CERTIFY CCW " < < LineEnding ;
TextStream < < LogoRefLine < < LineEnding ;
2019-09-21 18:47:33 +02:00
PrimFile . WriteBuffer ( FileData . constData ( ) , size_t ( FileData . size ( ) ) ) ;
PrimFile . Seek ( 0 , SEEK_SET ) ;
2020-12-15 00:08:43 +01:00
2019-09-21 18:47:33 +02:00
return true ;
}
2017-07-24 01:19:09 +02:00
bool lcPiecesLibrary : : LoadPrimitive ( lcLibraryPrimitive * Primitive )
2013-08-09 06:57:18 +02:00
{
2017-11-25 04:45:27 +01:00
mLoadMutex . lock ( ) ;
if ( Primitive - > mState = = lcPrimitiveState : : NOT_LOADED )
Primitive - > mState = lcPrimitiveState : : LOADING ;
else
{
mLoadMutex . unlock ( ) ;
while ( Primitive - > mState = = lcPrimitiveState : : LOADING )
lcSleeper : : msleep ( 5 ) ;
return Primitive - > mState = = lcPrimitiveState : : LOADED ;
}
mLoadMutex . unlock ( ) ;
2017-01-23 04:28:05 +01:00
2019-07-28 01:31:16 +02:00
lcMeshLoader MeshLoader ( Primitive - > mMeshData , true , nullptr , false ) ;
2013-08-09 06:57:18 +02:00
2019-09-21 18:47:33 +02:00
bool SetStudLogo = false ;
2014-05-08 00:58:59 +02:00
if ( mZipFiles [ LC_ZIPFILE_OFFICIAL ] )
2013-08-09 06:57:18 +02:00
{
2017-07-24 01:19:09 +02:00
lcLibraryPrimitive * LowPrimitive = nullptr ;
2015-05-24 06:36:25 +02:00
2019-09-21 18:47:33 +02:00
lcMemFile PrimFile ;
if ( Primitive - > mStud )
2015-05-24 06:36:25 +02:00
{
2020-03-23 04:18:52 +01:00
const bool OpenStud = ! strcmp ( Primitive - > mName , " stud2.dat " ) ;
2019-09-27 23:52:50 +02:00
if ( OpenStud | | ! strcmp ( Primitive - > mName , " stud.dat " ) )
2019-09-21 18:47:33 +02:00
{
2019-09-27 23:52:50 +02:00
Primitive - > mMeshData . mHasLogoStud = true ;
if ( mStudLogo )
2019-09-29 21:58:18 +02:00
SetStudLogo = GetStudLogoFile ( PrimFile , mStudLogo , OpenStud ) ;
}
if ( ! SetStudLogo & & strncmp ( Primitive - > mName , " 8/ " , 2 ) ) // todo: this is currently the only place that uses mName so use mFileName instead. this should also be done for the loose file libraries.
{
char Name [ LC_PIECE_NAME_LEN ] ;
strcpy ( Name , " 8/ " ) ;
strcat ( Name , Primitive - > mName ) ;
strupr ( Name ) ;
LowPrimitive = FindPrimitive ( Name ) ;
2019-09-21 18:47:33 +02:00
}
}
2013-08-09 06:57:18 +02:00
2019-09-21 18:47:33 +02:00
if ( ! SetStudLogo & & ! mZipFiles [ Primitive - > mZipFileType ] - > ExtractFile ( Primitive - > mZipFileIndex , PrimFile ) )
2013-08-09 06:57:18 +02:00
return false ;
2017-07-24 01:19:09 +02:00
if ( ! LowPrimitive )
2015-05-24 06:36:25 +02:00
{
2019-07-28 01:31:16 +02:00
if ( ! MeshLoader . LoadMesh ( PrimFile , LC_MESHDATA_SHARED ) )
2015-05-24 06:36:25 +02:00
return false ;
}
else
{
2019-07-28 01:31:16 +02:00
if ( ! MeshLoader . LoadMesh ( PrimFile , LC_MESHDATA_HIGH ) )
2015-05-24 06:36:25 +02:00
return false ;
if ( ! mZipFiles [ LowPrimitive - > mZipFileType ] - > ExtractFile ( LowPrimitive - > mZipFileIndex , PrimFile ) )
return false ;
2019-07-28 01:31:16 +02:00
if ( ! MeshLoader . LoadMesh ( PrimFile , LC_MESHDATA_LOW ) )
2015-05-24 06:36:25 +02:00
return false ;
}
2013-08-09 06:57:18 +02:00
}
else
{
2019-09-27 23:52:50 +02:00
if ( Primitive - > mStud )
2019-09-21 18:47:33 +02:00
{
2020-03-23 04:18:52 +01:00
const bool OpenStud = ! strcmp ( Primitive - > mName , " stud2.dat " ) ;
2019-09-24 01:22:11 +02:00
if ( OpenStud | | ! strcmp ( Primitive - > mName , " stud.dat " ) )
2019-09-21 18:47:33 +02:00
{
2019-09-27 23:52:50 +02:00
Primitive - > mMeshData . mHasLogoStud = true ;
if ( mStudLogo )
{
lcMemFile PrimFile ;
2019-09-29 21:58:18 +02:00
if ( GetStudLogoFile ( PrimFile , mStudLogo , OpenStud ) )
2019-09-27 23:52:50 +02:00
SetStudLogo = MeshLoader . LoadMesh ( PrimFile , LC_MESHDATA_SHARED ) ;
}
2019-09-21 18:47:33 +02:00
}
}
if ( ! SetStudLogo )
{
lcDiskFile PrimFile ( Primitive - > mFileName ) ;
if ( ! PrimFile . Open ( QIODevice : : ReadOnly ) | | ! MeshLoader . LoadMesh ( PrimFile , LC_MESHDATA_SHARED ) ) // todo: LOD like the zip files
return false ;
}
2013-08-09 06:57:18 +02:00
}
2017-11-25 04:45:27 +01:00
mLoadMutex . lock ( ) ;
Primitive - > mState = lcPrimitiveState : : LOADED ;
mLoadMutex . unlock ( ) ;
2013-08-09 06:57:18 +02:00
return true ;
}
2017-02-08 03:55:54 +01:00
bool lcPiecesLibrary : : PieceInCategory ( PieceInfo * Info , const char * CategoryKeywords ) const
2013-08-09 06:57:18 +02:00
{
2015-02-22 03:39:15 +01:00
if ( Info - > IsTemporary ( ) )
return false ;
2017-02-08 03:55:54 +01:00
const char * PieceName ;
2013-08-09 06:57:18 +02:00
if ( Info - > m_strDescription [ 0 ] = = ' ~ ' | | Info - > m_strDescription [ 0 ] = = ' _ ' )
PieceName = Info - > m_strDescription + 1 ;
else
PieceName = Info - > m_strDescription ;
2017-02-08 03:55:54 +01:00
return lcMatchCategory ( PieceName , CategoryKeywords ) ;
2013-08-09 06:57:18 +02:00
}
2013-08-16 01:43:18 +02:00
void lcPiecesLibrary : : GetCategoryEntries ( int CategoryIndex , bool GroupPieces , lcArray < PieceInfo * > & SinglePieces , lcArray < PieceInfo * > & GroupedPieces )
2013-08-09 06:57:18 +02:00
{
2020-01-11 02:40:14 +01:00
if ( CategoryIndex > = 0 & & CategoryIndex < static_cast < int > ( gCategories . size ( ) ) )
2017-02-08 18:41:48 +01:00
GetCategoryEntries ( gCategories [ CategoryIndex ] . Keywords . constData ( ) , GroupPieces , SinglePieces , GroupedPieces ) ;
2013-08-09 06:57:18 +02:00
}
2017-02-08 03:55:54 +01:00
void lcPiecesLibrary : : GetCategoryEntries ( const char * CategoryKeywords , bool GroupPieces , lcArray < PieceInfo * > & SinglePieces , lcArray < PieceInfo * > & GroupedPieces )
2013-08-09 06:57:18 +02:00
{
SinglePieces . RemoveAll ( ) ;
GroupedPieces . RemoveAll ( ) ;
2018-02-22 02:27:24 +01:00
for ( const auto & PieceIt : mPieces )
2013-08-09 06:57:18 +02:00
{
2017-07-24 04:35:18 +02:00
PieceInfo * Info = PieceIt . second ;
2013-08-09 06:57:18 +02:00
if ( ! PieceInCategory ( Info , CategoryKeywords ) )
continue ;
if ( ! GroupPieces )
{
SinglePieces . Add ( Info ) ;
continue ;
}
// Check if it's a patterned piece.
if ( Info - > IsPatterned ( ) )
{
PieceInfo * Parent ;
// Find the parent of this patterned piece.
char ParentName [ LC_PIECE_NAME_LEN ] ;
2017-07-27 18:21:55 +02:00
strcpy ( ParentName , Info - > mFileName ) ;
2013-08-09 06:57:18 +02:00
* strchr ( ParentName , ' P ' ) = ' \0 ' ;
2017-07-27 18:21:55 +02:00
strcat ( ParentName , " .dat " ) ;
2013-08-09 06:57:18 +02:00
2017-04-14 02:26:40 +02:00
Parent = FindPiece ( ParentName , nullptr , false , false ) ;
2013-08-09 06:57:18 +02:00
if ( Parent )
{
// Check if the parent was added as a single piece.
int Index = SinglePieces . FindIndex ( Parent ) ;
if ( Index ! = - 1 )
SinglePieces . RemoveIndex ( Index ) ;
Index = GroupedPieces . FindIndex ( Parent ) ;
if ( Index = = - 1 )
GroupedPieces . Add ( Parent ) ;
}
else
{
// Patterned pieces should have a parent but in case they don't just add them anyway.
SinglePieces . Add ( Info ) ;
}
}
else
{
// Check if this piece has already been added to this category by one of its children.
2020-03-23 04:18:52 +01:00
const int Index = GroupedPieces . FindIndex ( Info ) ;
2013-08-09 06:57:18 +02:00
if ( Index = = - 1 )
SinglePieces . Add ( Info ) ;
}
}
}
2013-08-16 01:43:18 +02:00
void lcPiecesLibrary : : GetPatternedPieces ( PieceInfo * Parent , lcArray < PieceInfo * > & Pieces ) const
2013-08-09 06:57:18 +02:00
{
char Name [ LC_PIECE_NAME_LEN ] ;
2017-07-27 18:21:55 +02:00
strcpy ( Name , Parent - > mFileName ) ;
char * Ext = strchr ( Name , ' . ' ) ;
if ( Ext )
* Ext = 0 ;
2013-08-09 06:57:18 +02:00
strcat ( Name , " P " ) ;
2017-07-27 18:21:55 +02:00
strupr ( Name ) ;
2013-08-09 06:57:18 +02:00
Pieces . RemoveAll ( ) ;
2018-02-22 02:27:24 +01:00
for ( const auto & PieceIt : mPieces )
2017-07-27 18:21:55 +02:00
if ( strncmp ( Name , PieceIt . first . c_str ( ) , strlen ( Name ) ) = = 0 )
Pieces . Add ( PieceIt . second ) ;
2013-08-09 06:57:18 +02:00
// Sometimes pieces with A and B versions don't follow the same convention (for example, 3040Pxx instead of 3040BPxx).
if ( Pieces . GetSize ( ) = = 0 )
{
2017-07-27 18:21:55 +02:00
strcpy ( Name , Parent - > mFileName ) ;
Ext = strchr ( Name , ' . ' ) ;
if ( Ext )
* Ext = 0 ;
2016-02-17 00:11:52 +01:00
size_t Len = strlen ( Name ) ;
2013-08-09 06:57:18 +02:00
if ( Name [ Len - 1 ] < ' 0 ' | | Name [ Len - 1 ] > ' 9 ' )
Name [ Len - 1 ] = ' P ' ;
2018-02-22 02:27:24 +01:00
for ( const auto & PieceIt : mPieces )
2017-07-27 18:21:55 +02:00
if ( strncmp ( Name , PieceIt . first . c_str ( ) , strlen ( Name ) ) = = 0 )
Pieces . Add ( PieceIt . second ) ;
2013-08-09 06:57:18 +02:00
}
}
2019-12-07 18:52:46 +01:00
void lcPiecesLibrary : : GetParts ( lcArray < PieceInfo * > & Parts ) const
2017-02-01 06:12:30 +01:00
{
2017-07-24 04:35:18 +02:00
Parts . SetSize ( 0 ) ;
Parts . AllocGrow ( mPieces . size ( ) ) ;
2018-02-22 02:27:24 +01:00
for ( const auto & PartIt : mPieces )
2017-07-24 04:35:18 +02:00
Parts . Add ( PartIt . second ) ;
2017-02-01 06:12:30 +01:00
}
2019-12-09 01:54:12 +01:00
std : : vector < PieceInfo * > lcPiecesLibrary : : GetPartsFromSet ( const std : : vector < std : : string > & PartIds ) const
2019-12-07 18:52:46 +01:00
{
std : : vector < PieceInfo * > Parts ;
2019-12-09 01:54:12 +01:00
Parts . reserve ( PartIds . size ( ) ) ;
2019-12-07 18:52:46 +01:00
2019-12-09 01:54:12 +01:00
for ( const std : : string & PartId : PartIds )
2019-12-07 18:52:46 +01:00
{
2019-12-09 01:54:12 +01:00
std : : map < std : : string , PieceInfo * > : : const_iterator PartIt = mPieces . find ( PartId ) ;
2019-12-07 18:52:46 +01:00
if ( PartIt ! = mPieces . end ( ) )
Parts . push_back ( PartIt - > second ) ;
}
return Parts ;
}
2019-12-09 01:54:12 +01:00
std : : string lcPiecesLibrary : : GetPartId ( const PieceInfo * Info ) const
2019-12-07 18:52:46 +01:00
{
std : : map < std : : string , PieceInfo * > : : const_iterator PartIt = std : : find_if ( mPieces . begin ( ) , mPieces . end ( ) , [ Info ] ( const std : : pair < std : : string , PieceInfo * > & PartIt )
{
return PartIt . second = = Info ;
} ) ;
2019-12-09 01:54:12 +01:00
if ( PartIt ! = mPieces . end ( ) )
return PartIt - > first ;
else
return std : : string ( ) ;
2019-12-07 18:52:46 +01:00
}
2014-09-11 21:55:34 +02:00
bool lcPiecesLibrary : : LoadBuiltinPieces ( )
2014-09-10 03:41:37 +02:00
{
2014-09-11 21:55:34 +02:00
QResource Resource ( " :/resources/library.zip " ) ;
2013-08-09 06:57:18 +02:00
2014-09-11 21:55:34 +02:00
if ( ! Resource . isValid ( ) )
return false ;
2013-08-09 06:57:18 +02:00
2020-12-13 16:40:29 +01:00
std : : unique_ptr < lcMemFile > File ( new lcMemFile ( ) ) ;
2014-09-11 21:55:34 +02:00
File - > WriteBuffer ( Resource . data ( ) , Resource . size ( ) ) ;
2013-08-09 06:57:18 +02:00
2020-12-19 20:12:31 +01:00
if ( ! OpenArchive ( std : : move ( File ) , LC_ZIPFILE_OFFICIAL ) )
2014-09-11 21:55:34 +02:00
return false ;
2014-09-10 03:41:37 +02:00
2014-09-11 21:55:34 +02:00
lcMemFile PieceFile ;
2013-08-09 06:57:18 +02:00
2018-02-22 02:27:24 +01:00
for ( const auto & PieceIt : mPieces )
2014-09-10 03:41:37 +02:00
{
2017-07-24 04:35:18 +02:00
PieceInfo * Info = PieceIt . second ;
2014-09-10 03:41:37 +02:00
2014-09-11 21:55:34 +02:00
mZipFiles [ Info - > mZipFileType ] - > ExtractFile ( Info - > mZipFileIndex , PieceFile , 256 ) ;
PieceFile . Seek ( 0 , SEEK_END ) ;
PieceFile . WriteU8 ( 0 ) ;
2014-09-10 03:41:37 +02:00
2014-09-11 21:55:34 +02:00
char * Src = ( char * ) PieceFile . mBuffer + 2 ;
char * Dst = Info - > m_strDescription ;
2013-08-09 06:57:18 +02:00
2014-09-11 21:55:34 +02:00
for ( ; ; )
2013-08-09 06:57:18 +02:00
{
2014-09-11 21:55:34 +02:00
if ( * Src ! = ' \r ' & & * Src ! = ' \n ' & & * Src & & Dst - Info - > m_strDescription < ( int ) sizeof ( Info - > m_strDescription ) - 1 )
2014-08-30 21:48:36 +02:00
{
2014-09-11 21:55:34 +02:00
* Dst + + = * Src + + ;
continue ;
2013-08-09 06:57:18 +02:00
}
2014-09-11 21:55:34 +02:00
* Dst = 0 ;
break ;
2013-08-09 06:57:18 +02:00
}
}
2014-09-11 21:55:34 +02:00
lcLoadDefaultColors ( ) ;
lcLoadDefaultCategories ( true ) ;
2016-02-29 21:13:54 +01:00
lcSynthInit ( ) ;
2014-09-11 21:55:34 +02:00
return true ;
2013-08-09 06:57:18 +02:00
}