package diff

// EditType represents the type of edit operation
type EditType string

const (
	EditTypeStrReplace EditType = "str_replace" // Search and replace content
	EditTypeInsert     EditType = "insert"      // Insert content after a line
	EditTypeCreate     EditType = "create"      // Create a new file
	EditTypeDelete     EditType = "delete"      // Delete a file (not supported by str_replace_editor)
	EditTypeRename     EditType = "rename"      // Rename/move a file
)

// Edit represents a single edit operation that can be applied to a file.
// This is a structured representation that maps directly to str_replace_editor commands.
type Edit struct {
	Type       EditType `json:"type"`                  // Type of edit operation
	Path       string   `json:"path"`                  // File path (target path for rename)
	OldPath    string   `json:"old_path,omitempty"`    // For rename: original file path
	OldContent string   `json:"old_content,omitempty"` // For str_replace: content to search for
	NewContent string   `json:"new_content,omitempty"` // For str_replace/insert/create: new content
	InsertLine int      `json:"insert_line,omitempty"` // For insert: line number to insert after
	StartLine  int      `json:"start_line,omitempty"`  // Approximate start line (for reference)
	EndLine    int      `json:"end_line,omitempty"`    // Approximate end line (for reference)
}

// CommitEdits represents all edits from a single commit
type CommitEdits struct {
	SHA            string `json:"sha"`
	Message        string `json:"message"`
	RefinedMessage string `json:"refined_message,omitempty"`
	Edits          []Edit `json:"edits"`
}

// FileEdits groups edits by file path
type FileEdits struct {
	Path  string `json:"path"`
	Edits []Edit `json:"edits"`
}

// TranslationResult contains the result of translating a diff patch
type TranslationResult struct {
	Success bool   `json:"success"`
	Error   string `json:"error,omitempty"`
	Edits   []Edit `json:"edits,omitempty"`
}

// TranslatePatchToEdits converts a diff patch string to a list of Edit structs.
// This is the main entry point for translating diffs to edits.
// It handles all edge cases including:
// - New file creation (from /dev/null)
// - File deletion (to /dev/null)
// - File rename/move
// - Addition-only hunks
// - Deletion-only hunks
// - Mixed changes
// - Empty patches (treated as new empty file creation)
func TranslatePatchToEdits(patch string, filePath string) TranslationResult {
	if patch == "" {
		// Empty patch typically means a new empty file was created
		// (GitHub API returns null/empty patch for empty files)
		// Generate a create edit for an empty file
		return TranslationResult{
			Success: true,
			Edits: []Edit{{
				Type:       EditTypeCreate,
				Path:       filePath,
				NewContent: "",
			}},
		}
	}

	diff, err := ParsePatch(patch)
	if err != nil {
		return TranslationResult{
			Success: false,
			Error:   err.Error(),
		}
	}

	if diff.IsBinary {
		return TranslationResult{
			Success: false,
			Error:   "binary file",
		}
	}

	// Handle special cases based on file paths in diff
	edits, err := translateDiffToEdits(diff, filePath)
	if err != nil {
		return TranslationResult{
			Success: false,
			Error:   err.Error(),
		}
	}

	return TranslationResult{
		Success: true,
		Edits:   edits,
	}
}

// translateDiffToEdits handles the translation logic for different diff types
func translateDiffToEdits(diff *FileDiff, filePath string) ([]Edit, error) {
	// Check for new file creation
	if diff.IsNewFile {
		return translateNewFile(diff, filePath)
	}

	// Check for file deletion
	if diff.IsDeletedFile {
		return translateDeletedFile(diff, filePath)
	}

	// Check for file rename
	if diff.IsRename {
		return translateRename(diff)
	}

	// Regular file modification - translate each hunk
	var edits []Edit
	for _, hunk := range diff.Hunks {
		edit := hunkToEdit(hunk, filePath)
		edits = append(edits, edit)
	}

	return edits, nil
}

// translateNewFile handles new file creation (from /dev/null)
func translateNewFile(diff *FileDiff, filePath string) ([]Edit, error) {
	// Collect all added lines from all hunks
	var allContent []string
	for _, hunk := range diff.Hunks {
		for _, line := range hunk.Lines {
			if line.Type == LineAdded {
				allContent = append(allContent, line.Content)
			}
		}
	}

	// Use the new path from diff if available, otherwise use provided filePath
	path := filePath
	if diff.NewPath != "" && diff.NewPath != "/dev/null" {
		path = diff.NewPath
	}

	return []Edit{{
		Type:       EditTypeCreate,
		Path:       path,
		NewContent: joinLines(allContent),
	}}, nil
}

// translateDeletedFile handles file deletion (to /dev/null)
func translateDeletedFile(diff *FileDiff, filePath string) ([]Edit, error) {
	// Use the old path from diff if available, otherwise use provided filePath
	path := filePath
	if diff.OldPath != "" && diff.OldPath != "/dev/null" {
		path = diff.OldPath
	}

	return []Edit{{
		Type: EditTypeDelete,
		Path: path,
	}}, nil
}

// translateRename handles file rename/move operations
func translateRename(diff *FileDiff) ([]Edit, error) {
	edits := []Edit{{
		Type:    EditTypeRename,
		Path:    diff.NewPath,
		OldPath: diff.OldPath,
	}}

	// If there are also content changes, add str_replace edits
	for _, hunk := range diff.Hunks {
		if hasChanges(hunk) {
			edit := hunkToEdit(hunk, diff.NewPath)
			edits = append(edits, edit)
		}
	}

	return edits, nil
}

// hasChanges checks if a hunk has actual changes (not just context)
func hasChanges(hunk Hunk) bool {
	for _, line := range hunk.Lines {
		if line.Type == LineAdded || line.Type == LineRemoved {
			return true
		}
	}
	return false
}

// hunkToEdit converts a single hunk to an Edit struct.
// It intelligently chooses the edit type based on the hunk content:
// - Addition-only at start of file -> insert at line 0
// - Addition-only with context -> str_replace with context
// - Deletion-only -> str_replace with empty replacement
// - Mixed changes -> str_replace
func hunkToEdit(hunk Hunk, filePath string) Edit {
	var oldLines []string
	var newLines []string
	var contextBeforeLines []string
	var contextAfterLines []string
	var addedLines []string
	var removedLines []string

	// Track whether we've seen any changes yet
	seenChanges := false

	// Separate lines by type and track structure
	for _, line := range hunk.Lines {
		switch line.Type {
		case LineContext:
			oldLines = append(oldLines, line.Content)
			newLines = append(newLines, line.Content)
			if !seenChanges {
				contextBeforeLines = append(contextBeforeLines, line.Content)
			} else {
				contextAfterLines = append(contextAfterLines, line.Content)
			}
		case LineRemoved:
			oldLines = append(oldLines, line.Content)
			removedLines = append(removedLines, line.Content)
			seenChanges = true
		case LineAdded:
			newLines = append(newLines, line.Content)
			addedLines = append(addedLines, line.Content)
			seenChanges = true
		}
	}

	// Calculate end line
	endLine := hunk.OldStart + hunk.OldCount - 1
	if endLine < hunk.OldStart {
		endLine = hunk.OldStart
	}

	// Determine the best edit type based on hunk structure
	hasRemovals := len(removedLines) > 0
	hasAdditions := len(addedLines) > 0
	hasContextBefore := len(contextBeforeLines) > 0

	// Special case: Addition-only at the very start of file (line 1, no context before)
	// This means we're prepending content to the file
	if !hasRemovals && hasAdditions && hunk.OldStart == 1 && !hasContextBefore {
		// Use insert at line 0 (prepend to file)
		return Edit{
			Type:       EditTypeInsert,
			Path:       filePath,
			NewContent: joinLines(addedLines),
			InsertLine: 0,
			StartLine:  1,
			EndLine:    len(addedLines),
		}
	}

	// Default: use str_replace for all other cases
	// This handles:
	// - Addition-only with context (context + new lines)
	// - Deletion-only (old lines -> empty or just context)
	// - Mixed changes (old lines -> new lines)
	return Edit{
		Type:       EditTypeStrReplace,
		Path:       filePath,
		OldContent: joinLines(oldLines),
		NewContent: joinLines(newLines),
		StartLine:  hunk.OldStart,
		EndLine:    endLine,
	}
}

// joinLines joins lines with newlines, handling empty slices
func joinLines(lines []string) string {
	if len(lines) == 0 {
		return ""
	}
	result := ""
	for i, line := range lines {
		if i > 0 {
			result += "\n"
		}
		result += line
	}
	return result
}

// TranslatePatchToEditsWithContext converts a diff patch with context expansion for uniqueness.
// This version uses the base file content to ensure search strings are unique.
// maxExpand specifies the maximum number of lines to add on each side for context expansion.
// A value of 10-20 is usually sufficient for most cases.
func TranslatePatchToEditsWithContext(patch string, filePath string, fileContent string, maxExpand int) TranslationResult {
	if patch == "" {
		// Empty patch typically means a new empty file was created
		// (GitHub API returns null/empty patch for empty files)
		// Generate a create edit for an empty file
		return TranslationResult{
			Success: true,
			Edits: []Edit{{
				Type:       EditTypeCreate,
				Path:       filePath,
				NewContent: "",
			}},
		}
	}

	diff, err := ParsePatch(patch)
	if err != nil {
		return TranslationResult{
			Success: false,
			Error:   err.Error(),
		}
	}

	if diff.IsBinary {
		return TranslationResult{
			Success: false,
			Error:   "binary file",
		}
	}

	// Handle special cases first (new file, deletion, rename)
	if diff.IsNewFile || diff.IsDeletedFile || diff.IsRename {
		edits, err := translateDiffToEdits(diff, filePath)
		if err != nil {
			return TranslationResult{
				Success: false,
				Error:   err.Error(),
			}
		}
		return TranslationResult{
			Success: true,
			Edits:   edits,
		}
	}

	// Regular file modification with context expansion
	var edits []Edit
	var ce *ContextExpander
	if fileContent != "" {
		ce = NewContextExpander(fileContent)
	}

	for _, hunk := range diff.Hunks {
		edit := hunkToEdit(hunk, filePath)

		// Only expand context for str_replace edits
		if edit.Type == EditTypeStrReplace && ce != nil && edit.OldContent != "" {
			if !ce.IsUnique(edit.OldContent) {
				expandedOld, expandedNew, err := ce.ExpandContext(
					edit.OldContent,
					edit.NewContent,
					hunk.OldStart,
					maxExpand,
				)
				if err == nil {
					edit.OldContent = expandedOld
					edit.NewContent = expandedNew
				}
				// If expansion fails, keep original (may cause issues when applying)
			}
		}

		edits = append(edits, edit)
	}

	return TranslationResult{
		Success: true,
		Edits:   edits,
	}
}
