import tkinter as tk
from tkinter import ttk
from tkinter.scrolledtext import ScrolledText


def on_entry_validate(P):
    """Allow input of numbers, decimal points, negative signs, and intermediate states"""
    allowed_chars = {'-', '.'}
    if all(c in "0123456789-." for c in P) and P.count('.') <= 1 and P.count('-') <= 1 and (
            P.find('-') <= 0 or P.find('-') == 0):
        return True
    return False


def focus_next_entry(event):
    """Enter to jump to the next input box"""
    event.widget.tk_focusNext().focus()
    return "break"


def generate_on_enter(event):
    """Enter the fourth input box to generate the result"""
    calculate_positions()


def calculate_positions():
    try:
        rows = int(row_entry.get())
        cols = int(col_entry.get())
        row_spacing = float(row_space_entry.get().replace(',', '.'))  # Compatible with comma input
        col_spacing = float(col_space_entry.get().replace(',', '.'))
    except ValueError:
        result_text.delete('1.0', tk.END)
        result_text.insert(tk.END, "Error: Please enter a valid number\\n")
        return

    total_width = (cols - 1) * col_spacing
    total_height = (rows - 1) * row_spacing
    start_x = -total_width / 2
    start_y = total_height / 2

    result_text.delete('1.0', tk.END)
    positions = []  # Store all location point information
    index = 1  # initial index
    
    for i in range(rows):
        for j in range(cols):
            dx = start_x + j * col_spacing
            dy = start_y - i * row_spacing
            # Only display the required number of digits
            dx_str = '{0:g}'.format(dx)
            dy_str = '{0:g}'.format(dy)
            # Add indices to the output
            result_text.insert(tk.END, f"{index}, {dx_str}, {dy_str}\\n\n")
            positions.append((i, j, dx, dy, index))  # Save row and column indexes and indices
            index += 1
    
    # Add explanatory text
    result_text.insert(tk.END, "note that indices above are for position reference only, do not reflect the real indices")
    
    # Draw a schematic diagram of points
    draw_points(positions, rows, cols, row_spacing, col_spacing)


def draw_points(positions, rows, cols, row_spacing, col_spacing):
    canvas.delete("all")  # reset contain
    

    canvas_width = canvas.winfo_width()
    canvas_height = canvas.winfo_height()
    
    # Calculate maximum range
    max_x = max(abs(pos[2]) for pos in positions) if positions else 0
    max_y = max(abs(pos[3]) for pos in positions) if positions else 0
    max_range = max(max_x, max_y) * 2 or 1  # Avoid dividing by zero
    
    # Calculate scaling ratio
    scale = min(canvas_width, canvas_height) * 0.7 / max_range
    
    # Draw coordinate axis
    center_x = canvas_width // 2
    center_y = canvas_height // 2
    canvas.create_line(0, center_y, canvas_width, center_y, fill="gray")  # X axis
    canvas.create_line(center_x, 0, center_x, canvas_height, fill="gray")  # Y axis
    
    # Draw the center point
    canvas.create_oval(center_x-3, center_y-3, center_x+3, center_y+3, fill="red")
    
    # Calculate label offset
    label_offset = 0
    point_label_offset = 5  
    
    # Draw all points
    point_size = 4
    for i, j, dx, dy, index in positions:
        # Convert actual coordinates to canvas coordinates
        x = center_x + dx * scale
        y = center_y - dy * scale  # reversing the Y-axis direction
        
        canvas.create_oval(
            x - point_size/2, y - point_size/2,
            x + point_size/2, y + point_size/2,
            fill="blue", outline="blue"
        )
        
        canvas.create_text(
            x , y - point_label_offset,
            text=str(index), fill="black", font=('Arial', 8)
        )




root = tk.Tk()
root.title("2-side coordinates generator")
root.geometry("850x600") 


main_frame = ttk.Frame(root)
main_frame.pack(fill='both', expand=True)


left_frame = ttk.Frame(main_frame)
left_frame.pack(side='left', fill='both', expand=True)


validate_cmd = root.register(on_entry_validate)


input_frame = ttk.Frame(left_frame, padding=10)
input_frame.pack(fill='x')


ttk.Label(input_frame, text="rows：").grid(row=0, column=0, sticky='w', padx=5)
row_entry = ttk.Entry(input_frame, validate="key",
                      validatecommand=(root.register(lambda P: P.isdigit() or P == ""), '%P'))
row_entry.grid(row=0, column=1, sticky='ew', padx=5, pady=2)
row_entry.bind('<Return>', focus_next_entry)


ttk.Label(input_frame, text="columns：").grid(row=1, column=0, sticky='w', padx=5)
col_entry = ttk.Entry(input_frame, validate="key",
                      validatecommand=(root.register(lambda P: P.isdigit() or P == ""), '%P'))
col_entry.grid(row=1, column=1, sticky='ew', padx=5, pady=2)
col_entry.bind('<Return>', focus_next_entry)


ttk.Label(input_frame, text="row spacing：").grid(row=2, column=0, sticky='w', padx=5)
row_space_entry = ttk.Entry(input_frame, validate="key", validatecommand=(validate_cmd, '%P'))
row_space_entry.grid(row=2, column=1, sticky='ew', padx=5, pady=2)
row_space_entry.bind('<Return>', focus_next_entry)


ttk.Label(input_frame, text="col spacing：").grid(row=3, column=0, sticky='w', padx=5)
col_space_entry = ttk.Entry(input_frame, validate="key", validatecommand=(validate_cmd, '%P'))
col_space_entry.grid(row=3, column=1, sticky='ew', padx=5, pady=2)
col_space_entry.bind('<Return>', generate_on_enter)


ttk.Label(input_frame, text="pin indices from left to right, top to bottom").grid(row=0, column=2, rowspan=4, sticky='w', padx=5)


generate_btn = ttk.Button(left_frame, text="generate", command=calculate_positions)
generate_btn.pack(pady=10)


result_text = ScrolledText(left_frame, width=45, height=20, font=('Consolas', 10))
result_text.pack(padx=10, pady=5, fill='both', expand=True)


right_frame = ttk.Frame(main_frame, width=350)
right_frame.pack(side='right', fill='both', expand=True)


ttk.Label(right_frame, text="pin coornidates（with indices）", font=('Arial', 12)).pack(pady=5)


canvas = tk.Canvas(right_frame, bg='white', bd=2, relief='groove')
canvas.pack(padx=10, pady=5, fill='both', expand=True)


row_entry.focus_set()

root.mainloop()