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 " " 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 } |