package main

import (
	"fmt"
	"gocv.io/x/gocv"
	"image"
	"image/color"
	"os"
	wrapper "rainbowsoft.ru/ml_wrapper"
	"strconv"
	"time"
)

func main() {
	xmlFile := "facedetect/data/haarcascade_frontalface_default.xml"

	if len(os.Args) < 3 {
		fmt.Println("How to run:\n\tfacedetect [camera ID] [FPS]")
		return
	}

	// parse args
	deviceID := os.Args[1]
	fixedFPS, err := strconv.Atoi(os.Args[2])
	if err != nil {
		fmt.Println(err)
		return
	}

	video := wrapper.NewVideoCapture()
	err = wrapper.VideoCaptureOpen(video, deviceID)
	if err != nil {
		fmt.Println("Ошибка открытия видео потока:", err)
		return
	}
	defer wrapper.VideoCaptureClose(video)

	// open display window
	window := gocv.NewWindow("Face Detect")
	defer window.Close()

	// color for the rect when faces detected
	blue := color.RGBA{0, 0, 255, 0}

	// load classifier to recognize faces
	classifier := gocv.NewCascadeClassifier()
	defer classifier.Close()

	if !classifier.Load(xmlFile) {
		fmt.Printf("Error reading cascade file: %v\n", xmlFile)
		return
	}

	var ticker *time.Ticker
	if fixedFPS != 0 {
		frameDuration := time.Second / time.Duration(fixedFPS)
		ticker = time.NewTicker(frameDuration)
		defer ticker.Stop()
	}

	lastTime := time.Now()

	fmt.Printf("Start reading device: %v\n", deviceID)
	for {
		start := time.Now()

		empty, err := wrapper.VideoCaptureReadFrame(video)
		if err != nil {
			fmt.Println(err)
			continue
		}
		if empty {
			continue
		}

		// detect faces
		rects := classifier.DetectMultiScale(*video.Img())
		// draw a rectangle around each face on the original image,
		// along with text identifing as "Human"

		// Вычисление FPS
		deltaTime := time.Since(lastTime).Seconds()
		fps := 1.0 / deltaTime
		lastTime = start

		// Вывод FPS на видео
		err = wrapper.VideoCaptureDrawText(video, fmt.Sprintf("FPS: %.2f", fps))
		if err != nil {
			fmt.Println(err)
			break
		}

		for _, r := range rects {
			gocv.Rectangle(video.Img(), r, blue, 3)

			size := gocv.GetTextSize("Human", gocv.FontHersheyPlain, 1.2, 2)
			pt := image.Pt(r.Min.X+(r.Min.X/2)-(size.X/2), r.Min.Y-2)
			gocv.PutText(video.Img(), "Human", pt, gocv.FontHersheyPlain, 1.2, blue, 2)
		}

		// Если задан FPS, ждем следующего тика таймера
		if fixedFPS != 0 {
			<-ticker.C
		}

		// show the image in the window, and wait 1 millisecond
		window.IMShow(*video.Img())
		if window.WaitKey(1) >= 0 {
			break
		}
	}
}