package diff

import (
	"fmt"
	"strings"
)

// ContextExpander helps ensure search strings are unique in the target file
type ContextExpander struct {
	fileContent string
	fileLines   []string
}

// NewContextExpander creates a new context expander with the base file content
func NewContextExpander(fileContent string) *ContextExpander {
	return &ContextExpander{
		fileContent: fileContent,
		fileLines:   strings.Split(fileContent, "\n"),
	}
}

// CountOccurrences counts how many times a string appears in the file
func (ce *ContextExpander) CountOccurrences(searchStr string) int {
	return strings.Count(ce.fileContent, searchStr)
}

// IsUnique checks if a search string appears exactly once in the file
func (ce *ContextExpander) IsUnique(searchStr string) bool {
	return ce.CountOccurrences(searchStr) == 1
}

// FindOccurrenceLines returns the line numbers (1-based) where the search string starts
func (ce *ContextExpander) FindOccurrenceLines(searchStr string) []int {
	var lines []int
	searchLines := strings.Split(searchStr, "\n")
	if len(searchLines) == 0 {
		return lines
	}

	firstLine := searchLines[0]
	for i, line := range ce.fileLines {
		if strings.Contains(line, firstLine) || line == firstLine {
			// Check if the full multi-line match works
			if ce.matchesAt(i, searchLines) {
				lines = append(lines, i+1) // 1-based
			}
		}
	}
	return lines
}

// matchesAt checks if searchLines match starting at lineIdx (0-based)
func (ce *ContextExpander) matchesAt(lineIdx int, searchLines []string) bool {
	if lineIdx+len(searchLines) > len(ce.fileLines) {
		return false
	}

	for i, searchLine := range searchLines {
		if ce.fileLines[lineIdx+i] != searchLine {
			return false
		}
	}
	return true
}

// ExpandContext expands the search string with additional context lines until unique
// Returns the expanded search string and the corresponding replacement string
// maxExpand is the maximum number of lines to add on each side
func (ce *ContextExpander) ExpandContext(oldStr, newStr string, hunkStartLine int, maxExpand int) (string, string, error) {
	// If already unique, return as-is
	if ce.IsUnique(oldStr) {
		return oldStr, newStr, nil
	}

	oldLines := strings.Split(oldStr, "\n")
	newLines := strings.Split(newStr, "\n")

	// Find the actual position in the file based on hunk start line
	// hunkStartLine is 1-based
	startIdx := hunkStartLine - 1
	if startIdx < 0 {
		startIdx = 0
	}
	// Clamp startIdx to file bounds
	if startIdx >= len(ce.fileLines) {
		startIdx = len(ce.fileLines) - 1
		if startIdx < 0 {
			startIdx = 0
		}
	}

	// Try expanding context
	for expand := 1; expand <= maxExpand; expand++ {
		// Calculate new boundaries
		newStartIdx := startIdx - expand
		if newStartIdx < 0 {
			newStartIdx = 0
		}
		newEndIdx := startIdx + len(oldLines) + expand
		if newEndIdx > len(ce.fileLines) {
			newEndIdx = len(ce.fileLines)
		}

		// Ensure valid slice bounds
		if newStartIdx >= newEndIdx {
			continue
		}

		// Build expanded old string from file content
		expandedOldLines := ce.fileLines[newStartIdx:newEndIdx]
		expandedOldStr := strings.Join(expandedOldLines, "\n")

		if ce.IsUnique(expandedOldStr) {
			// Build corresponding new string
			// Add prefix context (ensure valid bounds)
			var prefixLines []string
			if newStartIdx < startIdx && startIdx <= len(ce.fileLines) {
				prefixLines = ce.fileLines[newStartIdx:startIdx]
			}
			// Add suffix context
			suffixStartIdx := startIdx + len(oldLines)
			suffixEndIdx := newEndIdx
			var suffixLines []string
			if suffixStartIdx < suffixEndIdx && suffixStartIdx < len(ce.fileLines) && suffixEndIdx <= len(ce.fileLines) {
				suffixLines = ce.fileLines[suffixStartIdx:suffixEndIdx]
			}

			expandedNewLines := append(prefixLines, newLines...)
			expandedNewLines = append(expandedNewLines, suffixLines...)
			expandedNewStr := strings.Join(expandedNewLines, "\n")

			return expandedOldStr, expandedNewStr, nil
		}
	}

	// Could not make unique within maxExpand
	return oldStr, newStr, fmt.Errorf("could not make search string unique within %d lines of expansion", maxExpand)
}

// TranslateWithContext translates a diff with context expansion for uniqueness
func TranslateWithContext(diff *FileDiff, filePath string, fileContent string, maxExpand int) ([]*EditCommand, error) {
	if diff.IsBinary {
		return nil, fmt.Errorf("cannot translate binary file diff")
	}

	ce := NewContextExpander(fileContent)
	var commands []*EditCommand

	for _, hunk := range diff.Hunks {
		cmd, err := TranslateHunk(hunk, filePath)
		if err != nil {
			return nil, fmt.Errorf("failed to translate hunk: %w", err)
		}

		// Try to expand context if not unique
		if cmd.Command == "str_replace" && !ce.IsUnique(cmd.OldStr) {
			expandedOld, expandedNew, err := ce.ExpandContext(cmd.OldStr, cmd.NewStr, hunk.OldStart, maxExpand)
			if err != nil {
				// Log warning but continue with original
				fmt.Printf("[WARN] %s at line %d: %v\n", filePath, hunk.OldStart, err)
			} else {
				cmd.OldStr = expandedOld
				cmd.NewStr = expandedNew
			}
		}

		commands = append(commands, cmd)
	}

	return commands, nil
}

// TranslateAndFormatWithContext is a convenience function that handles context expansion
func TranslateAndFormatWithContext(patch string, filePath string, fileContent string, maxExpand int) (string, error) {
	diff, err := ParsePatch(patch)
	if err != nil {
		return "", fmt.Errorf("failed to parse patch: %w", err)
	}

	if diff.IsBinary {
		return "", fmt.Errorf("binary file, cannot translate")
	}

	var commands []*EditCommand
	if fileContent != "" {
		commands, err = TranslateWithContext(diff, filePath, fileContent, maxExpand)
	} else {
		commands, err = TranslateDiff(diff, filePath)
	}
	if err != nil {
		return "", err
	}

	var parts []string
	for _, cmd := range commands {
		parts = append(parts, FormatEditCommand(cmd))
	}

	return strings.Join(parts, "\n---\n\n"), nil
}

// ValidateEditCommand checks if an edit command can be applied to the given file content
func ValidateEditCommand(cmd *EditCommand, fileContent string) error {
	if cmd.Command != "str_replace" {
		return nil // Only validate str_replace for now
	}

	ce := NewContextExpander(fileContent)
	count := ce.CountOccurrences(cmd.OldStr)

	if count == 0 {
		return fmt.Errorf("search string not found in file")
	}
	if count > 1 {
		lines := ce.FindOccurrenceLines(cmd.OldStr)
		return fmt.Errorf("search string found %d times at lines %v", count, lines)
	}

	return nil
}

// ApplyEdit simulates applying an edit command to file content
// Returns the modified content
func ApplyEdit(cmd *EditCommand, fileContent string) (string, error) {
	if cmd.Command != "str_replace" {
		return "", fmt.Errorf("only str_replace is supported for simulation")
	}

	ce := NewContextExpander(fileContent)
	if !ce.IsUnique(cmd.OldStr) {
		return "", fmt.Errorf("search string is not unique")
	}

	return strings.Replace(fileContent, cmd.OldStr, cmd.NewStr, 1), nil
}
