SmartAceDesigns.ScriptoFormTemplates.psm1
<#
============================================================================================================================ Module: SmartAceDesigns.ScriptoFormTemplates Author: Smart Ace Designs Notes: This module provides a method for generating a new ScriptoForm project from a custom template using Plaster. ============================================================================================================================ #> function New-SADScriptoFormProject { <# .SYNOPSIS Generates a new ScriptoForm project from a custom template using the Plaster PowerShell module. .DESCRIPTION This function generates a new ScriptoForm project based on a custom template. A ScriptoForm is a PowerShell script that generates and displays a Microsoft WinForms application that can be used for a specific management or system administration task in a computer network environment. Typically, a ScriptoForm is compiled into an executable file which hides the PowerShell console window during execution and provides a more seamless and familiar experience to the user. A ScriptoForm project is the set of files and folders including the PowerShell script, typically stored in a GIT repository, that are used to compile the executable file using the Microsoft .NET CLI utility (dotnet.exe) which is available with any Microsoft .NET SDK. The ScriptoForm project includes a C# file which the compiler will use as the source for the executable, and a C# project file (csproj file) which provides the set of instructions used to compile the executable. A ScriptoForm project consists of the following folder structure: ScriptoFormName |--Build | Build.cs | Build.csproj | .gitignore | README.md | ScriptoFormName.ps1 If Visual Studio Code is installed, it will be opened to the new project directory location after the directory structure is created. .PARAMETER Size Specifies the initial size of the ScriptoForm to generate. The following pre-configred sizes are available to select: Large: Used to build a general purpose ScriptoForm using a large sized form Medium: Used to build a general purpose ScriptoForm using a medium sized form Small: Used to build a general purpose ScriptoForm using a small sized form .PARAMETER DestinationPath Specifies the root destination path of the new ScriptoForm project directory. A directory, using the value of the "Name" parameter, will be created under this path to hold the project files. .PARAMETER Name Specifies the name of PowerShell script as well as the name of the project directory. As a best practice, this name should describe what the script does in VERB-NOUN format. The name will be substituted into various locations within the project such as the Readme.md file and the comments section of the PowerShell script. .PARAMETER Author Specifies the ScriptoForm author that will be substituted into the comments section of the PowerShell script. By default this parameter will be set to the Git user name or the current user login name if Git is not installed. .PARAMETER NoLogo Specifies if the Plaster logo is shown during function execution. .EXAMPLE PS C:\>New-SADScriptoFormProject -Size Small -DestinationPath $HOME\Desktop -Name Show-SmallForm Description ----------- Deploys a new small sized ScriptoForm project named "Show-SmallForm" on the users desktop in the "Show-SmallForm" directory. If Git is installed, the Git user name will be used as the author name in the project. If Git is not installed, the current user login name will be used instead. .EXAMPLE PS C:\>New-SADScriptoFormProject -Size Medium -DestinationPath $HOME\Desktop -Name Show-MediumForm Description ----------- Deploys a new medium sized ScriptoForm project named "Show-MediumForm" on the users desktop in the "Show-MediumForm" directory. If Git is installed, the Git user name will be used as the author name in the project. If Git is not installed, the current user login name will be used instead. .LINK https://github.com/PowerShellOrg/Plaster #> [CmdletBinding()] Param ( [Parameter(Mandatory = $false, Position = 0)] [ValidateSet("Small","Medium","Large")] [string]$Size = "Medium", [Parameter(Mandatory = $false, Position = 1)] [string]$DestinationPath = "$HOME\Desktop", [Parameter(Mandatory = $false, Position = 2)] [string]$Name = "Show-DemoForm", [Parameter(Mandatory = $false, Position = 3)] [string]$Author, [Parameter(Mandatory = $false, Position = 4)] [switch]$NoLogo ) if (!$Author) { if (Get-Command git -ErrorAction SilentlyContinue) { $Author = git config user.name } else {$Author = $env:USERNAME} } switch ($Size) { Small {$FormHeight = 210} Medium {$FormHeight = 260} Large {$FormHeight = 370} } $PlasterParameters = @{ TemplatePath = "$PSScriptRoot\Templates\General" DestinationPath = "$DestinationPath\$Name" Name = $Name Author = $Author NoLogo = $NoLogo FormHeight = $FormHeight } Clear-Host Invoke-Plaster @PlasterParameters Set-Location -Path $DestinationPath\$Name if (Get-Command code -ErrorAction SilentlyContinue) { code $DestinationPath\$Name --profile "Default" } else { powershell_ise.exe "$Name.ps1" } } function Build-SADScriptoFormExecutable { <# .SYNOPSIS Compiles a ScriptoForm project located in the current working directory into a executable file. .DESCRIPTION This function performs the following steps for building a ScriptoForm assembly: - Searches the current working directory for PowerShell script files and digitally signs each one found if an appropriate code-signing certificate is available. - Searches the current working directory for the Microsoft .NET build files and compiles the specified PowerShell script into a executable for each build target specified. - Searches the current working directory for compiled executable files and digitally signs each one found if an appropriate code-signing certificate is available. .PARAMETER BuildTarget Specifies the .NET target framework to use for build the executable(s). Selecting "All" (the default value) will compile the an executable for all available frameworks. .PARAMETER BuildFolder Specifies the path to the Microsoft .NET build files. .PARAMETER CertificateFriendlyName Specifies the friendly name of the code-signing certificate. .PARAMETER CLI Specifies the path to the Microsoft .NET CLI used for compiling the executable(s). .PARAMETER ProjectFileName Specifies the path to the project file used to provide build instructions for the Microsoft .NET CLI. .PARAMETER ReleaseFolder Specifies the path to the target folder for the executable(s). .PARAMETER TimeStampServer Specifies the URL to the time stamp web server used for digitally signing files. .EXAMPLE PS C:\>Build-SADScriptoFormExecutable Description ----------- Invokes the build process using all default parameters and targets all available Microsoft .NET frameworks. .EXAMPLE PS C:\>Build-SADScriptoFormExecutable -CertificateFriendlyName "CodeSigningCert" Description ----------- Invokes the build process using a custom certificate with the friendly name "CodeSigningCert". .EXAMPLE PS C:\>Build-SADScriptoFormExecutable -BuildTarget Legacy Description ----------- Invokes the build process using all default parameters but targets only the legacy .NET 4 framework. #> [CmdletBinding()] Param ( [Parameter(Mandatory = $false, Position = 0)] [ValidateSet("All","Legacy","LTS","STS")] [string]$BuildTarget = "All", [Parameter(Mandatory = $false, Position = 1)] [string]$BuildFolder = "Build", [Parameter(Mandatory = $false, Position = 2)] [string]$CertificateFriendlyName = "PowerShell", [Parameter(Mandatory = $false, Position = 3)] [string]$CLI = "dotnet.exe", [Parameter(Mandatory = $false, Position = 4)] [string]$ProjectFileName = "Build.csproj", [Parameter(Mandatory = $false, Position = 5)] [string]$ReleaseFolder = "Release", [Parameter(Mandatory = $false, Position = 6)] [string]$TimeStampServer = "http://timestamp.digicert.com" ) if (Get-Command -Name $CLI -ErrorAction SilentlyContinue) { $Frameworks = @([PSCustomObject]@{Name = "Legacy";Version = "net48";Enabled=$false}, [PSCustomObject]@{Name = "LTS";Version = "net8.0-windows";Enabled=$false}, [PSCustomObject]@{Name = "STS";Version = "net9.0-windows";Enabled=$false}) if ($BuildTarget -eq "All") { foreach ($Framework in $Frameworks) {$Framework.Enabled = $true} } else { foreach ($Framework in $Frameworks) { if ($Framework.Name -eq $BuildTarget) {$Framework.Enabled = $true} } } } else { Clear-Host Write-Warning -Message "The Microsoft.NET SDK is not installed. Operation has been cancelled." break } Clear-Host if (Test-Path $BuildFolder\$ProjectFileName) { $Message = "Smart Ace Designs" $Width = $Host.UI.RawUI.WindowSize.Width Write-Host Write-Host ((" " * ($Width - $Message.Length)) + $Message) -ForegroundColor Yellow Write-Host ("=" * $Width) $SigningCertificate = Get-ChildItem Cert:\CurrentUser\My | Where-Object FriendlyName -eq $CertificateFriendlyName if ($SigningCertificate) { Write-Host "Signing script files" -ForegroundColor Yellow foreach ($ScriptFile in (Get-ChildItem -Filter "*.ps1" | Select-Object FullName -ExpandProperty FullName)) { [void](Set-AuthenticodeSignature -FilePath $ScriptFile -Certificate $SigningCertificate -TimestampServer $TimeStampServer -HashAlgorithm SHA256) } } Push-Location Set-Location $BuildFolder foreach ($Framework in $Frameworks) { if ($Framework.Enabled) { Write-Host "Building .NET $($Framework.Name) Assembly" -ForegroundColor Yellow (Start-Process -FilePath $CLI -ArgumentList "publish -f $($Framework.Version) -v q -nologo -o ..\$ReleaseFolder\$($Framework.Name)" -PassThru -WindowStyle Hidden).WaitForExit() (Start-Process -FilePath $CLI -ArgumentList "clean -f $($Framework.Version) -v q -nologo" -PassThru -WindowStyle Hidden).WaitForExit() } } Pop-Location if ($SigningCertificate) { if (Test-Path $ReleaseFolder) { Write-Host "Signing executable files" -ForegroundColor Yellow foreach ($File in (Get-ChildItem -Path $ReleaseFolder -Filter "*.exe" -Recurse | Select-Object FullName -ExpandProperty FullName)) { [void](Set-AuthenticodeSignature -FilePath $File -Certificate $SigningCertificate -TimestampServer $TimeStampServer -HashAlgorithm SHA256) } } } } else { Write-Warning -Message "The build files could could not be found. Operation has been cancelled." break } Write-Host Write-Host "Process complete" -ForegroundColor Yellow } |