Private/Output/WriteToFile.ps1

function Initialize-OutputBuilder {
    [CmdletBinding()]
    param (
        [Parameter(Mandatory=$true)]
        [TreeConfig]$TreeConfig,
        [boolean]$ShowExecutionStats
    )

    # Only create OutputBuilder if we need to save to a file
    if ([string]::IsNullOrEmpty($TreeConfig.OutFile)) {
        return $null
    }

    $outputBuilder = New-Object System.Text.StringBuilder
    
    # Add configuration details to the output file
    [void]$outputBuilder.AppendLine("# PowerTree Output")
    [void]$outputBuilder.AppendLine("# Generated: $(Get-Date)")
    [void]$outputBuilder.AppendLine("# Path: $($TreeConfig.Path)")
    [void]$outputBuilder.AppendLine("# Output File: $($TreeConfig.OutFile)")

    # Basic configuration options
    if ($TreeConfig.DirectoryOnly) {
        [void]$outputBuilder.AppendLine("# DirectoryOnly: True")
    }

    if ($TreeConfig.Quiet) {
        [void]$outputBuilder.AppendLine("# Quiet Mode: True")
    }

    if ($TreeConfig.PruneEmptyFolders) {
        [void]$outputBuilder.AppendLine("# PruneEmptyFolders: True")
    }

    if ($TreeConfig.ShowHiddenFiles) {
        [void]$outputBuilder.AppendLine("# ShowHiddenFiles: True")
    }

    # Depth control
    if ($TreeConfig.MaxDepth -ne -1) {
        [void]$outputBuilder.AppendLine("# Max Depth: $($TreeConfig.MaxDepth)")
    }
    
    # Excluded Directories
    if ($TreeConfig.ExcludeDirectories -and $TreeConfig.ExcludeDirectories.Count -gt 0) {
        [void]$outputBuilder.AppendLine("# ExcludedDirectories: $($TreeConfig.ExcludeDirectories -join ', ')")
    }
    
    # Sorting configuration
    $sortByText = if ([string]::IsNullOrEmpty($TreeConfig.SortBy)) { "Name" } else { $TreeConfig.SortBy }
    $direction = if ($TreeConfig.SortDescending) { "Descending" } else { "Ascending" }
    [void]$outputBuilder.AppendLine("# Sort By: $sortByText $direction")
    
    # File extensions filtering
    $includeExtensions = @()
    $excludeExtensions = @()
    
    if ($TreeConfig.ChildItemFileParams -and $TreeConfig.ChildItemFileParams.ContainsKey("Include")) {
        $includeExtensions = $TreeConfig.ChildItemFileParams["Include"]
    }
    
    if ($TreeConfig.ChildItemFileParams -and $TreeConfig.ChildItemFileParams.ContainsKey("Exclude")) {
        $excludeExtensions = $TreeConfig.ChildItemFileParams["Exclude"]
    }
    
    if ($includeExtensions -and $includeExtensions.Count -gt 0) {
        [void]$outputBuilder.AppendLine("# IncludedFileTypes: $($includeExtensions -join ', ')")
    }
    
    if ($excludeExtensions -and $excludeExtensions.Count -gt 0) {
        [void]$outputBuilder.AppendLine("# ExcludedFileTypes: $($excludeExtensions -join ', ')")
    }
    
    # File Size Bounds
    if ($TreeConfig.FileSizeBounds) {
        $lowerBound = $TreeConfig.FileSizeBounds.LowerBound
        $upperBound = $TreeConfig.FileSizeBounds.UpperBound
        $humanReadableLowerBound = if ($lowerBound -ge 0) { Get-HumanReadableSize -Bytes $lowerBound -Format "Compact" } else { $null }
        $humanReadableUpperBound = if ($upperBound -ge 0) { Get-HumanReadableSize -Bytes $upperBound -Format "Compact" } else { $null }

        $sizeFilterText = switch ($true) {
            (($lowerBound -ge 0) -and ($upperBound -ge 0)) { 
                "Between $humanReadableLowerBound and $humanReadableUpperBound" 
            }
            (($lowerBound -ge 0) -and ($upperBound -lt 0)) { 
                "Minimum $humanReadableLowerBound" 
            }
            (($lowerBound -lt 0) -and ($upperBound -ge 0)) { 
                "Maximum $humanReadableUpperBound" 
            }
            default { $null }
        }
        
        if ($sizeFilterText) {
            [void]$outputBuilder.AppendLine("# File Size Filter: $sizeFilterText")
        }
    }
    
    # Display configuration
    $displayColumns = @()
    foreach ($column in $TreeConfig.HeaderTable.HeaderColumns) {
        if ($column -ne "Hierarchy") {
            $displayColumns += $column
        }
    }
    
    if ($displayColumns -and $displayColumns.Count -gt 0) {
        [void]$outputBuilder.AppendLine("# Displayed Columns: $($displayColumns -join ', ')")
    }
    
    [void]$outputBuilder.AppendLine("")

    if($ShowExecutionStats){
        [void]$outputBuilder.AppendLine("Append the stats here later!!")
    }

    return $outputBuilder
}
function Write-ToFile {
    [CmdletBinding()]
    param(
        [Parameter(Mandatory=$true)]
        [string]$FilePath,
        
        [Parameter(Mandatory=$true, ValueFromPipeline=$true)]
        [string]$Content,
        
        [Parameter(Mandatory=$true)]
        [bool]$OpenOutputFileOnFinish
    )
    
    begin {
        try {
            # Ensure the directory exists
            $directory = Split-Path -Path $FilePath -Parent
            if (-not [string]::IsNullOrEmpty($directory) -and -not (Test-Path -Path $directory)) {
                New-Item -Path $directory -ItemType Directory -Force | Out-Null
                Write-Verbose "Created directory: $directory"
            }
            
            # Create or clear the file
            if (Test-Path -Path $FilePath) {
                Clear-Content -Path $FilePath
                Write-Verbose "Cleared existing file: $FilePath"
            } else {
                New-Item -Path $FilePath -ItemType File -Force | Out-Null
                Write-Verbose "Created new file: $FilePath"
            }
        } catch {
            Write-Error "Failed to initialize output file: $_"
            throw
        }
    }
    
    process {
        try {
            $Content | Add-Content -Path $FilePath
        } catch {
            Write-Error "Failed to write to output file: $_"
            throw
        }
    }
    
    end {
        Write-Verbose "Successfully wrote output to $FilePath"
        
        # Open the file after writing if requested
        if ($OpenOutputFileOnFinish) {
            try {
                # Try to resolve the path to handle relative paths
                $resolvedPath = Resolve-Path $FilePath -ErrorAction Stop
                Write-Verbose "Opening file: $resolvedPath"
                
                # Use the appropriate method to open the file based on OS
                if ($IsWindows -or $null -eq $IsWindows) {
                    # On Windows or PowerShell 5.1 where $IsWindows is not defined
                    Start-Process $resolvedPath
                } elseif ($IsMacOS) {
                    # On macOS
                    Start-Process "open" -ArgumentList $resolvedPath
                } elseif ($IsLinux) {
                    # On Linux, try xdg-open first
                    try {
                        Start-Process "xdg-open" -ArgumentList $resolvedPath
                    } catch {
                        # If xdg-open fails, try other common utilities
                        try { Start-Process "nano" -ArgumentList $resolvedPath } catch { 
                            Write-Verbose "Could not open file with xdg-open or nano" 
                        }
                    }
                }
            } catch {
                Write-Warning "Could not open file after writing: $_"
            }
        }
    }
}

# Adds a default .txt extension to a file path if it doesn't already have an extension
function Add-DefaultExtension {
    param (
        [string]$FilePath,
        [bool]$Quiet
    )
    
    if ([string]::IsNullOrEmpty($FilePath)) {
        # If no filepath is set and Quiet is true, use default "PowerTree.txt"
        if ($Quiet) {
            return "PowerTree.txt"
        }
        return $FilePath # meaning no outfile
    }
    
    if ([string]::IsNullOrEmpty([System.IO.Path]::GetExtension($FilePath))) {
        return "$FilePath.txt"
    }
    
    return $FilePath
}