PowerShellUniversal.Apps.Tools.psm1

function New-UDCenter {
    <#
    .SYNOPSIS
    Center items within a dashboard.
 
    .DESCRIPTION
    Center items within a dashboard.
 
    .PARAMETER Content
    The items to center.
 
    .EXAMPLE
    New-UDCenter -Content {
        New-UDTypography -Text 'Loading groups' -Variant h5
        New-UDProgress -Circular
    }
    #>

    param([ScriptBlock]$Content)

    New-UDElement -tag div -Content $Content -Attributes @{
        style = @{
            textAlign = 'center'
            width     = '100%'
        }
    }
}

function New-UDRight {
    <#
    .SYNOPSIS
    Pull items to the right
 
    .DESCRIPTION
    Pull items to the right
 
    .PARAMETER Content
    The content to move to the right.
    #>

    [CmdletBinding()]
    param([ScriptBlock]$Content)

    New-UDElement -Tag 'div' -Content $Content -Attributes @{
        style = @{
            "display"         = "flex"
            "justify-content" = "flex-end"
            "margin-left"     = "auto"
            "margin-right"    = "0"
            "align-items"     = "flex-end"
        }
    }
}

function New-UDConfirm {
    <#
    .SYNOPSIS
    Creates a confirmation dialog in Universal Dashboard.
 
    .DESCRIPTION
    The New-UDConfirm function creates a confirmation dialog in Universal Dashboard. It displays a modal dialog with a specified text and provides "Yes" and "No" buttons for the user to confirm or cancel the action.
 
    .PARAMETER Text
    Specifies the text to be displayed in the confirmation dialog. The default value is "Are you sure?".
 
    .EXAMPLE
    PS> if(New-UDConfirm -Text "Do you want to delete this item?") {
        # Delete the item
    }
 
    This example creates a confirmation dialog with the specified text "Do you want to delete this item?".
 
    .OUTPUTS
    System.Boolean
    Returns $true if the user clicks "Yes" and $false if the user clicks "No".
 
    #>

    [CmdletBinding()]
    param(
        [Parameter(Mandatory = $false,
            ValueFromPipeline = $true,
            ValueFromPipelineByPropertyName = $true)]
        [string]$Text = 'Are you sure?'
    )
    $Page:Result = $null
    Show-UDModal -Content {
        New-UDTypography -Text $Text
    } -Footer {
        New-UDButton -Text "Yes" -OnClick {
            $Page:Result = $true
            Hide-UDModal
        } -Style @{"border-radius" = "4px" }

        New-UDButton -Text "No" -OnClick {
            $Page:Result = $false
            Hide-UDModal
        } -Style @{"border-radius" = "4px" }

    } -Persistent
    while ($null -eq $Page:Result) {
        Start-Sleep -Milliseconds 100
    }
    return $Page:Result
}

function New-UDLineBreak {
    <#
    .SYNOPSIS
    Creates a line break element for Universal Dashboard.
 
    .DESCRIPTION
    The New-UDLineBreak function creates a line break element for Universal Dashboard. It can be used to add vertical spacing between elements.
 
    .PARAMETER Height
    Specifies the height of the line break element. The default value is "20px".
 
    .PARAMETER Fixed
    Indicates whether the height of the line break element is fixed using br. If this switch is specified, the height parameter is ignored.
 
    .EXAMPLE
    New-UDLineBreak
    Creates a line break element with a default height of 20 pixels.
 
    .EXAMPLE
    New-UDLineBreak -Height "30px"
    Creates a line break element with a height of 30 pixels.
 
    .EXAMPLE
    New-UDLineBreak -Fixed
    Creates a line break element with a fixed height.
 
    #>

    [CmdletBinding(DefaultParameterSetName = 'Variable Height')]
    param(
        [Parameter(ParameterSetName = 'Variable Height')]
        [string]$Height = "20px",
        [Parameter(ParameterSetName = 'Fixed Height')]
        [switch]$Fixed
    )
    Process {
        if ($Fixed) {
            New-UDElement -Tag 'div' -Content {
                New-UDElement -Tag 'br'
            }
        }
        else {
            New-UDElement -Tag 'div' -Content {} -Attributes @{
                style = @{
                    height = $Height
                }
            }
        }
    }
}


function Show-UDEventData {
    <#
    .SYNOPSIS
    Displays the event data.
 
    .DESCRIPTION
    The Show-UDEventData function is used to display the event data. It takes an optional parameter, $depth, which specifies the depth of the object to display.
 
    .PARAMETER depth
    Specifies the depth of the object to display. The default value is 2.
 
    .EXAMPLE
    Show-UDEventData -depth 3
    Displays the event data with a depth of 3.
 
    .INPUTS
    None.
 
    .OUTPUTS
    None.
    #>

    param($depth = 2)
    Show-UDObject -InputObject $EventData -depth $depth
}

function Reset-UDPage {
    <#
    .SYNOPSIS
    Reloads the current page.
 
    .DESCRIPTION
    Reloads the current page. This uses JavaScript directly.
    #>

    Invoke-UDJavaScript "window.location.reload()"
}

function Show-UDObject {
    <#
    .SYNOPSIS
    Displays an object in a modal dialog with syntax highlighting and copy functionality.
 
    .DESCRIPTION
    The Show-UDObject function displays an object in a modal dialog with syntax highlighting and copy functionality. It takes an input object and optional depth parameter to control the depth of the object's properties that will be displayed.
 
    .PARAMETER InputObject
    The input object to be displayed in the modal dialog.
 
    .PARAMETER Depth
    The maximum depth of the object's properties that will be displayed. Default value is 20.
 
    .EXAMPLE
    $myObject = Get-MyObject
    Show-UDObject -InputObject $myObject -Depth 10
 
    This example retrieves an object using the Get-MyObject function and displays it in a modal dialog with a maximum depth of 10.
    #>

    param(
        [Parameter(ValueFromPipeline, Mandatory)]
        $InputObject,
        $depth = 20
    )

    process {
        Show-UDModal -Header {
            New-UDTypography -Text $($inputObject.gettype()) -Variant h4
        } -Content {
            New-UDDynamic -LoadingComponent { New-UDProgress } -Content {
                New-UDSyntaxHighlighter -Code (ConvertTo-Json -InputObject $inputObject -Depth $depth) -Language json
            }
            New-UDButton -Text "Copy" -Icon Copy -OnClick { Set-UDClipboard -Data (ConvertTo-Json -InputObject $inputObject -Depth $depth) }
        }
    }
}

function Get-UDCache {
    <#
    .SYNOPSIS
    Returns all items in the $Cache: scope.
 
    .DESCRIPTION
    Returns all items in the $Cache: scope.
    #>

    [UniversalDashboard.Execution.GlobalCachedVariableProvider]::Cache
}

function Show-UDVariable {
    <#
    .SYNOPSIS
    Shows variables and their values in a modal.
 
    .DESCRIPTION
    Shows variables and their values in a modal.
 
    .PARAMETER Name
    A name. If not specified, all variables are returned.
 
    .EXAMPLE
    Show-UDVariable -Name 'EventData'
    #>

    param($Name)

    Show-UDModal -Content {
        New-UDDynamic -Content {
            $Variables = Get-Variable -Name "*$Name"

            New-UDTable -Title 'Variables' -Icon (New-UDIcon -Icon 'SquareRootVariable') -Data $Variables -Columns @(
                New-UDTableColumn -Property Name -ShowFilter
                New-UDTableColumn -Property Value -Render {
                    [string]$EventData.Value
                } -ShowFilter
            ) -ShowPagination -ShowFilter
        } -LoadingComponent {
            New-UDSkeleton
        }

    } -Footer {
        New-UDButton -Text 'Close' -OnClick {
            Hide-UDModal
        }
    } -FullScreen
}

function Show-UDThemeColorViewer {
    <#
    .SYNOPSIS
    Displays a theme color viewer for Universal Dashboard.
 
    .DESCRIPTION
    The Show-UDThemeColorViewer function displays a modal window that allows users to view and set different themes and their corresponding colors for Universal Dashboard. It provides options to set the default theme to either light or dark mode.
 
    .PARAMETER Filter
    Specifies an array of theme names to filter and display in the theme color viewer. Only the specified themes will be shown. If not specified, all available themes will be displayed.
 
    .PARAMETER defaultTheme
    Specifies the default theme to be set when the "Set Default - Light" or "Set Default - Dark" buttons are clicked. The default value is 'MaterialDesign'.
 
    .EXAMPLE
    Show-UDThemeColorViewer -Filter 'MaterialDesign', 'Desert', 'DoomOne' -defaultTheme 'Desert'
    Displays the theme color viewer with the specified themes filtered and sets the default theme to 'Desert'.
 
    .EXAMPLE
    Show-UDThemeColorViewer
    Displays the theme color viewer with all available themes and sets the default theme to 'MaterialDesign'.
    #>

    [cmdletbinding()]
    param(
        [string[]]$Filter = $null,
        $defaultTheme = 'MaterialDesign'
    )
    Show-UDModal -Header {
        New-UDStack -Direction row -Children {
            New-UDTypography -Variant h3 -Content {
                New-UDIcon -Icon  Images
                " Themes"
            }
            New-UDHtml -Markup "&nbsp;&nbsp;"
            New-UDButton -Text "Set Default - Light" -OnClick {
                Set-UDTheme -Theme (Get-UDTheme -Name $defaultTheme)
            }
            New-UDButton -Text "Set Default - Dark" -Color secondary -OnClick {
                Set-UDTheme -Theme (Get-UDTheme -Name $defaultTheme) -Variant dark
            }
        }
    } -Content {
        New-UDDynamic -Content {
            New-UDRow -Columns {
                $themes = @()
                if ($filter) {
                    $themes += 'MaterialDesign'
                    $themes += $filter | Sort-Object
                }
                else {
                    $themes = Get-UDTheme
                    $themes += 'MaterialDesign'
                    $themes = $themes | Where-Object { $_ -ne 'Cobalt Neon' -and $_ -ne 'duckbones' -and $_ -ne 'HaX0R_BLUE' -and $_ -ne 'HaX0R_GR33N' -and $_ -ne 'HaX0R_R3D' -and $_ -ne 'Retro' -and $_ -ne 'VibrantInk' -and $_ -ne 'C64' -and $_ -ne 'QB64 Super Dark Blue' -and $_ -ne 'kanagawabones' -and $_ -ne 'Darkside' -and $_ -ne 'Broadcast' -and $_ -ne 'Atom' -and $_ -ne 'primary' }  | Sort-Object
                }
                $themes | ForEach-Object {
                    New-UDColumn -SmallSize 4 -Content {
                        try {
                            $themeName = $_
                            $Theme = Get-UDTheme -Name $themeName -ErrorAction SilentlyContinue

                            New-UDStack -Direction row -Content {

                                New-UDCard -Title "$_ - Light" -Content {
                                    New-UDButton -Text "Set $_" -OnClick {
                                        Set-UDTheme -Theme $Theme -Variant light
                                        Write-Host ($Theme | ConvertTo-Json -Depth 20)
                                    }
                                    New-UDButton -Text "Set $_" -Color secondary -OnClick {
                                        Set-UDTheme -Theme $Theme -Variant light
                                        Write-Host ($Theme | ConvertTo-Json -Depth 20)
                                    }
                                    New-UDElement -Content {
                                        "Background"
                                    } -Attributes @{
                                        style = @{
                                            color           = $Theme.light.palette.text.primary
                                            backgroundColor = $Theme.light.overrides.MuiDrawer.paper.backgroundColor
                                        }
                                    }  -Tag 'div'

                                    New-UDElement -Content {
                                        "Primary Button"
                                    } -Attributes @{
                                        style = @{
                                            color           = $Theme.light.overrides.MuiButton.contained.color
                                            backgroundColor = $Theme.light.palette.primary.main
                                        }
                                    }  -Tag 'div'

                                    New-UDElement -Content {
                                        "Secondary Button"
                                    } -Attributes @{
                                        style = @{
                                            color           = $Theme.light.overrides.MuiButton.contained.color
                                            backgroundColor = $Theme.light.palette.secondary.main
                                        }
                                    } -Tag 'div'
                                }

                                New-UDCard -Title "$_ - Dark" -Content {
                                    New-UDButton -Text "Set $_" -OnClick {
                                        Set-UDTheme -Theme $Theme -Variant dark
                                        Write-Host ($Theme | ConvertTo-Json -Depth 20)
                                    }

                                    New-UDButton -Text "Set $_" -Color secondary -OnClick {
                                        Set-UDTheme -Theme $Theme -Variant dark
                                        Write-Host ($Theme | ConvertTo-Json -Depth 20)
                                    }

                                    New-UDElement -Content {
                                        "Primary Background"
                                    } -Attributes @{
                                        style = @{
                                            color           = $Theme.dark.palette.text.primary
                                            backgroundColor = $Theme.dark.overrides.MuiDrawer.paper.backgroundColor
                                        }
                                    }  -Tag 'div'

                                    New-UDElement -Content {
                                        "Primary Button"
                                    } -Attributes @{
                                        style = @{
                                            color           = $Theme.dark.overrides.MuiButton.contained.color
                                            backgroundColor = $Theme.dark.palette.primary.main
                                        }
                                    }  -Tag 'div'

                                    New-UDElement -Content {
                                        "Secondary Button"
                                    } -Attributes @{
                                        style = @{
                                            color           = $Theme.dark.overrides.MuiButton.contained.color
                                            backgroundColor = $Theme.dark.palette.secondary.main
                                        }
                                    } -Tag 'div'
                                }

                            }

                        }
                        catch {
                            Write-Host $themeName
                            Write-Host $_
                        }
                    }
                }
            }
        } -LoadingComponent {
            New-UDProgress
        }

    } -FullWidth -MaxWidth xl
}

Function ConvertTo-UDJson {
    <#
    .SYNOPSIS
    Converts PowerShell objects to JSON format.
 
    .DESCRIPTION
    The ConvertTo-UDJson function converts PowerShell objects to JSON format. It supports various types of objects, including enums, DateTime, DateTimeOffset, Type, strings, switches, value types, collections (IList and IDictionary), and custom objects.
 
    .PARAMETER InputObject
    Specifies the input object to be converted to JSON. This parameter is mandatory and accepts pipeline input.
 
    .PARAMETER Depth
    Specifies the depth of the conversion. The default value is 2. A depth of 0 or less will only return the type name of the object.
 
    .OUTPUTS
    The function outputs the converted JSON representation of the input object.
 
    .EXAMPLE
    Convert-OutputObject -InputObject $object -Depth 2
    Converts the $object to JSON format with a depth of 2.
 
    .NOTES
    This function is part of the Apps.PowerShell.Tools module.
    #>

    [CmdletBinding()]
    param (
        [Parameter(Mandatory = $true, ValueFromPipeline = $true)]
        [AllowNull()]
        [object]
        $InputObject,

        [Parameter(Mandatory = $true)]
        [int]
        $Depth = 2
    )

    begin {
        $childDepth = $Depth - 1

        $isType = {
            [CmdletBinding()]
            param (
                [Object]
                $InputObject,

                [Type]
                $Type
            )

            if ($InputObject -is $Type) {
                return $true
            }

            $psTypes = @($InputObject.PSTypeNames | ForEach-Object -Process {
                    $_ -replace '^Deserialized.'
                })

            $Type.FullName -in $psTypes
        }
    }

    process {
        if ($null -eq $InputObject) {
            $null
        }
        elseif ((&$isType -InputObject $InputObject -Type ([Enum])) -and $Depth -ge 0) {
            # ToString() gives the human readable value but I thought it better to give some more context behind
            # these types.
            @{
                Type   = ($InputObject.PSTypeNames[0] -replace '^Deserialized.')
                String = $InputObject.ToString()
                Value  = [int]$InputObject
            }
        }
        elseif ($InputObject -is [DateTime]) {
            # The offset is based on the Kind value
            # Unspecified leaves it off
            # UTC set it to Z
            # Local sets it to the local timezone
            $InputObject.ToString('o')
        }
        elseif (&$isType -InputObject $InputObject -Type ([DateTimeOffset])) {
            # If this is a deserialized object (from an executable) we need recreate a live DateTimeOffset
            if ($InputObject -isnot [DateTimeOffset]) {
                $InputObject = New-Object -TypeName DateTimeOffset $InputObject.DateTime, $InputObject.Offset
            }
            $InputObject.ToString('o')
        }
        elseif (&$isType -InputObject $InputObject -Type ([Type])) {
            if ($Depth -lt 0) {
                $InputObject.FullName
            }
            else {
                # This type is very complex with circular properties, only return somewhat useful properties.
                # BaseType might be a string (serialized output), try and convert it back to a Type if possible.
                $baseType = $InputObject.BaseType -as [Type]
                if ($baseType) {
                    $baseType = Convert-OutputObject -InputObject $baseType -Depth $childDepth
                }

                @{
                    Name                  = $InputObject.Name
                    FullName              = $InputObject.FullName
                    AssemblyQualifiedName = $InputObject.AssemblyQualifiedName
                    BaseType              = $baseType
                }
            }
        }
        elseif ($InputObject -is [string]) {
            $InputObject
        }
        elseif (&$isType -InputObject $InputObject -Type ([switch])) {
            $InputObject.IsPresent
        }
        elseif ($InputObject.GetType().IsValueType) {
            # We want to display just this value and not any properties it has (if any).
            $InputObject
        }
        elseif ($Depth -lt 0) {
            # This must occur after the above to ensure ints and other ValueTypes are preserved as is.
            [string]$InputObject
        }
        elseif ($InputObject -is [Collections.IList]) {
            , @(foreach ($obj in $InputObject) {
                    Convert-OutputObject -InputObject $obj -Depth $childDepth
                })
        }
        elseif ($InputObject -is [Collections.IDictionary]) {
            $newObj = @{}

            # Replicate ConvertTo-Json, props are replaced by keys if they share the same name. We only want ETS
            # properties as well.
            foreach ($prop in $InputObject.PSObject.Properties) {
                if ($prop.MemberType -notin @('AliasProperty', 'ScriptProperty', 'NoteProperty')) {
                    continue
                }
                $newObj[$prop.Name] = Convert-OutputObject -InputObject $prop.Value -Depth $childDepth
            }
            foreach ($kvp in $InputObject.GetEnumerator()) {
                $newObj[$kvp.Key] = Convert-OutputObject -InputObject $kvp.Value -Depth $childDepth
            }
            $newObj
        }
        else {
            $newObj = @{}
            foreach ($prop in $InputObject.PSObject.Properties) {
                $newObj[$prop.Name] = Convert-OutputObject -InputObject $prop.Value -Depth $childDepth
            }
            $newObj
        }
    }
}

function New-UDReactComponent {
    <#
    .SYNOPSIS
    Creates a new PowerShell Universal React component.
     
    .DESCRIPTION
    The New-UDReactComponent function creates a new PowerShell Universal React component. It creates a new directory with the specified name and copies the React template files to the directory. The function also creates a new module manifest and module script file for the component.
     
    .PARAMETER Name
    The name of the component.
     
    .PARAMETER Path
    The path to create the component.
     
    .EXAMPLE
    New-UDReactComponent -Name "react-icons" -Path "$PSScriptRoot\project"
    #>

    param(
        [Parameter(Mandatory = $true)]
        [string]$Name,
        [Parameter(Mandatory = $true)]
        [string]$Path
    )

    $NPM = Get-Command "npm" -ErrorAction SilentlyContinue 
    if (-not $NPM) {
        throw "NodeJS is not installed. Please install npm to use New-UDReactComponent."
    }

    if ((Test-Path $Path)) {
        throw "Path $Path already exists."
    }

    New-Item -ItemType Directory -Path $Path | Out-Null
    Copy-Item "$PSScriptRoot\ReactTemplate\*.*" -Destination $Path -Recurse

    $TemplateFiles = @("package.json", "index.js", "webpack.config.js")

    foreach ($TemplateFile in $TemplateFiles) {
        $Content = Get-Content "$Path\$TemplateFile"
        $Content = $Content -replace '{COMPONENT_NAME}', $Name
        Set-Content -Path "$Path\$TemplateFile" -Value $Content
    }

    New-ModuleManifest -Path "$Path\$Name.psd1" -RootModule "$Name.psm1"
    $ModuleContent = {
        $IndexJs = Get-ChildItem "$PSScriptRoot\index.*.bundle.js"
        $AssetId = [UniversalDashboard.Services.AssetService]::Instance.RegisterAsset($IndexJs.FullName)
        function New-UDComponent {
            param(
                [string]$Id = (New-Guid).ToString()
            )
            @{
                assetId  = $AssetId 
                isPlugin = $true 
                type     = "{COMPONENT_NAME}"
                id       = $Id
            }
        }
    }.ToString()

    $ModuleContent = $ModuleContent -replace '{COMPONENT_NAME}', $Name
    Set-Content -Path "$Path\$Name.psm1" -Value $ModuleContent
}

function Add-UDReactComponentLibrary {
    <#
    .SYNOPSIS
    Adds a NPM library to a PowerShell Universal React component.
     
    .DESCRIPTION
    The Add-UDReactComponentLibrary function adds a NPM library to a PowerShell Universal React component. It installs the specified library and saves it to the package.json file.
     
    .PARAMETER Path
    The path to the component.
     
    .PARAMETER Library
    The name of the library to install.
     
    .PARAMETER Version
    The version of the library to install.
     
    .EXAMPLE
    Add-UDReactComponentLibrary -Path "$PSScriptRoot\project" -Library "react-icons"
    #>

    param(
        [Parameter(Mandatory = $true)]
        [string]$Path, 
        [Parameter(Mandatory = $true)]
        [string]$Library, 
        [Parameter()]
        [string]$Version)

    $NPM = Get-Command "npm" -ErrorAction SilentlyContinue
    if (-not $NPM) {
        throw "NodeJS is not installed. Please install npm to use Add-UDReactComponentLibrary."
    }

    if (-not (Test-Path $Path)) {
        throw "Path $Path does not exist."
    }

    Push-Location $Path

    try {
        if ($Version) {
            & $NPM i $Library@$Version --save
        }
        else {
            & $NPM i $Library --save
        }
    }
    finally {
        Pop-Location
    }
}

function Invoke-UDReactComponentBuild {
    <#
    .SYNOPSIS
    Builds a PowerShell Universal React component.
     
    .DESCRIPTION
    The Invoke-UDReactComponentBuild function builds a PowerShell Universal React component. It installs the required NPM libraries, fixes any audit issues, and builds the component. The function copies the build output to the specified output path.
     
    .PARAMETER Path
    The path to the component project.
     
    .PARAMETER OutputPath
    The path to copy the build output to.
     
    .PARAMETER Force
    Forces the build output to be copied to the output path.
     
    .EXAMPLE
    Invoke-UDReactComponentBuild -Path "$PSScriptRoot\project" -OutputPath "$PSScriptRoot\output" -Force
    #>

    param(
        [Parameter(Mandatory = $true)]
        [string]$Path, 
        [Parameter(Mandatory = $true)]
        [string]$OutputPath, 
        [Parameter()]
        [Switch]$Force)

    $NPM = Get-Command "npm" -ErrorAction SilentlyContinue
    if (-not $NPM) {
        throw "NodeJS is not installed. Please install npm to use Invoke-UDReactComponentBuild."
    }

    if (-not (Test-Path $Path)) {
        throw "Path $Path does not exist."
    }

    if ((Test-Path $OutputPath)) {
        if ($Force) {
            Remove-Item $OutputPath -Recurse
        }
        else {
            throw "Path $OutputPath already exists."
        }
    }

    Push-Location $Path

    Remove-Item "$Path\public" -Recurse -ErrorAction SilentlyContinue

    try {
        & $NPM i --legacy-peer-deps
        & $NPM audit fix --force
        & $NPM run build 
    }
    finally {
        Pop-Location
    }

    Copy-Item "$Path\public" -Destination $OutputPath -Recurse
    Copy-Item -Path "$Path\*.psd1" -Destination $OutputPath
    Copy-Item -Path "$Path\*.psm1" -Destination $OutputPath
}