🏃‍♂️ Fast : An Efficient Way to Build Extensions

Oh No! Really? Oh no... that’s so disappointing. :cry:

I’ve been learning through web resources and AI.
(But in the end... :cry:~)
Could you please let me know what went wrong?

1 Like

I will fix it for you

1 Like

Here are a few open-source projects you might find helpful to learn from.

2 Likes

Update:
icon size 24/24px
jewelry

AndroidManifest.xml

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
  package="com.LLL.BluetoothHelper">

    <uses-permission android:name="android.permission.BLUETOOTH"/>
    <uses-permission android:name="android.permission.BLUETOOTH_ADMIN"/>
    <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"/>
    <uses-permission android:name="android.permission.BLUETOOTH_SCAN"/>
    <uses-permission android:name="android.permission.BLUETOOTH_CONNECT"/>


</manifest>

fast.yml

author: LLL

min_sdk: 7

proguard: true

R8: true

desugar_sources: false

desugar_deps: false

desugar_dex: true

gen_docs: true

auto_version: true

deannonate: true

filter_mit_classes: false

BluetoothHelper.java


package com.LLL.BluetoothHelper;

import android.app.Activity;
import android.bluetooth.BluetoothAdapter;
import android.bluetooth.BluetoothDevice;
import android.content.*;
import android.location.LocationManager;
import android.os.Build;
import android.provider.Settings;
import android.content.pm.PackageManager;
import com.google.appinventor.components.annotations.*;
import com.google.appinventor.components.common.ComponentCategory;
import com.google.appinventor.components.runtime.*;
import java.util.HashSet;
import java.util.ArrayList;
import java.util.List;

@DesignerComponent(version = 8,
        description = "Bluetooth Helper Extension - by LLL. Version 1.6 . Bluetooth Helper Extension for App Inventor.\n" +
                "  Includes: Bluetooth name lookup, GPS check, permission request,\n" +
                "  device discovery (no internal deduplication), events for Bluetooth/GPS state changes,\n" +
                "  and checking individual permissions.",
        category = ComponentCategory.EXTENSION,
        nonVisible = true,
        iconName = "jewelry.png")

public class BluetoothHelper extends AndroidNonvisibleComponent implements Component, ActivityResultListener {

    private final Activity activity;
    private final Context context;
    private final BluetoothAdapter bluetoothAdapter;
    private final BroadcastReceiver receiver;
    private final BroadcastReceiver stateChangeReceiver;

    public BluetoothHelper(ComponentContainer container) {
        super(container.$form());
        activity = container.$context();
        context = container.$context();
        bluetoothAdapter = BluetoothAdapter.getDefaultAdapter();

        receiver = new BroadcastReceiver() {
            public void onReceive(Context context, Intent intent) {
                String action = intent.getAction();
                if (BluetoothDevice.ACTION_FOUND.equals(action)) {
                    BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);
                    if (device != null) {
                        String name = device.getName();
                        String address = device.getAddress();
                        if (address != null) {
                            if (name == null || name.isEmpty()) {
                                name = "Unknown";
                            }
                            DeviceFound(name, address);
                        }
                    }
                }
            }
        };

        stateChangeReceiver = new BroadcastReceiver() {
            @Override
            public void onReceive(Context context, Intent intent) {
                if (BluetoothAdapter.ACTION_STATE_CHANGED.equals(intent.getAction())) {
                    int state = intent.getIntExtra(BluetoothAdapter.EXTRA_STATE, -1);
                    if (state == BluetoothAdapter.STATE_ON) {
                        BluetoothEnabled();
                    } else if (state == BluetoothAdapter.STATE_OFF) {
                        BluetoothDisabled();
                    }
                } else if (LocationManager.PROVIDERS_CHANGED_ACTION.equals(intent.getAction())) {
                    if (IsLocationEnabled()) {
                        LocationEnabled();
                    } else {
                        LocationDisabled();
                    }
                }
            }
        };

        IntentFilter filter = new IntentFilter();
        filter.addAction(BluetoothAdapter.ACTION_STATE_CHANGED);
        filter.addAction(LocationManager.PROVIDERS_CHANGED_ACTION);
        context.registerReceiver(stateChangeReceiver, filter);
    }

    @SimpleFunction(description = "Check if the specific permission is granted (e.g. android.permission.BLUETOOTH_CONNECT)")
    public boolean HasPermission(String permissionName) {
        return Build.VERSION.SDK_INT < 23 ||
                activity.checkSelfPermission(permissionName) == PackageManager.PERMISSION_GRANTED;
    }

    @SimpleFunction(description = "Get this device's Bluetooth name")
    public String GetBluetoothName() {
        return bluetoothAdapter != null ? bluetoothAdapter.getName() : "Unavailable";
    }

    @SimpleFunction(description = "Check if Bluetooth is enabled")
    public boolean IsBluetoothEnabled() {
        return bluetoothAdapter != null && bluetoothAdapter.isEnabled();
    }

    @SimpleFunction(description = "Request user to enable Bluetooth (system dialog)")
    public void RequestEnableBluetooth() {
        if (bluetoothAdapter != null && !bluetoothAdapter.isEnabled()) {
            Intent enableBtIntent = new Intent(BluetoothAdapter.ACTION_REQUEST_ENABLE);
            activity.startActivity(enableBtIntent);
        }
    }

    @SimpleFunction(description = "Check if location service (GPS) is enabled")
    public boolean IsLocationEnabled() {
        LocationManager lm = (LocationManager) context.getSystemService(Context.LOCATION_SERVICE);
        return lm.isProviderEnabled(LocationManager.GPS_PROVIDER);
    }

    @SimpleFunction(description = "Show location enable dialog (opens settings)")
    public void ShowLocationPromptDialog() {
        Intent intent = new Intent(Settings.ACTION_LOCATION_SOURCE_SETTINGS);
        activity.startActivity(intent);
    }

    @SimpleFunction(description = "Return list of required permissions by Android version")
    public List<String> GetRequiredPermissions() {
        List<String> permissions = new ArrayList<>();
        if (Build.VERSION.SDK_INT >= 31) {
            permissions.add("android.permission.BLUETOOTH_SCAN");
            permissions.add("android.permission.BLUETOOTH_CONNECT");
            permissions.add("android.permission.ACCESS_FINE_LOCATION");
        } else if (Build.VERSION.SDK_INT >= 23) {
            permissions.add("android.permission.ACCESS_FINE_LOCATION");
        }
        return permissions;
    }

    @SimpleFunction(description = "Start scanning for nearby Bluetooth devices")
    public void StartDiscovery() {
        if (bluetoothAdapter != null && bluetoothAdapter.isEnabled()) {
            IntentFilter filter = new IntentFilter(BluetoothDevice.ACTION_FOUND);
            context.registerReceiver(receiver, filter);
            bluetoothAdapter.startDiscovery();
        }
    }

    @SimpleFunction(description = "Stop scanning for Bluetooth devices")
    public void StopDiscovery() {
        if (bluetoothAdapter != null) {
            bluetoothAdapter.cancelDiscovery();
            try {
                context.unregisterReceiver(receiver);
            } catch (IllegalArgumentException e) {
                // receiver not registered
            }
        }
    }

    @SimpleFunction(description = "Request necessary permissions dynamically (if not already granted)")
    public void RequestRequiredPermissions() {
        List<String> permissions = GetRequiredPermissions();
        if (permissions != null && permissions.size() > 0) {
            activity.requestPermissions(permissions.toArray(new String[0]), 12345);
        }
    }

    @SimpleFunction(description = "Check if all required permissions have been granted")
    public boolean HasRequiredPermissions() {
        if (Build.VERSION.SDK_INT >= 31) {
            return activity.checkSelfPermission("android.permission.BLUETOOTH_SCAN") == 0 &&
                   activity.checkSelfPermission("android.permission.BLUETOOTH_CONNECT") == 0 &&
                   activity.checkSelfPermission("android.permission.ACCESS_FINE_LOCATION") == 0;
        } else if (Build.VERSION.SDK_INT >= 23) {
            return activity.checkSelfPermission("android.permission.ACCESS_FINE_LOCATION") == 0;
        }
        return true;
    }

    @SimpleEvent(description = "Triggered when a Bluetooth device is found (name and MAC address)")
    public void DeviceFound(String deviceName, String deviceAddress) {
        EventDispatcher.dispatchEvent(this, "DeviceFound", deviceName, deviceAddress);
    }

    @SimpleEvent(description = "Triggered when Bluetooth is successfully enabled")
    public void BluetoothEnabled() {
        EventDispatcher.dispatchEvent(this, "BluetoothEnabled");
    }

    @SimpleEvent(description = "Triggered when Bluetooth is turned OFF")
    public void BluetoothDisabled() {
        EventDispatcher.dispatchEvent(this, "BluetoothDisabled");
    }

    @SimpleEvent(description = "Triggered when GPS/Location service is enabled")
    public void LocationEnabled() {
        EventDispatcher.dispatchEvent(this, "LocationEnabled");
    }

    @SimpleEvent(description = "Triggered when GPS/Location service is turned OFF")
    public void LocationDisabled() {
        EventDispatcher.dispatchEvent(this, "LocationDisabled");
    }

    @Override
    public void resultReturned(int requestCode, int resultCode, Intent data) {
    }
}
2 Likes

Hi jewel, welldone for your work and dedication. By the way do you have any tutorial regarding the usage of your builder? Just bow i upgraded my pc to w11. So just share with me any tutorial.

Thanks

1 Like

Hi,
I am trying to create an extension for MIT App Inventor using Fast CLI and would like to integrate the OpenCV library. When converting the opencv.aar file to library.jar, the .so files are not included, causing the library not to work when the extension is used.

I understand that .so files are typically not included in a JAR, but is there any way to include .so files within the extension so it works correctly in MIT App Inventor?

Any suggestions or solutions would be greatly appreciated! Thank you.

It'll be possible once this PR is merged. I'm putting this feature in FAST's next major version since Niotron and AndroidBuilder already have it.

I plan to release several tutorial videos demonstrating FAST project creation, build processes, and installation procedures.

2 Likes
2 Likes

including the so file in extension is not possible maybe, since some so files are very big, that will make the aix too big to import to the ai2 server.

I have made an OpenCV extension, the so files can be downloaded from internet when first time run, or repack the apk file by adding so files to assets.

2 Likes

Thank you very much for the clarification!

I will try these methods and see which one works best for me. Thanks again for your help and for sharing your expertise

1 Like

Can i put file to assets folder of apk without uploading it in the assets of the project?