-
Notifications
You must be signed in to change notification settings - Fork 5
/
PoSHPF.ps1
269 lines (242 loc) · 10.9 KB
/
PoSHPF.ps1
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
# PoSHPF - Version 1.2
# Grab all resources (MahApps, etc), all XAML files, and any potential static resources
$Global:resources = Get-ChildItem -Path "$PSScriptRoot\Resources\*.dll" -ErrorAction SilentlyContinue
$Global:XAML = Get-ChildItem -Path "$PSScriptRoot\XAML\*.xaml" -ErrorAction SilentlyContinue
$Global:MediaResources = Get-ChildItem -Path "$PSScriptRoot\Media" -ErrorAction SilentlyContinue
# This class allows the synchronized hashtable to be available across threads,
# but also passes a couple of methods along with it to do GUI things via the
# object's dispatcher.
class SyncClass
{
#Hashtable containing all forms/windows and controls - automatically created when newing up
[hashtable]$SyncHash = [hashtable]::Synchronized(@{})
# method to close the window - pass window name
[void]CloseWindow($windowName){
$this.SyncHash.$windowName.Dispatcher.Invoke([action]{$this.SyncHash.$windowName.Close()},"Normal")
}
# method to update GUI - pass object name, property and value
[void]UpdateElement($object,$property,$value){
$this.SyncHash.$object.Dispatcher.Invoke([action]{ $this.SyncHash.$object.$property = $value },"Normal")
}
}
$Global:SyncClass = [SyncClass]::new() # create a new instance of this SyncClass to use.
###################
## Import Resources
###################
# Load WPF Assembly
Add-Type -assemblyName PresentationFramework
# Load Resources
foreach($dll in $resources) { [System.Reflection.Assembly]::LoadFrom("$($dll.FullName)") | out-null }
##############
## Import XAML
##############
$xp = '[^a-zA-Z_0-9]' # All characters that are not a-Z, 0-9, or _
$vx = @() # An array of XAML files loaded
foreach($x in $XAML) {
# Items from XAML that are known to cause issues
# when PowerShell parses them.
$xamlToRemove = @(
'mc:Ignorable="d"',
"x:Class=`"(.*?)`"",
"xmlns:local=`"(.*?)`""
)
$xaml = Get-Content $x.FullName # Load XAML
$xaml = $xaml -replace "x:N",'N' # Rename x:Name to just Name (for consumption in variables later)
foreach($xtr in $xamlToRemove){ $xaml = $xaml -replace $xtr } # Remove items from $xamlToRemove
# Create a new variable to store the XAML as XML
New-Variable -Name "xaml$(($x.BaseName) -replace $xp, '_')" -Value ($xaml -as [xml]) -Force
# Add XAML to list of XAML documents processed
$vx += "$(($x.BaseName) -replace $xp, '_')"
}
#######################
## Add Media Resources
#######################
$imageFileTypes = @(".jpg",".bmp",".gif",".tif",".png") # Supported image filetypes
$avFileTypes = @(".mp3",".wav",".wmv") # Supported audio/visual filetypes
$xp = '[^a-zA-Z_0-9]' # All characters that are not a-Z, 0-9, or _
if($MediaResources.Count -gt 0){
## Okay... the following code is just silly. I know
## but hear me out. Adding the nodes to the elements
## directly caused big issues - mainly surrounding the
## "x:" namespace identifiers. This is a hacky fix but
## it does the trick.
foreach($v in $vx)
{
$xml = ((Get-Variable -Name "xaml$($v)").Value) # Load the XML
# add the resources needed for strings
$xml.DocumentElement.SetAttribute("xmlns:sys","clr-namespace:System;assembly=System")
# if the document doesn't already have a "Window.Resources" create it
if($null -eq ($xml.DocumentElement.'Window.Resources')){
$fragment = "<Window.Resources>"
$fragment += "<ResourceDictionary>"
}
# Add each StaticResource with the key of the base name and source to the full name
foreach($sr in $MediaResources)
{
$srname = "$($sr.BaseName -replace $xp, '_')$($sr.Extension.Substring(1).ToUpper())" #convert name to basename + Uppercase Extension
if($sr.Extension -in $imageFileTypes){ $fragment += "<BitmapImage x:Key=`"$srname`" UriSource=`"$($sr.FullName)`" />" }
if($sr.Extension -in $avFileTypes){
$uri = [System.Uri]::new($sr.FullName)
$fragment += "<sys:Uri x:Key=`"$srname`">$uri</sys:Uri>"
}
}
# if the document doesn't already have a "Window.Resources" close it
if($null -eq ($xml.DocumentElement.'Window.Resources'))
{
$fragment += "</ResourceDictionary>"
$fragment += "</Window.Resources>"
$xml.DocumentElement.InnerXml = $fragment + $xml.DocumentElement.InnerXml
}
# otherwise just add the fragment to the existing resource dictionary
else
{
$xml.DocumentElement.'Window.Resources'.ResourceDictionary.InnerXml += $fragment
}
# Reset the value of the variable
(Get-Variable -Name "xaml$($v)").Value = $xml
}
}
#################
## Create "Forms"
#################
$forms = @()
foreach($x in $vx)
{
$Reader = (New-Object System.Xml.XmlNodeReader ((Get-Variable -Name "xaml$($x)").Value)) #load the xaml we created earlier into XmlNodeReader
New-Variable -Name "form$($x)" -Value ([Windows.Markup.XamlReader]::Load($Reader)) -Force #load the xaml into XamlReader
$forms += "form$($x)" #add the form name to our array
$SyncClass.SyncHash.Add("form$($x)", (Get-Variable -Name "form$($x)").Value) #add the form object to our synched hashtable
}
#################################
## Create Controls (Buttons, etc)
#################################
$controls = @()
$xp = '[^a-zA-Z_0-9]' # All characters that are not a-Z, 0-9, or _
foreach($x in $vx)
{
$xaml = (Get-Variable -Name "xaml$($x)").Value #load the xaml we created earlier
$xaml.SelectNodes("//*[@Name]") | %{ #find all nodes with a "Name" attribute
$cname = "form$($x)Control$(($_.Name -replace $xp, '_'))"
Set-Variable -Name "$cname" -Value $SyncClass.SyncHash."form$($x)".FindName($_.Name) #create a variale to hold the control/object
$controls += (Get-Variable -Name "form$($x)Control$($_.Name)").Name #add the control name to our array
$SyncClass.SyncHash.Add($cname, $SyncClass.SyncHash."form$($x)".FindName($_.Name)) #add the control directly to the hashtable
}
}
############################
## FORMS AND CONTROLS OUTPUT
############################
Write-Host -ForegroundColor Cyan "The following forms were created:"
$forms | %{ Write-Host -ForegroundColor Yellow " `$$_"} #output all forms to screen
if($controls.Count -gt 0){
Write-Host ""
Write-Host -ForegroundColor Cyan "The following controls were created:"
$controls | %{ Write-Host -ForegroundColor Yellow " `$$_"} #output all named controls to screen
}
#######################
## DISABLE A/V AUTOPLAY
#######################
foreach($x in $vx)
{
$carray = @()
$fts = $syncClass.SyncHash."form$($x)"
foreach($c in $fts.Content.Children)
{
if($c.GetType().Name -eq "MediaElement") #find all controls with the type MediaElement
{
$c.LoadedBehavior = "Manual" #Don't autoplay
$c.UnloadedBehavior = "Stop" #When the window closes, stop the music
$carray += $c #add the control to an array
}
}
if($carray.Count -gt 0)
{
New-Variable -Name "form$($x)PoSHPFCleanupAudio" -Value $carray -Force # Store the controls in an array to be accessed later
$syncClass.SyncHash."form$($x)".Add_Closed({
foreach($c in (Get-Variable "form$($x)PoSHPFCleanupAudio").Value)
{
$c.Source = $null #stops any currently playing media
}
})
}
}
#####################
## RUNSPACE FUNCTIONS
#####################
## Yo dawg... Runspace to clean up Runspaces
## Thank you Boe Prox / Stephen Owen
#region RSCleanup
$Script:JobCleanup = [hashtable]::Synchronized(@{})
$Script:Jobs = [system.collections.arraylist]::Synchronized((New-Object System.Collections.ArrayList)) #hashtable to store all these runspaces
$jobCleanup.Flag = $True #cleanup jobs
$newRunspace =[runspacefactory]::CreateRunspace() #create a new runspace for this job to cleanup jobs to live
$newRunspace.ApartmentState = "STA"
$newRunspace.ThreadOptions = "ReuseThread"
$newRunspace.Open()
$newRunspace.SessionStateProxy.SetVariable("jobCleanup",$jobCleanup) #pass the jobCleanup variable to the runspace
$newRunspace.SessionStateProxy.SetVariable("jobs",$jobs) #pass the jobs variable to the runspace
$jobCleanup.PowerShell = [PowerShell]::Create().AddScript({
#Routine to handle completed runspaces
Do {
Foreach($runspace in $jobs) {
If ($runspace.Runspace.isCompleted) { #if runspace is complete
[void]$runspace.powershell.EndInvoke($runspace.Runspace) #then end the script
$runspace.powershell.dispose() #dispose of the memory
$runspace.Runspace = $null #additional garbage collection
$runspace.powershell = $null #additional garbage collection
}
}
#Clean out unused runspace jobs
$temphash = $jobs.clone()
$temphash | Where {
$_.runspace -eq $Null
} | ForEach {
$jobs.remove($_)
}
Start-Sleep -Seconds 1 #lets not kill the processor here
} while ($jobCleanup.Flag)
})
$jobCleanup.PowerShell.Runspace = $newRunspace
$jobCleanup.Thread = $jobCleanup.PowerShell.BeginInvoke()
#endregion RSCleanup
#This function creates a new runspace for a script block to execute
#so that you can do your long running tasks not in the UI thread.
#Also the SyncClass is passed to this runspace so you can do UI
#updates from this thread as well.
function Start-BackgroundScriptBlock($scriptBlock){
$newRunspace =[runspacefactory]::CreateRunspace()
$newRunspace.ApartmentState = "STA"
$newRunspace.ThreadOptions = "ReuseThread"
$newRunspace.Open()
$newRunspace.SessionStateProxy.SetVariable("SyncClass",$SyncClass)
$PowerShell = [PowerShell]::Create().AddScript($scriptBlock)
$PowerShell.Runspace = $newRunspace
#Add it to the job list so that we can make sure it is cleaned up
[void]$Jobs.Add(
[pscustomobject]@{
PowerShell = $PowerShell
Runspace = $PowerShell.BeginInvoke()
}
)
}
########################
## WIRE UP YOUR CONTROLS
########################
# simple example: $formMainWindowControlButton.Add_Click({ your code })
#
# example with BackgroundScriptBlock and UpdateElement
# $formmainControlButton.Add_Click({
# $sb = {
# $SyncClass.UpdateElement("formmainControlProgress","Value",25)
# }
# Start-BackgroundScriptBlock $sb
# })
############################
###### DISPLAY DIALOG ######
############################
#[void]$formMainWindow.ShowDialog()
##########################
##### SCRIPT CLEANUP #####
##########################
$jobCleanup.Flag = $false #Stop Cleaning Jobs
$jobCleanup.PowerShell.Runspace.Close() #Close the runspace
$jobCleanup.PowerShell.Dispose() #Remove the runspace from memory