use wasm_bindgen::prelude::*;
use wasm_bindgen_test::*;
use wasm_bindgen::JsCast;
use js_sys::*;

#[wasm_bindgen]
extern {
    #[wasm_bindgen(js_name = max, js_namespace = Math)]
    static MAX: Function;

    type ArrayPrototype;
    #[wasm_bindgen(method, getter, structural)]
    pub fn push(this: &ArrayPrototype) -> Function;
    #[wasm_bindgen(js_name = prototype, js_namespace = Array)]
    static ARRAY_PROTOTYPE2: ArrayPrototype;
}

#[wasm_bindgen_test]
fn apply() {
    let args = Array::new();
    args.push(&1.into());
    args.push(&2.into());
    args.push(&3.into());
    assert_eq!(MAX.apply(&JsValue::undefined(), &args).unwrap(), 3);

    let arr = JsValue::from(Array::new());
    let args = Array::new();
    args.push(&1.into());
    ARRAY_PROTOTYPE2.push().apply(&arr, &args).unwrap();
    assert_eq!(Array::from(&arr).length(), 1);
}

#[wasm_bindgen(module = "tests/wasm/Function.js")]
extern {
    fn get_function_to_bind() -> Function;
    fn get_value_to_bind_to() -> JsValue;
    fn call_function(f: Function) -> JsValue;
}

#[wasm_bindgen_test]
fn bind() {
    let f = get_function_to_bind();
    let new_f = f.bind(&get_value_to_bind_to());
    assert_eq!(call_function(f), 1);
    assert_eq!(call_function(new_f), 2);
}

#[wasm_bindgen_test]
fn length() {
    assert_eq!(MAX.length(), 2);
    assert_eq!(ARRAY_PROTOTYPE2.push().length(), 1);
}

#[wasm_bindgen_test]
fn name() {
    assert_eq!(JsValue::from(MAX.name()), "max");
    assert_eq!(JsValue::from(ARRAY_PROTOTYPE2.push().name()), "push");
}

#[wasm_bindgen_test]
fn to_string() {
    assert!(MAX.to_string().length() > 0);
}

#[wasm_bindgen_test]
fn function_inheritance() {
    assert!(MAX.is_instance_of::<Function>());
    assert!(MAX.is_instance_of::<Object>());
}