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
use crate::{ir::*, target::Target};

use convert_case::{Case, Casing};
use lazy_static::lazy_static;
use quote::ToTokens;
use regex::Regex;

#[derive(Debug, Clone, Hash, Eq, PartialEq)]
pub struct IrTypeRustOpaque {
    pub inner_rust: String,
    pub inner_dart: String,
}

fn char_not_alphanumeric(c: char) -> bool {
    !c.is_alphanumeric()
}

fn rust_type_to_dart_type(rust: &str) -> String {
    lazy_static! {
        static ref OPAQUE_FILTER: Regex =
            Regex::new(r"(\bdyn|'static|\bDartSafe|\+ (Send|Sync|UnwindSafe|RefUnwindSafe))\b")
                .unwrap();
    }
    OPAQUE_FILTER
        .replace_all(rust, "")
        .replace(char_not_alphanumeric, "_")
        .to_case(Case::Pascal)
}

impl From<&syn::Type> for IrTypeRustOpaque {
    fn from(rust_ty: &syn::Type) -> Self {
        let inner_dart = match rust_ty {
            syn::Type::Tuple(tup) if tup.elems.is_empty() => "void".to_owned(),
            _ => rust_type_to_dart_type(&rust_ty.into_token_stream().to_string()),
        };

        Self {
            inner_rust: rust_ty.to_token_stream().to_string(),
            inner_dart,
        }
    }
}

impl From<String> for IrTypeRustOpaque {
    fn from(inner_rust: String) -> Self {
        let inner_dart = rust_type_to_dart_type(&inner_rust);
        Self {
            inner_rust,
            inner_dart,
        }
    }
}

impl IrTypeRustOpaque {
    pub fn new_unit() -> Self {
        Self {
            inner_rust: "()".to_owned(),
            inner_dart: "void".to_owned(),
        }
    }
}

impl IrTypeTrait for IrTypeRustOpaque {
    fn visit_children_types<F: FnMut(&IrType) -> bool>(&self, _f: &mut F, _ir_file: &IrFile) {
        // Do nothing.
    }

    fn safe_ident(&self) -> String {
        self.inner_dart.clone()
    }

    fn dart_api_type(&self) -> String {
        self.inner_dart.clone()
    }

    fn dart_wire_type(&self, target: crate::target::Target) -> String {
        if let Target::Wasm = target {
            "Object".to_owned()
        } else {
            self.rust_wire_type(target)
        }
    }

    fn rust_api_type(&self) -> String {
        format!("RustOpaque<{}>", self.inner_rust)
    }

    fn rust_wire_type(&self, target: crate::target::Target) -> String {
        if let Target::Wasm = target {
            "JsValue".into()
        } else {
            format!("wire_{}", self.safe_ident())
        }
    }
}