JEWEL
February 28, 2025, 12:25am
343
An update is available v2.5.0
Date built: 04.03.25.14.53
AnnotationProcessors now also allows non-YailType parameters. (ref )
The Maven resolver has been updated to the latest version.
2.4.2:
Added multi-threaded downloading technique on fast upgrade
.
Thanks to @Kevinkun for identifying and helping to resolve this issue .
4 Likes
Hi @JEWEL One more issue
I recently created an extension and add Helpers methods. But helpers doc not generate.
JEWEL
March 6, 2025, 10:20am
345
Please tell me more! How was it written, and how should it have been?
JEWEL
March 8, 2025, 11:48am
347
An update is available v2.6.0
Date built: 08.03.25.15.34
Added blocks generator for AppInventor
, Kodular
, Niotron
& AndroidBuilder
by @Patryk_F
UTF-8
encoding issues have been resolved.
7 Likes
Hi everyone, I’m a beginner in Java and currently trying to use Fast to create an .aix
extension for use in MIT App Inventor 2 . However, I encountered an issue when trying to compile the project into an .apk
.
Here are my .java
, fast.yml
, and assets/*.png
files:
Fast_BluetoothHelper_v1_6.zip - Google Drive
Issue summary:
When I import the .aix
into AI2, the extension’s icon doesn't display. (Could the image file be too large?)
The .aix
can be imported into the AI2 website and runs fine via the MIT AI2 Companion simulator,
but it fails to compile into an .apk
– even when there’s no code block used in the project.
I noticed that if I use the default .java
file provided by Fast instead of my own, the project can compile into an .apk
without any problem.
I would really appreciate it if any experienced developer could take a look and help me figure out what’s wrong. Thanks a lot in advance!
If possible, please help me check whether there’s anything wrong with my .java
or fast.yml
settings.
Hey, your URL miss fast.yml, and assets/*.png
And can u share with us the error message!
2 Likes
Sorry, I forgot to include some files in the previous upload.
Here is the updated compressed folder with all necessary files (.java
, fast.yml
, and assets/*.png
):
Fast_BluetoothHelper_v1_6.zip - Google Drive
U have a lot of problems.
Do u used ai to create this extension?
1 Like
Oh No! Really? Oh no... that’s so disappointing.
I’ve been learning through web resources and AI.
(But in the end... ~)
Could you please let me know what went wrong?
1 Like
JEWEL
March 31, 2025, 8:52am
355
Here are a few open-source projects you might find helpful to learn from.
2 Likes
Update:
icon size 24/24px
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.
JEWEL
March 31, 2025, 1:38pm
359
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.
mit-cml:master
← jarlisson2:edit
opened 05:37AM - 03 Jul 20 UTC
With this implementation it is possible to add an .aar type library in extension… . Just use the existing annotation "@UsesLibraries (libraries ="library.aar ") and add the library to the dependencies, just like the .jar files.
I plan to release several tutorial videos demonstrating FAST project creation, build processes, and installation procedures.
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?