aframe-rs

This is an Aframe library for rust. It's still fairly experimental and a lot might change. I started writing this for a bit of fun to see if I could play with aframe from inside a yew app. It started getting pretty large so I decided to abstract away all the yew-specific stuff and start making a library on its own. There's still a bunch missing and a bunch to do here, but what IS there is functional.

Setup

Currently, this crate doesn't contain any features to initialize Aframe itself, so in your HTML header you ought to include:

```html

```

Beyond that, you can either use this crate's Htmlify trait to output raw html, or use the yew-support feature to create a yew componment (described lower in this readme) to output your actual Aframe scene.

API

Scene

Instantiating a scene:
scene!

Components

Defining a new component:
component_def!

Declaring the structure of a defined component:
componentstruct!
simple
enum!
complex_enum!

Instantiating a component struct:
component!

See the component module for more information and for pre-defined component constants.

Entities & Primitives

Instantiating an entity or defined primitive:
entity!

Defining a new primitive:
primitive!

Shaders

Shader

Htmlify

The Htmlify trait is is to generate raw HTML from the structures provided in this crate. This may eventually be abstracted into a separate crate. (TBD: Is there a better crate in existence already?)

Htmlify

Assets

The assets! and mixin! macros are provided to define an Assets struct. Their signatures are as follows:

assets!
mixin!

Sys API

The lowest-level calls to Aframe are defined in the sys module:

registerPrimitive
registerComponent
registerShader

yew_support feature

The yew_support feature adds yew support to this crate. At its core, all this does is implement From<&Scene> for Html. This allows you to write a yew component as such:

```rust,ignore static INIT: AtomicBool = AtomicBool::new(false);

[derive(Clone, PartialEq, Properties)]

pub struct AframeProps { scene: aframe::Scene }

pub struct Aframe { props: AframeProps }

impl crate::utils::Component for Aframe { type Message = Msg; type Properties = AframeProps;

fn create(props: Self::Properties, _: ComponentLink<Self>) -> Self 
{
    // Register aframe stuff first time only
    if !INIT.load(Ordering::Relaxed)
    {
        unsafe 
        {
            // Code in this block registers shaders, components, and primitives with aframe
            shaders::register_shaders(); 
            component::register_components();
            primitive::register_primitives();
        }
        INIT.store(true, Ordering::Relaxed)
    }
    Self 
    { 
        props
    }
}

fn update(&mut self, _: Self::Message) -> ShouldRender 
{
    true
}

fn change(&mut self, _props: Self::Properties) -> ShouldRender 
{
    false
}

fn view(&self) -> Html 
{
    (&self.props.scene).into()
}

} ```

Example

Below is a full example of how a scene is constructed in yew (this also serves of a valid example of how to use the scene! macro even outside of a yew context):

```rust,ignore html! { , Cow<'static, str>); 1] = [(Cow::Borrowed("color"), Cow::Borrowed("lightblue"))]; scene! { // TODO: Some of these attributes are actually components, they need to be implemented in the library! attributes: ("inspector", "true"), ("embedded", "true"), ("cursor", "rayOrigin: mouse"), ("mixin", "intersectray"), ("style", "min-height: 50px;"), assets: assets! { // Assume we have a few assets available to use Image::new("ramen", "/pics/ramen.png"), Image::new("noise", "/pics/noise.bmp"), // Create a mixin for shadows to know what to interact with mixin! { "intersectray", ("raycaster", component! { RayCaster, objects: List(Cow::Borrowed(&[Cow::Borrowed("#ramen-cube, #water")])) }) } }, children: // The camera rig entity! { attributes: ("id", "rig"), components: ("position", component::Position { x: 0.0, y: 0.0, z: 0.0 }), ("geometry", component! { component::Geometry, primitive: component::GeometryPrimitive::Ring { radiusinner: 0.06, radiusouter: 0.2, segmentstheta: 32, segmentsphi: 8, thetastart: 0.0, thetalength: 360.0 } }), ("material", component! { component::Material, props: component::MaterialProps(Cow::Borrowed(&CURSORCOLOR)), opacity: 0.8 }), children: // The camera entity! { attributes: ("id", "camera"), components: ("position", component::Position { x: 0.0, y: 1.8, z: 0.0 }), ("camera", component!(component::Camera)), ("look-controls", component!(component::LookControls)) }, }, entity! { attributes: ("id", "cube-rig"), components: ("position", component::Position{x: 0.0, y: 2.5, z: -2.0}), ("sound", component! { component::Sound, src: Cow::Borrowed("#ambientmusic"), volume: 0.5 }), ("light", component! { component::Light, lighttype: component::LightType::Point { decay: 1.0, distance: 50.0, shadow: component::OptionalLocalShadow::NoCast{}, }, intensity: 0.0 }), ("animationmouseenter", component! { component::Animation, property: Cow::Borrowed("light.intensity"), to: Cow::Borrowed("1.0"), startevents: component::List(Cow::Borrowed(&[Cow::Borrowed("mouseenter")])), dur: 250 }), ("animationmouseleave", component! { component::Animation, property: Cow::Borrowed("light.intensity"), to: Cow::Borrowed("0.0"), start_events: component::List(Cow::Borrowed(&[Cow::Borrowed("mouseleave")])), dur: 250 }), // This assumes the existence of a primitive registered as "ramen-cube" children: entity! { primitive: "ramen-cube", attributes: ("id", "ramen-cube"), components: // None } },

        // Ambient light
        entity!
        {
            attributes: ("id", "ambient-light"),
            components: ("light", component!
            {
                component::Light,
                light_type: component::LightType::Ambient{},
                color: color::GREY73,
                intensity: 0.2
            })
        },

        // Directional light
        entity!
        {
            attributes: ("id", "directional-light"),
            components: 
            ("position", component::Position{ x: 0.5, y: 1.0, z: 1.0 }),
            ("light", component!
            {
                component::Light,
                light_type: component::LightType::Directional
                {
                    shadow: component::OptionalDirectionalShadow::Cast
                    {
                        shadow: component!
                        {
                            component::DirectionalShadow
                        }
                    }
                },
                color: color::WHITE,
                intensity: 0.1
            })
        },
        // The sky
        entity!
        {
            primitive: "a-sky",
            attributes: ("id", "sky"),
            components: ("material", component!
            {
                component::Material, 
                // This assumes the existence of a shader registered as "strobe"
                shader: Cow::Borrowed("strobe"),
                props: component::MaterialProps(Cow::Owned(vec!
                (
                    (Cow::Borrowed("color"), Cow::Borrowed("black")),
                    (Cow::Borrowed("color2"), Cow::Borrowed("#222222"))
                )))
            })
        },
        // The ocean
        entity!
        {
            primitive: "a-ocean",
            attributes: ("id", "water"), ("depth", "100"), ("width", "100"), ("amplitude", "0.5"),
            components: ("material", component!
            {
                component::Material, 
                // This assumes the existence of a shader registered as "water"
                shader: Cow::Borrowed("water"),
                props: component::MaterialProps(Cow::Owned(vec!((Cow::Borrowed("transparent"), Cow::Borrowed("true")))))
            })
        }
    }
} />

} ```