falcon4:tools:nsis_installer
This is an old revision of the document!
Table of Contents
NSIS Installer
NSIS Installer
Here in PMC Tactical we use NSIS installer to build our theater installers, also FreeFalcon/RedViper is using this method for their installation. Its very nice freeware open source program.
The scripting is relatively easy for a non-coder to figure out, but is still powerful and flexible. It has a strong user community and good open-source support. It also is “portable” in that there is a version which can run from almost any PC on USB memory stick.
Example Script
Here is the latest release PMC theater script, its from Vietnam theater.
!define TNAME "Vietnam" !define FRIENDLYNAME "PMC Vietnam Theater Installer" !define APP "VietnamInstaller" ;for some reason, this has to be #.#.#.# but I usually change the name after I build the exe !define VER "0.0.0.9" ; these establish some things about the installer the most important one to me is the outfile Name "${FRIENDLYNAME}" ; outfile is the name of the installer. Unless otherwise set, it will be the same directory as where the script is OutFile "PMC_${TNAME}_Theater_v${VER}.exe" Caption "${FRIENDLYNAME}" ; these VI definitions are what is displayed if you hover the mouse over the icon VIProductVersion "${VER}" VIAddVersionKey ProductName "${FRIENDLYNAME}" VIAddVersionKey Comments "Installs the ${TNAME} theater." VIAddVersionKey LegalCopyright "PMC Tactical" VIAddVersionKey FileDescription "${FRIENDLYNAME}" VIAddVersionKey FileVersion "${VER}" VIAddVersionKey ProductVersion "${VER}" VIAddVersionKey InternalName "${FRIENDLYNAME}" ;=== Runtime Switches ; the CRC check is something I use on bigger installers to make sure everything checks out. CRCCheck On ; this makes it more efficient AutoCloseWindow True ; this is something I use to make sure it uses the LZMA compression, which I believe is the best SetCompressor lzma ; branding text is just something that's displayed at the bottom of the installer BrandingText "http://tactical.nekromantix.com" ;=== Includes - this will include the necessary dlls and functions that I use in the installer ; MUI 1.67 compatible ------ !include "MUI.nsh" !include "Registry.nsh" !include "WinMessages.nsh" !include "TextFunc.nsh" !include "Sections.nsh" ; MUI Settings - the MUI is the Modern UI for NSIS, this section defines its parameters (there are alot more that I don't use) !define MUI_ABORTWARNING !define MUI_INSTALLCOLORS /windows ; Welcome page - this is the first page of the installer (I also have a snippet for a splashscreen, but we'll keep this simple) ;!define MUI_WELCOMEFINISHPAGE_BITMAP "@@:\FF4Installers\leftgraphic.bmp" ; using the variables means we don't have to remember to change all these when we use the template for another theater !define MUI_WELCOMEPAGE_TITLE "${FRIENDLYNAME} ${VER} Setup Wizard" !define MUI_WELCOMEPAGE_TEXT "Setup will install ${FRIENDLYNAME} ${VER} into your Falcon installation." !insertmacro MUI_PAGE_WELCOME !define MUI_LICENSEPAGE_RADIOBUTTONS !insertmacro MUI_PAGE_LICENSE "Theater_src_${TNAME}\Theaters\${TNAME}\EULA.txt" ; Instfiles page !define MUI_FINISHPAGE_NOAUTOCLOSE !insertmacro MUI_PAGE_INSTFILES ; Finish page - this is the last page people see !define MUI_FINISHPAGE_TITLE "${FRIENDLYNAME} ${VER} Installation completed." !define MUI_FINISHPAGE_TEXT "Click 'Finish' to close this wizard." !insertmacro MUI_PAGE_FINISH ; Language files !insertmacro MUI_LANGUAGE "English" ; MUI end ------ ; now we're ready for the actual installer! Section "Installer" SEC01 ; F4registry check - if no F4 registry, it kicks to the end - the $0 is a variable that holds the -1 (not found) or 0 (found) ; we can also start to determine which install we have ${registry::KeyExists} "HKLM\Software\MicroProse\Falcon\4.0" $0 ${registry::KeyExists} "HKLM\Software\Lead Pursuit\Battlefield Operations\Falcon" $1 ; IntCmp compares the integer in $0 with our number (0) and if it is =, it goes to label "F4Exists" IntCmp $0 0 F4Exists IntCmp $1 0 AFInstall ; this copies the string to $9 - I'm using that as an error label StrCpy $9 "No appropriate registry found." Goto Error F4Exists: ; this determines if there's an AF reg entry, and if so, it only advances 1 line, if not, it goes to F4Install IntCmp $1 0 0 F4Version ; at this point, we need to determine if the person wants to install over "original" F4 or RV Messagebox::show MB_DEFBUTTON3|MB_TOPMOST "" "" "Select where you want the theater installed." "OF/RV" "AF" IDCANCEL ; this places the response 1, 2, or 3, into $0 Pop $0 IntCmp $0 3 Finish IntCmp $0 2 AFInstall F4Version: ; now we know they want OF/RV so we need to differentiate - I'm only going to look for RV, if it isn't RV, I'm going to assume OF for now ; this will also validate the install a bit ; this reads the specified registry value into $0 ReadRegStr $0 HKLM "Software\MicroProse\Falcon\4.0" "basedir" ; if the file exists, go to next line (don't skip a line) if it doesn't, go to label "NoFileFound" IfFileExists $0\theater.lst 0 NoFileFound IfFileExists $0\terrdata\objects\KoreaObj.Dxl RVFound IfFileExists $0\terrdata\objects\KoreaObj.lod OFFound NoFileFound NoFileFound: StrCpy $9 "No F4 install found." Goto Error ; I want to set up a variable to differentiate OF and RV RVFound: ; we'll use this for the lstupdate variable StrCpy $8 "F4" StrCpy $7 "RV" Goto F4Install OFFound: StrCpy $8 "F4" StrCpy $7 "OF" Goto F4Install AFInstall: ; get the AF base directory into a variable ReadRegStr $0 HKLM "Software\Lead Pursuit\Battlefield Operations\Falcon" "basedir" ; sets the right campaign directory StrCpy $1 "$0\campaign\korea" ; campaing\save string StrCpy $2 "$0\Utilities\LxNormalFix.exe $$LP @${TNAME} @$0\Theaters\${TNAME}\terrdata" ; lxnormalfix string StrCpy $8 "AF" Goto CreateDirs F4Install: ReadRegStr $0 HKLM "Software\MicroProse\Falcon\4.0" "basedir" StrCpy $1 "$0\campaign\save" StrCpy $2 "$0\Utilities\LxNormalFix.exe @${TNAME} @$0\Theaters\${TNAME}\terrdata" CreateDirs: SetOutPath $0 CreateDirectory "$0\Theaters\${TNAME}\campaign\vietnam1\" CreateDirectory "$0\Theaters\${TNAME}\campaign\vietnam2\" CreateDirectory "$0\Theaters\${TNAME}\terrdata\weather\" CreateDirectory "$0\Theaters\${TNAME}\objects\" ; new intelligent campaing copy files list CopyFiles /filesonly "$1\*.pri" "$0\Theaters\${TNAME}\campaign\vietnam1\" CopyFiles /filesonly "$1\*.txt" "$0\Theaters\${TNAME}\campaign\vietnam1\" CopyFiles /filesonly "$1\*.b" "$0\Theaters\${TNAME}\campaign\vietnam1\" CopyFiles /filesonly "$1\*.db" "$0\Theaters\${TNAME}\campaign\vietnam1\" CopyFiles /filesonly "$1\*.ia" "$0\Theaters\${TNAME}\campaign\vietnam1\" CopyFiles /filesonly "$1\*.dfs" "$0\Theaters\${TNAME}\campaign\vietnam1\" CopyFiles /filesonly "$1\*.dat" "$0\Theaters\${TNAME}\campaign\vietnam1\" CopyFiles /filesonly "$1\*.gbd" "$0\Theaters\${TNAME}\campaign\vietnam1\" CopyFiles /filesonly "$1\*.lst" "$0\Theaters\${TNAME}\campaign\vietnam1\" CopyFiles /filesonly "$1\default.*" "$0\Theaters\${TNAME}\campaign\vietnam1\" CopyFiles /filesonly "$1\strings.*" "$0\Theaters\${TNAME}\campaign\vietnam1\" CopyFiles /filesonly "$1\atc.ini" "$0\Theaters\${TNAME}\campaign\vietnam1\" CopyFiles /filesonly "$1\falcon4.aii" "$0\Theaters\${TNAME}\campaign\vietnam1\" CopyFiles /filesonly "$1\falcon4.rt" "$0\Theaters\${TNAME}\campaign\vietnam1\" CopyFiles /filesonly "$1\falcon4.tt" "$0\Theaters\${TNAME}\campaign\vietnam1\" CopyFiles /filesonly "$1\validac.*" "$0\Theaters\${TNAME}\campaign\vietnam1\" ; another campaign dir CopyFiles /filesonly "$0\Theaters\${TNAME}\campaign\vietnam1\*.*" "$0\Theaters\${TNAME}\campaign\vietnam2\" ; database objects (same dir for all versions) ; this needs to be recursively because RV has koreaobj subdir. CopyFiles "$0\terrdata\objects\*.*" "$0\Theaters\${TNAME}\objects\" CopyFiles "$0\terrdata\korea\weather\*.*" "$0\Theaters\${TNAME}\terrdata\weather\" ; unpack the theater files ; this is the subdir in the developer's PC that has the files for AF File /r "Theater_src_${TNAME}\*.*" ; common campaign dirs CopyFiles /filesonly "$0\Theaters\${TNAME}\campaign\common\*.*" "$0\Theaters\${TNAME}\campaign\vietnam1\" CopyFiles /filesonly "$0\Theaters\${TNAME}\campaign\common\*.*" "$0\Theaters\${TNAME}\campaign\vietnam2\" ; beginning of falcon version DATABASE & UI ; database copy StrCmp $7 "OF" DataBaseOF StrCmp $7 "RV" DataBaseRV ; UI AF CopyFiles "$0\Theaters\${TNAME}\art\art-AF\*.*" "$0\Theaters\${TNAME}\art\art\" ; database AF CopyFiles "$0\Theaters\${TNAME}\objects\AF\*.*" "$0\Theaters\${TNAME}\objects\" CopyFiles "$0\Theaters\${TNAME}\campaign\common-AF\*.*" "$0\Theaters\${TNAME}\campaign\vietnam1\" CopyFiles "$0\Theaters\${TNAME}\campaign\common-AF\*.*" "$0\Theaters\${TNAME}\campaign\vietnam2\" goto EndOfDataBaseCopy DataBaseOF: ; UI OF ; copy only one \art\ as we have the art1024 dir there too, heh hmm. CopyFiles "$0\Theaters\${TNAME}\art\art-OF\*.*" "$0\Theaters\${TNAME}\art\" ; database OF CopyFiles "$0\Theaters\${TNAME}\objects\OF\*.*" "$0\Theaters\${TNAME}\objects\" CopyFiles "$0\Theaters\${TNAME}\campaign\common-OF\*.*" "$0\Theaters\${TNAME}\campaign\vietnam1\" CopyFiles "$0\Theaters\${TNAME}\campaign\common-OF\*.*" "$0\Theaters\${TNAME}\campaign\vietnam2\" goto EndOfDataBaseCopy DataBaseRV: ; UI RV CopyFiles "$0\Theaters\${TNAME}\art\art-RV\*.*" "$0\Theaters\${TNAME}\art\art\" ; database RV CopyFiles "$0\Theaters\${TNAME}\objects\RV\*.*" "$0\Theaters\${TNAME}\objects\" CopyFiles "$0\Theaters\${TNAME}\campaign\common-RV\*.*" "$0\Theaters\${TNAME}\campaign\vietnam1\" CopyFiles "$0\Theaters\${TNAME}\campaign\common-RV\*.*" "$0\Theaters\${TNAME}\campaign\vietnam2\" EndOfDataBaseCopy: RMDir /r "$0\Theaters\${TNAME}\objects\AF\" RMDir /r "$0\Theaters\${TNAME}\objects\RV\" RMDir /r "$0\Theaters\${TNAME}\objects\OF\" RMDir /r "$0\Theaters\${TNAME}\art\art-AF\" RMDir /r "$0\Theaters\${TNAME}\art\art-RV\" RMDir /r "$0\Theaters\${TNAME}\art\art-OF\" RMDir /r "$0\Theaters\${TNAME}\campaign\common-AF\" RMDir /r "$0\Theaters\${TNAME}\campaign\common-OF\" RMDir /r "$0\Theaters\${TNAME}\campaign\common-RV\" RMDir /r "$0\Theaters\${TNAME}\campaign\common\" ; end of falcon version DATABASE & UI StrCmp $8 "F4" F4curTheater TheaterAdmin StrCpy $9 "Unknown error before curTheater: $8." Goto Error F4curTheater: ; set current theater to new theater ; this reads the specified registry value into $2 for use later ReadRegStr $3 HKLM "Software\MicroProse\Falcon\4.0" "curTheater" ${registry::Write} "hklm\SOFTWARE\MicroProse\Falcon\4.0" "curTheater" "${TNAME}" "REG_SZ" $R0 ; if it is an OF install, skip straight to theater admin otherwise, do RV stuff StrCmp $7 "OF" TheaterAdmin ; this will be the area where we do stuff that it doesn't really matter which install it is TheaterAdmin: SetOutPath $0 ExecWait "$0\Utilities\lstupdate.exe $8 +Theaters\${TNAME}\${TNAME}1.tdf" ExecWait "$0\Utilities\lstupdate.exe $8 +Theaters\${TNAME}\${TNAME}2.tdf" ExecWait '$0\Utilities\SPTinstall.exe -auto "$0\Theaters\${TNAME}\terrdata"' ExecWait "$0\Utilities\LxNormalFix.exe $2" ; just compress textures if F4 then finish StrCmp $8 "AF" Finish ExecWait "$0\Utilities\SeasonSwitcher.exe -0" MessageBox MB_YESNO "Do you want to make the new theater the current one?" IDYES Finish IDNO OldReg OldReg: ${registry::Write} "hklm\SOFTWARE\MicroProse\Falcon\4.0" "curTheater" "$3" "REG_SZ" $R0 Goto Finish Error: Messagebox MB_OK $9 Finish: SectionEnd
Sorry for the long code, but thats the whole script we used to build Vietnam theater installer.
Check out our offical NSIS installer topic.
Notes
Homepage of NSIS.
falcon4/tools/nsis_installer.1184223389.txt.gz · Last modified: 2007/07/12 06:56 (external edit)