@CoWave-Fall
感谢您,完成了第一篇翻译贡献!
This commit is contained in:
Xingyu Wang 2022-05-21 11:07:53 +08:00
parent 6f3a16bd5c
commit 484d8c8eb6

View File

@ -3,66 +3,59 @@
[#]: author: (Edson Holanda Teixeira Junior https://opensource.com/users/edsonhtj) [#]: author: (Edson Holanda Teixeira Junior https://opensource.com/users/edsonhtj)
[#]: collector: (lujun9972) [#]: collector: (lujun9972)
[#]: translator: (CoWave-Fall) [#]: translator: (CoWave-Fall)
[#]: reviewer: ( ) [#]: reviewer: (wxy)
[#]: publisher: ( ) [#]: publisher: ( )
[#]: url: ( ) [#]: url: ( )
用 XML 和 Java 构建树莓派打印机的用户界面UI 用 XML 和 Java 构建树莓派打印机的用户界面
====== ======
使用 TotalCross 来快速构建嵌入式系统程序的用户界面。
![Tips and gears turning][1] > 使用 TotalCross 来快速构建嵌入式系统程序的用户界面。
从头开始构建 GUI 是一个非常耗时的过程,处理代码中的位置和对齐对于一些程序员来说确实很困难。所以在本文中,我将演示如何使用 XML 加快这一过程。 ![](https://img.linux.net.cn/data/attachment/album/202205/21/110711zv3t7n1o7hllhodt.jpg)
从头开始构建 GUI 是一个非常耗时的过程,以硬编码的方式处理所有的位置和对齐对于一些程序员来说确实很困难。所以在本文中,我将演示如何使用 XML 加快这一过程。
本项目使用 [TotalCross][2] 作为目标框架。 TotalCross 是一个开源的跨平台软件开发工具包 (SDK),旨在更快地为嵌入式设备创建 GUI。 TotalCross 无需在设备上运行 Java 即可提供 Java 的开发优势,因为它使用自己的字节码和虚拟机( <ruby>TC 字节码<rt>TC bytecode</rt></ruby> 和 TCVM来增强性能。 本项目使用 [TotalCross][2] 作为目标框架。TotalCross 是一个开源的跨平台软件开发工具包SDK旨在更快地为嵌入式设备创建 GUI。TotalCross 无需在设备上运行 Java 即可提供 Java 的开发优势,因为它使用自己的字节码和虚拟机(<ruby>TC 字节码<rt>TC bytecode</rt></ruby> 和 TCVM来增强性能。
我还使用了 Knowcode-XML这是一个用于 TotalCross 框架的开源 XML 解析器,它可以将 XML 文件转换为 TotalCross 组件。 我还使用了 Knowcode-XML这是一个用于 TotalCross 框架的开源 XML 解析器,它可以将 XML 文件转换为 TotalCross 组件。
### 项目需求 ### 项目需求
要重现此项目,需要: 要重现此项目,需要:
* [KnowCode-XML][3] * [KnowCode-XML][3]
* [VSCode][4] [或 VSCodium][5] * [VSCode][4] 或 [VSCodium][5]
* [一个 Android 开发环境][6] * [一个 Android 开发环境][6]
* [用于 VSCode 的 TotalCross 插件][7] * [用于 VSCode 的 TotalCross 插件][7]
* 适用于的开发平台([Linux][8]、[Mac][9] 或 [Windows][10])的 Java需要 Java 11或更高版本 * 适用于的开发平台([Linux][8]、[Mac][9] 或 [Windows][10])的 Java需要 Java 11或更高版本
* [Git][11] * [Git][11]
### 制作嵌入式应用程序 ### 制作嵌入式应用程序
该应用程序包含一个具有基本打印功能的嵌入式 GUI例如扫描、打印和复印 该应用程序由一个具有扫描、打印和复印等基本打印功能的嵌入式 GUI 组成
![打印机初始化画面][12] ![打印机初始化画面][12]
(图片提供Edson Holanda Teixeira Jr遵循[CC BY-SA 4.0][13]协议)
构建这个 GUI 需要几个步骤,包括使用 Android-XML 生成 GUI然后使用 Knowcode-XML 解析器在 TotalCross 框架上运行它。 构建这个 GUI 需要几个步骤,包括使用 Android-XML 生成 GUI然后使用 Knowcode-XML 解析器在 TotalCross 框架上运行它。
#### 1\. 生成 Android XML #### 1生成 Android XML
要创建 XML 文件,首先构建一个简单的 Android 屏幕,然后对其进行自定义。 如果你不知道如何编写 Android-XM或者你只是想简单尝试一下你可以从这个 [GitHub 项目][14] 中下载这个应用程序的 XML。 该项目还包含渲染 GUI 要用到的图片。 要创建 XML 文件,首先构建一个简单的 Android 屏幕,然后对其进行自定义。如果你不知道如何编写 Android-XML,或者你只是想简单尝试一下,你可以从这个 [GitHub 项目][14] 中下载这个应用程序的 XML。该项目还包含渲染 GUI 要用到的图片。
#### 2\. 调整 XML #### 2调整 XML
生成 XML 文件后,需要进行一些微调以确保所有内容都已经对齐、比例正确并且图像的路径正确。 生成 XML 文件后,需要进行一些微调以确保所有内容都已经对齐、比例正确并且图像的路径正确。
将 XML 布局添加到 **Layouts** 文件夹,将所有资源添加到 **Drawable** 文件夹。然后您就可以开始自定义 XML 了。 将 XML 布局添加到 `Layouts` 文件夹,将所有资源添加到 `Drawable` 文件夹。然后你就可以开始自定义 XML 了。
例如,如果想要更改 XML 对象的背景,可以更改 `android:background` 属性: 例如,如果想要更改 XML 对象的背景,可以更改 `android:background` 属性:
``` ```
`android:background="@drawable/scan"` android:background="@drawable/scan"
``` ```
您也可以使用 `tools:layout_editor_absoluteX``tools:layout_editor_absoluteY` 更改对象的位置: 你也可以使用 `tools:layout_editor_absoluteX``tools:layout_editor_absoluteY` 更改对象的位置:
``` ```
tools:layout_editor_absoluteX="830dp" tools:layout_editor_absoluteX="830dp"
@ -71,7 +64,6 @@ tools:layout_editor_absoluteY="511dp"
或者使用 `android:layout_width``android:layout_height` 更改对象的大小: 或者使用 `android:layout_width``android:layout_height` 更改对象的大小:
``` ```
android:layout_width="70dp" android:layout_width="70dp"
android:layout_height="70dp" android:layout_height="70dp"
@ -79,7 +71,6 @@ android:layout_height="70dp"
如果要在对象上放置文本,可以使用 `android:textSize`、`android:text`、`android:textStyle` 和 `android:textColor` 如果要在对象上放置文本,可以使用 `android:textSize`、`android:text`、`android:textStyle` 和 `android:textColor`
``` ```
android:textStyle="bold" android:textStyle="bold"
android:textColor="#000000" android:textColor="#000000"
@ -89,91 +80,83 @@ android:text="2:45PM"
下面是一个完整的 XML 对象的示例: 下面是一个完整的 XML 对象的示例:
``` ```
    <ImageButton <ImageButton
           android:id="@+id/ImageButton" android:id="@+id/ImageButton"
           android:layout_width="70dp" android:layout_width="70dp"
           android:layout_height="70dp" android:layout_height="70dp"
           tools:layout_editor_absoluteX="830dp" tools:layout_editor_absoluteX="830dp"
           tools:layout_editor_absoluteY="511dp" tools:layout_editor_absoluteY="511dp"
           android:background="@drawable/home_config" android:background="@drawable/home_config" />
/>;
``` ```
#### 3\. 在 TotalCross 上运行 GUI #### 3在 TotalCross 上运行 GUI
完成所有 XML 调整后,就可以在 TotalCross 上运行它了。 在 TotalCross 扩展(译注:在 VSCode 里面)上创建一个新项目,并将 **XML****Drawable** 文件夹添加到 **Main** 文件夹里。 如果您仍然不确定如何创建 TotalCross 项目,请参阅我们的 [入门指南][15]。 完成所有 XML 调整后,就可以在 TotalCross 上运行它了。在 TotalCross 扩展(LCTT 译注:在 VSCode 里面)上创建一个新项目,并将 `XML``Drawable` 文件夹添加到 `Main` 文件夹里。如果你仍然不确定如何创建 TotalCross 项目,请参阅我们的 [入门指南][15]。
配置好环境后,使用 `totalcross.knowcode.parse.XmlContainerFactory``import totalcross.knowcode.parse.XmlContainerLayout` 在 TotalCross 框架上使用 XML GUI。 可以在其 [GitHub 页面][3] 上找到更多关于使用 KnowCode-XML 的信息。 配置好环境后,使用 `totalcross.knowcode.parse.XmlContainerFactory``import totalcross.knowcode.parse.XmlContainerLayout` 在 TotalCross 框架上使用 XML GUI。 可以在其 [GitHub 页面][3] 上找到更多关于使用 KnowCode-XML 的信息。
#### 4\. 添加过渡效果 #### 4添加过渡效果
这个项目的平滑过渡效果是由 `SlidingNavigator` 类创建的它使用TotalCross的 `ControlAnimation` 类从一个屏幕滑到另一个屏幕。 这个项目的平滑过渡效果是由 `SlidingNavigator` 类创建的,它使用 TotalCross `ControlAnimation` 类从一个屏幕滑到另一个屏幕。
`XMLpresenter` 类上调用 `SlidingNavigator` `XMLpresenter` 类上调用 `SlidingNavigator`
``` ```
`new SlidingNavigator(this).present(HomePresenter.class);` new SlidingNavigator(this).present(HomePresenter.class);
``` ```
`SlidingNavigator` 类上实现 `present` 函数: `SlidingNavigator` 类上实现 `present` 函数:
``` ```
public void present(Class&lt;? extends XMLPresenter&gt; presenterClass) public void present(Class<? extends XMLPresenter> presenterClass)
         throws [InstantiationException][16], [IllegalAccessException][17] { throws InstantiationException, IllegalAccessException {
      final XMLPresenter presenter = cache.containsKey(presenterClass) ? cache.get(presenterClass) final XMLPresenter presenter = cache.containsKey(presenterClass) ? cache.get(presenterClass)
            : presenterClass.newInstance(); : presenterClass.newInstance();
      if (!cache.containsKey(presenterClass)) { if (!cache.containsKey(presenterClass)) {
         cache.put(presenterClass, presenter); cache.put(presenterClass, presenter);
      } }
      if (presenters.isEmpty()) { if (presenters.isEmpty()) {
         window.add(presenter.content, LEFT, TOP, FILL, FILL); window.add(presenter.content, LEFT, TOP, FILL, FILL);
      } else { } else {
         XMLPresenter previous = presenters.lastElement(); XMLPresenter previous = presenters.lastElement();
         window.add(presenter.content, AFTER, TOP, SCREENSIZE, SCREENSIZE, previous.content); window.add(presenter.content, AFTER, TOP, SCREENSIZE, SCREENSIZE, previous.content);
``` ```
使用动画控件中的 `PathAnimation` 来创建从一个屏幕到另一个屏幕的滑动动画: 使用动画控件中的 `PathAnimation` 来创建从一个屏幕到另一个屏幕的滑动动画:
``` ```
         PathAnimation.create(previous.content, -Settings.screenWidth, 0, new ControlAnimation.AnimationFinished() { PathAnimation.create(previous.content, -Settings.screenWidth, 0, new ControlAnimation.AnimationFinished() {
            @Override @Override
            public void onAnimationFinished(ControlAnimation anim) { public void onAnimationFinished(ControlAnimation anim) {
               window.remove(previous.content); window.remove(previous.content);
            } }
         }, 1000).with(PathAnimation.create(presenter.content, 0, 0, new ControlAnimation.AnimationFinished() { }, 1000).with(PathAnimation.create(presenter.content, 0, 0, new ControlAnimation.AnimationFinished() {
            @Override @Override
            public void onAnimation Finished(Control Animation anim) { public void onAnimation Finished(Control Animation anim) {
               presenter.content.setRect(LEFT, TOP, FILL, FILL); presenter.content.setRect(LEFT, TOP, FILL, FILL);
            } }
         }, 1000)).start(); }, 1000)).start();
      } }
      presenter.setNavigator(this); presenter.setNavigator(this);
      presenters.push(presenter); presenters.push(presenter);
      presenter.bind2(); presenter.bind2();
      if (presenter.isFirstPresent) { if (presenter.isFirstPresent) {
         presenter.onPresent(); presenter.onPresent();
         presenter.isFirstPresent = false; presenter.isFirstPresent = false;
      } }
``` ```
#### 5\. 加载 <ruby> 环形进度条 <rt>spinner</rt> </ruby> #### 5、加载环形进度条
打印机应用程序的另一个不错的功能是显示进度的加载屏幕动画。它包括文本和旋转动画。 打印机应用程序的另一个不错的功能是显示进度的加载屏幕动画。它包括文本和旋转动画。
![加载环形进度条][18] ![加载环形进度条][18]
(图片提供Edson Holanda Teixeira Jr遵循[CC BY-SA 4.0][13]协议)
通过添加定时器和定时器监听器来更新进度标签,然后调用函数 `spinner.start()` 来实现此功能。所有的动画都是由 TotalCross 和 KnowCode 自动生成的: 通过添加定时器和定时器监听器来更新进度标签,然后调用函数 `spinner.start()` 来实现此功能。所有的动画都是由 TotalCross 和 KnowCode 自动生成的:
``` ```
public void startSpinner() { public void startSpinner() {
time = content.addTimer(500); time = content.addTimer(500);
@ -190,8 +173,7 @@ public void startSpinner() {
} }
``` ```
这里的环形进度条被实例化为对 XML 文件中描述的 `XmlContainerLayout` <ruby> 环形进度条 <rt>spinner</rt> </ruby>的引用: 这里的环形进度条被实例化为对 XML 文件中描述的 `XmlContainerLayout` `spinner` 的引用:
``` ```
<ProgressBar <ProgressBar
@ -204,47 +186,33 @@ tools:layout_editor_absoluteY="198dp"
style="?android:attr/progressBarStyle" /> style="?android:attr/progressBarStyle" />
``` ```
#### 6\. 构建应用程序 #### 6构建应用程序
是时候构建应用程序了。可以在 `pom.xml` 中查看和更改<ruby>目标系统<rt>target systems</rt></ruby>。 请确保 **Linux Arm** 目标可用。 是时候构建应用程序了。可以在 `pom.xml` 中查看和更改<ruby>目标系统<rt>target systems</rt></ruby>。 请确保 `Linux Arm` 目标可用。
如果您使用的是 VSCode请按下键盘上的 **F1** 键,选择 **TotalCross: Package** 并等待完成。 然后就可以在 **Target** 文件夹中看到安装文件了。 如果你使用的是 VSCode请按下键盘上的 `F1` 键,选择 `TotalCross: Package` 并等待完成。 然后就可以在 `Target` 文件夹中看到安装文件了。
#### 7\. 在树莓派上部署和运行应用程序 #### 7在树莓派上部署和运行应用程序
要使用 SSH 协议在 [树莓派][19] 上部署应用程序,请按键盘上的 **F1**。 选择 **TotalCross: Deploy&amp;Run** 并提供有关您的 SSH 连接的信息用户名、IP地址、密码和应用程序路径。 要使用 SSH 协议在 [树莓派][19] 上部署应用程序,请按键盘上的 `F1`。选择 `TotalCross: Deploy&Run` 并提供有关你的 SSH 连接的信息用户名、IP地址、密码和应用程序路径。
![TotalCross部署与运行][20] ![TotalCross部署与运行][20]
(图片提供Edson Holanda Teixeira Jr遵循[CC BY-SA 4.0][13]协议)
![配置 SSH 用户名][21] ![配置 SSH 用户名][21]
(图片提供Edson Holanda Teixeira Jr遵循[CC BY-SA 4.0][13]协议)
![配置 IP 地址][22] ![配置 IP 地址][22]
(图片提供Edson Holanda Teixeira Jr遵循[CC BY-SA 4.0][13]协议)
![输入密码][23] ![输入密码][23]
(图片提供Edson Holanda Teixeira Jr遵循[CC BY-SA 4.0][13]协议)
![配置路径][24] ![配置路径][24]
(图片提供Edson Holanda Teixeira Jr遵循[CC BY-SA 4.0][13]协议)
这是应用程序在机器上运行的样子。
<video height="480" src="https://drive.google.com/file/d/1Reubm-B7E7MniAjsuLYw2Egqdt5Sq2IG/preview" width="640"></iframe>
### 总结 ### 总结
KnowCode 让使用 Java 创建和管理应用程序屏幕变得更加容易。 Knowcode-XML 将您的 XML 转换为 TotalCross GUI 界面,然后生成二进制文件以在您的树莓派上运行。 KnowCode 让使用 Java 创建和管理应用程序屏幕变得更加容易。Knowcode-XML 将你的 XML 转换为 TotalCross GUI 界面,然后生成二进制文件以在你的树莓派上运行。
将 KnowCode 技术与 TotalCross 相结合,使您能够更快地创建嵌入式应用程序。 您可以访问我们在 GitHub 上的 [嵌入式示例][25] 并编辑您自己的应用程序,了解您还可以做什么。 将 KnowCode 技术与 TotalCross 相结合,使你能够更快地创建嵌入式应用程序。 你可以访问我们在 GitHub 上的 [嵌入式示例][25] 并编辑你自己的应用程序,了解你还可以做什么。
如果有问题、需要帮助,或者只是想与其他嵌入式 GUI 开发人员互动,请随时加入我们的 [Telegram][26] 小组,讨论任何框架上的嵌入式应用程序。 如果有问题、需要帮助,或者只是想与其他嵌入式 GUI 开发人员互动,请随时加入我们的 [Telegram][26] 小组,讨论任何框架上的嵌入式应用程序。
-------------------------------------------------------------------------------- --------------------------------------------------------------------------------
@ -253,7 +221,7 @@ via: https://opensource.com/article/21/3/raspberry-pi-totalcross
作者:[Edson Holanda Teixeira Junior][a] 作者:[Edson Holanda Teixeira Junior][a]
选题:[lujun9972][b] 选题:[lujun9972][b]
译者:[CoWave-Fall](https://github.com/CoWave-Fall) 译者:[CoWave-Fall](https://github.com/CoWave-Fall)
校对:[校对者ID](https://github.com/校对者ID) 校对:[wxy](https://github.com/wxy)
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出 本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出