1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
use crate::{
    clipper_clipper64, clipper_clipper64_add_clip, clipper_clipper64_add_open_subject,
    clipper_clipper64_add_subject, clipper_clipper64_clear, clipper_clipper64_execute,
    clipper_clipper64_execute_tree_with_open, clipper_clipper64_get_preserve_collinear,
    clipper_clipper64_get_reverse_solution, clipper_clipper64_set_preserve_collinear,
    clipper_clipper64_set_reverse_solution, clipper_clipper64_size, clipper_delete_clipper64,
    clipper_delete_paths64, clipper_delete_polytree64, clipper_polytree64, clipper_polytree64_size,
    malloc, ClipType, ClipperClipper64, FillRule, Paths64, PolyTree64,
};

pub struct Clipper64 {
    ptr: *mut ClipperClipper64,
}

impl Clipper64 {
    pub fn new() -> Self {
        let ptr = unsafe {
            let mem = malloc(clipper_clipper64_size());
            clipper_clipper64(mem)
        };
        Self { ptr: ptr }
    }

    pub fn set_preserve_collinear(&self, value: bool) {
        unsafe { clipper_clipper64_set_preserve_collinear(self.ptr, if value { 1 } else { 0 }) }
    }

    pub fn get_preserve_collinear(&self) -> bool {
        unsafe { clipper_clipper64_get_preserve_collinear(self.ptr) == 1 }
    }

    pub fn set_reverse_solution(&self, value: bool) {
        unsafe { clipper_clipper64_set_reverse_solution(self.ptr, if value { 1 } else { 0 }) }
    }

    pub fn get_reverse_solution(&self) -> bool {
        unsafe { clipper_clipper64_get_reverse_solution(self.ptr) == 1 }
    }

    pub fn clear(&self) {
        unsafe { clipper_clipper64_clear(self.ptr) }
    }

    pub fn add_open_subject(&self, open_subject: Paths64) {
        unsafe { clipper_clipper64_add_open_subject(self.ptr, open_subject.get_clipper_paths()) }
    }

    pub fn add_subject(&self, subject: Paths64) {
        unsafe { clipper_clipper64_add_subject(self.ptr, subject.get_clipper_paths()) }
    }

    pub fn add_clip(&self, clip: Paths64) {
        unsafe { clipper_clipper64_add_clip(self.ptr, clip.get_clipper_paths()) }
    }

    pub fn boolean_operation(&self, clip_type: ClipType, fill_rule: FillRule) -> Paths64 {
        let closed_path = Paths64::new(&vec![]).get_clipper_paths();
        let open_path = Paths64::new(&vec![]).get_clipper_paths();
        unsafe {
            let is_success = clipper_clipper64_execute(
                self.ptr,
                clip_type.into(),
                fill_rule.into(),
                closed_path,
                open_path,
            );

            let path = Paths64::from(closed_path);
            clipper_delete_paths64(closed_path);
            clipper_delete_paths64(open_path);
            path
        }
    }

    pub fn boolean_operation_tree(&self, clip_type: ClipType, fill_rule: FillRule) -> PolyTree64 {
        let tree_ptr = unsafe {
            let mem = malloc(clipper_polytree64_size());
            clipper_polytree64(mem, std::ptr::null_mut())
        };
        let open_path = Paths64::new(&vec![]).get_clipper_paths();
        unsafe {
            let is_success = clipper_clipper64_execute_tree_with_open(
                self.ptr,
                clip_type.into(),
                fill_rule.into(),
                tree_ptr,
                open_path,
            );
            clipper_delete_paths64(open_path);
            let tree = PolyTree64::from(tree_ptr);
            clipper_delete_polytree64(tree_ptr);
            tree
        }
    }
}

impl Drop for Clipper64 {
    fn drop(&mut self) {
        unsafe { clipper_delete_clipper64(self.ptr) }
    }
}