2012-03-02 04:41:29 +02:00
# include <iostream>
# include <boost/program_options.hpp>
2010-07-21 13:52:28 +02:00
# include <components/esm/esm_reader.hpp>
# include <components/esm/records.hpp>
2010-02-19 09:12:49 +01:00
2012-03-02 04:41:29 +02:00
# define ESMTOOL_VERSION 1.1
2010-02-19 14:23:22 +01:00
2010-02-18 13:15:43 +01:00
using namespace std ;
2010-02-19 14:23:22 +01:00
using namespace ESM ;
2010-02-18 13:15:43 +01:00
2012-03-02 04:41:29 +02:00
// Create a local alias for brevity
namespace bpo = boost : : program_options ;
2010-02-19 09:12:49 +01:00
void printRaw ( ESMReader & esm ) ;
2010-03-04 15:42:59 +01:00
void loadCell ( Cell & cell , ESMReader & esm , bool quiet ) ;
2010-02-18 14:58:50 +01:00
2012-03-02 04:41:29 +02:00
// Based on the legacy struct
struct Arguments
2010-02-18 13:15:43 +01:00
{
2012-03-02 04:41:29 +02:00
unsigned int raw_given ;
unsigned int quiet_given ;
unsigned int loadcells_given ;
std : : string encoding ;
std : : string filename ;
} ;
2010-02-19 09:12:49 +01:00
2012-03-02 04:41:29 +02:00
bool parseOptions ( int argc , char * * argv , Arguments & info )
{
bpo : : options_description desc ( " Inspect and extract from Morrowind ES files (ESM, ESP, ESS) \n Syntax: esmtool [options] file \n Allowed options " ) ;
desc . add_options ( )
( " help,h " , " print help message. " )
( " version,v " , " print version information and quit. " )
( " raw,r " , " Show an unformattet list of all records and subrecords. " )
( " quiet,q " , " Supress all record information. Useful for speed tests. " )
( " loadcells,C " , " Browse through contents of all cells. " )
( " encoding,e " , bpo : : value < std : : string > ( & ( info . encoding ) ) - >
default_value ( " win1252 " ) ,
" Character encoding used in ESMTool: \n "
" \n \t win1250 - Central and Eastern European such as Polish, Czech, Slovak, Hungarian, Slovene, Bosnian, Croatian, Serbian (Latin script), Romanian and Albanian languages \n "
" \n \t win1251 - Cyrillic alphabet such as Russian, Bulgarian, Serbian Cyrillic and other languages \n "
" \n \t win1252 - Western European (Latin) alphabet, used by default " )
;
2010-02-19 09:12:49 +01:00
2012-03-02 04:41:29 +02:00
std : : string finalText = " \n If no option is given, the default action is to parse all records in the archive \n and display diagnostic information. " ;
// input-file is hidden and used as a positional argument
bpo : : options_description hidden ( " Hidden Options " ) ;
hidden . add_options ( )
( " input-file,i " , bpo : : value < vector < std : : string > > ( ) , " input file " )
;
bpo : : positional_options_description p ;
p . add ( " input-file " , - 1 ) ;
// there might be a better way to do this
bpo : : options_description all ;
all . add ( desc ) . add ( hidden ) ;
bpo : : parsed_options valid_opts = bpo : : command_line_parser ( argc , argv )
. options ( all ) . positional ( p ) . run ( ) ;
bpo : : variables_map variables ;
bpo : : store ( valid_opts , variables ) ;
bpo : : notify ( variables ) ;
if ( variables . count ( " help " ) )
{
std : : cout < < desc < < finalText < < std : : endl ;
return false ;
}
if ( variables . count ( " version " ) )
2010-02-18 13:15:43 +01:00
{
2012-03-02 04:41:29 +02:00
std : : cout < < " ESMTool version " < < ESMTOOL_VERSION < < std : : endl ;
return false ;
2010-02-18 13:15:43 +01:00
}
2012-03-02 04:41:29 +02:00
if ( ! variables . count ( " input-file " ) )
{
std : : cout < < " \n ERROR: missing ES file \n \n " ;
std : : cout < < desc < < finalText < < std : : endl ;
return false ;
}
// handling gracefully the user adding multiple files
if ( variables [ " input-file " ] . as < vector < std : : string > > ( ) . size ( ) > 1 )
{
std : : cout < < " \n ERROR: more than one ES file specified \n \n " ;
std : : cout < < desc < < finalText < < std : : endl ;
return false ;
}
info . filename = variables [ " input-file " ] . as < vector < std : : string > > ( ) [ 0 ] ;
info . raw_given = variables . count ( " raw " ) ;
info . quiet_given = variables . count ( " quiet " ) ;
info . loadcells_given = variables . count ( " loadcells " ) ;
// Font encoding settings
info . encoding = variables [ " encoding " ] . as < std : : string > ( ) ;
if ( info . encoding = = " win1250 " )
{
std : : cout < < " Using Central and Eastern European font encoding. " < < std : : endl ;
}
else if ( info . encoding = = " win1251 " )
{
std : : cout < < " Using Cyrillic font encoding. " < < std : : endl ;
}
else
{
if ( info . encoding ! = " win1252 " )
{
std : : cout < < info . encoding < < " is not a valid encoding option. " < < std : : endl ;
info . encoding = " win1252 " ;
}
std : : cout < < " Using default (English) font encoding. " < < std : : endl ;
}
return true ;
}
int main ( int argc , char * * argv )
{
Arguments info ;
if ( ! parseOptions ( argc , argv , info ) )
return 1 ;
2010-02-18 13:15:43 +01:00
ESMReader esm ;
2012-03-02 04:41:29 +02:00
esm . setEncoding ( info . encoding ) ;
string filename = info . filename ;
2010-02-19 09:12:49 +01:00
cout < < " \n File: " < < filename < < endl ;
2010-03-04 15:42:59 +01:00
try {
2010-02-19 09:12:49 +01:00
if ( info . raw_given )
{
cout < < " RAW file listing: \n " ;
esm . openRaw ( filename ) ;
printRaw ( esm ) ;
return 0 ;
}
2010-02-28 09:18:48 +01:00
bool quiet = info . quiet_given ;
2010-03-04 15:42:59 +01:00
bool loadCells = info . loadcells_given ;
2010-02-28 09:18:48 +01:00
2010-02-19 09:12:49 +01:00
esm . open ( filename ) ;
2010-02-18 13:15:43 +01:00
cout < < " Author: " < < esm . getAuthor ( ) < < endl ;
cout < < " Description: " < < esm . getDesc ( ) < < endl ;
cout < < " File format version: " < < esm . getFVer ( ) < < endl ;
cout < < " Special flag: " < < esm . getSpecial ( ) < < endl ;
cout < < " Masters: \n " ;
ESMReader : : MasterList m = esm . getMasters ( ) ;
2010-07-22 14:15:02 +02:00
for ( unsigned int i = 0 ; i < m . size ( ) ; i + + )
2010-02-18 13:15:43 +01:00
cout < < " " < < m [ i ] . name < < " , " < < m [ i ] . size < < " bytes \n " ;
// Loop through all records
while ( esm . hasMoreRecs ( ) )
{
NAME n = esm . getRecName ( ) ;
esm . getRecHeader ( ) ;
2010-02-19 09:12:49 +01:00
string id = esm . getHNOString ( " NAME " ) ;
2010-02-28 09:18:48 +01:00
if ( ! quiet )
cout < < " \n Record: " < < n . toString ( )
< < " ' " < < id < < " ' \n " ;
2010-02-18 13:15:43 +01:00
switch ( n . val )
{
2010-02-20 21:55:51 +01:00
case REC_ACTI :
{
Activator ac ;
ac . load ( esm ) ;
2010-02-28 09:18:48 +01:00
if ( quiet ) break ;
2010-02-20 21:55:51 +01:00
cout < < " Name: " < < ac . name < < endl ;
cout < < " Mesh: " < < ac . model < < endl ;
cout < < " Script: " < < ac . script < < endl ;
break ;
}
2010-02-28 09:18:48 +01:00
case REC_ALCH :
{
Potion p ;
p . load ( esm ) ;
if ( quiet ) break ;
cout < < " Name: " < < p . name < < endl ;
break ;
}
case REC_APPA :
{
Apparatus p ;
p . load ( esm ) ;
if ( quiet ) break ;
cout < < " Name: " < < p . name < < endl ;
break ;
}
2010-02-18 14:58:50 +01:00
case REC_ARMO :
{
Armor am ;
am . load ( esm ) ;
2010-02-28 09:18:48 +01:00
if ( quiet ) break ;
2010-02-18 14:58:50 +01:00
cout < < " Name: " < < am . name < < endl ;
cout < < " Mesh: " < < am . model < < endl ;
cout < < " Icon: " < < am . icon < < endl ;
cout < < " Script: " < < am . script < < endl ;
cout < < " Enchantment: " < < am . enchant < < endl ;
cout < < " Type: " < < am . data . type < < endl ;
cout < < " Weight: " < < am . data . weight < < endl ;
break ;
}
2010-02-18 13:15:43 +01:00
case REC_BODY :
{
BodyPart bp ;
bp . load ( esm ) ;
2010-02-28 09:18:48 +01:00
if ( quiet ) break ;
2010-02-18 13:15:43 +01:00
cout < < " Name: " < < bp . name < < endl ;
cout < < " Mesh: " < < bp . model < < endl ;
break ;
}
2010-02-28 09:18:48 +01:00
case REC_BOOK :
{
Book b ;
b . load ( esm ) ;
if ( quiet ) break ;
cout < < " Name: " < < b . name < < endl ;
cout < < " Mesh: " < < b . model < < endl ;
break ;
}
2010-02-19 14:23:22 +01:00
case REC_BSGN :
{
BirthSign bs ;
bs . load ( esm ) ;
2010-02-28 09:18:48 +01:00
if ( quiet ) break ;
2010-02-19 14:23:22 +01:00
cout < < " Name: " < < bs . name < < endl ;
cout < < " Texture: " < < bs . texture < < endl ;
cout < < " Description: " < < bs . description < < endl ;
break ;
}
2010-02-28 09:18:48 +01:00
case REC_CELL :
{
Cell b ;
b . load ( esm ) ;
2010-03-04 15:42:59 +01:00
if ( ! quiet )
{
cout < < " Name: " < < b . name < < endl ;
cout < < " Region: " < < b . region < < endl ;
}
if ( loadCells )
loadCell ( b , esm , quiet ) ;
2010-02-28 09:18:48 +01:00
break ;
}
case REC_CLAS :
{
Class b ;
b . load ( esm ) ;
if ( quiet ) break ;
cout < < " Name: " < < b . name < < endl ;
cout < < " Description: " < < b . description < < endl ;
break ;
}
case REC_CLOT :
{
Clothing b ;
b . load ( esm ) ;
if ( quiet ) break ;
cout < < " Name: " < < b . name < < endl ;
break ;
}
case REC_CONT :
{
Container b ;
b . load ( esm ) ;
if ( quiet ) break ;
cout < < " Name: " < < b . name < < endl ;
break ;
}
case REC_CREA :
{
Creature b ;
2010-08-08 14:09:09 +02:00
b . load ( esm , id ) ;
2010-02-28 09:18:48 +01:00
if ( quiet ) break ;
cout < < " Name: " < < b . name < < endl ;
break ;
}
case REC_DIAL :
{
Dialogue b ;
b . load ( esm ) ;
break ;
}
2010-02-19 14:23:22 +01:00
case REC_DOOR :
{
Door d ;
d . load ( esm ) ;
2010-02-28 09:18:48 +01:00
if ( quiet ) break ;
2010-02-19 14:23:22 +01:00
cout < < " Name: " < < d . name < < endl ;
cout < < " Mesh: " < < d . model < < endl ;
cout < < " Script: " < < d . script < < endl ;
cout < < " OpenSound: " < < d . openSound < < endl ;
cout < < " CloseSound: " < < d . closeSound < < endl ;
break ;
}
2010-02-28 09:18:48 +01:00
case REC_ENCH :
{
Enchantment b ;
b . load ( esm ) ;
break ;
}
case REC_GMST :
{
GameSetting b ;
b . id = id ;
b . load ( esm ) ;
if ( quiet ) break ;
cout < < " Value: " ;
if ( b . type = = VT_String )
cout < < " ' " < < b . str < < " ' (string) " ;
else if ( b . type = = VT_Float )
cout < < b . f < < " (float) " ;
else if ( b . type = = VT_Int )
cout < < b . i < < " (int) " ;
cout < < " \n Dirty: " < < b . dirty < < endl ;
break ;
}
case REC_INFO :
{
DialInfo p ;
p . load ( esm ) ;
if ( quiet ) break ;
cout < < " Id: " < < p . id < < endl ;
cout < < " Text: " < < p . response < < endl ;
break ;
}
2010-02-19 14:23:22 +01:00
case REC_SOUN :
{
Sound d ;
d . load ( esm ) ;
2010-02-28 09:18:48 +01:00
if ( quiet ) break ;
2010-02-19 14:23:22 +01:00
cout < < " Sound: " < < d . sound < < endl ;
cout < < " Volume: " < < ( int ) d . data . volume < < endl ;
break ;
}
2010-02-20 21:55:51 +01:00
case REC_SPEL :
{
Spell s ;
s . load ( esm ) ;
2010-02-28 09:18:48 +01:00
if ( quiet ) break ;
2010-02-20 21:55:51 +01:00
cout < < " Name: " < < s . name < < endl ;
break ;
}
2010-02-18 13:15:43 +01:00
default :
esm . skipRecord ( ) ;
2010-02-28 09:18:48 +01:00
if ( quiet ) break ;
cout < < " Skipping \n " ;
2010-02-18 13:15:43 +01:00
}
}
2010-03-04 15:42:59 +01:00
} catch ( exception & e )
{
cout < < " \n ERROR: \n \n " < < e . what ( ) < < endl ;
return 1 ;
}
2010-02-18 13:15:43 +01:00
return 0 ;
}
2010-02-19 09:12:49 +01:00
2010-03-04 15:42:59 +01:00
void loadCell ( Cell & cell , ESMReader & esm , bool quiet )
{
// Skip back to the beginning of the reference list
cell . restore ( esm ) ;
// Loop through all the references
CellRef ref ;
if ( ! quiet ) cout < < " References: \n " ;
while ( cell . getNextRef ( esm , ref ) )
{
if ( quiet ) continue ;
cout < < " Refnum: " < < ref . refnum < < endl ;
cout < < " ID: ' " < < ref . refID < < " ' \n " ;
cout < < " Owner: ' " < < ref . owner < < " ' \n " ;
cout < < " INTV: " < < ref . intv < < " NAM9: " < < ref . intv < < endl ;
}
}
2010-02-19 09:12:49 +01:00
void printRaw ( ESMReader & esm )
{
while ( esm . hasMoreRecs ( ) )
{
NAME n = esm . getRecName ( ) ;
cout < < " Record: " < < n . toString ( ) < < endl ;
esm . getRecHeader ( ) ;
while ( esm . hasMoreSubs ( ) )
{
uint64_t offs = esm . getOffset ( ) ;
esm . getSubName ( ) ;
esm . skipHSub ( ) ;
n = esm . retSubName ( ) ;
cout < < " " < < n . toString ( ) < < " - " < < esm . getSubSize ( )
< < " bytes @ 0x " < < hex < < offs < < " \n " ;
}
}
}