ruffle/core/src/avm1/globals.rs

566 lines
18 KiB
Rust
Raw Normal View History

2019-09-20 19:11:33 +00:00
use crate::avm1::fscommand;
use crate::avm1::function::{Executable, FunctionObject};
use crate::avm1::listeners::SystemListeners;
use crate::avm1::return_value::ReturnValue;
use crate::avm1::{Avm1, Error, Object, ScriptObject, TObject, UpdateContext, Value};
use crate::backend::navigator::NavigationMethod;
use enumset::EnumSet;
use gc_arena::MutationContext;
use rand::Rng;
2019-11-22 22:32:57 +00:00
use std::f64;
mod array;
2020-01-20 08:32:32 +00:00
pub(crate) mod boolean;
pub(crate) mod button;
2020-01-04 03:48:16 +00:00
mod color;
pub(crate) mod display_object;
mod function;
mod key;
mod math;
mod matrix;
pub(crate) mod mouse;
pub(crate) mod movie_clip;
mod movie_clip_loader;
2020-01-20 08:19:55 +00:00
pub(crate) mod number;
mod object;
mod point;
mod rectangle;
2019-12-22 23:32:32 +00:00
mod sound;
mod stage;
pub(crate) mod string;
pub(crate) mod text_field;
mod text_format;
2019-12-20 20:28:49 +00:00
mod xml;
2019-09-20 19:11:33 +00:00
#[allow(non_snake_case, unused_must_use)] //can't use errors yet
2019-09-01 22:40:32 +00:00
pub fn getURL<'a, 'gc>(
avm: &mut Avm1<'gc>,
context: &mut UpdateContext<'a, 'gc, '_>,
_this: Object<'gc>,
2019-09-01 22:40:32 +00:00
args: &[Value<'gc>],
) -> Result<ReturnValue<'gc>, Error> {
//TODO: Error behavior if no arguments are present
if let Some(url_val) = args.get(0) {
let swf_version = avm.current_swf_version();
let url = url_val.clone().into_string(swf_version);
2019-09-20 19:11:33 +00:00
if let Some(fscommand) = fscommand::parse(&url) {
fscommand::handle(fscommand, avm, context);
return Ok(Value::Undefined.into());
2019-09-20 19:11:33 +00:00
}
2019-09-26 18:45:45 +00:00
let window = args.get(1).map(|v| v.clone().into_string(swf_version));
let method = match args.get(2) {
Some(Value::String(s)) if s == "GET" => Some(NavigationMethod::GET),
Some(Value::String(s)) if s == "POST" => Some(NavigationMethod::POST),
2019-09-17 03:37:11 +00:00
_ => None,
};
2019-10-08 14:34:08 +00:00
let vars_method = method.map(|m| (m, avm.locals_into_form_values(context)));
2019-09-17 03:37:11 +00:00
context.navigator.navigate_to_url(url, window, vars_method);
2019-09-01 22:40:32 +00:00
}
Ok(Value::Undefined.into())
2019-09-01 22:40:32 +00:00
}
pub fn random<'gc>(
_avm: &mut Avm1<'gc>,
action_context: &mut UpdateContext<'_, 'gc, '_>,
_this: Object<'gc>,
args: &[Value<'gc>],
) -> Result<ReturnValue<'gc>, Error> {
match args.get(0) {
Some(Value::Number(max)) => Ok(action_context.rng.gen_range(0.0f64, max).floor().into()),
_ => Ok(Value::Undefined.into()), //TODO: Shouldn't this be an error condition?
}
}
pub fn is_nan<'gc>(
avm: &mut Avm1<'gc>,
action_context: &mut UpdateContext<'_, 'gc, '_>,
_this: Object<'gc>,
args: &[Value<'gc>],
) -> Result<ReturnValue<'gc>, Error> {
if let Some(val) = args.get(0) {
Ok(val.as_number(avm, action_context)?.is_nan().into())
} else {
Ok(true.into())
}
}
2019-11-22 22:32:57 +00:00
pub fn get_infinity<'gc>(
avm: &mut Avm1<'gc>,
_action_context: &mut UpdateContext<'_, 'gc, '_>,
_this: Object<'gc>,
2019-11-22 22:32:57 +00:00
_args: &[Value<'gc>],
) -> Result<ReturnValue<'gc>, Error> {
if avm.current_swf_version() > 4 {
Ok(f64::INFINITY.into())
} else {
Ok(Value::Undefined.into())
}
}
pub fn get_nan<'gc>(
avm: &mut Avm1<'gc>,
_action_context: &mut UpdateContext<'_, 'gc, '_>,
_this: Object<'gc>,
2019-11-22 22:32:57 +00:00
_args: &[Value<'gc>],
) -> Result<ReturnValue<'gc>, Error> {
if avm.current_swf_version() > 4 {
Ok(f64::NAN.into())
} else {
Ok(Value::Undefined.into())
}
}
/// This structure represents all system builtins that are used regardless of
/// whatever the hell happens to `_global`. These are, of course,
/// user-modifiable.
#[derive(Clone)]
pub struct SystemPrototypes<'gc> {
pub button: Object<'gc>,
pub object: Object<'gc>,
pub function: Object<'gc>,
pub movie_clip: Object<'gc>,
2019-12-22 23:32:32 +00:00
pub sound: Object<'gc>,
pub text_field: Object<'gc>,
pub text_format: Object<'gc>,
pub array: Object<'gc>,
2019-12-22 03:50:03 +00:00
pub xml_node: Object<'gc>,
pub string: Object<'gc>,
2020-01-20 08:19:55 +00:00
pub number: Object<'gc>,
2020-01-20 08:32:32 +00:00
pub boolean: Object<'gc>,
pub matrix: Object<'gc>,
pub point: Object<'gc>,
pub rectangle: Object<'gc>,
pub rectangle_constructor: Object<'gc>,
}
unsafe impl<'gc> gc_arena::Collect for SystemPrototypes<'gc> {
#[inline]
fn trace(&self, cc: gc_arena::CollectionContext) {
self.object.trace(cc);
self.function.trace(cc);
self.movie_clip.trace(cc);
self.button.trace(cc);
2019-12-22 23:32:32 +00:00
self.sound.trace(cc);
self.text_field.trace(cc);
self.text_format.trace(cc);
self.array.trace(cc);
2019-12-22 03:50:03 +00:00
self.xml_node.trace(cc);
self.string.trace(cc);
2020-01-20 08:19:55 +00:00
self.number.trace(cc);
2020-01-20 08:32:32 +00:00
self.boolean.trace(cc);
self.matrix.trace(cc);
self.point.trace(cc);
self.rectangle.trace(cc);
self.rectangle_constructor.trace(cc);
}
}
/// Initialize default global scope and builtins for an AVM1 instance.
pub fn create_globals<'gc>(
gc_context: MutationContext<'gc, '_>,
) -> (SystemPrototypes<'gc>, Object<'gc>, SystemListeners<'gc>) {
let object_proto = ScriptObject::object_cell(gc_context, None);
let function_proto = function::create_proto(gc_context, object_proto);
object::fill_proto(gc_context, object_proto, function_proto);
let button_proto: Object<'gc> = button::create_proto(gc_context, object_proto, function_proto);
let movie_clip_proto: Object<'gc> =
movie_clip::create_proto(gc_context, object_proto, function_proto);
let movie_clip_loader_proto: Object<'gc> =
movie_clip_loader::create_proto(gc_context, object_proto, function_proto);
2019-12-22 23:32:32 +00:00
let sound_proto: Object<'gc> = sound::create_proto(gc_context, object_proto, function_proto);
let text_field_proto: Object<'gc> =
text_field::create_proto(gc_context, object_proto, function_proto);
let text_format_proto: Object<'gc> =
text_format::create_proto(gc_context, object_proto, function_proto);
let array_proto: Object<'gc> = array::create_proto(gc_context, object_proto, function_proto);
2020-01-04 03:48:16 +00:00
let color_proto: Object<'gc> = color::create_proto(gc_context, object_proto, function_proto);
2019-12-20 20:28:49 +00:00
let xmlnode_proto: Object<'gc> =
xml::create_xmlnode_proto(gc_context, object_proto, function_proto);
2020-01-04 03:48:16 +00:00
let xml_proto: Object<'gc> = xml::create_xml_proto(gc_context, xmlnode_proto, function_proto);
let string_proto: Object<'gc> = string::create_proto(gc_context, object_proto, function_proto);
2020-01-20 08:19:55 +00:00
let number_proto: Object<'gc> = number::create_proto(gc_context, object_proto, function_proto);
2020-01-20 08:32:32 +00:00
let boolean_proto: Object<'gc> =
boolean::create_proto(gc_context, object_proto, function_proto);
let matrix_proto: Object<'gc> = matrix::create_proto(gc_context, object_proto, function_proto);
let point_proto: Object<'gc> = point::create_proto(gc_context, object_proto, function_proto);
let rectangle_proto: Object<'gc> =
rectangle::create_proto(gc_context, object_proto, function_proto);
//TODO: These need to be constructors and should also set `.prototype` on each one
2020-01-20 21:46:46 +00:00
let object = object::create_object_object(gc_context, object_proto, function_proto);
let button = FunctionObject::function(
gc_context,
Executable::Native(button::constructor),
Some(function_proto),
Some(button_proto),
);
let color = FunctionObject::function(
2020-01-04 03:48:16 +00:00
gc_context,
Executable::Native(color::constructor),
Some(function_proto),
Some(color_proto),
);
let function = FunctionObject::function(
gc_context,
Executable::Native(function::constructor),
Some(function_proto),
Some(function_proto),
);
let movie_clip = FunctionObject::function(
gc_context,
Executable::Native(movie_clip::constructor),
Some(function_proto),
Some(movie_clip_proto),
);
let movie_clip_loader = FunctionObject::function(
gc_context,
Executable::Native(movie_clip_loader::constructor),
Some(function_proto),
Some(movie_clip_loader_proto),
);
let sound = FunctionObject::function(
2019-12-22 23:32:32 +00:00
gc_context,
Executable::Native(sound::constructor),
Some(function_proto),
Some(sound_proto),
);
let text_field = FunctionObject::function(
gc_context,
Executable::Native(text_field::constructor),
Some(function_proto),
Some(text_field_proto),
);
let text_format = FunctionObject::function(
gc_context,
Executable::Native(text_format::constructor),
Some(function_proto),
Some(text_format_proto),
);
let array = array::create_array_object(gc_context, Some(array_proto), Some(function_proto));
let xmlnode = FunctionObject::function(
2019-12-20 20:28:49 +00:00
gc_context,
Executable::Native(xml::xmlnode_constructor),
Some(function_proto),
Some(xmlnode_proto),
);
let xml = FunctionObject::function(
gc_context,
Executable::Native(xml::xml_constructor),
Some(function_proto),
Some(xml_proto),
);
2020-01-18 22:48:34 +00:00
let string = string::create_string_object(gc_context, Some(string_proto), Some(function_proto));
2020-01-20 08:19:55 +00:00
let number = number::create_number_object(gc_context, Some(number_proto), Some(function_proto));
2020-01-20 08:32:32 +00:00
let boolean =
boolean::create_boolean_object(gc_context, Some(boolean_proto), Some(function_proto));
let flash = ScriptObject::object(gc_context, Some(object_proto));
let geom = ScriptObject::object(gc_context, Some(object_proto));
let matrix = matrix::create_matrix_object(gc_context, Some(matrix_proto), Some(function_proto));
let point = point::create_point_object(gc_context, Some(point_proto), Some(function_proto));
let rectangle =
rectangle::create_rectangle_object(gc_context, Some(rectangle_proto), Some(function_proto));
flash.define_value(gc_context, "geom", geom.into(), EnumSet::empty());
geom.define_value(gc_context, "Matrix", matrix.into(), EnumSet::empty());
geom.define_value(gc_context, "Point", point.into(), EnumSet::empty());
geom.define_value(gc_context, "Rectangle", rectangle.into(), EnumSet::empty());
let listeners = SystemListeners::new(gc_context, Some(array_proto));
let mut globals = ScriptObject::bare_object(gc_context);
globals.define_value(gc_context, "flash", flash.into(), EnumSet::empty());
globals.define_value(gc_context, "Array", array.into(), EnumSet::empty());
globals.define_value(gc_context, "Button", button.into(), EnumSet::empty());
2020-01-04 03:48:16 +00:00
globals.define_value(gc_context, "Color", color.into(), EnumSet::empty());
globals.define_value(gc_context, "Object", object.into(), EnumSet::empty());
globals.define_value(gc_context, "Function", function.into(), EnumSet::empty());
globals.define_value(gc_context, "MovieClip", movie_clip.into(), EnumSet::empty());
globals.define_value(
gc_context,
"MovieClipLoader",
movie_clip_loader.into(),
EnumSet::empty(),
);
2019-12-22 23:32:32 +00:00
globals.define_value(gc_context, "Sound", sound.into(), EnumSet::empty());
globals.define_value(gc_context, "TextField", text_field.into(), EnumSet::empty());
globals.define_value(
gc_context,
"TextFormat",
text_format.into(),
EnumSet::empty(),
);
2019-12-20 20:28:49 +00:00
globals.define_value(gc_context, "XMLNode", xmlnode.into(), EnumSet::empty());
globals.define_value(gc_context, "XML", xml.into(), EnumSet::empty());
globals.define_value(gc_context, "String", string.into(), EnumSet::empty());
2020-01-20 08:19:55 +00:00
globals.define_value(gc_context, "Number", number.into(), EnumSet::empty());
2020-01-20 08:32:32 +00:00
globals.define_value(gc_context, "Boolean", boolean.into(), EnumSet::empty());
globals.define_value(
gc_context,
"Math",
Value::Object(math::create(
gc_context,
Some(object_proto),
Some(function_proto),
)),
EnumSet::empty(),
);
globals.define_value(
gc_context,
"Mouse",
Value::Object(mouse::create_mouse_object(
gc_context,
Some(object_proto),
Some(function_proto),
&listeners.mouse,
)),
EnumSet::empty(),
);
globals.define_value(
gc_context,
"Key",
Value::Object(key::create_key_object(
gc_context,
Some(object_proto),
Some(function_proto),
)),
EnumSet::empty(),
);
globals.define_value(
gc_context,
"Stage",
Value::Object(stage::create_stage_object(
gc_context,
Some(object_proto),
Some(array_proto),
Some(function_proto),
)),
EnumSet::empty(),
);
globals.force_set_function(
"isNaN",
is_nan,
gc_context,
EnumSet::empty(),
Some(function_proto),
);
globals.force_set_function(
"getURL",
getURL,
gc_context,
EnumSet::empty(),
Some(function_proto),
);
globals.force_set_function(
"random",
random,
gc_context,
EnumSet::empty(),
Some(function_proto),
);
2019-11-03 21:52:10 +00:00
globals.force_set_function(
"ASSetPropFlags",
object::as_set_prop_flags,
gc_context,
EnumSet::empty(),
Some(function_proto),
);
2019-11-22 22:32:57 +00:00
globals.add_property(
gc_context,
"NaN",
Executable::Native(get_nan),
None,
EnumSet::empty(),
);
globals.add_property(
gc_context,
"Infinity",
2019-11-22 22:32:57 +00:00
Executable::Native(get_infinity),
None,
EnumSet::empty(),
);
(
SystemPrototypes {
button: button_proto,
object: object_proto,
function: function_proto,
movie_clip: movie_clip_proto,
2019-12-22 23:32:32 +00:00
sound: sound_proto,
text_field: text_field_proto,
text_format: text_format_proto,
array: array_proto,
2019-12-22 03:50:03 +00:00
xml_node: xmlnode_proto,
string: string_proto,
2020-01-20 08:19:55 +00:00
number: number_proto,
2020-01-20 08:32:32 +00:00
boolean: boolean_proto,
matrix: matrix_proto,
point: point_proto,
rectangle: rectangle_proto,
rectangle_constructor: rectangle,
},
globals.into(),
listeners,
)
2019-09-17 03:37:11 +00:00
}
2019-10-10 12:28:14 +00:00
#[cfg(test)]
#[allow(clippy::unreadable_literal)]
2019-10-10 12:28:14 +00:00
mod tests {
use super::*;
fn setup<'gc>(_avm: &mut Avm1<'gc>, context: &mut UpdateContext<'_, 'gc, '_>) -> Object<'gc> {
create_globals(context.gc_context).1
2019-10-10 12:28:14 +00:00
}
test_method!(boolean_function, "Boolean", setup,
[19] => {
[true] => true,
[false] => false,
[10.0] => true,
[-10.0] => true,
[0.0] => false,
[std::f64::INFINITY] => true,
[std::f64::NAN] => false,
[""] => false,
["Hello"] => true,
[" "] => true,
["0"] => true,
["1"] => true,
2020-01-20 22:06:15 +00:00
[Value::Undefined] => false,
[Value::Null] => false,
[] => Value::Undefined
},
[6] => {
[true] => true,
[false] => false,
[10.0] => true,
[-10.0] => true,
[0.0] => false,
[std::f64::INFINITY] => true,
[std::f64::NAN] => false,
[""] => false,
["Hello"] => false,
[" "] => false,
["0"] => false,
["1"] => true,
2020-01-20 22:06:15 +00:00
[Value::Undefined] => false,
[Value::Null] => false,
[] => Value::Undefined
}
2019-10-10 12:28:14 +00:00
);
test_method!(is_nan_function, "isNaN", setup,
[19] => {
[true] => false,
[false] => false,
[10.0] => false,
[-10.0] => false,
[0.0] => false,
[std::f64::INFINITY] => false,
[std::f64::NAN] => true,
[""] => true,
["Hello"] => true,
[" "] => true,
[" 5 "] => true,
["0"] => false,
["1"] => false,
["Infinity"] => true,
["100a"] => true,
["0x10"] => false,
["0xhello"] => true,
["0x1999999981ffffff"] => false,
["0xUIXUIDFKHJDF012345678"] => true,
["123e-1"] => false,
[] => true
}
);
test_method!(number_function, "Number", setup,
[5, 6] => {
[true] => 1.0,
[false] => 0.0,
[10.0] => 10.0,
[-10.0] => -10.0,
["true"] => std::f64::NAN,
["false"] => std::f64::NAN,
[1.0] => 1.0,
[0.0] => 0.0,
[0.000] => 0.0,
["0.000"] => 0.0,
["True"] => std::f64::NAN,
["False"] => std::f64::NAN,
[std::f64::NAN] => std::f64::NAN,
[std::f64::INFINITY] => std::f64::INFINITY,
[std::f64::NEG_INFINITY] => std::f64::NEG_INFINITY,
[" 12"] => 12.0,
[" \t\r\n12"] => 12.0,
["\u{A0}12"] => std::f64::NAN,
[" 0x12"] => std::f64::NAN,
["01.2"] => 1.2,
[""] => std::f64::NAN,
["Hello"] => std::f64::NAN,
[" "] => std::f64::NAN,
[" 5 "] => std::f64::NAN,
["0"] => 0.0,
["1"] => 1.0,
["Infinity"] => std::f64::NAN,
["100a"] => std::f64::NAN,
["0xhello"] => std::f64::NAN,
["123e-1"] => 12.3,
["0xUIXUIDFKHJDF012345678"] => std::f64::NAN,
[] => 0.0
},
[5] => {
["0x12"] => std::f64::NAN,
["0x10"] => std::f64::NAN,
["0x1999999981ffffff"] => std::f64::NAN,
["010"] => 10,
["-010"] => -10,
["+010"] => 10,
[" 010"] => 10,
[" -010"] => -10,
[" +010"] => 10,
["037777777777"] => 37777777777.0,
["-037777777777"] => -37777777777.0
},
[6, 7] => {
["0x12"] => 18.0,
["0x10"] => 16.0,
["-0x10"] => std::f64::NAN,
["0x1999999981ffffff"] => -2113929217.0,
["010"] => 8,
["-010"] => -8,
["+010"] => 8,
[" 010"] => 10,
[" -010"] => -10,
[" +010"] => 10,
["037777777777"] => -1,
["-037777777777"] => 1
},
[5, 6] => {
[Value::Undefined] => 0.0,
[Value::Null] => 0.0
},
[7] => {
[Value::Undefined] => std::f64::NAN,
[Value::Null] => std::f64::NAN
}
);
2019-10-10 12:28:14 +00:00
}