floui-rs

Rust bindings for floui, pronounced "flowy", a proof-of-concept single header C++17 lib inspired by SwiftUI, which wraps native iOS and Android controls/widgets, and integrates into the de facto build environments of each platform (XCode and Android Studio).

image

Currently available controls:

Usage

You can check out the floui-rs-template, which is structured to be able to build for both ios or android from the command-line.

Otherwise, if you would like to do it manually:

Build your library as a static-lib: ```toml

Cargo.toml

[lib] crate-type = ["static-lib"]

[dependencies] floui = "0.1" ``` Build against the android and ios architectures you might need.

Rust

```rust use floui::{enums::, prelude::, widgets::*}; use std::cell::RefCell; use std::rc::Rc;

fn mygui(vc: &ViewController) -> MainView { let count = Rc::from(RefCell::from(0)); MainView::new( &vc, &[ &Button::new("Increment").foreground(Color::Blue).action({ let count = count.clone(); move || { log("Increment clicked"); let mut c = count.borrowmut(); *c += 1; let t: Text = fromid("mytext").unwrap(); t.text(&format!("{}", c)); } }), &Text::new("0").id("mytext").center().bold(), &Button::new("Decrement") .foreground(Color::Red) .action(move || { log("Increment clicked"); let mut c = count.borrowmut(); *c -= 1; let t: Text = fromid("mytext").unwrap(); t.text(&format!("{}", c)); }), ], ) }

use std::os::raw::c_void;

[no_mangle]

extern "C" fn flouihandleevents(arg1: *mut cvoid) { unsafe { ViewController::handleevents(arg1); } }

[no_mangle]

extern "C" fn flouimain(arg1: *mut cvoid, arg2: *mut cvoid, arg3: *mut cvoid) -> *mut c_void { let vc = unsafe { ViewController::new(arg1, arg2, arg3) }; mygui(&vc).underlying() as _ } ```

Notes on certain usages

Creating new widgets

Wrapping platform widgets doesn't require using C++, it requires that the widget implements WidgetExt, and the underlying (JNI jobject pointer for Android, UIView on iOS) can be retrieved. You can use the jni-rs crate and objc crates for such purposes.

Target-specific structure

iOS

extern void *floui_main(void *, void *, void *);

@interface ViewController ()

@end

@implementation ViewController - (void)viewDidLoad { [super viewDidLoad]; floui_main((void *)CFBridgingRetain(self), nil, nil); }

@end ```

Android

find_library(log-lib log)

addlibrary(myapplication SHARED native-lib.cpp) addlibrary(rust-lib STATIC IMPORTED)

if (ANDROIDABI STREQUAL x86) set(RUSTARCH i686-linux-android) elseif (ANDROIDABI STREQUAL armeabi-v7a) set(RUSTARCH armv7-linux-androideabi) elseif (ANDROIDABI STREQUAL arm64-v8a) set(RUSTARCH aarch64-linux-android) elseif (ANDROIDABI STREQUAL x8664) set(RUSTARCH x8664-linux-android) else () message(FATAL "Unknown architecture") endif ()

setproperty(TARGET rust-lib PROPERTY IMPORTEDLOCATIONDEBUG ${CMAKECURRENTLISTDIR}/app/target/${RUSTARCH}/debug/libapp.a) setproperty(TARGET rust-lib PROPERTY IMPORTEDLOCATIONRELEASE ${CMAKECURRENTLISTDIR}/app/target/${RUSTARCH}/release/libapp.a)

targetlinklibraries(myapplication android rust-lib ${log-lib}) - Modify your C++ file to just call the Rust lib. c++

include

include

extern "C" void *floui_main(void *, void *, void *);

extern "C" void flouihandleevents(void *);

extern "C" JNIEXPORT jobject JNICALL JavacomexamplemyapplicationMainActivitymainView(JNIEnv* env, jobject mainactivity, jobject view) { return (jobject) flouimain(env, mainactivity, view); }

extern "C" JNIEXPORT void JNICALL JavacomexamplemyapplicationMainActivityhandleEvent(JNIEnv *env, jobject thiz, jobject view) { flouihandle_events(view); } - Modify your MainActivity.java to look like: java package com.example.myapplication;

import androidx.annotation.NonNull; import androidx.appcompat.app.AppCompatActivity; import androidx.constraintlayout.widget.ConstraintLayout;

import android.os.Bundle; import android.view.View;

import com.google.android.material.slider.Slider;

public class MainActivity extends AppCompatActivity implements View.OnClickListener, Slider.OnChangeListener { static { System.loadLibrary("myapplication"); }

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    ConstraintLayout layout = new ConstraintLayout(this);
    setContentView(layout);
    mainView(layout);
}
public native View mainView(View view);
public native void handleEvent(View view);

@Override
public void onClick(View view) {
    handleEvent(view);
}

@Override
public void onValueChange(@NonNull Slider slider, float value, boolean fromUser) {
    handleEvent(slider);
}

} ```