2

我将我的 JavaFX 应用程序移植到我的 Android 设备上。我希望我的应用程序能够读取传入的 SMS 消息并将其存储在数据库中。我在 StackOverflow 中发现了几个问题,但我不知道如何在 JavaFX 方法中实现。请帮忙!

4

1 回答 1

6

这些是创建 JavaFX 应用程序并将其移植到 Android 设备所需的步骤,因此您可以跟踪 SMS 消息,从而允许:

  • 向您键入的号码发送短信。警告:这可能会导致您的移动帐户产生费用。
  • 阅读收件箱中的所有短信列表。
  • 收听收到的短信,并在出现新短信时显示内容。

步骤1

使用 NetBeans 的 Gluon插件创建一个新的 JavaFX 项目。我们称它为 SMSTracker,带有 main 类org.jpereda.sms.SMSTrackerFX。上 build.gradle,将 jfxmobile 插件版本更新为 b9:

dependencies {
    classpath 'org.javafxports:jfxmobile-plugin:1.0.0-b9'
}

首先让SMSMessage我们用我们的模型创建一个 JavaFX pojo:

public class SMSMessage {

    private final StringProperty id;
    private final StringProperty address;
    private final StringProperty msg;
    private final StringProperty readState; //"0" not read, "1" read sms
    private final StringProperty time;
    private final StringProperty folderName;

    public SMSMessage(String id, String address, String msg, String readState, String time, String folderName){
        this.id = new SimpleStringProperty(id);
        this.address = new SimpleStringProperty(address);
        this.msg = new SimpleStringProperty(msg);
        this.readState = new SimpleStringProperty(readState);
        this.time = new SimpleStringProperty(time);
        this.folderName = new SimpleStringProperty(folderName);
    }

    public String getId() {
        return id.get();
    }

    public StringProperty idProperty() {
        return id;
    }

    public String getAddress() {
        return address.get();
    }

    public StringProperty addressProperty() {
        return address;
    }

    public String getMsg() {
        return msg.get();
    }

    public StringProperty msgProperty() {
        return msg;
    }

    public String getReadState() {
        return readState.get();
    }

    public StringProperty readStateProperty() {
        return readState;
    }

    public String getTime() {
        return time.get();
    }

    public StringProperty timeProperty() {
        return time;
    }

    public String getFolderName() {
        return folderName.get();
    }

    public StringProperty folderNameProperty() {
        return folderName;
    }

    @Override
    public String toString(){
        return id.get()+ ": " + address.get() + ": " + msg.get();
    }
}

使用 ScenicBuilder 和 FXML 或通过代码创建您的 UI。对于此示例,它将是一个包含三个主要区域的简单 UI,用于上述三个选项。

public class SMSTrackerFX extends Application {

    @Override
    public void start(Stage stage) {
        BorderPane root = new BorderPane();

        /*
        TOP :: Sending SMS
        Warning: This may by subjected to costs to your mobile account
        */
        Button buttonSend = new Button("Send SMS");

        TextField number = new TextField();
        number.setPromptText("Insert number");
        HBox.setHgrow(number, Priority.ALWAYS);
        HBox hbox = new HBox(10,buttonSend, number);

        TextField message = new TextField();
        message.setPromptText("Insert text");
        HBox.setHgrow(message, Priority.ALWAYS);

        VBox vboxTop = new VBox(10,hbox,message);

        buttonSend.disableProperty().bind(Bindings.createBooleanBinding(()->{
                return number.textProperty().isEmpty()
                        .or(message.textProperty().isEmpty()).get();
            }, number.textProperty(),message.textProperty()));

        vboxTop.setPadding(new Insets(10));
        root.setTop(vboxTop);

        /*
        CENTER :: Reading SMS Inbox
        */
        Button button = new Button("Read SMS Inbox");

        ListView<SMSMessage> view = new ListView<>();
        view.setCellFactory(data -> new SMSListCell());
        VBox.setVgrow(view, Priority.ALWAYS);

        VBox vboxCenter = new VBox(10,button,view);
        vboxCenter.setPadding(new Insets(10));
        root.setCenter(vboxCenter);

        /*
        BOTTOM :: Listening to incoming SMS
        */
        Label incoming = new Label("No messages");

        VBox vboxBottom = new VBox(10,new Label("Incoming SMS"),incoming);
        vboxBottom.setPadding(new Insets(10));

        root.setBottom(vboxBottom);

        Rectangle2D visualBounds = Screen.getPrimary().getVisualBounds();
        Scene scene = new Scene(root, visualBounds.getWidth(), visualBounds.getHeight());
        stage.setScene(scene);
        stage.show();
    }

    private static class SMSListCell extends ListCell<SMSMessage> {

        @Override
        protected void updateItem(SMSMessage item, boolean empty) {
            super.updateItem(item, empty);
            if (item != null && !empty) {
                setGraphic(new Label(item.getId()+ ": " + item.getMsg()));
            } else {
                setGraphic(null);
            }
        }
    }

}

这就是我们现在所拥有的:

短信1

第2步

在 HelloPlatform示例之后,我们将添加一个PlatformService类和一个PlatformProvider接口,以及所需的服务:

public interface PlatformProvider {

    void sendSMS(String number, String message);

    List<SMSMessage> readSMSs();

    void listenToIncomingSMS();

    ObjectProperty<SMSMessage> messageProperty();

}

该接口将在三个平台(桌面、iOS 和 Android)中的每一个上实现。显然,我们将只关注 Android 实现。

这是 NetBeans 上的项目视图,其中包含所有涉及的包和文件:

短信2

第 3 步

现在让我们在 Android 上实现这些服务。为此,我在 SO 上遵循了几个很好的答案:发送SMS、阅读收件箱和收听传入的SMS

从最后一个开始,我们可以创建一个类,每次收到ObjectProperty一个对象时都设置一个对象:SMSMessage

public class SmsListener extends BroadcastReceiver {

    private final ObjectProperty<SMSMessage> messages = new SimpleObjectProperty<>();

    @Override
    public void onReceive(Context cntxt, Intent intent) {
        if(intent.getAction().equals(Intents.SMS_RECEIVED_ACTION)){
            for (SmsMessage smsMessage : Intents.getMessagesFromIntent(intent)) {
                SMSMessage sms = new SMSMessage("0", smsMessage.getOriginatingAddress(),
                        smsMessage.getMessageBody(), smsMessage.getStatus()==1?"read":"not read", 
                        Long.toString(smsMessage.getTimestampMillis()), "inbox");
                messages.set(sms);
            }
        }
    }

    public ObjectProperty<SMSMessage> messagesProperty() {
        return messages;
    }

}

为了启动这个监听器,我们使用FXActivity扩展 Android Context 类并提供对 Android 服务的访问的类来注册一个实例SmsListener

public class AndroidPlatformProvider implements PlatformProvider {

    private final SmsListener receiver = new SmsListener();

    @Override
    public void listenToIncomingSMS() {
        FXActivity.getInstance().registerReceiver(receiver, new IntentFilter(Intents.SMS_RECEIVED_ACTION));
    }

    @Override
    public ObjectProperty<SMSMessage> messagesProperty() {
        return receiver.messagesProperty();
    }

}

第4步

最后,我们需要做的就是将属性与我们在 UI 上的标签绑定并开始广播:

    @Override
    public void start(Stage stage) {
        ...    
    PlatformService.getInstance().messageProperty().addListener(
        (obs,s,s1)->{
            Platform.runLater(()->incoming.setText(s1.toString()));
        });

        // start broadcast
        PlatformService.getInstance().listenToIncomingSMS();
    }    

注意Platform.runLater()更新标签的使用:广播线程与 JavaFX 线程不同。

第 5 步

构建 apk 之前的最后一步是修改AndroidManifest.xml文件以请求所需的权限。

由于 jfxmobile-plugin 默认创建一个,gradlew android在这个阶段运行会生成它。它位于SMSTracker\build\javafxports\tmp\android文件夹下。

将其复制到另一个位置 ( SMSTracker\lib) 并将其引用包含在build.gradle文件中:

jfxmobile {
    android {
        manifest = 'lib/AndroidManifest.xml'
    }
}

现在编辑文件并添加所需的权限和接收者:

<?xml version="1.0" encoding="UTF-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android" package="org.jpereda.sms" android:versionCode="1" android:versionName="1.0">
        <supports-screens android:xlargeScreens="true"/>
        <uses-permission android:name="android.permission.READ_SMS"/>
        <uses-permission android:name="android.permission.SEND_SMS"/>
        <uses-permission android:name="android.permission.RECEIVE_SMS" />
        <uses-sdk android:minSdkVersion="4" android:targetSdkVersion="21"/>
        <application android:label="SMSTracker" android:name="android.support.multidex.MultiDexApplication">
                <activity android:name="javafxports.android.FXActivity" android:label="SMSTracker" android:configChanges="orientation|screenSize">
                        <meta-data android:name="main.class" android:value="org.jpereda.sms.SMSTrackerFX"/>
                        <meta-data android:name="debug.port" android:value="0"/>
                        <intent-filter>
                                <action android:name="android.intent.action.MAIN"/>
                                <category android:name="android.intent.category.LAUNCHER"/>
                        </intent-filter>
                </activity>
                <receiver android:name=".SmsListener">
                    <intent-filter android:priority="2147483647">
                        <action android:name="android.provider.Telephony.SMS_RECEIVED" />
                    </intent-filter>
                </receiver>
        </application>
</manifest>

保存、构建并运行gradlew androidInstall以将 apk 上传到您的设备。

完整项目

此应用程序的所有代码都可以在这里找到,包括准备安装在 Android 设备上的apk 。

于 2015-06-17T22:22:18.473 回答