functions/Get-Calendar.ps1

Function Get-Calendar {
    [cmdletbinding(DefaultParameterSetName = "month")]
    [OutputType([System.String])]
    [Alias("cal")]

    Param(
        [Parameter(Position = 0, ParameterSetName = "month")]
        [ValidateNotNullOrEmpty()]
        [ValidateScript({
                $names = _getMonthsByCulture
                if ($names -contains $_) {
                    $True
                }
                else {
                    Throw "You entered an invalid month. Valid choices are $($names -join ',')"
                    $False
                }
            })]
        [string]$Month = (Get-Date -Format MMMM),

        [Parameter(
            Mandatory,
            Position = 0,
            ParameterSetName = "quarter",
            HelpMessage = "Specify a calendar year quarter to display."
        )]
        [ValidateRange(1,4)]
        [int]$Quarter,

        [Parameter(Position = 1,ParameterSetName = "month")]
        [Parameter(Position = 1,ParameterSetName = "quarter")]
        [ValidatePattern('^\d{4}$')]
        [int]$Year = (Get-Date).Year,

        [Parameter(
            Mandatory,
            HelpMessage = "Enter the start of a month like 1/1/2020 that is correct for your culture.",
            ParameterSetName = "span"
        )]
        [ValidateNotNullOrEmpty()]
        [string]$Start,

        [Parameter(
            Mandatory,
            HelpMessage = "Enter an ending date for the month like 3/1/2020 that is correct for your culture.",
            ParameterSetName = "span"
            )]
        [ValidateNotNullOrEmpty()]
        [string]$End,

        [Parameter(HelpMessage = "Specify a collection of dates to highlight in the calendar display.")]
        [ValidateNotNullOrEmpty()]
        [string[]]$HighLightDate,

        [Parameter(HelpMessage = "Specify the first day of the week.")]
        [ValidateNotNullOrEmpty()]
        [System.DayOfWeek]$FirstDay = ([System.Globalization.CultureInfo]::CurrentCulture).DateTimeFormat.FirstDayOfWeek,

        [Parameter(HelpMessage = "Do not use any ANSI formatting.")]
        [switch]$NoANSI,

        [Parameter(HelpMessage = "Do not show any leading or trailing days.")]
        [switch]$MonthOnly,

        [Parameter(
            Mandatory,
            HelpMessage = "Enter a year between 1000 and 3000 to display in calendar view.",
            ParameterSetName = "calyear"
        )]
        [ValidateRange(1000,3000)]
        [int]$CalendarYear
    )

    Begin {
        #display the module version defined in the psm1 file
        Write-Verbose "Starting: $($MyInvocation.MyCommand) [v$modVer]"
        Write-Verbose "Using PowerShell version: $($PSVersionTable.PSVersion)"
        Write-Verbose "Running in PowerShell host: $($host.name)"
        #Call .NET for better results when testing this command in different cultures
        $currCulture = [System.Globalization.CultureInfo]::CurrentCulture
        Write-Verbose "Using culture: $($currCulture.DisplayName) [$($currCulture.name)]"
    }
    Process {
        Write-Verbose "Using parameter set: $($PSCmdlet.ParameterSetName)"
        Write-Verbose "Getting start date using pattern $($currCulture.DateTimeFormat.ShortDatePattern)"

        #validate $End Issue #26
        if ($PSCmdlet.ParameterSetName -eq 'span') {
            if ( [DateTime]$end -lt [DateTime]$Start) {
                Write-Verbose "Validating End ($end) compared to Start ($Start)"
                Throw "[Validation Error] The end date ($end) must be later than the start date ($start)"
            }
        }

        Write-Debug "Using PSBoundParameters:"
        Write-Debug ($PSBoundParameters | Out-String).trim()

        Switch ($PSCmdlet.ParameterSetName) {
            "month" {
                Write-Verbose "Using month $month and year $year"

                #get month number
                Write-Verbose "Parsing $month to number"
                $monthInt = [DateTime]::parse("1 $month $year").month
                Write-Verbose "Returned month number $monthInt"
                $startD = [DateTime]::new($year, $monthInt, 1)
                $endD = $startD.date
            }
            "span"  {
                #Figure out the first day of the start and end months
                # Write-Verbose "Calculating start from month $($start.month) year $($start.year)"
                # $start = [DateTime]::new($start.year, $start.Month, 1)
                Write-Verbose "Treating $start as [DateTime]"
                $startD = $start -as [DateTime]
                # Write-Verbose "Calculating end from month $($end.month) year $($end.year)"
                # $end = [DateTime]::new($end.year, $end.month, 1)
                Write-Verbose "Treating $end as [DateTime]"
                $endD = $end -as [DateTime]
            }
            "calyear" {
                Write-Verbose "Getting calendar for $CalendarYear"
                $startD = [DateTime]::new($year, 1, 1)
                $endD = [DateTime]::new($year, 12, 1)
            }
            "quarter" {
                Write-Verbose "Getting calendar for Q$Quarter $Year"
                switch ($Quarter) {
                    1 { $startD = [DateTime]::new($year, 1, 1); $endD = [DateTime]::new($year, 3, 1) }
                    2 { $startD = [DateTime]::new($year, 4, 1); $endD = [DateTime]::new($year, 6, 1) }
                    3 { $startD = [DateTime]::new($year, 7, 1); $endD = [DateTime]::new($year, 9, 1) }
                    4 { $startD = [DateTime]::new($year, 10, 1); $endD = [DateTime]::new($year,12, 1) }
                }
            }
            default {
                #this should never get called, but if it does, something is coded wrong
                Write-Warning "Detected parameter set $($PSCmdlet.ParameterSetName) and I don't know what to do!"
            }
        }

        Write-Verbose "Starting at $($startD.toString())"
        Write-Verbose "Ending at $($endD.toString())"
        Write-Verbose "Highlighting: $($HighLightDate-join ',')"
        Write-Verbose "Go through the requested months."
        while ($startD -le $endD) {
            Write-Debug "Looping from $($startD.DateTime)"
            $paramHash = @{
                start          = $startD
                HighLightDates = $HighLightDate
                firstday       = $FirstDay
                noAnsi         = $NoANSI
                monthOnly      = $monthOnly
            }

            #enforce NoAnsi if running in the PowerShell ISE [Issue #30]
            if ($host.name -Match "ISE Host") {
                $paramHash.NoAnsi = $True
            }
            _getCalendar @paramHash

            #And now move onto the next month
            $startD = $startD.AddMonths(1)
        }
    } #process

    End {
        Write-Verbose "Ending: $($MyInvocation.MyCommand)"
    }
}