import cv2
import numpy as np
from PIL import Image
import matplotlib.pyplot as plt

# Constants
VIDEO_FILE = "./Wood2/.mp4" # Video address
FAIL_THRESHOLD = 0.9
MOVING_AVERAGE_WINDOW = 1 # The size of the moving average window
POLYNOMIAL_DEGREE = 1  # The order of polynomial fitting

# Lists to store similarity values and corresponding frame numbers
similarities = []
frame_numbers = []

def d_hash(image):
    """Calculate the difference hash for the given image."""
    hash_bits = []
    for i in range(8):
        for j in range(8):
            hash_bits.append(1 if image[i, j] > image[i, j + 1] else 0)
    return hash_bits

def hamming_distance(hash1, hash2):
    """Calculate the Hamming distance between two hashes."""
    return sum(1 for x, y in zip(hash1, hash2) if x != y)

def process_frame(frame):
    """Process frame to convert it to a format suitable for hash computation."""
    return np.array(Image.fromarray(frame).resize((9, 8), Image.LANCZOS).convert('L'), 'f')

def moving_average(data, window_size):
    """Compute the moving average of the data."""
    return np.convolve(data, np.ones(window_size) / window_size, mode='valid')

def main():
    video = cv2.VideoCapture(VIDEO_FILE)
    edit_detected = False
    frame_count = 0  # Frame counter

    # Read the first frame
    success, prev_frame = video.read()
    if not success:
        print("Failed to read the video file.")
        return

    while frame_count<150:
        # Read the next frame
        success, current_frame = video.read()
        if not success:
            break

        # Process frames
        prev_frame_processed = process_frame(prev_frame)
        current_frame_processed = process_frame(current_frame)

        # Calculate hashes and distance
        hash_prev = d_hash(prev_frame_processed)
        hash_current = d_hash(current_frame_processed)
        distance = hamming_distance(hash_prev, hash_current)
        similarity = 1.0 - distance / 64.0

        # Store similarity and frame number
        frame_numbers.append(frame_count)
        similarities.append(similarity)

        # Check similarity against the threshold
        if similarity < FAIL_THRESHOLD:
            print(f"Frame {frame_count} detected with similarity {similarity}")
            edit_detected = True

        # Update previous frame and increment frame counter
        prev_frame = current_frame
        frame_count += 1

    if not edit_detected:
        print("No edit detected.")

    # Apply moving average to smooth the data
    smoothed_similarities = moving_average(similarities, MOVING_AVERAGE_WINDOW)
    smoothed_frame_numbers = frame_numbers[:len(smoothed_similarities)]  # Adjust frame numbers to match the smoothed data

    # Plot smoothed similarity over frame numbers
    plt.figure(figsize=(10, 6))
    plt.plot(smoothed_frame_numbers, smoothed_similarities, label="Smoothed Similarity", color="blue")
    plt.axhline(FAIL_THRESHOLD, color="red", linestyle="--", label=f"Threshold ({FAIL_THRESHOLD})")
    plt.xlabel("Frame Number")
    plt.ylabel("Similarity")
    plt.title("Smoothed Frame Similarity Over Frame Numbers")
    plt.legend()
    plt.grid(True)
    plt.show()

if __name__ == "__main__":
    main()
