Module.psm1

#Requires -Version 5.1

$ErrorActionPreference = "Stop"

# Only suported on Windows PowerShell. For the time being.
if ($IsCoreCLR) { Throw "Due to limitations with the Selenium PowerShell Module. This module is not supported on PowerShell Core yet." }

Write-Host "Selenium MVP Module Loaded. Use 'ConnectTo-MVPPortal' to get started." -ForegroundColor Green

#CompiledByBuildScript
#Resource: 00_HTMLElements.ps1
Data LocalizedDataHTMLElements {
    ConvertFrom-StringData @'
    Date=DateOfActivity
    Title=TitleOfActivity
    URL=ReferenceUrl
    Description=Description
    AnnualQuantity=AnnualQuantity
    SecondAnnualQuantity=SecondAnnualQuantity
    AnnualReach=AnnualReach
'@

}

#Resource: 01_HTMLFormStrucuture.ps1
# Read-Only
New-Variable -Name HTMLFormStructure -Scope Global -Option ReadOnly -Force -Value @(
    @{
        Name = "Default"
        Properties = @(
            @{
                Name = 'Date'
                Element = $LocalizedDataHTMLElements.Date
                isRequired = $true
            }
            @{
                Name = 'Title'
                Element = $LocalizedDataHTMLElements.Title
                isRequired = $true
            }
            @{
                Name = 'URL'
                Element = $LocalizedDataHTMLElements.URL
                isRequired = $false
            }
            @{
                Name = 'Description'
                Element = $LocalizedDataHTMLElements.Description
            }            
        )
    }
    @{
        Name = "Article"
        Properties = @(
            @{
                Name = 'Number of Articles'
                Element = $LocalizedDataHTMLElements.AnnualQuantity
                isRequired = $true
            },
            @{
                Name = 'Number of Views'
                Element = $LocalizedDataHTMLElements.AnnualReach
            }           
        )
    }
    @{
        Name = "Blog/WebSite Post"
        Properties = @(
            @{
                Name = 'URL'
                Element = $LocalizedDataHTMLElements.URL
                isRequired = $true
            }
            @{
                Name = 'Number of Posts'
                Element = $LocalizedDataHTMLElements.AnnualQuantity
                isRequired = $true
            },
            @{
                Name = 'Number of Subscribers'
                Element = $LocalizedDataHTMLElements.SecondAnnualQuantity
            },
            @{
                Name = 'Annual Unique Visitors'
                Element = $LocalizedDataHTMLElements.AnnualReach
            }                       
        )
    }
    @{
        Name = "Book (Author)"
        Properties = @(
            @{
                Name = 'Number of Books'
                Element = $LocalizedDataHTMLElements.AnnualQuantity
                isRequired = $true
            },
            @{
                Name = 'Copies Sold'
                Element = $LocalizedDataHTMLElements.AnnualReach
            }                       
        )
    }
    @{
        Name = "Book (Co-Author)"
        Properties = @(
            @{
                Name = 'Number of Books'
                Element = $LocalizedDataHTMLElements.AnnualQuantity
                isRequired = $true
            },
            @{
                Name = 'Copies Sold'
                Element = $LocalizedDataHTMLElements.AnnualReach
            }                       
        )
    }
    @{
        Name = "Conference (Staffing)"
        Properties = @(
            @{
                Name = 'Number of Conferences'
                Element = $LocalizedDataHTMLElements.AnnualQuantity
                isRequired = $true
            },
            @{
                Name = 'Number of Visitors'
                Element = $LocalizedDataHTMLElements.AnnualReach
            }                       
        )
    }
    @{
        Name = "Docs.Microsoft.com Contribution"
        Properties = @(
            @{
                Name = 'Pull Requests/Issues/Submissions'
                Element = $LocalizedDataHTMLElements.AnnualQuantity
                isRequired = $true
            }                     
        )
    }
    @{
        Name = "Forum Moderator"
        Properties = @(
            @{
                Name = 'Number of Threads moderated'
                Element = $LocalizedDataHTMLElements.AnnualQuantity
                isRequired = $true
            }                     
        )
    }
    @{
        Name = "Forum Participation (3rd Party forums)"
        Properties = @(
            @{
                Name = 'URL'
                Element = $LocalizedDataHTMLElements.URL
                isRequired = $true
            }
            @{
                Name = 'Number of Answers'
                Element = $LocalizedDataHTMLElements.AnnualQuantity
                isRequired = $true
            },
            @{
                Name = 'Number of Posts'
                Element = $LocalizedDataHTMLElements.SecondAnnualQuantity
                isRequired = $true
            },
            @{
                Name = 'Views of answers'
                Element = $LocalizedDataHTMLElements.AnnualReach
            }                                   
        )
    }
    @{
        Name = "Forum Participation (Microsoft Forums)"
        Properties = @(
            @{
                Name = 'URL'
                Element = $LocalizedDataHTMLElements.URL
                isRequired = $true
            },            
            @{
                Name = 'Number of Answers'
                Element = $LocalizedDataHTMLElements.AnnualQuantity
                isRequired = $true
            },
            @{
                Name = 'Number of Posts'
                Element = $LocalizedDataHTMLElements.SecondAnnualQuantity
            },
            @{
                Name = 'Views of answers'
                Element = $LocalizedDataHTMLElements.AnnualReach
            }                                   
        )
    }
    @{
        Name = "Mentorship"
        Properties = @(
            @{
                Name = 'Mentoring Topic'
                Element = $LocalizedDataHTMLElements.Description
            }             
            @{
                Name = 'Number of Mentorship Activity'
                Element = $LocalizedDataHTMLElements.AnnualQuantity
                isRequired = $true
            }
            @{
                Name = 'Number of Mentees'
                Element = $LocalizedDataHTMLElements.AnnualReach
            }                                 
        )
    }
    @{
        Name = "Microsoft Open Source Projects"
        Properties = @(           
            @{
                Name = 'Number of Projects'
                Element = $LocalizedDataHTMLElements.AnnualQuantity
                isRequired = $true
            }                             
        )
    }
    @{
        Name = "Non-Microsoft Open Source Projects"
        Properties = @(
            @{
                Name = 'Projects(s)'
                Element = $LocalizedDataHTMLElements.AnnualQuantity
                isRequired = $true
            }
            @{
                Name = 'Contribution(s)'
                Element = $LocalizedDataHTMLElements.SecondAnnualQuantity
            }                                            
        )
    }
    @{
        Name = "Organizer (User Group/Meetup/Local Events)"
        Properties = @(
            @{
                Name = 'Meetings'
                Element = $LocalizedDataHTMLElements.AnnualQuantity
                isRequired = $true
            }
            @{
                Name = 'Members'
                Element = $LocalizedDataHTMLElements.AnnualReach
            }                                            
        )
    }
    @{
        Name = "Organizer of Conference"
        Properties = @(
            @{
                Name = 'Number of Conferences'
                Element = $LocalizedDataHTMLElements.AnnualQuantity
                isRequired = $true
            }
            @{
                Name = 'Number of Attendees'
                Element = $LocalizedDataHTMLElements.AnnualReach
            }                                            
        )
    }
    @{
        Name = "Other"
        Properties = @(
            @{
                Name = 'Annual quantity'
                Element = $LocalizedDataHTMLElements.AnnualQuantity
                isRequired = $true
            }
            @{
                Name = 'Annual reach'
                Element = $LocalizedDataHTMLElements.AnnualReach
            }                                            
        )
    }  
    @{
        Name = "Product Group Feedback"
        Properties = @(
            @{
                Name = 'Number of Events Participated'
                Element = $LocalizedDataHTMLElements.AnnualQuantity
                isRequired = $true
            }
            @{
                Name = 'Number of Feedbacks Provided'
                Element = $LocalizedDataHTMLElements.SecondAnnualQuantity
            }                                            
        )
    }
    @{
        Name = "Sample Code/Projects/Tools"
        Properties = @(
            @{
                Name = 'URL'
                Element = $LocalizedDataHTMLElements.URL
                isRequired = $true
            }            
            @{
                Name = 'Number of Samples'
                Element = $LocalizedDataHTMLElements.AnnualQuantity
                isRequired = $true
            }
            @{
                Name = 'Number of Downloads'
                Element = $LocalizedDataHTMLElements.AnnualReach
            }                                            
        )
    }
    @{
        Name = "Site Owner"
        Properties = @(          
            @{
                Name = 'Number of Sites'
                Element = $LocalizedDataHTMLElements.AnnualQuantity
                isRequired = $true
            }
            @{
                Name = 'Number of Visitors'
                Element = $LocalizedDataHTMLElements.AnnualReach
            }                                            
        )
    }
    @{
        Name = "Speaking (Conference)"
        Properties = @(          
            @{
                Name = 'Number of Talks'
                Element = $LocalizedDataHTMLElements.AnnualQuantity
                isRequired = $true
            }
            @{
                Name = 'Attendees of Talks'
                Element = $LocalizedDataHTMLElements.AnnualReach
            }                                            
        )
    }
    @{
        Name = "Speaking (User Group/Meetup/Local events)"
        Properties = @(          
            @{
                Name = 'Number of Talks'
                Element = $LocalizedDataHTMLElements.AnnualQuantity
                isRequired = $true
            }
            @{
                Name = 'Attendees of Talks'
                Element = $LocalizedDataHTMLElements.AnnualReach
            }                                            
        )
    }
    @{
        Name = "Technical Social Media (Twitter, Facebook, LinkedIn...)"
        Properties = @(
            @{
                Name = 'URL'
                Element = $LocalizedDataHTMLElements.URL
                isRequired = $true
            }                    
            @{
                Name = 'Number of Talks'
                Element = $LocalizedDataHTMLElements.AnnualQuantity
                isRequired = $true
            }
            @{
                Name = 'Number of Followers'
                Element = $LocalizedDataHTMLElements.AnnualReach
            }                                            
        )
    }
    @{
        Name = "Translation Review, Feedback and Editing"
        Properties = @(          
            @{
                Name = 'Annual quantity'
                Element = $LocalizedDataHTMLElements.AnnualQuantity
                isRequired = $true
            }                                         
        )
    }
    @{
        Name = "Video/Webcast/Podcast"
        Properties = @(
            @{
                Name = 'URL'
                Element = $LocalizedDataHTMLElements.URL
                isRequired = $true
            }                   
            @{
                Name = 'Number of Videos'
                Element = $LocalizedDataHTMLElements.AnnualQuantity
                isRequired = $true
            }
            @{
                Name = 'Number of Views'
                Element = $LocalizedDataHTMLElements.AnnualReach
            }                                            
        )
    }
    @{
        Name = "Workshop/Volunteer/Proctor"
        Properties = @(          
            @{
                Name = 'Number of Events'
                Element = $LocalizedDataHTMLElements.AnnualQuantity
                isRequired = $true
            }                                          
        )
    }                       
)
#Resource: 02_HTMLFormStructureFormatting.ps1
New-Variable -Name HTMLFormStructureValidate -Scope Global -Option ReadOnly -Force -Value @(

    @{
        Name = "Date"
        Format = {
            param($date)
            $currentCultureDate = [DateTime]::Parse($date, [cultureinfo]::New((Get-Culture).Name))                    
            write-output ($currentCultureDate.ToString("MM/dd/yyyy"))
        }
    }

    @{
        Name = "URL"
        Format = {
            param($url)
            Write-Output ([URI]::New($url).AbsoluteUri)
        }
    }    

)
#Resource: 03_LocalizedData.ps1
data LocalizedData {
    ConvertFrom-StringData @'
    ErrorConnectToMVPPortal=An Error occured when attempting to connect to the portal. Details: '{0}'
    ErrorNestedMVPActivity=Nested MVPActivity. MVPActivity is only requried for the top-level declaration. Refer to usage.
    ErrorMissingMVPActivityArea=Missing 'Area' Statement.
    ErrorMissingMVPActivityAreaMultiple=Multiple Statements of 'Area' was detected. Only a single instance is permitted.
    ErrorMissingMVPActivityContributionArea=Missing 'ContributionArea' Statement.
    ErrorExceedMVPActivityContributionArea=Exceeded 'ContributionArea' Limit. Limit is 2. Count was '{0}'.
    ErrorMissingMVPActivityValue=Missing 'Value' Statement.
    ErrorMissingActivityType=Unable to enumerate ActivityTypes from source HTML.
    ErrorMissingContributionType=Unable to enumerate ContributionAreas from source HTML.
    ErrorMissingDriver=Missing Selinum Driver. Use: ConnectTo-Selenium to connect.
    ErrorTryTentitiveCommand=Try-TentativeCommand: Exceeded Retry Limit.
    ErrorMissingSelectedValue=The Parameter '-SelectedValue' value '{0}', could not be found
    ErrorTooManySelectedValue=The Parameter '-SelectedValue' value '{0}', returned too many items. Count '{1}'
    ErrorJavaScriptTimeout=Javascript Timeout.
    ErrorNoActivityButton=Cannot Select Add New Activity Button.
    ErrorAreaNotNested=Error. 'Area' is not nested within MVPActivity.
    ErrorContributionAreaNotNested=Error. 'ContributionArea' is not nested within MVPActivity.
    ErrorHTMLFormStructureDefaultParameter=Error. The 'default' Parameter Value is prohibited.
    ErrorHTMLFormStructureMissingName=Error. Could not match name '{0}' with HTMLFormStructure.
    ErrorCannotFindHTMLElement=Error. Could not match HTML element '{0}' with HTMLFormStructure.
    ErrorTooManyHTMLElements=Error. Too many results with HTML element '{0} with HTMLFormStructure. Count '{1}'
    ErrorFieldValidationError=Error. A field validation error was found within the form. Error: '{0}'"
    ErrorMissingRequiredEntries=Error. Missing Required HTML Fields: '{0}'
    ErrorMissingMVPActivityInCallStack=Error. '{0}' can only be called from within MVPActivity.
    ErrorFormattingValue=Error. There was an error raised when attempting to format '{0}' with '{1}'. Error Message: {2}
    ErrorSavingMVPActivity=Error. There was an error raised when attempting to save the MVP activity. Error: '{0}'
    ErrorSelectingDropDown=Error. There was an error raised when attempting to select the dropDown '{0}' with the value: '{1}'. Error: '{2}'
    ErrorStopMVPActivity=Error. There was an error raised when attempting to cancel the MVP activity. Error: '{0}'
    ErrorAreaFailure=Error. Could not set MVP Area within HTML Form, unable to continue.
    ErrorTestCSVSchemaMissingCSVFile="Missing CSV File: '{0}'"
    ErrorTestCSVSchemaImportCSVFile=An error occured when importing the CSV File: Error '{0}'
    ErrorTestCSVSchemaMissingColumns=Error. Validation Failed: Missing Columns '{0}'
    ErrorTestCSVSchemaDifferentAreaColumn=Formatting Error. Validation Failed: Cannot have different Areas ('{0}') within the same CSV File.
    ErrorTestCSVSchemaDuplicateContributionArea=Duplicate Contribution Area Found with SecondContributionArea and ThirdContributionArea columns within the CSV File. Please correct this issue and try again!
    ErrorParseValueCheck=Parser Check Failed. Please ensure that 'Value' has the correct names added that are appropriate for the respective 'Area'. Please run "Get-AreaNamedValues -AreaName '{0}'" to get correct 'Value' names. Alternatively, you can run: 'New-MVPFixture -AreaName '{0}' to generate an example template.
    ErrorParseContributionAreaCheck=Parser Check Failed. Please ensure that 'ContributionArea' has the correct names added. Please run "Get-Help ContributionArea" for a list of valid Contribution Areas.
    Error500=Error 500 Detected on the MVP Portal. This is an intermittent error.
    WarningEntryWasNotSaved=An Error occured when attempting to add the entry. THE ENTRY WAS NOT SAVED.
    ElementIdActivityType=activityTypeSelector
    ElementIdContributionArea=select_contributionAreasDDL
    ElementButtonNewActivity=addNewActivityBtn
    ElementButtonCancelActivity=submitCloseButton
    ElementButtonSubmitActivity=submitActivityButton
    ElementValueArticle=e36464de-179a-e411-bbc8-6c3be5a82b68
    ElementValueChefPuppetInDataCenter=b803f4ef-066b-e511-810b-fc15b428ced0
    ElementFieldValidationError=field-validation-error
    ElementVisibilityBoxXPath=//a[contains(@class,'Txt') and contains(@role,'navigation')]
    VisibilityListItem=//li[contains(@class,'Item') and contains(@txt,'{0}')]
    VariableSaveActivitySleepCounter=SaveActivitySleepCounter
    TestActivityRegexMVPActivity=^MVPActivity
    TestActivityRegexMVPArea=^Area
    TestActivityRegexMVPContributionArea=^ContributionArea
    TestActivityRegexMVPVisibility=^Visibility
    TestActivityRegexMVPValue=^Value
    ConnectToMVPPortalRegexURLMatch=^https:\/\/login\.((microsoftonline)|(live))\.com
    MVPPortal500Error=^https:\/\/mvp\.microsoft\.com\/Error\/500\?.+
'@

}
#Resource: 04_HTMLContributionAreas.ps1
# Read-Only
New-Variable -Name HTMLContributionAreas -Scope Global -Option ReadOnly -Force -Value @(
'Chef/Puppet in Datacenter'
'Container Management'
'Datacenter Management'
'Group Policy'
'High Availability'
'Hyper-V'
'Linux in System Center/Operations Management Suite'
'Linux on Hyper-V'
'Networking'
'PowerShell'
'Storage'
'Windows Server for Small & Medium Business'
'Azure Bot Service'
'Azure Cognitive Search'
'Azure Cognitive Services'
'Azure Machine Learning'
'Dynamics 365'
'Power Apps'
'Power Automate'
'Power Pages'
'Power Virtual Agents'
'Azure Arc Enabled Data Services'
'Azure Cosmos DB'
'Azure Data Catalog'
'Azure Data Explorer'
'Azure Data Lake'
'Azure Database for MySQL'
'Azure Database for PostgreSQL'
'Azure Databricks'
'Azure HDInsight and Hadoop & Spark on Azure'
'Azure Search'
'Azure SQL (Database, Pools, Serverless, Hyperscale, Managed Instance, Virtual Machines)'
'Azure SQL Edge'
'Azure Stream Analytics'
'Azure Synapse Analytics'
'Big Data Clusters'
'Cortana Intelligence Suite'
'Data Warehousing (Azure SQL Data Warehouse, Fast Track & APS)'
'Information Management (ADF, SSIS, & Data Sync)'
'Microsoft Purview'
'Power BI'
'SQL Server (on Windows, Linux, Containers)'
'SQL Server Machine Learning Services (R, Python)'
'SQL Server Reporting Services & Analysis Services'
'Tools & Connectivity'
'.NET'
'Accessibility'
'ASP.NET/IIS'
'C++'
'Developer Security'
'Front End Web Dev'
'Github & Azure DevOps'
'Java'
'Javascript/Typescript'
'Node.js'
'Python'
'Quantum'
'Unity'
'Visual Studio Code'
'Visual Studio Extensibility'
'Xamarin'
'Information Protection'
'Microsoft Intune'
'Previous Expertise: Directory Services'
'Remote Desktop Services'
'Azure Edge Devices'
'Azure IoT Services & Development'
'Access'
'Excel'
'Exchange'
'Microsoft Stream'
'Microsoft Teams'
'Microsoft Viva'
'Office 365'
'OneDrive'
'OneNote'
'Outlook'
'PowerPoint'
'Project'
'SharePoint'
'Skype for Business'
'Visio'
'Word'
'Yammer'
'Microsoft Graph'
'Microsoft Teams Development'
'Outlook Development'
'SharePoint Development'
'W/X/P Development'
'ARM Templates (Infra as Code)'
'Azure API Management'
'Azure App Service'
'Azure Arc Enabled Infrastructure'
'Azure Backup & Disaster Recovery'
'Azure Blockchain'
'Azure Core Compute (VMSS, Confidential Computing, Platform Deployment)'
'Azure Cost Management'
'Azure Functions'
'Azure Hybrid'
'Azure Kubernetes, Container Instances, Docker'
'Azure Lighthouse'
'Azure Logic Apps'
'Azure Migrate'
'Azure Monitor'
'Azure Networking'
'Azure Policy & Governance'
'Azure SDK (Software Development Kit) and CLIs ( Az CLI, PowerShell, Terraform, Ansible)'
'Azure Service Fabric'
'Azure Storage'
'Distributed App Runtime (Dapr), Open App Model (OAM), Open Service Mesh (OSM)'
'Previous Expertise: Microsoft Azure'
'Service Bus, Event Hubs, Event Grid, Relay'
'D365 Mixed Reality'
'MR Design'
'MR Development'
'Cloud Security'
'Identity & Access'
'SIEM & XDR'
'Surface for IT'
'Windows 365'
'Windows for IT'
'Windows Design'
'Windows Development'
)
#BuildFileName: Get-ActivityTypes.ps1
Function Get-ActivityTypes {
    [CmdletBinding()]
    param()

    Write-Verbose "Get-ActivityType Called:"

    Test-SEDriver

    # If the Activity Types have already been cached, then return the cache
    if (Test-SEActivityTypes) {
        Write-Verbose "[Get-ActivityType] Returning Cache:"
        return $Global:SEActivityTypes
    }

    # Load and Parse the HTML
    $HTMLDoc = [HtmlAgilityPack.HtmlDocument]::new()
    $HTMLDoc.LoadHtml($Global:MVPDriver.PageSource)

    # Retrive the Contribution Areas
    $ActivityTypes = $HTMLDoc.GetElementbyId("activityTypeSelector").ChildNodes.Where{$_.Name -eq 'option'}

    if ($ActivityTypes.count -eq 0) { $PSCmdlet.ThrowTerminatingError($LocalizedData.ErrorMissingActivityType)}

    # Build Custom Object
    $Global:SEActivityTypes = ($ActivityTypes | Select-Object @{
                                        Name="Name"
                                        Expression={$_.InnerText}}, 
                                    @{
                                        Name="Value"
                                        Expression={($_.Attributes.Where{$_.Name -eq 'value'}).Value}
                                    })

    Write-Debug ("[Get-ActivityType] Global:SEActivityTypes: {0}" -f ($Global:SEActivityTypes | ConvertTo-Json))

    Write-Output $Global:SEActivityTypes

}

#BuildFileName: Get-ASTInstanceValues.ps1
function Get-ASTInstanceValues {
    [CmdletBinding()]
    param(
        [Parameter(Mandatory)]
        [System.Management.Automation.Language.CommandAst[]]
        $InstanceValues,
        [Parameter()]
        [string]
        $ParameterName
    )
 
    $values = [System.Collections.Generic.List[String]]::New()

    # Split Instances with ParameterNames and Without ParameterNames
    $InstanceValuesWithParameterNames,$InstanceValuesWithoutParameterNames = $InstanceValues.Where({
        $_.CommandElements.Where{$_.ParameterName}
    }, 'Split')

    # Return a list of cmdlet's that have parsed parameters into the input and capture their values
    forEach ($valueInstance in $InstanceValuesWithParameterNames) {

        $Result = 0..$valueInstance.CommandElements.count | Where-Object { 
            ($valueInstance.CommandElements[$_].ParameterName -eq $ParameterName) 
        }

        # Get the following item in the array, which will be the value of the parameter name
        $values.Add($valueInstance.CommandElements[$Result+1].Value)

    }

    $InstanceValuesWithoutParameterNames | ForEach-Object { $values.Add( $_.CommandElements[1].Value )  }

    return $values

}
#BuildFileName: Get-ContributionAreas.ps1
Function Get-ContributionAreas {
    [CmdletBinding()]
    param()

    Write-Verbose "Get-ContributionAreas Called:"

    Test-SEDriver

    # If the Activity Types have already been cached, then return the cache
    if (Test-SEContributionAreas) {
        Write-Verbose "[Get-ContributionAreas] Returning Cache:"
        return $Global:SEContributionAreas
    }

    # Load and Parse the HTML
    $HTMLDoc = [HtmlAgilityPack.HtmlDocument]::new()
    $HTMLDoc.LoadHtml($Global:MVPDriver.PageSource)

    # Retrive the Contribution Areas
    $ContributionAreas = $HTMLDoc.GetElementbyId("select_contributionAreasDDL").ChildNodes.Where{$_.Name -eq "optgroup"}

    # Filter by Non-Disabled Attributes. Disabled Attributes are headers
    $OptionElements = $ContributionAreas.ChildNodes.Where{$_.Attributes.Name -notcontains 'disabled'}

    if ($OptionElements.count -eq 0) { $PSCmdlet.ThrowTerminatingError($LocalizedData.ErrorMissingContributionType)}

    # HTML Sanitization
    $htmlSanitizer = {
        param($value)

        $value = $value -replace '&', '&'
        $value = $value -replace ' ', ''

        Write-Output $value.trim()

    }

    # Build Custom Object
    $Global:SEContributionAreas = ($OptionElements | Select-Object @{
                                        Name="Name"
                                        Expression={ $htmlSanitizer.Invoke($_.InnerText) }
                                    }, 
                                    @{
                                        Name="Value"
                                        Expression={($_.Attributes.Where{$_.Name -eq 'data-contributionid'}).Value}
                                    })
    
    Write-Debug ("[Get-ContributionAreas] Global:SEContributionAreas: {0}" -f ($Global:SEActivityTypes | ConvertTo-Json))
    Write-Output $Global:SEContributionAreas

}

#BuildFileName: Get-HTMLFormStructure.ps1
function Get-HTMLFormStructure {
    [CmdletBinding()]
    param (
       # Parameter help description
       [Parameter(Mandatory)]
       [String]
       $Name
    )

    Write-Verbose "Get-HTMLFormStructure Started:"

    # Trim the input so that we can be more forgiving.
    $Name = $Name.Trim()

    $output = [System.Collections.Generic.List[PSCustomObject]]::New()

    # The Default Parameter Value is Prohibited
    if ($Name -eq "Default") { Throw $LocalizedData.ErrorHTMLFormStructureDefaultParameter }

    $defaultProperties = $Global:HTMLFormStructure.Where{$_.Name -eq 'Default'}
    [Array]$matched = $Global:HTMLFormStructure.Where{($_.Name -eq $Name) -or ($_.Element -eq $Name)}

    # If the Either The Element of the String name can't be found,
    # then Throw a terminating error.
    if ($matched.count -eq 0) {
        Throw ($LocalizedData.ErrorHTMLFormStructureMissingName -f $Name)
    }

    # If there is more then 1 result, data duplicate. Throw temrinating error.
    if ($matched.count -ne 1) {
        Throw ($LocalizedData.ErrorHTMLFormStructureMissingName -f $Name)
    }

    # Join the Default data with the matched data
    forEach ($Property in $defaultProperties.Properties) {
        # Test if the Element Already Exists. If so, skip.
        if ($matched.Properties.Element -contains $Property.Element) { 
            Write-Debug ("[Get-HTMLFormStructure] Duplicate Element '{0}'" -f $Property.Element)
            continue 
        }

        $obj = [PSCustomObject]@{
            Name = $Property.Name
            Element = $Property.Element
            isRequired = $(
                if ($Property.Keys -notcontains "isRequired") { $false }
                else { $Property.isRequired }
            )
            isSet = $false
        }

        Write-Debug ("[Get-HTMLFormStructure] Adding Default Element '{0}' " -f ($obj | ConvertTo-Json))
        $output.Add($obj)     
    }

    forEach ($Property in $matched.properties) {

        $obj = [PSCustomObject]@{
                    Name = $Property.Name
                    Element = $Property.Element
                    isRequired = $(
                        if ($Property.Keys -notcontains "isRequired") { $false }
                        else { $Property.isRequired }
                    )
                    isSet = $false
                }

        Write-Debug ("[Get-HTMLFormStructure] Adding Element '{0}' " -f ($obj | ConvertTo-Json))
        $output.Add($obj)

    }

    #
    # Join the Formatting Scriptblock form the HTMLFormStructureValidate Variable.
    $output = $output | Select-Object *, @{
        Name = "Format"
        Expression = { 
            $Name = $_.Name
            try { ($Global:HTMLFormStructureValidate.Where{$_.Name -eq $Name}).Format } catch {}
        }
    }

    Write-Output $output      
    Write-Verbose "Get-HTMLFormStructure Completed:"

}
#BuildFileName: Invoke-MVPActivity.ps1
function Invoke-MVPActivity {
    [CmdletBinding()]
    param()

    # Update Streams
    Write-Verbose "Invoke-MVPActivity:"

    # Update Debug Stream
    Write-Debug "[Invoke-MVPActivity] Calling Test-SEDriver:"

    # Test if the Driver is active. If not throw a terminating error.
    Test-SEDriver

    # Update Debug Stream
    Write-Debug "[Invoke-MVPActivity] Calling Try-TentativeCommand:"
    Write-Debug ("[Invoke-MVPActivity] Try-TentativeCommand Params: {0}" -f ($params | ConvertTo-Json))

    $result = ttry {

        # Try and click "Add New Activity"

        $ActivityButton = Find-SeElement -Driver $Global:MVPDriver -Id $LocalizedData.ElementButtonNewActivity
        Invoke-SeClick -Element $ActivityButton

        # Update Debug
        Write-Debug "[Invoke-MVPActivity:] Tentative-Try: Success"

        Write-Output $true

    } -catch {

        # Update Debug
        Write-Debug ("[Invoke-MVPActivity] Tentative-Try: Error Raised {0}" -f $_.Exception)
        # Try and close the activity if the window is already open.
        Stop-MVPActivity
        # Sleep for a second
        Start-Sleep -Seconds 1

    } -RetryLimit 4

    if ($null -eq $result) {
        # Update Verbose Stream
        Write-Verbose "Invoke-MVPActivity: Failure"

        Throw $LocalizedData.ErrorNoActivityButton
    }

    # Update Verbose Stream
    Write-Verbose "Invoke-MVPActivity: Success"

    Start-Sleep -Seconds 1

}
#BuildFileName: New-CSVArguments.ps1
function New-CSVArguments {
    param(
        [Parameter(Mandatory)]
        [String]
        $LiteralPath
    )

    # Import the CSV File
    $CSV = Import-Csv @PSBoundParameters

    # Get the CSV Column Names, excluding 'Area' & ContributionArea
    $CSVColumnNames = ($CSV | Get-Member -MemberType NoteProperty).Name

    # Get the HTMLForm Structure and Get the Column Names Match from the CSV File
    $FormStructure = Get-HTMLFormStructure @($CSV.Area)[0] | Where-Object { $_.Name -in $CSVColumnNames }

    $ArgumentList = $CSV | ForEach-Object {

        $row = $_

        $params = @{
            Area = $row.Area
            ContributionArea = ($row.ContributionArea, $row.SecondContributionArea, $row.ThirdContributionArea | Where-Object {$null -ne $_})
        }

        if ($CSVColumnNames -contains 'Visibility') {
            $params.Visibility = $row.Visibility
        }

        # Iterate Through the Matched Items and Add them
        $FormStructure | ForEach-Object {
            $params.('{0}' -f $_.Name) = $row."$($_.Name)"
        }

        Write-Output $params

    }

    Write-Output $ArgumentList
    
}
#BuildFileName: New-CSVFixture.ps1
Function New-CSVFixture {
    param(
        [Parameter(Mandatory)]
        [String]
        $LiteralPath
    )

    # Import the CSV File
    [Array]$CSV = Import-Csv @PSBoundParameters

    # Single CSV Entry
    if ($CSV.Count -eq 1) {
        $Area = $CSV.Area
    } else {
        $Area = $CSV.Area[0]
    }

    # Get the CSV Column Names, excluding 'Area' & ContributionArea
    $CSVColumnNames = ($CSV | Get-Member -MemberType NoteProperty).Name

    # Get the HTMLForm Structure and Get the Column Names Match from the CSV File
    $FormStructure = Get-HTMLFormStructure $Area | Where-Object { $_.Name -in $CSVColumnNames }

    # If the Visability is present, manually interpolate.
    $VisabilityParameter = $(
        if ($CSVColumnNames -contains 'Visibility') { Write-Output ',$Visibility'}
        else { Write-Output '' }
    )

    # Param Block
    $ParamBlock = 'param($Area,$ContributionArea,{0}{1});' -f (($FormStructure.Name | ForEach-Object { "`${$_}" }) -join ','), $VisabilityParameter

    # Create the Scriptblock
    $ScriptBlock = $FormStructure | ForEach-Object -Begin {
        $ParamBlock
    } -Process {
        "Value '{0}' {1}" -f $_.Name, "`${$($_.Name)};"
    }

    # Return our Fixture Goodness
    return ([scriptblock]::Create($ScriptBlock))

}
#BuildFileName: New-MVPActivity.ps1
function New-MVPActivity {
    [CmdletBinding(DefaultParameterSetName="Default")]
    param (
        # Scriptblock of the Activity
        [Parameter(Mandatory,Position=1,ParameterSetName="Arguments")]
        [Parameter(Mandatory,Position=1,ParameterSetName="Default")]
        [ScriptBlock]
        $Fixture,
        # ArgumentList of the Activity
        [Parameter(Position=2,ParameterSetName="Arguments")]
        [HashTable]
        $ArgumentList      
    )
    
    begin {

        Write-Debug "[New-MVPActivity] BEGIN: Buildup of Activity"

        #
        # Activity Setup

        $invokeTearDown = $false
        $isPreParse = $true
        # Setup of Contributions and Areas
        $Script:MVPArea = $null
        $Script:ContributionAreas = [System.Collections.Generic.List[PSCustomObject]]::New()
        $Script:MVPHTMLFormStructure = $null

        try {

            #
            # Test the configuration is valid.
            $parameterCmdlets = Test-ActivityScriptBlock @PSBoundParameters
            # Test that the driver is active
            Test-SEDriver

            $isPreParse = $false

            # Test that the MVPActivity elements is present.
            Wait-ForMVPElement
            # Create new MVPActivity. Click the Button.
            Invoke-MVPActivity    
             
        } catch {
            Write-Debug "[MVPActivity] Failed MVP Validation Error Below:"
            Write-Error $_
            $invokeTearDown = $true
        }

        Write-Debug "[MVPActivity] BEGIN: Buildup Complete"
   
    }
    
    process {
        
        if ($invokeTearDown) { return }

        Write-Debug "[New-MVPActivity] PROCESS:"

        # Invoke the Fixture

        try {

            # If the Parameter is used instead of the cmdlet itself, invoke the cmdlet parsing in the parametervalue
            if ($parameterCmdlets.ParametrizedArea) {
                Area $ArgumentList.Area
            }
            
            if ($parameterCmdlets.ParametrizedContributionArea) {
                ContributionArea $ArgumentList.ContributionArea
            }

            if ($parameterCmdlets.ParametrizedVisibility) {
                Visibility $ArgumentList.Visibility
            }

            if ($ArgumentList) {
                # Invoke the Fixture Splatting the Args in as parameters.
                & $Fixture @ArgumentList
            } else {
                $null = $Fixture.Invoke()
            }
            # Save the MVP Activity
            Save-MVPActivity
            
        } catch {
            Write-Error $_
            Write-Warning $LocalizedData.WarningEntryWasNotSaved
            Write-Debug "Error was triggered, initiate the teardown of the MVPActivity"
            $invokeTearDown = $true
        }
        
    }
    end {

        Write-Debug "[New-MVPActivity] END: Beginning Teardown"

        #
        # Activity Tear Down
        #

        try {

            if ($invokeTearDown -and (-not $isPreParse)) {
                # Close the MVP Activity
                try {
                    Stop-MVPActivity
                } catch {
                    Write-Warning $_
                }
            }

        } finally {
            # TearDown of Contributions and Areas
            $Script:MVPArea = $null
            $Script:ContributionAreas = [System.Collections.Generic.List[PSCustomObject]]::New()
            $Script:MVPHTMLFormStructure = $null               
        }

        Write-Debug "[New-MVPActivity] END: Completed"

        #
        # Wait for the Window to be closed
        #

        Wait-ForActivityWindow

        #
        # Return to the Caller
        #

        return $null
        
    }

}
#BuildFileName: Save-MVPActivity.ps1
function Save-MVPActivity {
    [cmdletbinding()]
    param()

    # Test if the Driver is active. If not throw a terminating error.
    Test-SEDriver
    
    # Validate that HTML Form Strucuture Prior to Submission
    # Ensure that all required items are set.
    $missingRequiredEntries = $Script:MVPHTMLFormStructure.Where{($_.isSet -eq $false) -and ($_.isRequired -eq $true)}
    if ($missingRequiredEntries) {
        Throw ($LocalizedData.ErrorMissingRequiredEntries -f (-join $missingRequiredEntries.Name))
        # Fail the Submission and Tidy Up
    }
    
    #
    # Search for Field Validation Errors

    $fieldValidationErrors = Find-SeElement -Target $Global:MVPDriver -By ClassName -Selection $LocalizedData.ElementFieldValidationError
    if ($fieldValidationErrors) {
        Throw ($LocalizedData.ErrorFieldValidationError -f $fieldValidationErrors.Text)
    }

    #
    # Save the Activity

    $GetDriverParams = @{
        Driver = $Global:MVPDriver
        Id = $LocalizedData.ElementButtonSubmitActivity
    }

    try {
        # Click the Save Button
        $SaveButton = Find-SeElement -Driver $Global:MVPDriver -Id $LocalizedData.ElementButtonSubmitActivity
        Invoke-SeClick -Element $SaveButton
        # Look for error 500's. The URL will approx redirect to:
        # https://mvp.microsoft.com/Error/500?aspxerrorpath=/
        # There are MSFT Issues. Stop
        if ($Global:MVPDriver.Url -match $LocalizedData.MVPPortal500Error) { Throw $LocalizedData.Error500 }
    } catch {
        Throw ($LocalizedData.ErrorSavingMVPActivity -f $_)
    }
    
}
#BuildFileName: Select-DropDown.ps1
function Select-DropDown {
    [CmdletBinding()]
    param (
        [Parameter(Mandatory)]
        [string]
        $elementId,
        [Parameter(Mandatory)]
        [string]
        $selectedValue
    )

    try {
        $ActivityType = Find-SeElement -Driver $Global:MVPDriver -Id $elementId
        $SelectElement = New-Object -TypeName OpenQA.Selenium.Support.UI.SelectElement $ActivityType
        $SelectElement.SelectByValue($selectedValue)
    } catch {
        $PSCmdlet.ThrowTerminatingError($LocalizedData.ErrorSelectingDropDown -f $elementId, $selectedValue, $_)
    }

}
#BuildFileName: Stop-MVPActivity.ps1
function Stop-MVPActivity {
    [CmdletBinding()]
    param (
    )

    # Test if the Driver is active. If not throw a terminating error.
    Test-SEDriver
    
    try {
        $CancelButton = Find-SeElement -Driver $Global:MVPDriver -Id $LocalizedData.ElementButtonCancelActivity
        Invoke-SeClick -Element $CancelButton 
    } catch {
        Throw ($LocalizedData.ErrorStopMVPActivity -f $_)
    }
       
}
#BuildFileName: Test-ActivityScriptBlock.ps1
function Test-ActivityScriptBlock {
    [CmdletBinding()]
    param (
        # Scriptblock to invoke
        [Parameter(Mandatory)]
        [ScriptBlock]
        $Fixture,
        [Parameter()]
        [HashTable]
        $ArgumentList
    )

    Write-Debug "[Test-ActivityScriptBlock] Validating Scriptblock:"

    #
    # Return an Object
    $resultObject = [PSCustomObject]@{
        ParametrizedArea = $false
        ParametrizedContributionArea = $false
        ParametrizedVisibility = $false
    }

    #
    # Return a list of Commands
    $CommandAst = $Fixture.ast.FindAll({$args[0] -is [System.Management.Automation.Language.CommandAst]}, $true)
    $AreaValue = $null
    $ContributionAreaValues = $null

    # Requirement 1:
    # Ensure that there are no additional MVPActivity scriptblocks within it.
    if ($CommandAst -match $LocalizedData.TestActivityRegexMVPActivity) {
        Throw $LocalizedData.ErrorNestedMVPActivity
    }

    # Requirement 2:
    # Ensure that a Area is present and it's a single instance.
    $areaInstance = $CommandAst -match $LocalizedData.TestActivityRegexMVPArea
    if (($areaInstance.count -eq 0) -and ($null -eq $ArgumentList.Area)) {
        # No Instances were found
        Throw $LocalizedData.ErrorMissingMVPActivityArea
    } elseif (($areaInstance.count -eq 0) -and ($null -ne $ArgumentList.Area)) {
        $resultObject.ParametrizedArea = $true
        $AreaValue = $ArgumentList.Area
    } elseif ($areaInstance.count -ne 1 ) {
        # Multiple Statements of Area was defined
        Throw $LocalizedData.ErrorMissingMVPActivityAreaMultiple
    } elseif (($areaInstance.count -ne 0) -and ($null -eq $ArgumentList.Area)) {
        $AreaValue = Get-ASTInstanceValues -InstanceValues $areaInstance -ParameterName 'Name'
    }

    # Requirement 3:
    # Ensure that the ContributionArea is present and that there is a maximum of two contributions.
    $ContributionAreaInstance = $CommandAst -match $LocalizedData.TestActivityRegexMVPContributionArea
    if ((($ContributionAreaInstance).Count -eq 0) -and ($null -eq $ArgumentList.ContributionArea)) {
        Throw $LocalizedData.ErrorMissingMVPActivityContributionArea
    } elseif (($ContributionAreaInstance.count -eq 0) -and ($null -ne $ArgumentList.ContributionArea)) {
        $resultObject.ParametrizedContributionArea = $true
        $ContributionAreaValues = $ArgumentList.ContributionArea        
    } elseif (($ContributionAreaInstance).Count -gt 3)  {
        Throw ($LocalizedData.ErrorExceedMVPActivityContributionArea -f $ContributionAreaInstance.Count)
    } elseif (($ContributionAreaInstance.count -ne 0) -and ($null -eq $ArgumentList.ContributionArea)) {
        $ContributionAreaValues = Get-ASTInstanceValues -InstanceValues $ContributionAreaInstance -ParameterName 'SelectedValue'
    }

    # Requirement 4:
    # Ensure that the Value Cmdlet is present.
    [Array]$valueInstances = $CommandAst -match $LocalizedData.TestActivityRegexMVPValue
    if ($valueInstances.Count -eq 0) {
        Throw $LocalizedData.ErrorMissingMVPActivityValue
    }

    # Requirement 5:
    # Visibility dosen't need to be included into the statement, however ensure that there are only one of them.
    [Array]$visibilityInstances = $CommandAst -match $LocalizedData.TestActivityRegexMVPVisibility
    if ($visibilityInstances.Count -gt 1) {
        Throw $LocalizedData.ErrorMultipleVisibilityStatements
    } 
    # If Visability is included as a parameter, process it.
    elseif (($visibilityInstances.count -eq 0) -and ($null -ne $ArgumentList.Visibility)) {
        $resultObject.ParametrizedVisibility = $true          
    }

    # Requirement 6:
    # Ensure that all Values added to the fixture are (required) AND are correct
    $HTMLFormStructure = Get-HTMLFormStructure -Name $AreaValue
    $ASTInstanceValues = Get-ASTInstanceValues -InstanceValues $valueInstances -ParameterName 'Name'

    $testMandatoryValues = $HTMLFormStructure | Where-Object { (
        ($_.isRequired) -and 
        (
            ($_.Name -notin $ASTInstanceValues) -and 
            ($_.Element -notin $ASTInstanceValues)
        )    
    )}

    $testMisnamedValues = $ASTInstanceValues | Where-Object {
        $_ -notin $HTMLFormStructure.Name -and 
        $_ -notin $HTMLFormStructure.Element
    }

    if ($testMandatoryValues -or $testMisnamedValues) {
        Throw ($LocalizedData.ErrorParseValueCheck -f $AreaValue)
    }

    # Requirement 5:
    # Ensure that all ContributionAreas are added to the fixture are (required) AND are correct

    if ($ContributionAreaValues | Where-Object { $_ -notin $Global:HTMLContributionAreas}) {
        Throw ($LocalizedData.ErrorParseContributionAreaCheck -f $AreaValue)
    }

    #
    # Finished

    Write-Output $resultObject
    
    Write-Debug "[Test-ActivityScriptBlock] All Tests Passed:"

}
#BuildFileName: Test-CallStack.ps1
function Test-CallStack {
    [CmdletBinding()]
    param(
        # Parameter help description
        [Parameter(Mandatory)]
        [String]
        $Name
    )

    if ((Get-PSCallStack).Command -notcontains "MVPActivity") {
        Throw ($LocalizedData.ErrorMissingMVPActivityInCallStack -f $Name)
    }

}
#BuildFileName: Test-CSVSchema.ps1
Function Test-CSVSchema {
    [cmdletbinding()]
    param(
        [Parameter(Mandatory)]
        [String]
        $LiteralPath
    )

    $RequiredTopLevelColumns = @('Area','ContributionArea')

    # Test-Path
    if (-not(Test-Path @PSBoundParameters -ErrorAction SilentlyContinue)) {
        Throw ($LocalizedData.ErrorTestCSVSchemaMissingCSVFile -f $LiteralPath)
    }

    try {
        # Import the CSV File
        $CSV = Import-Csv @PSBoundParameters
    } catch {
        Throw ($LocalizedData.ErrorTestCSVSchemaImportCSVFile -f $_)
    }

    # Get the CSV Column Names
    $CSVColumnNames = ($CSV | Get-Member).Name

    # Validate if the Area/ ContributionArea Columns are present.
    $MissingTopLevelColumns = $RequiredTopLevelColumns | Where-Object {
        $CSVColumnNames -notcontains $_
    }
    # If items were matched then throw a terminating error.
    if ($MissingTopLevelColumns.Count -ne 0) {
        Throw ($LocalizedData.ErrorTestCSVSchemaMissingColumns -f ($MissingTopLevelColumns -join ' , '))
    }

    # Ensure that the Areas are all the same. Group by the Property Area
    $GroupedItems = $CSV | Group-Object -Property Area

    # Retrive the Area and Contribution Areas
    if ($GroupedItems.Length -ne 1) {
        Throw ($LocalizedData.ErrorTestCSVSchemaDifferentAreaColumn -f $GroupedItems.Name -join ' , ')
    }

    $Area = $GroupedItems | Select-Object -First 1 -ExpandProperty Name

    # Perform a Lookup to workout what columns we need
    $FormStructure = Get-HTMLFormStructure $Area | Where-Object {$_.isRequired}

    # Validate that the required Name or Element is Present within the CSV File
    $MissingCSVColumns = $FormStructure | Where-Object { $_.Name -notin $CSVColumnNames }
    if ($MissingCSVColumns.Count -ne 0) {
        Throw ($LocalizedData.ErrorTestCSVSchemaMissingColumns -f $MissingCSVColumns.Name -join ' , ')
    }    

    # Validate no duplicate Contribution Area's are present

    ForEach($Row in $CSV) {

        if ((-not([String]::IsNullOrEmpty($Row.SecondContributionArea))) -or (-not($Row.ThirdContributionArea))) {
            $arr = $Row.ContributionArea, $Row.SecondContributionArea, $Row.ThirdContributionArea | Group-Object | Where-Object {$_.Count -ne 1}
            
            if ($arr.Count -ne 0) {
                Throw ($LocalizedData.ErrorTestCSVSchemaDuplicateContributionArea)
            }
            
        }

    }

}
#BuildFileName: Test-MVPDriverisMicrosoftLogin.ps1
function Test-MVPDriverisMicrosoftLogin {
    [CmdletBinding()]
    param (
        [Parameter()]
        [Switch]
        $waitUntilLoaded,
        [Parameter()]
        [Switch]
        $isCompleted
    )
    # The parameters provide pester guidance on how to mock this function
    Write-Output ($Global:MVPDriver.Url -match $LocalizedData.ConnectToMVPPortalRegexURLMatch)
}
#BuildFileName: Test-SEActivityTypes.ps1
function Test-SEActivityTypes {
    
    $result = $false
    # If there Driver Variable is $null. Throw a Terminating Error
    if ($null -ne $Global:SEActivityTypes) { $result = $true }
    Write-Output $result

}
#BuildFileName: Test-SEContributionAreas.ps1
function Test-SEContributionAreas {
    [cmdletbinding()]
    param()
    $result = $false
    # If there Driver Variable is $null. Throw a Terminating Error
    if ($null -ne $Global:SEContributionAreas) { $result = $true }
    Write-Output $result

}
#BuildFileName: Test-SEDriver.ps1
function Test-SEDriver {
    [CmdletBinding()]
    param()
    # If there Driver Variable is $null. Throw a Terminating Error
    if ($null -eq $Global:MVPDriver) {
        Throw ($LocalizedData.ErrorMissingDriver) 
    } elseif ($null -eq $Global:MVPDriver.URL) {
        Throw ($LocalizedData.ErrorMissingDriver) 
    }

}
#BuildFileName: Try-TentitiveCommand.ps1
function ttry {
    [CmdletBinding()]
    param (
        [Parameter(Mandatory, Position = 0)]
        [Scriptblock]
        $Try,
        [Parameter(Mandatory,  Position = 1)]
        [Scriptblock]
        $Catch,
        [Parameter(Position = 3)]
        [Int]
        $RetryLimit = 3
    )

    $Count = 0
    Do {

        Try {
            $Result = $Try.Invoke()
            break
        } Catch {            
            $Result = $Catch.Invoke()
            $Count++
        }

    } Until ($Count -eq $RetryLimit)

    if ($Count -eq $RetryLimit) {
        Write-Error $LocalizedData.ErrorTryTentitiveCommand
    }

    Write-Debug "[Try-TentativeCommand] Returning:"
    
    return $Result

}

#Yup it's gotta be flipped, because I need to mock ttry and I can't mock and alias.
Set-Alias -Name "Try-TentativeCommand" -Value "ttry"
#BuildFileName: Wait-ForActivityWindow.ps1
function Wait-ForActivityWindow {
    [CmdletBinding()]
    param (
    )

    # Test the Selenium Driver
    Test-SEDriver

    # Start Verbose
    Write-Verbose "[Wait-ForActivityWindow] Started:"

    # Wait for the MVPActivity Window to be closed.
    try {
        $DialogElement = Find-SeElement -Driver ($Global:MVPDriver) -ClassName 'mvp-dialog'
    } Catch {
        Throw $_
    }

    While (($null -ne $DialogElement) -and ($DialogElement.Displayed)) {
        Write-Verbose "[Wait-ForActivityWindow] Waiting for 'Add a new activity' to close:"
        Start-Sleep -Seconds 1
    }

    # Start Verbose
    Write-Verbose "[Wait-ForActivityWindow] Finished:"

}
#BuildFileName: Wait-ForJavascript.ps1
Function Wait-ForJavascript {
    [CmdletBinding()]
    param (
        [Parameter(Mandatory)]
        [String]
        $ElementTextToSearch       
    )
    
    Test-SEDriver

    #
    # Sometimes the Javascript dosen't load and add elements
    # to the HTML page. This checks to see if the element text is present
    # and if so, then continue.

    $retry = 0

    Do {
        if (($Global:MVPDriver).PageSource -like ("*{0}*" -f $ElementTextToSearch)) { break }
        Start-Sleep -Seconds 1
        $retry++
    } Until ($retry -eq 3)
    
    if ($retry -eq 3) {
        Throw $LocalizedData.ErrorJavaScriptTimeout
    }
    
}
#BuildFileName: Wait-ForMVPElement.ps1
function Wait-ForMVPElement {
    [CmdletBinding()]
    param (    
    )

    Write-Verbose "[Wait-ForMVPElement] Started:"

    ttry {
        $ActivityButton = Find-SeElement -Driver $Global:MVPDriver -Id $LocalizedData.ElementButtonNewActivity
        if ($null -eq $ActivityButton) {
            Enter-SeUrl 'https://mvp.microsoft.com/en-us/MyProfile/EditActivity' -Driver $Global:MVPDriver             
            Throw "Missing"
        }
    } -catch {
        # Look for error 500's. The URL will approx redirect to:
        # https://mvp.microsoft.com/Error/500?aspxerrorpath=/
        # There are MSFT Issues. Stop
        if ($Global:MVPDriver.Url -match $LocalizedData.MVPPortal500Error) { Throw $LocalizedData.Error500 }

        Write-Debug ("Waiting for New Activity Button. {0}" -f $LocalizedData.ElementButtonNewActivity)
        Write-Warning "Waiting for New Activity Button."        
    } -RetryLimit 10

    Write-Verbose "[Wait-ForMVPElement] Completed:"
    
}
#BuildFileName: Area.ps1
function Area {
<#
.Description
Used inside of the MVPActivity block, the ‘Area’ Describes the Contribution Area.
The Current List of Areas are:

+ Article
+ Blog/WebSite Post
+ Book (Author)
+ Book (Co-Author)
+ Conference (Staffing)
+ Docs.Microsoft.com Contribution
+ Forum Moderator
+ Forum Participation (3rd Party forums)
+ Forum Participation (Microsoft Forums)
+ Mentorship
+ Microsoft Open Source Projects
+ Non-Microsoft Open Source Projects
+ Organizer (User Group/Meetup/Local Events)
+ Organizer of Conference
+ Other
+ Product Group Feedback
+ Sample Code/Projects/Tools
+ Site Owner
+ Speaking (Conference)
+ Speaking (User Group/Meetup/Local events)
+ Technical Social Media (Twitter, Facebook, LinkedIn...)
+ Translation Review, Feedback and Editing
+ Video/Webcast/Podcast
+ Workshop/Volunteer/Proctor

.PARAMETER SelectedValue
The MVP Contribution Area

.EXAMPLE
Area 'Article'

.SYNOPSIS
'Area' defines the MVP Contribution Area within the Portal. Area can only be used within MVPActivity.
#>

    [CmdletBinding()]
    param(
        # Selected Value
        [Parameter(Mandatory=$true)]
        [String]
        $SelectedValue
    )
   
    begin {

        # Test if the Driver is active. If not throw a terminating error.
        Test-SEDriver
        # Test the Callstack.
        Test-CallStack $PSCmdlet.MyInvocation.MyCommand.Name
        
    }

    Process {


        # Validate the $SelectedValue Attributes
        [Array]$matchedActivityType = Get-ActivityTypes | Where-Object { $_.Name -eq $SelectedValue }
        if ($matchedActivityType.Count -eq 0) { Throw ($LocalizedData.ErrorMissingSelectedValue -f $SelectedValue) }
        if ($matchedActivityType.Count -ne 1) { Throw ($LocalizedData.ErrorTooManySelectedValue -f $SelectedValue, $matchedActivityType.count) }
        
        # If the Activity Type matches, we need to do a lookup to get the HTML Form structure
        $HTMLFormStructure = Get-HTMLFormStructure $SelectedValue

        # Select the Element
        $output = ttry {

            Select-DropDown -elementId $LocalizedData.ElementIdActivityType -selectedValue $matchedActivityType.Value

            # Iterate through the HTML Structure and validate if the fields exist.
            $HTMLFormStructure | ForEach-Object {
                $Element = Find-SeElement -Driver ($Global:MVPDriver) -Id $_.Element -Timeout 1
                if (-not($Element)) {
                    Throw $LocalizedData.ErrorJavaScriptTimeout
                }
            }

            # Update the Area
            $Script:MVPArea = $SelectedValue
            $Script:MVPHTMLFormStructure = $HTMLFormStructure

            Write-Output $true

        } -Catch {       
            # If the Javascript Fails to Populate the Sub entries within the form
            # it will retrigger by select the "Article"
            Start-Sleep -Seconds 1
            Select-DropDown -elementId $LocalizedData.ElementIdActivityType -selectedValue $LocalizedData.ElementValueArticle   
        } -RetryLimit 5
        
        # If it failed to select the Area, we need to fail the cmdlet
        if ($output -ne $true) {
            # Throw a terminating error
            Throw $LocalizedData.ErrorAreaFailure
        }

    }

}
#BuildFileName: ConnectTo-MVPPortal.ps1
function ConnectTo-MVPPortal {
<#
.Description
Connects to the MVP Portal and waits for the login to complete.

.PARAMETER URLPath
A custom URL provided in the MVP email. Otherwise, if excluded, it will prompt the user for a URL.

.PARAMETER DriverType
You can specify different browser types. The default is Firefox. Options are: Firefox, Chrome, Edge, OldEdge

.EXAMPLE
Connect to Portal:

ConnectTo-MVPPortal -URLPath "https://Your-URL-Path-Goes-Here.com"
.EXAMPLE
Connect to Portal:

ConnectTo-MVPPortal -URLPath "https://Your-URL-Path-Goes-Here.com" -DriverType ('Firefox','Chrome' or 'Edge')
.SYNOPSIS
Connects to the MVP Portal and waits for the login to complete.
#>
    
    [CmdletBinding()]
    param (
        [Parameter()]
        [String]
        $URLPath=(Read-Host "Enter in URL Path"),
        [ValidateSet("Firefox","Chrome","Edge", "OldEdge")]
        [String]
        $DriverType = 'Firefox'
    )

    try {

        switch ($DriverType) {
            'Firefox' { $Global:MVPDriver = Start-SeFirefox -StartURL $URLPath }
            'Chrome'  { $Global:MVPDriver = Start-SeChrome  -StartURL $URLPath }
            'Edge'    { $Global:MVPDriver = Start-SeNewEdge -StartURL $URLPath }
            'OldEdge' { $Global:MVPDriver = Start-SeEdge -StartURL $URLPath    }
        }
                
    } catch {
        $Global:MVPDriver = $null
        Throw ($LocalizedData.ErrorConnectToMVPPortal -f $_)       
    }

    # Wait until the Login Screen Loads
    do {
        Test-SEDriver
        Write-Verbose "Waiting for the URL to redirect to the Microsoft Login"
        Start-Sleep -Seconds 1
    } Until (Test-MVPDriverisMicrosoftLogin -waitUntilLoaded)

    # Then Pause Execution until login process has completed
    while (Test-MVPDriverisMicrosoftLogin -isCompleted) {
        Test-SEDriver
        Write-Verbose "Waiting for Login to complete:"
        Start-Sleep -Seconds 1
    }

    Write-Verbose "Success! Logged in."
}
#BuildFileName: ContributionArea.ps1
Function ContributionArea {
<#
.Description
The ContributionArea defines the Contribution Areas within a contribution.
ContributionArea can be used as a String Array (ContributionArea 'Area1','Area2','Area3'), or by specifying multiple ContributionArea statements.

Similarly to Area, if ContributionArea is included within the Param() block, it's not required to be defined and will be automatically invoked.

The current list of 'Contribution Areas' are:
+ Chef/Puppet in Datacenter
+ Container Management
+ Datacenter Management
+ Group Policy
+ High Availability
+ Hyper-V
+ Linux in System Center/Operations Management Suite
+ Linux on Hyper-V
+ Networking
+ PowerShell
+ Storage
+ Windows Server for Small & Medium Business
+ Azure Bot Service
+ Azure Cognitive Search
+ Azure Cognitive Services
+ Azure Machine Learning
+ Dynamics 365
+ Power Apps
+ Power Automate
+ Power Pages
+ Power Virtual Agents
+ Azure Arc Enabled Data Services
+ Azure Cosmos DB
+ Azure Data Catalog
+ Azure Data Explorer
+ Azure Data Lake
+ Azure Database for MySQL
+ Azure Database for PostgreSQL
+ Azure Databricks
+ Azure HDInsight and Hadoop & Spark on Azure
+ Azure Search
+ Azure SQL (Database, Pools, Serverless, Hyperscale, Managed Instance, Virtual Machines)
+ Azure SQL Edge
+ Azure Stream Analytics
+ Azure Synapse Analytics
+ Big Data Clusters
+ Cortana Intelligence Suite
+ Data Warehousing (Azure SQL Data Warehouse, Fast Track & APS)
+ Information Management (ADF, SSIS, & Data Sync)
+ Microsoft Purview
+ Power BI
+ SQL Server (on Windows, Linux, Containers)
+ SQL Server Machine Learning Services (R, Python)
+ SQL Server Reporting Services & Analysis Services
+ Tools & Connectivity
+ .NET
+ Accessibility
+ ASP.NET/IIS
+ C++
+ Developer Security
+ Front End Web Dev
+ Github & Azure DevOps
+ Java
+ Javascript/Typescript
+ Node.js
+ Python
+ Quantum
+ Unity
+ Visual Studio Code
+ Visual Studio Extensibility
+ Xamarin
+ Information Protection
+ Microsoft Intune
+ Previous Expertise: Directory Services
+ Remote Desktop Services
+ Azure Edge Devices
+ Azure IoT Services & Development
+ Access
+ Excel
+ Exchange
+ Microsoft Stream
+ Microsoft Teams
+ Microsoft Viva
+ Office 365
+ OneDrive
+ OneNote
+ Outlook
+ PowerPoint
+ Project
+ SharePoint
+ Skype for Business
+ Visio
+ Word
+ Yammer
+ Microsoft Graph
+ Microsoft Teams Development
+ Outlook Development
+ SharePoint Development
+ W/X/P Development
+ ARM Templates (Infra as Code)
+ Azure API Management
+ Azure App Service
+ Azure Arc Enabled Infrastructure
+ Azure Backup & Disaster Recovery
+ Azure Blockchain
+ Azure Core Compute (VMSS, Confidential Computing, Platform Deployment)
+ Azure Cost Management
+ Azure Functions
+ Azure Hybrid
+ Azure Kubernetes, Container Instances, Docker
+ Azure Lighthouse
+ Azure Logic Apps
+ Azure Migrate
+ Azure Monitor
+ Azure Networking
+ Azure Policy & Governance
+ Azure SDK (Software Development Kit) and CLIs ( Az CLI, PowerShell, Terraform, Ansible)
+ Azure Service Fabric
+ Azure Storage
+ Distributed App Runtime (Dapr), Open App Model (OAM), Open Service Mesh (OSM)
+ Previous Expertise: Microsoft Azure
+ Service Bus, Event Hubs, Event Grid, Relay
+ D365 Mixed Reality
+ MR Design
+ MR Development
+ Cloud Security
+ Identity & Access
+ SIEM & XDR
+ Surface for IT
+ Windows 365
+ Windows for IT
+ Windows Design
+ Windows Development

.PARAMETER SelectedValue
The Contribution Areas that you wish to add.

.EXAMPLE
Single Use:

ContributionArea 'Area1'
.EXAMPLE
ContributionArea can be stacked like so:

ContributionArea 'Area1'
ContributionArea 'Area2'
ContributionArea 'Area3'

.EXAMPLE
ContributionArea can placed in an array:

ContributionArea 'Area1','Area2','Area3'

.SYNOPSIS
The ContributionArea is a command that defines the Contribution Areas within the MVP Portal.

#>

    [cmdletbinding()]
    param(
        # Parameter help description
        [Parameter(Mandatory)]
        [String[]]
        $SelectedValue
    )

    begin {

        # Test if the Driver is active. If not throw a terminating error.
        Test-SEDriver
        # Test the Callstack.
        Test-CallStack $PSCmdlet.MyInvocation.MyCommand.Name
        
    }

    process {

        # Multiple Contribution Areas can be parsed in.
        ForEach ($Param in $SelectedValue) {

            # Fetch the Contribution Areas (sometimes the browser is too quick)
            $HTMLContributionAreas = ttry {
                Get-ContributionAreas
            } -Catch { 
                Start-Sleep -Milliseconds 500
            } -RetryLimit 5

            # Validate the $SelectedValue Parameters
            [Array]$matchedActivityType = $HTMLContributionAreas | Where-Object { $_.Name -eq $Param }
            if ($matchedActivityType.Count -eq 0) { Throw ($LocalizedData.ErrorMissingSelectedValue -f $Param) }
            if ($matchedActivityType.Count -ne 1) { Throw ($LocalizedData.ErrorTooManySelectedValue -f $Param, $matchedActivityType.count) }
            
            # Generate the ElementId
            $elementId = $(
                if ($Script:ContributionAreas.Count -eq 0) { $LocalizedData.ElementIdContributionArea }
                else { "{0}{1}" -f $LocalizedData.ElementIdContributionArea, ($Script:ContributionAreas.Count + 1) }
            )

            # Add to the Contribution Areas
            $Script:ContributionAreas.Add(
                [PSCustomObject]@{
                    elementId = $matchedActivityType.Name
                    selectedValue = $matchedActivityType.Value
                }
            )

            #
            # If the $Script:ContributionAreas count is gt then 3, then select the 'add more' link

            if ($Script:ContributionAreas.count -ge 3) {
                $AddMoreButton = Find-SeElement -Driver $Global:MVPDriver -By ClassName -Selection 'add' 
                Invoke-SeClick -Element $AddMoreButton 
            }

            #
            # You will need to inspect the elements on the HTML to add additional drop down settings.
            # Make sure you set the id as the value

            ttry {
                Select-DropDown -elementId $elementId -selectedValue $matchedActivityType.Value
                # We are using Views of answers to dertmine if the Javascript has ran
                Wait-ForJavascript -ElementText 'Views of answers'
            } -Catch {       
                # If the Javascript Fails to Populate the Sub entries within the form
                # it will retrigger by select the "Chef/Puppet in Datacenter"
                Start-Sleep -Seconds 1
                Select-DropDown -elementId $Script:ContributionAreas[-1].elementId -selectedValue $LocalizedData.ElementValueChefPuppetInDataCenter   
            } -RetryLimit 10     
        }   
    }
}
#BuildFileName: Get-AreaNamedValues.ps1
function Get-AreaNamedValues {
<#
.Description
This command is used to retrieve the list of fields that are required for an MVP submission.
Since different MVP Area's have other fields, you can use Get-AreaNamedValues 'AreaName' to identify the fields.

In the example below, we will use Get-AreaNamedValues 'Article' to get the Value names:

Get-AreaNamedValues Article

Name Mandatory
---- ---------
Number of Articles True
Title True
Date True
Number of Views False
URL False
Description False

Once the list is returned, we can construct the data structure using the output:

MVPActivity "Test" {
    
    # Let's set the Area and the Contribution Area
    Area 'Article'
    ContributionArea 'PowerShell'

    # We can set the mandatory parameters
    Value "Number of Articles" 1
    Value Title "Test Entry"
    Value Date "30/11/2020"

}

.PARAMETER AreaName
The MVP Contribution Area name to lookup.

.EXAMPLE
In the example below, we will use Get-AreaNamedValues 'Article' to get the Value names:

Get-AreaNamedValues Article

Name Mandatory
---- ---------
Number of Articles True
Title True
Date True
Number of Views False
URL False
Description False

.SYNOPSIS
This command is used to retrive the list of fields that are required for a MVP submission.
#>

    [CmdletBinding()]
    param (
        [Parameter()]
        [String]
        $AreaName
    )

    # Call the HTML FormStrucutre and Parse in the Article Name
    Get-HTMLFormStructure $AreaName | Select-Object Name, @{Name='Mandatory';Exp={$_.isRequired}} | Sort-Object -Property Mandatory -Descending

}
#BuildFileName: MVPActivity.ps1
function MVPActivity {
<#
.Description
MVPActivity is the top-level definition command, which groups the MVP contribution types into their respective areas.

This command is responsible for buildup (creation) and (tear-down) of the contribution. There are two types of input with this command:

1. CSV Import and,
2. The Domain Specific Language.

The CSV Import is a 'simplified' input parameter, however *each* CSV import is limited to a seperate 'Area'.
If you choose to use the Domain Specific Language type, commands: 'Area', 'ContributionArea' and 'Value' are mandatory.
MVPActivity also performs input validation ensuring that input meets the requirements of the HTML form.

.PARAMETER Name
Name is the name of the MVPActivity. This provides a visual grouping of data that is being added to the MVP Portal.

.PARAMETER CSVPath
Import the CSV File and Parse it.
Please Note: *The headers 'Area' & 'ContributionArea' are mandatory.*
Adding Secondary and Tertiary 'ContributionAreas' is done by using: 'SecondContributionArea' and 'ThirdContributionArea' (See Example Below)

    MVPActivity -CSVPath 'Path to CSV File'

Example CSV File:

Date,Title,URL,Description,Number of Articles,Number of Views,Area,ContributionArea,SecondContributionArea,ThirdContributionArea
28/11/2020,TEST,https://www.google.com,TEST,1,1,Article,PowerShell,Networking,Storage
28/11/2020,TEST,https://www.google.com,TEST,1,1,Article,PowerShell,Networking,Storage

.PARAMETER ArgumentList
MVPActivity supports arguments being parsed into it using the -ArgumentList parameter, similarly to the -TestCases parameter within Pester.
Inputs are defined as a [HashTable] or a [HashTable[]] with the name of the parameter being the [HashTable] key and it's value being the item.
To use the parameters that have been parsed into the fixture, use the param() block at the beginning of the fixture with the ParameterName.

In the example below we declare the parameters and parse them into MVPActivity:

# Declare the parameters:
$param = @{
    Parameter1 = 'Test'
}

# Parse the Arguments by using the -ArgumentList parameter:
MVPActivity "Test Entry" -ArgumentList $param {
    # To get the value from $Parameter1, add a param() within the fixture/scriptblock.
    param($Parameter1)

    # We can reference the value by referencing the variable name '$Parameter1'
    Area $Parameter1 'Value'

}

The -ArgumentList parameter also supports an array of hashtables ([HashTable[]]) with common values, providing a way to parse multiple parameters into the same fixture.
For Example:

# Declare the parameters as an array:
$param = @(
    @{
        Parameter1 = 'One Value'
    }
    @{
        Parameter1 = 'Another Value'
    }
)

MVPActivity "Test Entry" -ArgumentList $param {
    param($Parameter1)

    Area $Parameter1 'Value'

}

This will process 'Parameter1' as a seperate MVP Contribution.

.PARAMETER Fixture
The Fixture/Scriptblock contains the DSL PowerShell code that adds the MVP entry to the portal.
The Fixture is a regular Powershell scriptblock, however Area, ContributionArea and Value are mandatory commands.

*Please ensure that 'Area', 'ContributionArea' are prefixed before 'Value'*.

.EXAMPLE
Import the CSV File and Parse it:

MVPActivity -CSVPath 'Path to CSV File'

.EXAMPLE
Standard Usage:

    MVPActivity "Reddit Contribution" {

        Area 'Article'
        # ContributionArea can accept an Array
        ContributionArea 'PowerShell','Yammer','Word'

        Value 'Date' $date
        Value 'Title' 'TEST'
        Value 'URL' 'https:\\test.com'
        Value 'Description' 'THIS IS A TEST'
        Value 'Number of Posts' '1'
        # Or you can use the HTML DivId
        Value 'Number of Subscribers' '1'
        Value 'Annual Unique Visitors' '1'

    }

.EXAMPLE
In this example below, we will parse a hashtable of arguments into the fixture:

    $params = @{
        Area = "Blog/Website Post"
        ContributionArea = "PowerShell","Yammer","Word"
        date = '26/10/2020'
    }

    # Define Activity
    MVPActivity "Name of Activity" -ArgumentList $params {
        # If the Parameters -Area or -ContributionArea are defined
        # in the param block, the cmdlet is not required (Area or ContributionArea).
        param($Area, $ContributionArea, $date)

        # You can use the String Name
        Value 'Date' $date
        Value 'Title' 'TEST'
        Value 'URL' 'https:\\test.com'
        Value 'Description' 'THIS IS A TEST'
        Value 'Number of Posts' '1'
        # Or you can use the HTML Div Id
        Value 'Number of Subscribers' '1'
        Value 'Annual Unique Visitors' '1'
        # Still can execute PowerShell within the script block
        Start-Sleep -Seconds 3

   }
.EXAMPLE

For those who love complexity, the fixture supports an array of HashTables to be parsed into the fixture.

    $params = @(
        @{
            Area = "Blog/Website Post"
            ContributionArea = "PowerShell","Yammer","Word"
            date = '26/10/2020'
        }
        @{
            Area = "Blog/Website Post"
            ContributionArea = "PowerShell","Yammer","Word"
            date = '15/10/2020'
        }
    )

    # Define Activity
    MVPActivity "Name of Activity" -ArgumentList $params {
        # If the Parameters -Area or -ContributionArea are defined in the param block,
        # the cmdlet is not required (Area or ContributionArea).
        param($Area, $ContributionArea, $date)

        # You can use the String Name
        Value 'Date' $date
        Value 'Title' 'TEST'
        Value 'URL' 'https:\\test.com'
        Value 'Description' 'THIS IS A TEST'
        Value 'Number of Posts' '1'
        # Or you can use the HTML Div Id
        Value 'Number of Subscribers' '1'
        Value 'Annual Unique Visitors' '1'
        # Still can execute PowerShell within the script block
        Start-Sleep -Seconds 3

   }


.SYNOPSIS
MVPActivity is the top-level definition command, which groups the MVP contribution types into their respective areas.
#>
    
    [CmdletBinding(DefaultParameterSetName="Default")]
    param (        
        # Scriptblock of the Name of the Contribution
        [Parameter(Mandatory,Position=1,ParameterSetName="Default")]
        [Parameter(Mandatory,Position=1,ParameterSetName="Arguments")]
        [Parameter(Position=1,ParameterSetName="CSVFile")]
        [String]
        $Name,
        # Scriptblock of the Activity
        [Parameter(Mandatory,Position=2,ParameterSetName="Arguments")]
        [Parameter(Mandatory,Position=2,ParameterSetName="Default")]
        [ScriptBlock]
        $Fixture,
        # ArgumentList of the Activity
        [Parameter(Position=3,ParameterSetName="Arguments")]
        [HashTable[]]
        $ArgumentList,
        # CSV File
        [Parameter(Position=2,ParameterSetName="CSVFile")]
        [String]
        $CSVPath
    )

    # Construct the Parameters to Invoke
    $params = @{}

    switch ($PSCmdlet.ParameterSetName) {
        "Arguments" {

            $params.Fixture = $Fixture
            
        }
        "CSVFile" {

            # Test the CSV Schema to ensure that all the columns are present
            Test-CSVSchema $CSVPath
            # Create the Fixture
            $params.Fixture = New-CSVFixture $CSVPath
            # Create the Arguements for the Fixture
            $ArgumentList = New-CSVArguments $CSVPath
            
        }
        default {

            # If the Default Execution is run, execute the Fixture and then return.
            $params.Fixture = $Fixture

        }
    }

    # Write-Host
    if ($Name) { Write-Host ('Executing MVPActivity: "{0}"' -f $Name) -ForegroundColor Green }
    else { Write-Host ('Executing MVPActivity: (From CSV File) "{0}"' -f $CSVPath) -ForegroundColor Green }

    # If Arguments are Present, then iterate each Fixture
    if ($ArgumentList) {
        # Iterate Through Each of the Arguments and Create the MVPActivity
        foreach($Argument in $ArgumentList) {

            $params.ArgumentList = $Argument
            
            ttry {
                
                #Adding more feedback to the user.
                Write-Host ("[ADDING] TITLE: '{0}' - DATE: '{1}'" -f $params.ArgumentList.Title, $params.ArgumentList.Date)
                New-MVPActivity @params

            } -catch {

                Write-Warning "[ERROR] Failed to Add Activity (Most likley to bad Javascript). Retrying."
                # Refresh the page
                Enter-SeUrl 'https://mvp.microsoft.com/en-us/MyProfile/EditActivity' -Driver $Global:MVPDriver
                Start-Sleep -Seconds 5

            }
            
        }

    } else {

        # Otherwise Execute the MVP Activity, since there is not additional fixtures.
        New-MVPActivity @params

    }


}
#BuildFileName: New-MVPFixture.ps1
function New-MVPFixture {
<#
.Description
New-MVPFixture is a utility that generates MVPActivity ScriptBlocks.

.PARAMETER AreaName
The MVP Contribution Area name to lookup.

.PARAMETER MVPActivityName
The MVP Activity Name.

.EXAMPLE
Standard Execution:

New-MVPFixture -AreaName 'Article' -MVPActivityName 'Test'

.SYNOPSIS
New-MVPFixture is a utility that generates MVPActivity ScriptBlocks.
#>

    [CmdletBinding()]
    param (
        [String]
        $AreaName=(Read-Host "Enter in an AreaName"),
        [String]
        $MVPActivityName=(Read-Host "Enter in an MVPActivity Name")
    )

    $FormStructure = Get-AreaNamedValues -AreaName $AreaName

    return ("
+==================== START COPY ====================+
MVPActivity '$MVPActivityName' {
    # Param Block for arguments to be parsed in.
    param()

    # Define the Area
    Area '$AreaName'
    # Define the Contribution Area
    ContributionArea 'Contribution Area 1', 'Contribution Area 2', 'Contribution Area 3'

    # Define the Input Values:

    $(
        $FormStructure | ForEach-Object {
            "`n",
            " #Mandatory: $($_.Mandatory) `n",
            " Value '$($_.Name)' '' `n"
        }
    )
}
+==================== STOP COPY ====================+
    "
)
    
}
#BuildFileName: Value.ps1
Function Value {
<#
The 'Value' command is used as input entries for the HTML form, using 'Name' (Being the HTML Div Element ID or Text Name) and 'Value' (Corresponding Value) syntax.
Since different 'Area''s have various fields, you can use Get-AreaNamedValues 'AreaName' to locate those fields.

    Value is mandatory within MVPActivity and won't be automatically invoked when specified within the Param() bock.

Value supports auto-formatting of inputted data to meet the requirements of the MVP Portal. Currently, the following Portal dependencies are auto-formatted:

    Date (Required to be US Date Format) (MM-DD-YYYY)
    URL (Required to meet URL format) (https://site.com)

To view the list of Values for an MVP Contribution Area, use: Get-AreaNamedValues 'Name' (Get-AreaNamedValues 'Article')

.PARAMETER Name
The HTML Div Element ID or Text Name

.PARAMETER Value
The input value.

.EXAMPLE
In the example, 'Value' is being used to set the HTML values for the 'Article' MVP Contribution Area.

MVPActivity "Test" {
    
    # Let's set the Area and the Contribution Area
    Area 'Article'

    # We can set the mandatory parameters
    Value "Number of Articles" 1
    Value Title "Test Entry"
    Value Date "30/11/2020"

}

.SYNOPSIS
The Value command is used to input the data into the HTML form, using the 'Name' (Being the HTML Div Element ID or Text Name) and 'Value' (Corresponding Value) syntax.
#>
       
    [CmdletBinding()]
    param (
        [Parameter(Mandatory)]
        [String]
        $Name,
        [Parameter(Mandatory)]
        [String]
        $Value
    )
      
    # Test if the Driver is active. If not throw a terminating error.
    Test-SEDriver
    # Test the Callstack.
    Test-CallStack $PSCmdlet.MyInvocation.MyCommand.Name   

    # Get the HTML Element Data
    $htmlElementData = $Script:MVPHTMLFormStructure

    # Attempt to Locate the HTML Element that is needed to update.
    # If there is no value, then there is no data
    [array]$matched = $htmlElementData.Where{($_.Name -eq $Name) -or ($_.Element -eq $Name)}
    # If the Either The Element of the String name can't be found,
    # then Throw a terminating error.
    if ($matched.count -eq 0) {
        Throw ($LocalizedData.ErrorCannotFindHTMLElement -f $Name)
    }
    # If there is more then 1 result, data duplicate. Throw temrinating error.
    if ($matched.count -ne 1) {
        Throw ($LocalizedData.ErrorTooManyHTMLElements -f $Name, $matched.count)
    }    
    
    $params = @{
        Keys = $Value
    }

    # If there is a formatting property, then we invoke the scriptblock and format the data
    if ($matched.Format) {

        try {
            # Invoke the Scriptblock
            $params.Keys = $matched.Format.Invoke($Value)
        } catch {
            # Throw an Error if there was a problem.
            Throw ($LocalizedData.ErrorFormattingValue -f $Name, $Value, $_.ToString())
        }
        
    }

    # Fetches the Element and Updates the Field
    $Element = Find-SeElement -Driver ($Global:MVPDriver) -Id $matched[0].Element
    if ($null -eq $Element) { return }
    Send-SeKeys $Element @params

    # Locate and Update the HTMLElement
    $matched[0].isSet = $true

}
#BuildFileName: Visibility.ps1
Function Visibility {
    <#
    The 'Visibility' Parameter sets the visibility of the Activity.
    This statement can be included within the fixture or executed within the Param Block, similar to 'Area' and 'Contribution Area'.
    Multiple statements aren't allowed within the Fixture.
    
    .PARAMETER Name
    The Visibility Name: Values can be: 'Everyone','MVP Community' & 'Microsoft'.
    
    .EXAMPLE
    In the example, 'Visibility' is set within the Fixture:
    
    MVPActivity "Test" {
        
        # Let's set the Area and the Contribution Area
        Area 'Article'
        Visibility 'Microsoft'
    
        # We can set the mandatory parameters
        Value "Number of Articles" 1
        Value Title "Test Entry"
        Value Date "30/11/2020"
    
    }
    
    .EXAMPLE
    In the example, 'Visibility' is set within the param block:
    
    MVPActivity "Test" {
        param($Visibility)
        
        # Let's set the Area and the Contribution Area
        Area 'Article'
    
        # We can set the mandatory parameters
        Value "Number of Articles" 1
        Value Title "Test Entry"
        Value Date "30/11/2020"
    
    }
    
    .SYNOPSIS
    Set's the Visibility of the entry.
    
    #>
     
    [CmdletBinding()]
    param (
        [Parameter(Mandatory)]
        [String]
        [ValidateSet('Everyone','MVP Community','Microsoft')]
        $Name
    )
    # Test if the Driver is active. If not throw a terminating error.
    Test-SEDriver
    # Test the Callstack.
    Test-CallStack $PSCmdlet.MyInvocation.MyCommand.Name   

    Write-Verbose "[Visibility()] Invoked. Parameter `$Name: $Name"

    # Locate the ListItem Permission
    $PermissionsParams = @{
        Driver = $Global:MVPDriver
        By = 'XPath'
        Selection = $LocalizedData.VisibilityListItem -f $Name
    }

    $sleepTimer = 250

    ttry {

        # Locate the Visibility Element.
        $VisibilityElement = Find-SeElement -Driver ($Global:MVPDriver) -By XPath -Selection $LocalizedData.ElementVisibilityBoxXPath

        Write-Verbose "[Visibility()] `$VisibilityElement isNull: $($null -eq $VisibilityElement)"
        Write-Verbose "[Visibility()] `$VisibilityElement.Text : $($VisibilityElement.Text)"

        if ($null -eq $VisibilityElement) { return }
        if ($Name -eq $VisibilityElement.Text) { return }

        Write-Verbose "[Visibility()] Selecting VisibilityElement"

        # Select the Element and Select the Correct Item
        Invoke-SeClick -Element $VisibilityElement

        Start-Sleep -Milliseconds $sleepTimer

        Write-Verbose "[Visibility()] Finding VisibilityListItem:"
        $PermissionElement = Find-SeElement @PermissionsParams

        Write-Verbose "[Visibility()] `$VisibilityListItem isNull: $($null -eq $PermissionElement)"

        Start-Sleep -Milliseconds $sleepTimer
        # Select the DropDown Item

        Write-Verbose "[Visibility()] Selecting Visibility From Dropdown : $Name"
        Invoke-SeClick -Element $PermissionElement

    } -catch {

        Write-Warning "[ERROR]. Failed to set Visibility. Retrying:"
        $SleepTimer += 100

    } -RetryLimit 5
        
}