functions/private.ps1
function _getCalendar { [cmdletbinding()] Param( [DateTime]$start = (Get-Date), [System.DayOfWeek]$FirstDay = "Sunday", [string[]]$HighLightDates, [switch]$NoANSI, [switch]$MonthOnly ) $global:mm=@() # https://fmoralesdev.com/2019/03/21/c-DateTime-examples/ $currCulture = [System.Globalization.CultureInfo]::CurrentCulture Write-Debug "Building calendar for $($currCulture.Name)" #Need to rebuild HighLightDate to respect culture? if ($HighLightDates) { $HighLightDates = foreach ($item in $HighLightDates) { Write-Debug "Casting $item as [DateTime]" $item -as [DateTime] } Write-Debug "Detected $($HighLightDates.count) highlight dates" $HighLightDates | ForEach-Object { Write-Debug $_.toString()} } $mo = $start.month $yr = $start.year $max = $currCulture.DateTimeFormat.Calendar.GetDaysInMonth($yr, $mo) Write-Debug "Totals days in $mo/$yr is $max" $end = Get-Date -Year $yr -Month $mo -Day $max Write-Debug "Ending $end" #$DateTimeFormat = $currCulture.DateTimeFormat $fd = $FirstDay.value__ #$DateTimeFormat.FirstDayOfWeek.value__ Write-Debug "First day of the week is $FirstDay [$fd]" $CurrentDay = $start $day0 = @() $day1 = @() $day2 = @() $day3 = @() $day4 = @() $day5 = @() $day6 = @() #adjust for the beginning of the month while ($currentDay.DayOfWeek.value__ -ne $fd) { $CurrentDay =$CurrentDay.AddDays(-1) } While ($currentDay.date -le $end.date) { [DateTime]$aDay =$CurrentDay Switch ($aDay.DayOfWeek.value__) { 0 { $day0 += $aDay } 1 { $day1 += $aDay } 2 { $day2 += $aDay } 3 { $day3 += $aDay } 4 { $day4 += $aDay } 5 { $day5 += $aDay } 6 { $day6 += $aDay } } $CurrentDay =$CurrentDay.AddDays(1) } #add enough days to finish the week While ($currentDay.DayOfWeek.value__ -ne 0) { [DateTime]$aDay =$CurrentDay #Write-Debug "adding $aDay" Switch ($aDay.DayOfWeek.value__) { 0 { $day0 += $aDay } 1 { $day1 += $aDay } 2 { $day2 += $aDay } 3 { $day3 += $aDay } 4 { $day4 += $aDay } 5 { $day5 += $aDay } 6 { $day6 += $aDay } } $CurrentDay =$CurrentDay.AddDays(1) } #fix for issue #32 if ($fd) { $day0 += [DateTime]$currentDay } if ($fd -eq 0) { $mo = [PSCustomObject]@{ PSTypeName = "PSCalendarMonth" Month = "{0:MMMM}" -f $start Year = $start.year D0 = $day0 D1 = $day1 D2 = $day2 D3 = $day3 D4 = $day4 D5 = $day5 D6 = $day6 } } else { $mo = [PSCustomObject]@{ PSTypeName = "PSCalendarMonth" Month = "{0:MMMM}" -f $start Year = $start.year D1 = $day1 D2 = $day2 D3 = $day3 D4 = $day4 D5 = $day5 D6 = $day6 D0 = $day0 } } $dow = $mo.PSObject.Properties.name | Where-Object { $_ -notmatch "Month|Year" } #Build an array of short day names $abbreviated = $currCulture.DateTimeFormat.AbbreviatedDayNames #code suggestion from @scriptingstudio Issue #32 $days = [System.Collections.Generic.list[string]]::new() $underline = 4 $addDay = { $d = $abbreviated[$args[0]].PadLeft($underline, " ") if ($NoANSI) { $days.add($d) } else { $days.add(("{0}{1}{2}" -f $PScalendarConfiguration.DayOfWeek, $d, "$esc[0m")) } } $n = if ($fd -eq 0) {0} else {1} for ($n; $n -lt $abbreviated.count; $n++) {. $addDay $n} if ($fd) {. $addDay 0} $plainHead = "$($mo.Month) $($mo.Year)" if ($NoANSI) { $head = $plainHead } else { $head = "{0}{1}{2}" -f $pscalendarConfiguration.title, $plainHead, "$esc[0m" } $dayHead = $days -join ' ' Write-Debug "Using day heading $dayHead" $month = for ($i = 0; $i -lt 6; $i++) { $wk = for ($k = 0; $k -lt $dow.count; $k++) { $theDay = ($mo.$($dow[$k])[$i]) -as [DateTime] #Write-Debug "Adding $theDay" if ($theDay) { if (($start.month -ne $theDay.month) -AND $MonthOnly) { $theDay = $null $d = ' ' } else { $d = $theDay.day } $value = $d.toString().PadLeft($underline, ' ') #$value = $d.toString().PadLeft(4, ' ') if (($theDay.date -eq (Get-Date).date) -AND (-Not $NoANSI)) { "{0}{1}{2}" -f $PScalendarConfiguration.Today, $value, "$esc[0m" } elseif ( ($HighLightDates -contains $theDay.date) -AND (-Not $NoANSI)) { "{0}{1}{2}" -f $PScalendarConfiguration.Highlight, $value, "$esc[0m" } else { $value } } } Write-Debug "Adding week $wk" $wk -join ' ' } #writing a single string object Function makemonth { #this is a hack function to write all the strings to the pipeline #separately #code suggestion from @scriptingstudio Issue #32 [int]$pad = (10*$underline - $plainHead.Length) / 2 + 1 #[int]$pad = (40 - $plainHead.Length) / 2 + 1 $p = " " * $pad "`n$p$head`n" $dayHead $month } #join all the strings into a single string makemonth #| Out-String $global:mm += makemonth } #_getCalendar function _getMonthsByCulture { [cmdletbinding()] Param([string]$Culture = ([system.threading.thread]::currentThread).CurrentCulture) Write-Debug "Getting months for culture $Culture" [CultureInfo]::GetCultureInfo($culture).DateTimeFormat.MonthNames } function _getMonthNumber { [cmdletbinding()] Param([string]$MonthName) _getMonthsByCulture | ForEach-Object -Begin { $i = 0 } -Process { $i++; if ($_ -eq $MonthName) { return $i } } } Function New-RunspaceCleanupJob { <# You use this function like this: $newrunspace = <code> $pscmd = [powershell]::create() add commands to $pscmd $pscmd.runspace = $newrunspace $handle = $pscmd.beginInvoke() Start a thread job to test if runspace is being used and close it if it is finished New-RunspaceCleanUpJob -handle $handle -powershell $pscmd -sleepinterval 30 #> [cmdletbinding()] [OutputType("None", "ThreadJob")] Param( [Parameter(Mandatory, HelpMessage = "This should be the System.Management.Automation.Runspaces.AsyncResult object from the BeginInvoke() method.")] [ValidateNotNullOrEmpty()] [object]$Handle, [Parameter(Mandatory)] [ValidateNotNullOrEmpty()] [System.Management.Automation.PowerShell]$PowerShell, [Parameter(HelpMessage = "Specify a sleep interval in seconds")] [ValidateRange(5, 600)] [int32]$SleepInterval = 10, [Parameter(HelpMessage = "Pass the thread job object to the pipeline")] [switch]$Passthru ) $job = Start-ThreadJob -ScriptBlock { param($handle, $ps, $sleep) #the Write-Host lines are so that if you look at the results of the thread job #you'll see something you can use for debugging or troubleshooting. Write-Host "[$(Get-Date)] Sleeping in $sleep second loops" Write-Host "Watching this runspace" Write-Host ($ps.runspace | Select-Object -Property * | Out-String) #loop until the handle shows as completed, sleeping the the specified #number of seconds do { Start-Sleep -Seconds $sleep } Until ($handle.IsCompleted) Write-Host "[$(Get-Date)] Closing runspace" $ps.runspace.close() Write-Host "[$(Get-Date)] Disposing runspace" $ps.runspace.Dispose() Write-Host "[$(Get-Date)] Disposing PowerShell" $ps.dispose() Write-Host "[$(Get-Date)] Ending job" } -ArgumentList $Handle, $PowerShell, $SleepInterval if ($passthru) { #Write the ThreadJob object to the pipeline $job } } |