Compare commits
11 Commits
a2c6061172
...
46c1db5ac3
Author | SHA1 | Date | |
---|---|---|---|
46c1db5ac3 | |||
087fc9e0aa | |||
0e0cef6eba | |||
726b7db95f | |||
02a326e60e | |||
fdbc629877 | |||
9481c802f1 | |||
a4372a12fc | |||
5ecb12a679 | |||
ea68167afb | |||
efa0d27041 |
2
.bundle/config
Normal file
@ -0,0 +1,2 @@
|
||||
BUNDLE_PATH: "vendor/bundle"
|
||||
BUNDLE_FORCE_RUBY_PLATFORM: 1
|
18
.claude/settings.local.json
Normal file
@ -0,0 +1,18 @@
|
||||
{
|
||||
"permissions": {
|
||||
"allow": [
|
||||
"Bash(npm install:*)",
|
||||
"Bash(mkdir -p:*)",
|
||||
"Bash(npm install:*)",
|
||||
"Bash(*:*)",
|
||||
"Bash(npm uninstall react-native-paper react-native-paper-dropdown)",
|
||||
"Bash(npm uninstall react-native-paper react-native-paper-dropdown --legacy-peer-deps)",
|
||||
"Bash(npx react-native start --reset-cache)",
|
||||
"Bash(npm run lint)"
|
||||
],
|
||||
"deny": [
|
||||
"WebSearch"
|
||||
],
|
||||
"defaultMode": "bypassPermissions"
|
||||
}
|
||||
}
|
4
.eslintrc.js
Normal file
@ -0,0 +1,4 @@
|
||||
module.exports = {
|
||||
root: true,
|
||||
extends: '@react-native',
|
||||
};
|
75
.gitignore
vendored
Normal file
@ -0,0 +1,75 @@
|
||||
# OSX
|
||||
#
|
||||
.DS_Store
|
||||
|
||||
# Xcode
|
||||
#
|
||||
build/
|
||||
*.pbxuser
|
||||
!default.pbxuser
|
||||
*.mode1v3
|
||||
!default.mode1v3
|
||||
*.mode2v3
|
||||
!default.mode2v3
|
||||
*.perspectivev3
|
||||
!default.perspectivev3
|
||||
xcuserdata
|
||||
*.xccheckout
|
||||
*.moved-aside
|
||||
DerivedData
|
||||
*.hmap
|
||||
*.ipa
|
||||
*.xcuserstate
|
||||
**/.xcode.env.local
|
||||
|
||||
# Android/IntelliJ
|
||||
#
|
||||
build/
|
||||
.idea
|
||||
.gradle
|
||||
local.properties
|
||||
*.iml
|
||||
*.hprof
|
||||
.cxx/
|
||||
*.keystore
|
||||
!debug.keystore
|
||||
.kotlin/
|
||||
|
||||
# node.js
|
||||
#
|
||||
node_modules/
|
||||
npm-debug.log
|
||||
yarn-error.log
|
||||
|
||||
# fastlane
|
||||
#
|
||||
# It is recommended to not store the screenshots in the git repo. Instead, use fastlane to re-generate the
|
||||
# screenshots whenever they are needed.
|
||||
# For more information about the recommended setup visit:
|
||||
# https://docs.fastlane.tools/best-practices/source-control/
|
||||
|
||||
**/fastlane/report.xml
|
||||
**/fastlane/Preview.html
|
||||
**/fastlane/screenshots
|
||||
**/fastlane/test_output
|
||||
|
||||
# Bundle artifact
|
||||
*.jsbundle
|
||||
|
||||
# Ruby / CocoaPods
|
||||
**/Pods/
|
||||
/vendor/bundle/
|
||||
|
||||
# Temporary files created by Metro to check the health of the file watcher
|
||||
.metro-health-check*
|
||||
|
||||
# testing
|
||||
/coverage
|
||||
|
||||
# Yarn
|
||||
.yarn/*
|
||||
!.yarn/patches
|
||||
!.yarn/plugins
|
||||
!.yarn/releases
|
||||
!.yarn/sdks
|
||||
!.yarn/versions
|
5
.prettierrc.js
Normal file
@ -0,0 +1,5 @@
|
||||
module.exports = {
|
||||
arrowParens: 'avoid',
|
||||
singleQuote: true,
|
||||
trailingComma: 'all',
|
||||
};
|
1
.watchmanconfig
Normal file
@ -0,0 +1 @@
|
||||
{}
|
18
App.tsx
Normal file
@ -0,0 +1,18 @@
|
||||
import React from 'react';
|
||||
import { NavigationContainer } from '@react-navigation/native';
|
||||
import AppNavigator from './src/navigation/AppNavigator';
|
||||
|
||||
import { TasksProvider } from './src/context/TasksContext';
|
||||
import { ThemeProvider } from '@rneui/themed';
|
||||
|
||||
export default function App() {
|
||||
return (
|
||||
<ThemeProvider>
|
||||
<TasksProvider>
|
||||
<NavigationContainer>
|
||||
<AppNavigator />
|
||||
</NavigationContainer>
|
||||
</TasksProvider>
|
||||
</ThemeProvider>
|
||||
);
|
||||
}
|
27
CLAUDE.md
Normal file
@ -0,0 +1,27 @@
|
||||
# CLAUDE.md
|
||||
|
||||
This file provides guidance to Claude Code (claude.ai/code) when working with code in this repository.
|
||||
|
||||
## Commands
|
||||
|
||||
- **Start Metro bundler**: `npm start`
|
||||
- **Run on Android**: `npm run android`
|
||||
- **Run on iOS**: `npm run ios`
|
||||
- Before running on iOS for the first time, or after updating native dependencies, you need to install CocoaPods dependencies:
|
||||
- `bundle install`
|
||||
- `bundle exec pod install`
|
||||
- **Run tests**: `npm test`
|
||||
- **Lint files**: `npm run lint`
|
||||
|
||||
## Architecture
|
||||
|
||||
This is a React Native application for task management.
|
||||
|
||||
- **Navigation**: The app uses `react-navigation`. The main navigator is a bottom tab navigator (`AppNavigator.tsx`) with four tabs: "主页" (Home), "运行" (Run), "编辑" (Edit), and "设置" (Settings). The "主页" tab contains a stack navigator for the task list and task editing screens.
|
||||
- **State Management**: Task data is managed globally using React Context (`src/context/TasksContext.tsx`). The `TasksProvider` provides tasks and functions to interact with them (`getTaskById`, `updateTask`, `runTask`).
|
||||
- **Data**: Mock task data is used for development, located in `src/data/mockData.ts`.
|
||||
- **Components**: Reusable components are located in `src/components`.
|
||||
- **Screens**: Each screen of the application is a separate component in the `src/screens` directory.
|
||||
- **Types**: TypeScript types, like the `Task` type, are defined in the `src/types` directory.
|
||||
|
||||
The root component of the application is `App.tsx`, which sets up the `ThemeProvider`, `TasksProvider`, and `NavigationContainer`.
|
16
Gemfile
Normal file
@ -0,0 +1,16 @@
|
||||
source 'https://rubygems.org'
|
||||
|
||||
# You may use http://rbenv.org/ or https://rvm.io/ to install and use this version
|
||||
ruby ">= 2.6.10"
|
||||
|
||||
# Exclude problematic versions of cocoapods and activesupport that causes build failures.
|
||||
gem 'cocoapods', '>= 1.13', '!= 1.15.0', '!= 1.15.1'
|
||||
gem 'activesupport', '>= 6.1.7.5', '!= 7.1.0'
|
||||
gem 'xcodeproj', '< 1.26.0'
|
||||
gem 'concurrent-ruby', '< 1.3.4'
|
||||
|
||||
# Ruby 3.4.0 has removed some libraries from the standard library.
|
||||
gem 'bigdecimal'
|
||||
gem 'logger'
|
||||
gem 'benchmark'
|
||||
gem 'mutex_m'
|
100
README.md
@ -1,3 +1,101 @@
|
||||
This is a new [**React Native**](https://reactnative.dev) project, bootstrapped using [`@react-native-community/cli`](https://github.com/react-native-community/cli).
|
||||
|
||||
# Getting Started
|
||||
|
||||
> **Note**: Make sure you have completed the [Set Up Your Environment](https://reactnative.dev/docs/set-up-your-environment) guide before proceeding.
|
||||
|
||||
## Step 1: Start Metro
|
||||
|
||||
First, you will need to run **Metro**, the JavaScript build tool for React Native.
|
||||
|
||||
To start the Metro dev server, run the following command from the root of your React Native project:
|
||||
|
||||
```sh
|
||||
# Using npm
|
||||
npm start
|
||||
|
||||
# OR using Yarn
|
||||
yarn start
|
||||
```
|
||||
|
||||
## Step 2: Build and run your app
|
||||
|
||||
With Metro running, open a new terminal window/pane from the root of your React Native project, and use one of the following commands to build and run your Android or iOS app:
|
||||
|
||||
### Android
|
||||
|
||||
```sh
|
||||
# Using npm
|
||||
npm run android
|
||||
|
||||
# OR using Yarn
|
||||
yarn android
|
||||
```
|
||||
|
||||
### iOS
|
||||
|
||||
For iOS, remember to install CocoaPods dependencies (this only needs to be run on first clone or after updating native deps).
|
||||
|
||||
The first time you create a new project, run the Ruby bundler to install CocoaPods itself:
|
||||
|
||||
```sh
|
||||
bundle install
|
||||
```
|
||||
|
||||
Then, and every time you update your native dependencies, run:
|
||||
|
||||
```sh
|
||||
bundle exec pod install
|
||||
```
|
||||
|
||||
For more information, please visit [CocoaPods Getting Started guide](https://guides.cocoapods.org/using/getting-started.html).
|
||||
|
||||
```sh
|
||||
# Using npm
|
||||
npm run ios
|
||||
|
||||
# OR using Yarn
|
||||
yarn ios
|
||||
```
|
||||
|
||||
If everything is set up correctly, you should see your new app running in the Android Emulator, iOS Simulator, or your connected device.
|
||||
|
||||
This is one way to run your app — you can also build it directly from Android Studio or Xcode.
|
||||
|
||||
## Step 3: Modify your app
|
||||
|
||||
Now that you have successfully run the app, let's make changes!
|
||||
|
||||
Open `App.tsx` in your text editor of choice and make some changes. When you save, your app will automatically update and reflect these changes — this is powered by [Fast Refresh](https://reactnative.dev/docs/fast-refresh).
|
||||
|
||||
When you want to forcefully reload, for example to reset the state of your app, you can perform a full reload:
|
||||
|
||||
- **Android**: Press the <kbd>R</kbd> key twice or select **"Reload"** from the **Dev Menu**, accessed via <kbd>Ctrl</kbd> + <kbd>M</kbd> (Windows/Linux) or <kbd>Cmd ⌘</kbd> + <kbd>M</kbd> (macOS).
|
||||
- **iOS**: Press <kbd>R</kbd> in iOS Simulator.
|
||||
|
||||
## Congratulations! :tada:
|
||||
|
||||
You've successfully run and modified your React Native App. :partying_face:
|
||||
|
||||
### Now what?
|
||||
|
||||
- If you want to add this new React Native code to an existing application, check out the [Integration guide](https://reactnative.dev/docs/integration-with-existing-apps).
|
||||
- If you're curious to learn more about React Native, check out the [docs](https://reactnative.dev/docs/getting-started).
|
||||
|
||||
# Troubleshooting
|
||||
|
||||
If you're having issues getting the above steps to work, see the [Troubleshooting](https://reactnative.dev/docs/troubleshooting) page.
|
||||
|
||||
# Learn More
|
||||
|
||||
To learn more about React Native, take a look at the following resources:
|
||||
|
||||
- [React Native Website](https://reactnative.dev) - learn more about React Native.
|
||||
- [Getting Started](https://reactnative.dev/docs/environment-setup) - an **overview** of React Native and how setup your environment.
|
||||
- [Learn the Basics](https://reactnative.dev/docs/getting-started) - a **guided tour** of the React Native **basics**.
|
||||
- [Blog](https://reactnative.dev/blog) - read the latest official React Native **Blog** posts.
|
||||
- [`@facebook/react-native`](https://github.com/facebook/react-native) - the Open Source; GitHub **repository** for React Native.
|
||||
|
||||
# webapp
|
||||
|
||||
手持安卓端app
|
||||
手持安卓端 app
|
||||
|
13
__tests__/App.test.tsx
Normal file
@ -0,0 +1,13 @@
|
||||
/**
|
||||
* @format
|
||||
*/
|
||||
|
||||
import React from 'react';
|
||||
import ReactTestRenderer from 'react-test-renderer';
|
||||
import App from '../App';
|
||||
|
||||
test('renders correctly', async () => {
|
||||
await ReactTestRenderer.act(() => {
|
||||
ReactTestRenderer.create(<App />);
|
||||
});
|
||||
});
|
121
android/app/build.gradle
Normal file
@ -0,0 +1,121 @@
|
||||
apply plugin: "com.android.application"
|
||||
apply plugin: "org.jetbrains.kotlin.android"
|
||||
apply plugin: "com.facebook.react"
|
||||
|
||||
/**
|
||||
* This is the configuration block to customize your React Native Android app.
|
||||
* By default you don't need to apply any configuration, just uncomment the lines you need.
|
||||
*/
|
||||
react {
|
||||
/* Folders */
|
||||
// The root of your project, i.e. where "package.json" lives. Default is '../..'
|
||||
// root = file("../../")
|
||||
// The folder where the react-native NPM package is. Default is ../../node_modules/react-native
|
||||
// reactNativeDir = file("../../node_modules/react-native")
|
||||
// The folder where the react-native Codegen package is. Default is ../../node_modules/@react-native/codegen
|
||||
// codegenDir = file("../../node_modules/@react-native/codegen")
|
||||
// The cli.js file which is the React Native CLI entrypoint. Default is ../../node_modules/react-native/cli.js
|
||||
// cliFile = file("../../node_modules/react-native/cli.js")
|
||||
|
||||
/* Variants */
|
||||
// The list of variants to that are debuggable. For those we're going to
|
||||
// skip the bundling of the JS bundle and the assets. By default is just 'debug'.
|
||||
// If you add flavors like lite, prod, etc. you'll have to list your debuggableVariants.
|
||||
// debuggableVariants = ["liteDebug", "prodDebug"]
|
||||
|
||||
/* Bundling */
|
||||
// A list containing the node command and its flags. Default is just 'node'.
|
||||
// nodeExecutableAndArgs = ["node"]
|
||||
//
|
||||
// The command to run when bundling. By default is 'bundle'
|
||||
// bundleCommand = "ram-bundle"
|
||||
//
|
||||
// The path to the CLI configuration file. Default is empty.
|
||||
// bundleConfig = file(../rn-cli.config.js)
|
||||
//
|
||||
// The name of the generated asset file containing your JS bundle
|
||||
// bundleAssetName = "MyApplication.android.bundle"
|
||||
//
|
||||
// The entry file for bundle generation. Default is 'index.android.js' or 'index.js'
|
||||
// entryFile = file("../js/MyApplication.android.js")
|
||||
//
|
||||
// A list of extra flags to pass to the 'bundle' commands.
|
||||
// See https://github.com/react-native-community/cli/blob/main/docs/commands.md#bundle
|
||||
// extraPackagerArgs = []
|
||||
|
||||
/* Hermes Commands */
|
||||
// The hermes compiler command to run. By default it is 'hermesc'
|
||||
// hermesCommand = "$rootDir/my-custom-hermesc/bin/hermesc"
|
||||
//
|
||||
// The list of flags to pass to the Hermes compiler. By default is "-O", "-output-source-map"
|
||||
// hermesFlags = ["-O", "-output-source-map"]
|
||||
|
||||
/* Autolinking */
|
||||
autolinkLibrariesWithApp()
|
||||
}
|
||||
|
||||
/**
|
||||
* Set this to true to Run Proguard on Release builds to minify the Java bytecode.
|
||||
*/
|
||||
def enableProguardInReleaseBuilds = false
|
||||
|
||||
/**
|
||||
* The preferred build flavor of JavaScriptCore (JSC)
|
||||
*
|
||||
* For example, to use the international variant, you can use:
|
||||
* `def jscFlavor = io.github.react-native-community:jsc-android-intl:2026004.+`
|
||||
*
|
||||
* The international variant includes ICU i18n library and necessary data
|
||||
* allowing to use e.g. `Date.toLocaleString` and `String.localeCompare` that
|
||||
* give correct results when using with locales other than en-US. Note that
|
||||
* this variant is about 6MiB larger per architecture than default.
|
||||
*/
|
||||
def jscFlavor = 'io.github.react-native-community:jsc-android:2026004.+'
|
||||
|
||||
android {
|
||||
ndkVersion rootProject.ext.ndkVersion
|
||||
buildToolsVersion rootProject.ext.buildToolsVersion
|
||||
compileSdk rootProject.ext.compileSdkVersion
|
||||
|
||||
namespace "com.myreactnativeapp"
|
||||
defaultConfig {
|
||||
applicationId "com.myreactnativeapp"
|
||||
minSdkVersion rootProject.ext.minSdkVersion
|
||||
targetSdkVersion rootProject.ext.targetSdkVersion
|
||||
versionCode 1
|
||||
versionName "1.0"
|
||||
}
|
||||
signingConfigs {
|
||||
debug {
|
||||
storeFile file('debug.keystore')
|
||||
storePassword 'android'
|
||||
keyAlias 'androiddebugkey'
|
||||
keyPassword 'android'
|
||||
}
|
||||
}
|
||||
buildTypes {
|
||||
debug {
|
||||
signingConfig signingConfigs.debug
|
||||
}
|
||||
release {
|
||||
// Caution! In production, you need to generate your own keystore file.
|
||||
// see https://reactnative.dev/docs/signed-apk-android.
|
||||
signingConfig signingConfigs.debug
|
||||
minifyEnabled enableProguardInReleaseBuilds
|
||||
proguardFiles getDefaultProguardFile("proguard-android.txt"), "proguard-rules.pro"
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
apply from: file("../../node_modules/react-native-vector-icons/fonts.gradle")
|
||||
|
||||
dependencies {
|
||||
// The version of react-native is set by the React Native Gradle Plugin
|
||||
implementation("com.facebook.react:react-android")
|
||||
|
||||
if (hermesEnabled.toBoolean()) {
|
||||
implementation("com.facebook.react:hermes-android")
|
||||
} else {
|
||||
implementation jscFlavor
|
||||
}
|
||||
}
|
BIN
android/app/debug.keystore
Normal file
10
android/app/proguard-rules.pro
vendored
Normal file
@ -0,0 +1,10 @@
|
||||
# Add project specific ProGuard rules here.
|
||||
# By default, the flags in this file are appended to flags specified
|
||||
# in /usr/local/Cellar/android-sdk/24.3.3/tools/proguard/proguard-android.txt
|
||||
# You can edit the include path and order by changing the proguardFiles
|
||||
# directive in build.gradle.
|
||||
#
|
||||
# For more details, see
|
||||
# http://developer.android.com/guide/developing/tools/proguard.html
|
||||
|
||||
# Add any project specific keep options here:
|
9
android/app/src/debug/AndroidManifest.xml
Normal file
@ -0,0 +1,9 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
xmlns:tools="http://schemas.android.com/tools">
|
||||
|
||||
<application
|
||||
android:usesCleartextTraffic="true"
|
||||
tools:targetApi="28"
|
||||
tools:ignore="GoogleAppIndexingWarning"/>
|
||||
</manifest>
|
26
android/app/src/main/AndroidManifest.xml
Normal file
@ -0,0 +1,26 @@
|
||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
|
||||
<uses-permission android:name="android.permission.INTERNET" />
|
||||
|
||||
<application
|
||||
android:name=".MainApplication"
|
||||
android:label="@string/app_name"
|
||||
android:icon="@mipmap/ic_launcher"
|
||||
android:roundIcon="@mipmap/ic_launcher_round"
|
||||
android:allowBackup="false"
|
||||
android:theme="@style/AppTheme"
|
||||
android:supportsRtl="true">
|
||||
<activity
|
||||
android:name=".MainActivity"
|
||||
android:label="@string/app_name"
|
||||
android:configChanges="keyboard|keyboardHidden|orientation|screenLayout|screenSize|smallestScreenSize|uiMode"
|
||||
android:launchMode="singleTask"
|
||||
android:windowSoftInputMode="adjustResize"
|
||||
android:exported="true">
|
||||
<intent-filter>
|
||||
<action android:name="android.intent.action.MAIN" />
|
||||
<category android:name="android.intent.category.LAUNCHER" />
|
||||
</intent-filter>
|
||||
</activity>
|
||||
</application>
|
||||
</manifest>
|
@ -0,0 +1,22 @@
|
||||
package com.myreactnativeapp
|
||||
|
||||
import com.facebook.react.ReactActivity
|
||||
import com.facebook.react.ReactActivityDelegate
|
||||
import com.facebook.react.defaults.DefaultNewArchitectureEntryPoint.fabricEnabled
|
||||
import com.facebook.react.defaults.DefaultReactActivityDelegate
|
||||
|
||||
class MainActivity : ReactActivity() {
|
||||
|
||||
/**
|
||||
* Returns the name of the main component registered from JavaScript. This is used to schedule
|
||||
* rendering of the component.
|
||||
*/
|
||||
override fun getMainComponentName(): String = "MyReactNativeApp"
|
||||
|
||||
/**
|
||||
* Returns the instance of the [ReactActivityDelegate]. We use [DefaultReactActivityDelegate]
|
||||
* which allows you to enable New Architecture with a single boolean flags [fabricEnabled]
|
||||
*/
|
||||
override fun createReactActivityDelegate(): ReactActivityDelegate =
|
||||
DefaultReactActivityDelegate(this, mainComponentName, fabricEnabled)
|
||||
}
|
@ -0,0 +1,38 @@
|
||||
package com.myreactnativeapp
|
||||
|
||||
import android.app.Application
|
||||
import com.facebook.react.PackageList
|
||||
import com.facebook.react.ReactApplication
|
||||
import com.facebook.react.ReactHost
|
||||
import com.facebook.react.ReactNativeApplicationEntryPoint.loadReactNative
|
||||
import com.facebook.react.ReactNativeHost
|
||||
import com.facebook.react.ReactPackage
|
||||
import com.facebook.react.defaults.DefaultReactHost.getDefaultReactHost
|
||||
import com.facebook.react.defaults.DefaultReactNativeHost
|
||||
|
||||
class MainApplication : Application(), ReactApplication {
|
||||
|
||||
override val reactNativeHost: ReactNativeHost =
|
||||
object : DefaultReactNativeHost(this) {
|
||||
override fun getPackages(): List<ReactPackage> =
|
||||
PackageList(this).packages.apply {
|
||||
// Packages that cannot be autolinked yet can be added manually here, for example:
|
||||
// add(MyReactNativePackage())
|
||||
}
|
||||
|
||||
override fun getJSMainModuleName(): String = "index"
|
||||
|
||||
override fun getUseDeveloperSupport(): Boolean = BuildConfig.DEBUG
|
||||
|
||||
override val isNewArchEnabled: Boolean = BuildConfig.IS_NEW_ARCHITECTURE_ENABLED
|
||||
override val isHermesEnabled: Boolean = BuildConfig.IS_HERMES_ENABLED
|
||||
}
|
||||
|
||||
override val reactHost: ReactHost
|
||||
get() = getDefaultReactHost(applicationContext, reactNativeHost)
|
||||
|
||||
override fun onCreate() {
|
||||
super.onCreate()
|
||||
loadReactNative(this)
|
||||
}
|
||||
}
|
37
android/app/src/main/res/drawable/rn_edit_text_material.xml
Normal file
@ -0,0 +1,37 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<!-- Copyright (C) 2014 The Android Open Source Project
|
||||
|
||||
Licensed under the Apache License, Version 2.0 (the "License");
|
||||
you may not use this file except in compliance with the License.
|
||||
You may obtain a copy of the License at
|
||||
|
||||
http://www.apache.org/licenses/LICENSE-2.0
|
||||
|
||||
Unless required by applicable law or agreed to in writing, software
|
||||
distributed under the License is distributed on an "AS IS" BASIS,
|
||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
See the License for the specific language governing permissions and
|
||||
limitations under the License.
|
||||
-->
|
||||
<inset xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
android:insetLeft="@dimen/abc_edit_text_inset_horizontal_material"
|
||||
android:insetRight="@dimen/abc_edit_text_inset_horizontal_material"
|
||||
android:insetTop="@dimen/abc_edit_text_inset_top_material"
|
||||
android:insetBottom="@dimen/abc_edit_text_inset_bottom_material"
|
||||
>
|
||||
|
||||
<selector>
|
||||
<!--
|
||||
This file is a copy of abc_edit_text_material (https://bit.ly/3k8fX7I).
|
||||
The item below with state_pressed="false" and state_focused="false" causes a NullPointerException.
|
||||
NullPointerException:tempt to invoke virtual method 'android.graphics.drawable.Drawable android.graphics.drawable.Drawable$ConstantState.newDrawable(android.content.res.Resources)'
|
||||
|
||||
<item android:state_pressed="false" android:state_focused="false" android:drawable="@drawable/abc_textfield_default_mtrl_alpha"/>
|
||||
|
||||
For more info, see https://bit.ly/3CdLStv (react-native/pull/29452) and https://bit.ly/3nxOMoR.
|
||||
-->
|
||||
<item android:state_enabled="false" android:drawable="@drawable/abc_textfield_default_mtrl_alpha"/>
|
||||
<item android:drawable="@drawable/abc_textfield_activated_mtrl_alpha"/>
|
||||
</selector>
|
||||
|
||||
</inset>
|
BIN
android/app/src/main/res/mipmap-hdpi/ic_launcher.png
Normal file
After Width: | Height: | Size: 3.0 KiB |
BIN
android/app/src/main/res/mipmap-hdpi/ic_launcher_round.png
Normal file
After Width: | Height: | Size: 4.9 KiB |
BIN
android/app/src/main/res/mipmap-mdpi/ic_launcher.png
Normal file
After Width: | Height: | Size: 2.0 KiB |
BIN
android/app/src/main/res/mipmap-mdpi/ic_launcher_round.png
Normal file
After Width: | Height: | Size: 2.8 KiB |
BIN
android/app/src/main/res/mipmap-xhdpi/ic_launcher.png
Normal file
After Width: | Height: | Size: 4.5 KiB |
BIN
android/app/src/main/res/mipmap-xhdpi/ic_launcher_round.png
Normal file
After Width: | Height: | Size: 6.9 KiB |
BIN
android/app/src/main/res/mipmap-xxhdpi/ic_launcher.png
Normal file
After Width: | Height: | Size: 6.3 KiB |
BIN
android/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.png
Normal file
After Width: | Height: | Size: 10 KiB |
BIN
android/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png
Normal file
After Width: | Height: | Size: 9.0 KiB |
BIN
android/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.png
Normal file
After Width: | Height: | Size: 15 KiB |
3
android/app/src/main/res/values/strings.xml
Normal file
@ -0,0 +1,3 @@
|
||||
<resources>
|
||||
<string name="app_name">MyReactNativeApp</string>
|
||||
</resources>
|
9
android/app/src/main/res/values/styles.xml
Normal file
@ -0,0 +1,9 @@
|
||||
<resources>
|
||||
|
||||
<!-- Base application theme. -->
|
||||
<style name="AppTheme" parent="Theme.AppCompat.DayNight.NoActionBar">
|
||||
<!-- Customize your theme here. -->
|
||||
<item name="android:editTextBackground">@drawable/rn_edit_text_material</item>
|
||||
</style>
|
||||
|
||||
</resources>
|
21
android/build.gradle
Normal file
@ -0,0 +1,21 @@
|
||||
buildscript {
|
||||
ext {
|
||||
buildToolsVersion = "35.0.0"
|
||||
minSdkVersion = 24
|
||||
compileSdkVersion = 35
|
||||
targetSdkVersion = 35
|
||||
ndkVersion = "27.1.12297006"
|
||||
kotlinVersion = "2.1.20"
|
||||
}
|
||||
repositories {
|
||||
google()
|
||||
mavenCentral()
|
||||
}
|
||||
dependencies {
|
||||
classpath("com.android.tools.build:gradle")
|
||||
classpath("com.facebook.react:react-native-gradle-plugin")
|
||||
classpath("org.jetbrains.kotlin:kotlin-gradle-plugin")
|
||||
}
|
||||
}
|
||||
|
||||
apply plugin: "com.facebook.react.rootproject"
|
39
android/gradle.properties
Normal file
@ -0,0 +1,39 @@
|
||||
# Project-wide Gradle settings.
|
||||
|
||||
# IDE (e.g. Android Studio) users:
|
||||
# Gradle settings configured through the IDE *will override*
|
||||
# any settings specified in this file.
|
||||
|
||||
# For more details on how to configure your build environment visit
|
||||
# http://www.gradle.org/docs/current/userguide/build_environment.html
|
||||
|
||||
# Specifies the JVM arguments used for the daemon process.
|
||||
# The setting is particularly useful for tweaking memory settings.
|
||||
# Default value: -Xmx512m -XX:MaxMetaspaceSize=256m
|
||||
org.gradle.jvmargs=-Xmx2048m -XX:MaxMetaspaceSize=512m
|
||||
|
||||
# When configured, Gradle will run in incubating parallel mode.
|
||||
# This option should only be used with decoupled projects. More details, visit
|
||||
# http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects
|
||||
# org.gradle.parallel=true
|
||||
|
||||
# AndroidX package structure to make it clearer which packages are bundled with the
|
||||
# Android operating system, and which are packaged with your app's APK
|
||||
# https://developer.android.com/topic/libraries/support-library/androidx-rn
|
||||
android.useAndroidX=true
|
||||
|
||||
# Use this property to specify which architecture you want to build.
|
||||
# You can also override it from the CLI using
|
||||
# ./gradlew <task> -PreactNativeArchitectures=x86_64
|
||||
reactNativeArchitectures=armeabi-v7a,arm64-v8a,x86,x86_64
|
||||
|
||||
# Use this property to enable support to the new architecture.
|
||||
# This will allow you to use TurboModules and the Fabric render in
|
||||
# your application. You should enable this flag either if you want
|
||||
# to write custom TurboModules/Fabric components OR use libraries that
|
||||
# are providing them.
|
||||
newArchEnabled=true
|
||||
|
||||
# Use this property to enable or disable the Hermes JS engine.
|
||||
# If set to false, you will be using JSC instead.
|
||||
hermesEnabled=true
|
BIN
android/gradle/wrapper/gradle-wrapper.jar
vendored
Normal file
7
android/gradle/wrapper/gradle-wrapper.properties
vendored
Normal file
@ -0,0 +1,7 @@
|
||||
distributionBase=GRADLE_USER_HOME
|
||||
distributionPath=wrapper/dists
|
||||
distributionUrl=https\://services.gradle.org/distributions/gradle-8.14.1-bin.zip
|
||||
networkTimeout=10000
|
||||
validateDistributionUrl=true
|
||||
zipStoreBase=GRADLE_USER_HOME
|
||||
zipStorePath=wrapper/dists
|
251
android/gradlew
vendored
Normal file
@ -0,0 +1,251 @@
|
||||
#!/bin/sh
|
||||
|
||||
#
|
||||
# Copyright © 2015-2021 the original authors.
|
||||
#
|
||||
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||
# you may not use this file except in compliance with the License.
|
||||
# You may obtain a copy of the License at
|
||||
#
|
||||
# https://www.apache.org/licenses/LICENSE-2.0
|
||||
#
|
||||
# Unless required by applicable law or agreed to in writing, software
|
||||
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
# See the License for the specific language governing permissions and
|
||||
# limitations under the License.
|
||||
#
|
||||
# SPDX-License-Identifier: Apache-2.0
|
||||
#
|
||||
|
||||
##############################################################################
|
||||
#
|
||||
# Gradle start up script for POSIX generated by Gradle.
|
||||
#
|
||||
# Important for running:
|
||||
#
|
||||
# (1) You need a POSIX-compliant shell to run this script. If your /bin/sh is
|
||||
# noncompliant, but you have some other compliant shell such as ksh or
|
||||
# bash, then to run this script, type that shell name before the whole
|
||||
# command line, like:
|
||||
#
|
||||
# ksh Gradle
|
||||
#
|
||||
# Busybox and similar reduced shells will NOT work, because this script
|
||||
# requires all of these POSIX shell features:
|
||||
# * functions;
|
||||
# * expansions «$var», «${var}», «${var:-default}», «${var+SET}»,
|
||||
# «${var#prefix}», «${var%suffix}», and «$( cmd )»;
|
||||
# * compound commands having a testable exit status, especially «case»;
|
||||
# * various built-in commands including «command», «set», and «ulimit».
|
||||
#
|
||||
# Important for patching:
|
||||
#
|
||||
# (2) This script targets any POSIX shell, so it avoids extensions provided
|
||||
# by Bash, Ksh, etc; in particular arrays are avoided.
|
||||
#
|
||||
# The "traditional" practice of packing multiple parameters into a
|
||||
# space-separated string is a well documented source of bugs and security
|
||||
# problems, so this is (mostly) avoided, by progressively accumulating
|
||||
# options in "$@", and eventually passing that to Java.
|
||||
#
|
||||
# Where the inherited environment variables (DEFAULT_JVM_OPTS, JAVA_OPTS,
|
||||
# and GRADLE_OPTS) rely on word-splitting, this is performed explicitly;
|
||||
# see the in-line comments for details.
|
||||
#
|
||||
# There are tweaks for specific operating systems such as AIX, CygWin,
|
||||
# Darwin, MinGW, and NonStop.
|
||||
#
|
||||
# (3) This script is generated from the Groovy template
|
||||
# https://github.com/gradle/gradle/blob/HEAD/platforms/jvm/plugins-application/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt
|
||||
# within the Gradle project.
|
||||
#
|
||||
# You can find Gradle at https://github.com/gradle/gradle/.
|
||||
#
|
||||
##############################################################################
|
||||
|
||||
# Attempt to set APP_HOME
|
||||
|
||||
# Resolve links: $0 may be a link
|
||||
app_path=$0
|
||||
|
||||
# Need this for daisy-chained symlinks.
|
||||
while
|
||||
APP_HOME=${app_path%"${app_path##*/}"} # leaves a trailing /; empty if no leading path
|
||||
[ -h "$app_path" ]
|
||||
do
|
||||
ls=$( ls -ld "$app_path" )
|
||||
link=${ls#*' -> '}
|
||||
case $link in #(
|
||||
/*) app_path=$link ;; #(
|
||||
*) app_path=$APP_HOME$link ;;
|
||||
esac
|
||||
done
|
||||
|
||||
# This is normally unused
|
||||
# shellcheck disable=SC2034
|
||||
APP_BASE_NAME=${0##*/}
|
||||
# Discard cd standard output in case $CDPATH is set (https://github.com/gradle/gradle/issues/25036)
|
||||
APP_HOME=$( cd -P "${APP_HOME:-./}" > /dev/null && printf '%s\n' "$PWD" ) || exit
|
||||
|
||||
# Use the maximum available, or set MAX_FD != -1 to use that value.
|
||||
MAX_FD=maximum
|
||||
|
||||
warn () {
|
||||
echo "$*"
|
||||
} >&2
|
||||
|
||||
die () {
|
||||
echo
|
||||
echo "$*"
|
||||
echo
|
||||
exit 1
|
||||
} >&2
|
||||
|
||||
# OS specific support (must be 'true' or 'false').
|
||||
cygwin=false
|
||||
msys=false
|
||||
darwin=false
|
||||
nonstop=false
|
||||
case "$( uname )" in #(
|
||||
CYGWIN* ) cygwin=true ;; #(
|
||||
Darwin* ) darwin=true ;; #(
|
||||
MSYS* | MINGW* ) msys=true ;; #(
|
||||
NONSTOP* ) nonstop=true ;;
|
||||
esac
|
||||
|
||||
CLASSPATH="\\\"\\\""
|
||||
|
||||
|
||||
# Determine the Java command to use to start the JVM.
|
||||
if [ -n "$JAVA_HOME" ] ; then
|
||||
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
|
||||
# IBM's JDK on AIX uses strange locations for the executables
|
||||
JAVACMD=$JAVA_HOME/jre/sh/java
|
||||
else
|
||||
JAVACMD=$JAVA_HOME/bin/java
|
||||
fi
|
||||
if [ ! -x "$JAVACMD" ] ; then
|
||||
die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
|
||||
|
||||
Please set the JAVA_HOME variable in your environment to match the
|
||||
location of your Java installation."
|
||||
fi
|
||||
else
|
||||
JAVACMD=java
|
||||
if ! command -v java >/dev/null 2>&1
|
||||
then
|
||||
die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
|
||||
|
||||
Please set the JAVA_HOME variable in your environment to match the
|
||||
location of your Java installation."
|
||||
fi
|
||||
fi
|
||||
|
||||
# Increase the maximum file descriptors if we can.
|
||||
if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then
|
||||
case $MAX_FD in #(
|
||||
max*)
|
||||
# In POSIX sh, ulimit -H is undefined. That's why the result is checked to see if it worked.
|
||||
# shellcheck disable=SC2039,SC3045
|
||||
MAX_FD=$( ulimit -H -n ) ||
|
||||
warn "Could not query maximum file descriptor limit"
|
||||
esac
|
||||
case $MAX_FD in #(
|
||||
'' | soft) :;; #(
|
||||
*)
|
||||
# In POSIX sh, ulimit -n is undefined. That's why the result is checked to see if it worked.
|
||||
# shellcheck disable=SC2039,SC3045
|
||||
ulimit -n "$MAX_FD" ||
|
||||
warn "Could not set maximum file descriptor limit to $MAX_FD"
|
||||
esac
|
||||
fi
|
||||
|
||||
# Collect all arguments for the java command, stacking in reverse order:
|
||||
# * args from the command line
|
||||
# * the main class name
|
||||
# * -classpath
|
||||
# * -D...appname settings
|
||||
# * --module-path (only if needed)
|
||||
# * DEFAULT_JVM_OPTS, JAVA_OPTS, and GRADLE_OPTS environment variables.
|
||||
|
||||
# For Cygwin or MSYS, switch paths to Windows format before running java
|
||||
if "$cygwin" || "$msys" ; then
|
||||
APP_HOME=$( cygpath --path --mixed "$APP_HOME" )
|
||||
CLASSPATH=$( cygpath --path --mixed "$CLASSPATH" )
|
||||
|
||||
JAVACMD=$( cygpath --unix "$JAVACMD" )
|
||||
|
||||
# Now convert the arguments - kludge to limit ourselves to /bin/sh
|
||||
for arg do
|
||||
if
|
||||
case $arg in #(
|
||||
-*) false ;; # don't mess with options #(
|
||||
/?*) t=${arg#/} t=/${t%%/*} # looks like a POSIX filepath
|
||||
[ -e "$t" ] ;; #(
|
||||
*) false ;;
|
||||
esac
|
||||
then
|
||||
arg=$( cygpath --path --ignore --mixed "$arg" )
|
||||
fi
|
||||
# Roll the args list around exactly as many times as the number of
|
||||
# args, so each arg winds up back in the position where it started, but
|
||||
# possibly modified.
|
||||
#
|
||||
# NB: a `for` loop captures its iteration list before it begins, so
|
||||
# changing the positional parameters here affects neither the number of
|
||||
# iterations, nor the values presented in `arg`.
|
||||
shift # remove old arg
|
||||
set -- "$@" "$arg" # push replacement arg
|
||||
done
|
||||
fi
|
||||
|
||||
|
||||
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
|
||||
DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"'
|
||||
|
||||
# Collect all arguments for the java command:
|
||||
# * DEFAULT_JVM_OPTS, JAVA_OPTS, and optsEnvironmentVar are not allowed to contain shell fragments,
|
||||
# and any embedded shellness will be escaped.
|
||||
# * For example: A user cannot expect ${Hostname} to be expanded, as it is an environment variable and will be
|
||||
# treated as '${Hostname}' itself on the command line.
|
||||
|
||||
set -- \
|
||||
"-Dorg.gradle.appname=$APP_BASE_NAME" \
|
||||
-classpath "$CLASSPATH" \
|
||||
-jar "$APP_HOME/gradle/wrapper/gradle-wrapper.jar" \
|
||||
"$@"
|
||||
|
||||
# Stop when "xargs" is not available.
|
||||
if ! command -v xargs >/dev/null 2>&1
|
||||
then
|
||||
die "xargs is not available"
|
||||
fi
|
||||
|
||||
# Use "xargs" to parse quoted args.
|
||||
#
|
||||
# With -n1 it outputs one arg per line, with the quotes and backslashes removed.
|
||||
#
|
||||
# In Bash we could simply go:
|
||||
#
|
||||
# readarray ARGS < <( xargs -n1 <<<"$var" ) &&
|
||||
# set -- "${ARGS[@]}" "$@"
|
||||
#
|
||||
# but POSIX shell has neither arrays nor command substitution, so instead we
|
||||
# post-process each arg (as a line of input to sed) to backslash-escape any
|
||||
# character that might be a shell metacharacter, then use eval to reverse
|
||||
# that process (while maintaining the separation between arguments), and wrap
|
||||
# the whole thing up as a single "set" statement.
|
||||
#
|
||||
# This will of course break if any of these variables contains a newline or
|
||||
# an unmatched quote.
|
||||
#
|
||||
|
||||
eval "set -- $(
|
||||
printf '%s\n' "$DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS" |
|
||||
xargs -n1 |
|
||||
sed ' s~[^-[:alnum:]+,./:=@_]~\\&~g; ' |
|
||||
tr '\n' ' '
|
||||
)" '"$@"'
|
||||
|
||||
exec "$JAVACMD" "$@"
|
99
android/gradlew.bat
vendored
Normal file
@ -0,0 +1,99 @@
|
||||
@REM Copyright (c) Meta Platforms, Inc. and affiliates.
|
||||
@REM
|
||||
@REM This source code is licensed under the MIT license found in the
|
||||
@REM LICENSE file in the root directory of this source tree.
|
||||
|
||||
@rem
|
||||
@rem Copyright 2015 the original author or authors.
|
||||
@rem
|
||||
@rem Licensed under the Apache License, Version 2.0 (the "License");
|
||||
@rem you may not use this file except in compliance with the License.
|
||||
@rem You may obtain a copy of the License at
|
||||
@rem
|
||||
@rem https://www.apache.org/licenses/LICENSE-2.0
|
||||
@rem
|
||||
@rem Unless required by applicable law or agreed to in writing, software
|
||||
@rem distributed under the License is distributed on an "AS IS" BASIS,
|
||||
@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
@rem See the License for the specific language governing permissions and
|
||||
@rem limitations under the License.
|
||||
@rem
|
||||
@rem SPDX-License-Identifier: Apache-2.0
|
||||
@rem
|
||||
|
||||
@if "%DEBUG%"=="" @echo off
|
||||
@rem ##########################################################################
|
||||
@rem
|
||||
@rem Gradle startup script for Windows
|
||||
@rem
|
||||
@rem ##########################################################################
|
||||
|
||||
@rem Set local scope for the variables with windows NT shell
|
||||
if "%OS%"=="Windows_NT" setlocal
|
||||
|
||||
set DIRNAME=%~dp0
|
||||
if "%DIRNAME%"=="" set DIRNAME=.
|
||||
@rem This is normally unused
|
||||
set APP_BASE_NAME=%~n0
|
||||
set APP_HOME=%DIRNAME%
|
||||
|
||||
@rem Resolve any "." and ".." in APP_HOME to make it shorter.
|
||||
for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi
|
||||
|
||||
@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
|
||||
set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m"
|
||||
|
||||
@rem Find java.exe
|
||||
if defined JAVA_HOME goto findJavaFromJavaHome
|
||||
|
||||
set JAVA_EXE=java.exe
|
||||
%JAVA_EXE% -version >NUL 2>&1
|
||||
if %ERRORLEVEL% equ 0 goto execute
|
||||
|
||||
echo. 1>&2
|
||||
echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 1>&2
|
||||
echo. 1>&2
|
||||
echo Please set the JAVA_HOME variable in your environment to match the 1>&2
|
||||
echo location of your Java installation. 1>&2
|
||||
|
||||
goto fail
|
||||
|
||||
:findJavaFromJavaHome
|
||||
set JAVA_HOME=%JAVA_HOME:"=%
|
||||
set JAVA_EXE=%JAVA_HOME%/bin/java.exe
|
||||
|
||||
if exist "%JAVA_EXE%" goto execute
|
||||
|
||||
echo. 1>&2
|
||||
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% 1>&2
|
||||
echo. 1>&2
|
||||
echo Please set the JAVA_HOME variable in your environment to match the 1>&2
|
||||
echo location of your Java installation. 1>&2
|
||||
|
||||
goto fail
|
||||
|
||||
:execute
|
||||
@rem Setup the command line
|
||||
|
||||
set CLASSPATH=
|
||||
|
||||
|
||||
@rem Execute Gradle
|
||||
"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" -jar "%APP_HOME%\gradle\wrapper\gradle-wrapper.jar" %*
|
||||
|
||||
:end
|
||||
@rem End local scope for the variables with windows NT shell
|
||||
if %ERRORLEVEL% equ 0 goto mainEnd
|
||||
|
||||
:fail
|
||||
rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
|
||||
rem the _cmd.exe /c_ return code!
|
||||
set EXIT_CODE=%ERRORLEVEL%
|
||||
if %EXIT_CODE% equ 0 set EXIT_CODE=1
|
||||
if not ""=="%GRADLE_EXIT_CONSOLE%" exit %EXIT_CODE%
|
||||
exit /b %EXIT_CODE%
|
||||
|
||||
:mainEnd
|
||||
if "%OS%"=="Windows_NT" endlocal
|
||||
|
||||
:omega
|
6
android/settings.gradle
Normal file
@ -0,0 +1,6 @@
|
||||
pluginManagement { includeBuild("../node_modules/@react-native/gradle-plugin") }
|
||||
plugins { id("com.facebook.react.settings") }
|
||||
extensions.configure(com.facebook.react.ReactSettingsExtension){ ex -> ex.autolinkLibrariesFromCommand() }
|
||||
rootProject.name = 'MyReactNativeApp'
|
||||
include ':app'
|
||||
includeBuild('../node_modules/@react-native/gradle-plugin')
|
4
app.json
Normal file
@ -0,0 +1,4 @@
|
||||
{
|
||||
"name": "MyReactNativeApp",
|
||||
"displayName": "MyReactNativeApp"
|
||||
}
|
3
babel.config.js
Normal file
@ -0,0 +1,3 @@
|
||||
module.exports = {
|
||||
presets: ['module:@react-native/babel-preset'],
|
||||
};
|
709
config.json
Normal file
@ -0,0 +1,709 @@
|
||||
{
|
||||
"version": "1.0.0",
|
||||
"locations": [
|
||||
{ "label": "AP-1", "value": "AP-1" },
|
||||
{ "label": "AP-2", "value": "AP-2" },
|
||||
{ "label": "AP-3", "value": "AP-3" },
|
||||
{ "label": "AP-4", "value": "AP-4" },
|
||||
{ "label": "AP-5", "value": "AP-5" },
|
||||
{ "label": "炉前缓存区", "value": "FURNACE_BUFFER" },
|
||||
{ "label": "热处理上料交接区", "value": "HEAT_TREATMENT_LOADING" },
|
||||
{ "label": "成品存储区", "value": "FINISHED_STORAGE" },
|
||||
{ "label": "原料仓库", "value": "RAW_MATERIAL_WAREHOUSE" },
|
||||
{ "label": "质检区", "value": "QUALITY_INSPECTION" }
|
||||
],
|
||||
"locationsBays": [
|
||||
{ "label": "AS2_2_001", "value": "AS2_2_001" },
|
||||
{ "label": "AS2_2_002", "value": "AS2_2_002" },
|
||||
{ "label": "AS2_2_003", "value": "AS2_2_003" },
|
||||
{ "label": "AS2_2_004", "value": "AS2_2_004" },
|
||||
{ "label": "AS2_2_005", "value": "AS2_2_005" }
|
||||
],
|
||||
"payloads": [
|
||||
{ "label": "满料架-A1", "value": "满料架-A1" },
|
||||
{ "label": "满料架-A2", "value": "满料架-A2" },
|
||||
{ "label": "空料架-B1", "value": "空料架-B1" },
|
||||
{ "label": "空料架-B2", "value": "空料架-B2" },
|
||||
{ "label": "空车", "value": "空车" },
|
||||
{ "label": "钢材-Q235", "value": "钢材-Q235" },
|
||||
{ "label": "铝合金-6061", "value": "铝合金-6061" }
|
||||
],
|
||||
"robotActions": [
|
||||
{ "label": "运输", "value": "TRANSPORT" },
|
||||
{ "label": "取货", "value": "PICKUP" },
|
||||
{ "label": "卸货", "value": "DROPOFF" },
|
||||
{ "label": "等待", "value": "WAIT" },
|
||||
{ "label": "充电", "value": "CHARGE" },
|
||||
{ "label": "清洁", "value": "CLEAN" }
|
||||
],
|
||||
"tasks": [
|
||||
{
|
||||
"id": "task-001",
|
||||
"name": "炉前缓存区到热处理上料交接区运输",
|
||||
"status": "IDLE",
|
||||
"createdAt": "2024-01-15T08:30:00.000Z",
|
||||
"parameters": {
|
||||
"startLocation": {
|
||||
"label": "起点",
|
||||
"type": "Select",
|
||||
"value": "FURNACE_BUFFER",
|
||||
"required": true,
|
||||
"options": [
|
||||
{ "label": "AP-1", "value": "AP-1" },
|
||||
{ "label": "AP-2", "value": "AP-2" },
|
||||
{ "label": "AP-3", "value": "AP-3" },
|
||||
{ "label": "AP-4", "value": "AP-4" },
|
||||
{ "label": "AP-5", "value": "AP-5" },
|
||||
{ "label": "炉前缓存区", "value": "FURNACE_BUFFER" },
|
||||
{ "label": "热处理上料交接区", "value": "HEAT_TREATMENT_LOADING" },
|
||||
{ "label": "成品存储区", "value": "FINISHED_STORAGE" },
|
||||
{ "label": "原料仓库", "value": "RAW_MATERIAL_WAREHOUSE" },
|
||||
{ "label": "质检区", "value": "QUALITY_INSPECTION" }
|
||||
]
|
||||
},
|
||||
"endLocation": {
|
||||
"label": "终点",
|
||||
"type": "Select",
|
||||
"value": "HEAT_TREATMENT_LOADING",
|
||||
"required": true,
|
||||
"options": [
|
||||
{ "label": "AP-1", "value": "AP-1" },
|
||||
{ "label": "AP-2", "value": "AP-2" },
|
||||
{ "label": "AP-3", "value": "AP-3" },
|
||||
{ "label": "AP-4", "value": "AP-4" },
|
||||
{ "label": "AP-5", "value": "AP-5" },
|
||||
{ "label": "炉前缓存区", "value": "FURNACE_BUFFER" },
|
||||
{ "label": "热处理上料交接区", "value": "HEAT_TREATMENT_LOADING" },
|
||||
{ "label": "成品存储区", "value": "FINISHED_STORAGE" },
|
||||
{ "label": "原料仓库", "value": "RAW_MATERIAL_WAREHOUSE" },
|
||||
{ "label": "质检区", "value": "QUALITY_INSPECTION" }
|
||||
]
|
||||
},
|
||||
"robotAction": {
|
||||
"label": "机器人动作",
|
||||
"type": "Select",
|
||||
"value": "TRANSPORT",
|
||||
"required": true,
|
||||
"options": [
|
||||
{ "label": "运输", "value": "TRANSPORT" },
|
||||
{ "label": "取货", "value": "PICKUP" },
|
||||
{ "label": "卸货", "value": "DROPOFF" },
|
||||
{ "label": "等待", "value": "WAIT" },
|
||||
{ "label": "充电", "value": "CHARGE" },
|
||||
{ "label": "清洁", "value": "CLEAN" }
|
||||
]
|
||||
},
|
||||
"payload": {
|
||||
"label": "载荷",
|
||||
"type": "Select",
|
||||
"value": "满料架-A1",
|
||||
"required": false,
|
||||
"options": [
|
||||
{ "label": "满料架-A1", "value": "满料架-A1" },
|
||||
{ "label": "满料架-A2", "value": "满料架-A2" },
|
||||
{ "label": "空料架-B1", "value": "空料架-B1" },
|
||||
{ "label": "空料架-B2", "value": "空料架-B2" },
|
||||
{ "label": "空车", "value": "空车" },
|
||||
{ "label": "钢材-Q235", "value": "钢材-Q235" },
|
||||
{ "label": "铝合金-6061", "value": "铝合金-6061" }
|
||||
]
|
||||
},
|
||||
"locationBay": {
|
||||
"label": "库位",
|
||||
"type": "Select",
|
||||
"value": "AS2_2_001",
|
||||
"required": false,
|
||||
"options": [
|
||||
{ "label": "AS2_2_001", "value": "AS2_2_001" },
|
||||
{ "label": "AS2_2_002", "value": "AS2_2_002" },
|
||||
{ "label": "AS2_2_003", "value": "AS2_2_003" },
|
||||
{ "label": "AS2_2_004", "value": "AS2_2_004" },
|
||||
{ "label": "AS2_2_005", "value": "AS2_2_005" }
|
||||
]
|
||||
}
|
||||
}
|
||||
},
|
||||
{
|
||||
"id": "task-002",
|
||||
"name": "原料仓库到质检区取货",
|
||||
"status": "IDLE",
|
||||
"createdAt": "2024-01-15T09:15:00.000Z",
|
||||
"parameters": {
|
||||
"startLocation": {
|
||||
"label": "起点",
|
||||
"type": "Select",
|
||||
"value": "RAW_MATERIAL_WAREHOUSE",
|
||||
"required": true,
|
||||
"options": [
|
||||
{ "label": "AP-1", "value": "AP-1" },
|
||||
{ "label": "AP-2", "value": "AP-2" },
|
||||
{ "label": "AP-3", "value": "AP-3" },
|
||||
{ "label": "AP-4", "value": "AP-4" },
|
||||
{ "label": "AP-5", "value": "AP-5" },
|
||||
{ "label": "炉前缓存区", "value": "FURNACE_BUFFER" },
|
||||
{ "label": "热处理上料交接区", "value": "HEAT_TREATMENT_LOADING" },
|
||||
{ "label": "成品存储区", "value": "FINISHED_STORAGE" },
|
||||
{ "label": "原料仓库", "value": "RAW_MATERIAL_WAREHOUSE" },
|
||||
{ "label": "质检区", "value": "QUALITY_INSPECTION" }
|
||||
]
|
||||
},
|
||||
"endLocation": {
|
||||
"label": "终点",
|
||||
"type": "Select",
|
||||
"value": "QUALITY_INSPECTION",
|
||||
"required": true,
|
||||
"options": [
|
||||
{ "label": "AP-1", "value": "AP-1" },
|
||||
{ "label": "AP-2", "value": "AP-2" },
|
||||
{ "label": "AP-3", "value": "AP-3" },
|
||||
{ "label": "AP-4", "value": "AP-4" },
|
||||
{ "label": "AP-5", "value": "AP-5" },
|
||||
{ "label": "炉前缓存区", "value": "FURNACE_BUFFER" },
|
||||
{ "label": "热处理上料交接区", "value": "HEAT_TREATMENT_LOADING" },
|
||||
{ "label": "成品存储区", "value": "FINISHED_STORAGE" },
|
||||
{ "label": "原料仓库", "value": "RAW_MATERIAL_WAREHOUSE" },
|
||||
{ "label": "质检区", "value": "QUALITY_INSPECTION" }
|
||||
]
|
||||
},
|
||||
"robotAction": {
|
||||
"label": "机器人动作",
|
||||
"type": "Select",
|
||||
"value": "PICKUP",
|
||||
"required": true,
|
||||
"options": [
|
||||
{ "label": "运输", "value": "TRANSPORT" },
|
||||
{ "label": "取货", "value": "PICKUP" },
|
||||
{ "label": "卸货", "value": "DROPOFF" },
|
||||
{ "label": "等待", "value": "WAIT" },
|
||||
{ "label": "充电", "value": "CHARGE" },
|
||||
{ "label": "清洁", "value": "CLEAN" }
|
||||
]
|
||||
},
|
||||
"payload": {
|
||||
"label": "载荷",
|
||||
"type": "Select",
|
||||
"value": "钢材-Q235",
|
||||
"required": false,
|
||||
"options": [
|
||||
{ "label": "满料架-A1", "value": "满料架-A1" },
|
||||
{ "label": "满料架-A2", "value": "满料架-A2" },
|
||||
{ "label": "空料架-B1", "value": "空料架-B1" },
|
||||
{ "label": "空料架-B2", "value": "空料架-B2" },
|
||||
{ "label": "空车", "value": "空车" },
|
||||
{ "label": "钢材-Q235", "value": "钢材-Q235" },
|
||||
{ "label": "铝合金-6061", "value": "铝合金-6061" }
|
||||
]
|
||||
},
|
||||
"locationBay": {
|
||||
"label": "库位",
|
||||
"type": "Select",
|
||||
"value": "AS2_2_002",
|
||||
"required": false,
|
||||
"options": [
|
||||
{ "label": "AS2_2_001", "value": "AS2_2_001" },
|
||||
{ "label": "AS2_2_002", "value": "AS2_2_002" },
|
||||
{ "label": "AS2_2_003", "value": "AS2_2_003" },
|
||||
{ "label": "AS2_2_004", "value": "AS2_2_004" },
|
||||
{ "label": "AS2_2_005", "value": "AS2_2_005" }
|
||||
]
|
||||
}
|
||||
}
|
||||
},
|
||||
{
|
||||
"id": "task-003",
|
||||
"name": "质检区到成品存储区卸货",
|
||||
"status": "COMPLETED",
|
||||
"createdAt": "2024-01-15T10:00:00.000Z",
|
||||
"parameters": {
|
||||
"startLocation": {
|
||||
"label": "起点",
|
||||
"type": "Select",
|
||||
"value": "QUALITY_INSPECTION",
|
||||
"required": true,
|
||||
"options": [
|
||||
{ "label": "AP-1", "value": "AP-1" },
|
||||
{ "label": "AP-2", "value": "AP-2" },
|
||||
{ "label": "AP-3", "value": "AP-3" },
|
||||
{ "label": "AP-4", "value": "AP-4" },
|
||||
{ "label": "AP-5", "value": "AP-5" },
|
||||
{ "label": "炉前缓存区", "value": "FURNACE_BUFFER" },
|
||||
{ "label": "热处理上料交接区", "value": "HEAT_TREATMENT_LOADING" },
|
||||
{ "label": "成品存储区", "value": "FINISHED_STORAGE" },
|
||||
{ "label": "原料仓库", "value": "RAW_MATERIAL_WAREHOUSE" },
|
||||
{ "label": "质检区", "value": "QUALITY_INSPECTION" }
|
||||
]
|
||||
},
|
||||
"endLocation": {
|
||||
"label": "终点",
|
||||
"type": "Select",
|
||||
"value": "FINISHED_STORAGE",
|
||||
"required": true,
|
||||
"options": [
|
||||
{ "label": "AP-1", "value": "AP-1" },
|
||||
{ "label": "AP-2", "value": "AP-2" },
|
||||
{ "label": "AP-3", "value": "AP-3" },
|
||||
{ "label": "AP-4", "value": "AP-4" },
|
||||
{ "label": "AP-5", "value": "AP-5" },
|
||||
{ "label": "炉前缓存区", "value": "FURNACE_BUFFER" },
|
||||
{ "label": "热处理上料交接区", "value": "HEAT_TREATMENT_LOADING" },
|
||||
{ "label": "成品存储区", "value": "FINISHED_STORAGE" },
|
||||
{ "label": "原料仓库", "value": "RAW_MATERIAL_WAREHOUSE" },
|
||||
{ "label": "质检区", "value": "QUALITY_INSPECTION" }
|
||||
]
|
||||
},
|
||||
"robotAction": {
|
||||
"label": "机器人动作",
|
||||
"type": "Select",
|
||||
"value": "DROPOFF",
|
||||
"required": true,
|
||||
"options": [
|
||||
{ "label": "运输", "value": "TRANSPORT" },
|
||||
{ "label": "取货", "value": "PICKUP" },
|
||||
{ "label": "卸货", "value": "DROPOFF" },
|
||||
{ "label": "等待", "value": "WAIT" },
|
||||
{ "label": "充电", "value": "CHARGE" },
|
||||
{ "label": "清洁", "value": "CLEAN" }
|
||||
]
|
||||
},
|
||||
"payload": {
|
||||
"label": "载荷",
|
||||
"type": "Select",
|
||||
"value": "铝合金-6061",
|
||||
"required": false,
|
||||
"options": [
|
||||
{ "label": "满料架-A1", "value": "满料架-A1" },
|
||||
{ "label": "满料架-A2", "value": "满料架-A2" },
|
||||
{ "label": "空料架-B1", "value": "空料架-B1" },
|
||||
{ "label": "空料架-B2", "value": "空料架-B2" },
|
||||
{ "label": "空车", "value": "空车" },
|
||||
{ "label": "钢材-Q235", "value": "钢材-Q235" },
|
||||
{ "label": "铝合金-6061", "value": "铝合金-6061" }
|
||||
]
|
||||
},
|
||||
"locationBay": {
|
||||
"label": "库位",
|
||||
"type": "Select",
|
||||
"value": "AS2_2_003",
|
||||
"required": false,
|
||||
"options": [
|
||||
{ "label": "AS2_2_001", "value": "AS2_2_001" },
|
||||
{ "label": "AS2_2_002", "value": "AS2_2_002" },
|
||||
{ "label": "AS2_2_003", "value": "AS2_2_003" },
|
||||
{ "label": "AS2_2_004", "value": "AS2_2_004" },
|
||||
{ "label": "AS2_2_005", "value": "AS2_2_005" }
|
||||
]
|
||||
}
|
||||
}
|
||||
},
|
||||
{
|
||||
"id": "task-004",
|
||||
"name": "AP-1到AP-2空车运输",
|
||||
"status": "IDLE",
|
||||
"createdAt": "2024-01-15T11:30:00.000Z",
|
||||
"parameters": {
|
||||
"startLocation": {
|
||||
"label": "起点",
|
||||
"type": "Select",
|
||||
"value": "AP-1",
|
||||
"required": true,
|
||||
"options": [
|
||||
{ "label": "AP-1", "value": "AP-1" },
|
||||
{ "label": "AP-2", "value": "AP-2" },
|
||||
{ "label": "AP-3", "value": "AP-3" },
|
||||
{ "label": "AP-4", "value": "AP-4" },
|
||||
{ "label": "AP-5", "value": "AP-5" },
|
||||
{ "label": "炉前缓存区", "value": "FURNACE_BUFFER" },
|
||||
{ "label": "热处理上料交接区", "value": "HEAT_TREATMENT_LOADING" },
|
||||
{ "label": "成品存储区", "value": "FINISHED_STORAGE" },
|
||||
{ "label": "原料仓库", "value": "RAW_MATERIAL_WAREHOUSE" },
|
||||
{ "label": "质检区", "value": "QUALITY_INSPECTION" }
|
||||
]
|
||||
},
|
||||
"endLocation": {
|
||||
"label": "终点",
|
||||
"type": "Select",
|
||||
"value": "AP-2",
|
||||
"required": true,
|
||||
"options": [
|
||||
{ "label": "AP-1", "value": "AP-1" },
|
||||
{ "label": "AP-2", "value": "AP-2" },
|
||||
{ "label": "AP-3", "value": "AP-3" },
|
||||
{ "label": "AP-4", "value": "AP-4" },
|
||||
{ "label": "AP-5", "value": "AP-5" },
|
||||
{ "label": "炉前缓存区", "value": "FURNACE_BUFFER" },
|
||||
{ "label": "热处理上料交接区", "value": "HEAT_TREATMENT_LOADING" },
|
||||
{ "label": "成品存储区", "value": "FINISHED_STORAGE" },
|
||||
{ "label": "原料仓库", "value": "RAW_MATERIAL_WAREHOUSE" },
|
||||
{ "label": "质检区", "value": "QUALITY_INSPECTION" }
|
||||
]
|
||||
},
|
||||
"robotAction": {
|
||||
"label": "机器人动作",
|
||||
"type": "Select",
|
||||
"value": "TRANSPORT",
|
||||
"required": true,
|
||||
"options": [
|
||||
{ "label": "运输", "value": "TRANSPORT" },
|
||||
{ "label": "取货", "value": "PICKUP" },
|
||||
{ "label": "卸货", "value": "DROPOFF" },
|
||||
{ "label": "等待", "value": "WAIT" },
|
||||
{ "label": "充电", "value": "CHARGE" },
|
||||
{ "label": "清洁", "value": "CLEAN" }
|
||||
]
|
||||
},
|
||||
"payload": {
|
||||
"label": "载荷",
|
||||
"type": "Select",
|
||||
"value": "空车",
|
||||
"required": false,
|
||||
"options": [
|
||||
{ "label": "满料架-A1", "value": "满料架-A1" },
|
||||
{ "label": "满料架-A2", "value": "满料架-A2" },
|
||||
{ "label": "空料架-B1", "value": "空料架-B1" },
|
||||
{ "label": "空料架-B2", "value": "空料架-B2" },
|
||||
{ "label": "空车", "value": "空车" },
|
||||
{ "label": "钢材-Q235", "value": "钢材-Q235" },
|
||||
{ "label": "铝合金-6061", "value": "铝合金-6061" }
|
||||
]
|
||||
},
|
||||
"locationBay": {
|
||||
"label": "库位",
|
||||
"type": "Select",
|
||||
"value": "AS2_2_004",
|
||||
"required": false,
|
||||
"options": [
|
||||
{ "label": "AS2_2_001", "value": "AS2_2_001" },
|
||||
{ "label": "AS2_2_002", "value": "AS2_2_002" },
|
||||
{ "label": "AS2_2_003", "value": "AS2_2_003" },
|
||||
{ "label": "AS2_2_004", "value": "AS2_2_004" },
|
||||
{ "label": "AS2_2_005", "value": "AS2_2_005" }
|
||||
]
|
||||
}
|
||||
}
|
||||
},
|
||||
{
|
||||
"id": "task-005",
|
||||
"name": "机器人充电任务",
|
||||
"status": "IDLE",
|
||||
"createdAt": "2024-01-15T12:00:00.000Z",
|
||||
"parameters": {
|
||||
"startLocation": {
|
||||
"label": "起点",
|
||||
"type": "Select",
|
||||
"value": "AP-3",
|
||||
"required": true,
|
||||
"options": [
|
||||
{ "label": "AP-1", "value": "AP-1" },
|
||||
{ "label": "AP-2", "value": "AP-2" },
|
||||
{ "label": "AP-3", "value": "AP-3" },
|
||||
{ "label": "AP-4", "value": "AP-4" },
|
||||
{ "label": "AP-5", "value": "AP-5" },
|
||||
{ "label": "炉前缓存区", "value": "FURNACE_BUFFER" },
|
||||
{ "label": "热处理上料交接区", "value": "HEAT_TREATMENT_LOADING" },
|
||||
{ "label": "成品存储区", "value": "FINISHED_STORAGE" },
|
||||
{ "label": "原料仓库", "value": "RAW_MATERIAL_WAREHOUSE" },
|
||||
{ "label": "质检区", "value": "QUALITY_INSPECTION" }
|
||||
]
|
||||
},
|
||||
"endLocation": {
|
||||
"label": "终点",
|
||||
"type": "Select",
|
||||
"value": "AP-3",
|
||||
"required": true,
|
||||
"options": [
|
||||
{ "label": "AP-1", "value": "AP-1" },
|
||||
{ "label": "AP-2", "value": "AP-2" },
|
||||
{ "label": "AP-3", "value": "AP-3" },
|
||||
{ "label": "AP-4", "value": "AP-4" },
|
||||
{ "label": "AP-5", "value": "AP-5" },
|
||||
{ "label": "炉前缓存区", "value": "FURNACE_BUFFER" },
|
||||
{ "label": "热处理上料交接区", "value": "HEAT_TREATMENT_LOADING" },
|
||||
{ "label": "成品存储区", "value": "FINISHED_STORAGE" },
|
||||
{ "label": "原料仓库", "value": "RAW_MATERIAL_WAREHOUSE" },
|
||||
{ "label": "质检区", "value": "QUALITY_INSPECTION" }
|
||||
]
|
||||
},
|
||||
"robotAction": {
|
||||
"label": "机器人动作",
|
||||
"type": "Select",
|
||||
"value": "CHARGE",
|
||||
"required": true,
|
||||
"options": [
|
||||
{ "label": "运输", "value": "TRANSPORT" },
|
||||
{ "label": "取货", "value": "PICKUP" },
|
||||
{ "label": "卸货", "value": "DROPOFF" },
|
||||
{ "label": "等待", "value": "WAIT" },
|
||||
{ "label": "充电", "value": "CHARGE" },
|
||||
{ "label": "清洁", "value": "CLEAN" }
|
||||
]
|
||||
},
|
||||
"payload": {
|
||||
"label": "载荷",
|
||||
"type": "Select",
|
||||
"value": "空车",
|
||||
"required": false,
|
||||
"options": [
|
||||
{ "label": "满料架-A1", "value": "满料架-A1" },
|
||||
{ "label": "满料架-A2", "value": "满料架-A2" },
|
||||
{ "label": "空料架-B1", "value": "空料架-B1" },
|
||||
{ "label": "空料架-B2", "value": "空料架-B2" },
|
||||
{ "label": "空车", "value": "空车" },
|
||||
{ "label": "钢材-Q235", "value": "钢材-Q235" },
|
||||
{ "label": "铝合金-6061", "value": "铝合金-6061" }
|
||||
]
|
||||
}
|
||||
}
|
||||
},
|
||||
{
|
||||
"id": "task-006",
|
||||
"name": "成品存储区清洁任务",
|
||||
"status": "RUNNING",
|
||||
"createdAt": "2024-01-15T13:15:00.000Z",
|
||||
"parameters": {
|
||||
"startLocation": {
|
||||
"label": "起点",
|
||||
"type": "Select",
|
||||
"value": "FINISHED_STORAGE",
|
||||
"required": true,
|
||||
"options": [
|
||||
{ "label": "AP-1", "value": "AP-1" },
|
||||
{ "label": "AP-2", "value": "AP-2" },
|
||||
{ "label": "AP-3", "value": "AP-3" },
|
||||
{ "label": "AP-4", "value": "AP-4" },
|
||||
{ "label": "AP-5", "value": "AP-5" },
|
||||
{ "label": "炉前缓存区", "value": "FURNACE_BUFFER" },
|
||||
{ "label": "热处理上料交接区", "value": "HEAT_TREATMENT_LOADING" },
|
||||
{ "label": "成品存储区", "value": "FINISHED_STORAGE" },
|
||||
{ "label": "原料仓库", "value": "RAW_MATERIAL_WAREHOUSE" },
|
||||
{ "label": "质检区", "value": "QUALITY_INSPECTION" }
|
||||
]
|
||||
},
|
||||
"endLocation": {
|
||||
"label": "终点",
|
||||
"type": "Select",
|
||||
"value": "FINISHED_STORAGE",
|
||||
"required": true,
|
||||
"options": [
|
||||
{ "label": "AP-1", "value": "AP-1" },
|
||||
{ "label": "AP-2", "value": "AP-2" },
|
||||
{ "label": "AP-3", "value": "AP-3" },
|
||||
{ "label": "AP-4", "value": "AP-4" },
|
||||
{ "label": "AP-5", "value": "AP-5" },
|
||||
{ "label": "炉前缓存区", "value": "FURNACE_BUFFER" },
|
||||
{ "label": "热处理上料交接区", "value": "HEAT_TREATMENT_LOADING" },
|
||||
{ "label": "成品存储区", "value": "FINISHED_STORAGE" },
|
||||
{ "label": "原料仓库", "value": "RAW_MATERIAL_WAREHOUSE" },
|
||||
{ "label": "质检区", "value": "QUALITY_INSPECTION" }
|
||||
]
|
||||
},
|
||||
"robotAction": {
|
||||
"label": "机器人动作",
|
||||
"type": "Select",
|
||||
"value": "CLEAN",
|
||||
"required": true,
|
||||
"options": [
|
||||
{ "label": "运输", "value": "TRANSPORT" },
|
||||
{ "label": "取货", "value": "PICKUP" },
|
||||
{ "label": "卸货", "value": "DROPOFF" },
|
||||
{ "label": "等待", "value": "WAIT" },
|
||||
{ "label": "充电", "value": "CHARGE" },
|
||||
{ "label": "清洁", "value": "CLEAN" }
|
||||
]
|
||||
},
|
||||
"payload": {
|
||||
"label": "载荷",
|
||||
"type": "Select",
|
||||
"value": "空车",
|
||||
"required": false,
|
||||
"options": [
|
||||
{ "label": "满料架-A1", "value": "满料架-A1" },
|
||||
{ "label": "满料架-A2", "value": "满料架-A2" },
|
||||
{ "label": "空料架-B1", "value": "空料架-B1" },
|
||||
{ "label": "空料架-B2", "value": "空料架-B2" },
|
||||
{ "label": "空车", "value": "空车" },
|
||||
{ "label": "钢材-Q235", "value": "钢材-Q235" },
|
||||
{ "label": "铝合金-6061", "value": "铝合金-6061" }
|
||||
]
|
||||
}
|
||||
}
|
||||
},
|
||||
{
|
||||
"id": "task-007",
|
||||
"name": "满料架A2从原料仓库到AP-4运输",
|
||||
"status": "IDLE",
|
||||
"createdAt": "2024-01-15T14:20:00.000Z",
|
||||
"parameters": {
|
||||
"startLocation": {
|
||||
"label": "起点",
|
||||
"type": "Select",
|
||||
"value": "RAW_MATERIAL_WAREHOUSE",
|
||||
"required": true,
|
||||
"options": [
|
||||
{ "label": "AP-1", "value": "AP-1" },
|
||||
{ "label": "AP-2", "value": "AP-2" },
|
||||
{ "label": "AP-3", "value": "AP-3" },
|
||||
{ "label": "AP-4", "value": "AP-4" },
|
||||
{ "label": "AP-5", "value": "AP-5" },
|
||||
{ "label": "炉前缓存区", "value": "FURNACE_BUFFER" },
|
||||
{ "label": "热处理上料交接区", "value": "HEAT_TREATMENT_LOADING" },
|
||||
{ "label": "成品存储区", "value": "FINISHED_STORAGE" },
|
||||
{ "label": "原料仓库", "value": "RAW_MATERIAL_WAREHOUSE" },
|
||||
{ "label": "质检区", "value": "QUALITY_INSPECTION" }
|
||||
]
|
||||
},
|
||||
"endLocation": {
|
||||
"label": "终点",
|
||||
"type": "Select",
|
||||
"value": "AP-4",
|
||||
"required": true,
|
||||
"options": [
|
||||
{ "label": "AP-1", "value": "AP-1" },
|
||||
{ "label": "AP-2", "value": "AP-2" },
|
||||
{ "label": "AP-3", "value": "AP-3" },
|
||||
{ "label": "AP-4", "value": "AP-4" },
|
||||
{ "label": "AP-5", "value": "AP-5" },
|
||||
{ "label": "炉前缓存区", "value": "FURNACE_BUFFER" },
|
||||
{ "label": "热处理上料交接区", "value": "HEAT_TREATMENT_LOADING" },
|
||||
{ "label": "成品存储区", "value": "FINISHED_STORAGE" },
|
||||
{ "label": "原料仓库", "value": "RAW_MATERIAL_WAREHOUSE" },
|
||||
{ "label": "质检区", "value": "QUALITY_INSPECTION" }
|
||||
]
|
||||
},
|
||||
"waypoint": {
|
||||
"label": "途经点",
|
||||
"type": "Select",
|
||||
"value": "QUALITY_INSPECTION",
|
||||
"required": false,
|
||||
"options": [
|
||||
{ "label": "AP-1", "value": "AP-1" },
|
||||
{ "label": "AP-2", "value": "AP-2" },
|
||||
{ "label": "AP-3", "value": "AP-3" },
|
||||
{ "label": "AP-4", "value": "AP-4" },
|
||||
{ "label": "AP-5", "value": "AP-5" },
|
||||
{ "label": "炉前缓存区", "value": "FURNACE_BUFFER" },
|
||||
{ "label": "热处理上料交接区", "value": "HEAT_TREATMENT_LOADING" },
|
||||
{ "label": "成品存储区", "value": "FINISHED_STORAGE" },
|
||||
{ "label": "原料仓库", "value": "RAW_MATERIAL_WAREHOUSE" },
|
||||
{ "label": "质检区", "value": "QUALITY_INSPECTION" }
|
||||
]
|
||||
},
|
||||
"robotAction": {
|
||||
"label": "机器人动作",
|
||||
"type": "Select",
|
||||
"value": "TRANSPORT",
|
||||
"required": true,
|
||||
"options": [
|
||||
{ "label": "运输", "value": "TRANSPORT" },
|
||||
{ "label": "取货", "value": "PICKUP" },
|
||||
{ "label": "卸货", "value": "DROPOFF" },
|
||||
{ "label": "等待", "value": "WAIT" },
|
||||
{ "label": "充电", "value": "CHARGE" },
|
||||
{ "label": "清洁", "value": "CLEAN" }
|
||||
]
|
||||
},
|
||||
"payload": {
|
||||
"label": "载荷",
|
||||
"type": "Select",
|
||||
"value": "满料架-A2",
|
||||
"required": false,
|
||||
"options": [
|
||||
{ "label": "满料架-A1", "value": "满料架-A1" },
|
||||
{ "label": "满料架-A2", "value": "满料架-A2" },
|
||||
{ "label": "空料架-B1", "value": "空料架-B1" },
|
||||
{ "label": "空料架-B2", "value": "空料架-B2" },
|
||||
{ "label": "空车", "value": "空车" },
|
||||
{ "label": "钢材-Q235", "value": "钢材-Q235" },
|
||||
{ "label": "铝合金-6061", "value": "铝合金-6061" }
|
||||
]
|
||||
},
|
||||
"locationBay": {
|
||||
"label": "库位",
|
||||
"type": "Select",
|
||||
"value": "AS2_2_005",
|
||||
"required": false,
|
||||
"options": [
|
||||
{ "label": "AS2_2_001", "value": "AS2_2_001" },
|
||||
{ "label": "AS2_2_002", "value": "AS2_2_002" },
|
||||
{ "label": "AS2_2_003", "value": "AS2_2_003" },
|
||||
{ "label": "AS2_2_004", "value": "AS2_2_004" },
|
||||
{ "label": "AS2_2_005", "value": "AS2_2_005" }
|
||||
]
|
||||
}
|
||||
}
|
||||
},
|
||||
{
|
||||
"id": "task-008",
|
||||
"name": "AP-5等待任务",
|
||||
"status": "ERROR",
|
||||
"createdAt": "2024-01-15T15:45:00.000Z",
|
||||
"parameters": {
|
||||
"startLocation": {
|
||||
"label": "起点",
|
||||
"type": "Select",
|
||||
"value": "AP-5",
|
||||
"required": true,
|
||||
"options": [
|
||||
{ "label": "AP-1", "value": "AP-1" },
|
||||
{ "label": "AP-2", "value": "AP-2" },
|
||||
{ "label": "AP-3", "value": "AP-3" },
|
||||
{ "label": "AP-4", "value": "AP-4" },
|
||||
{ "label": "AP-5", "value": "AP-5" },
|
||||
{ "label": "炉前缓存区", "value": "FURNACE_BUFFER" },
|
||||
{ "label": "热处理上料交接区", "value": "HEAT_TREATMENT_LOADING" },
|
||||
{ "label": "成品存储区", "value": "FINISHED_STORAGE" },
|
||||
{ "label": "原料仓库", "value": "RAW_MATERIAL_WAREHOUSE" },
|
||||
{ "label": "质检区", "value": "QUALITY_INSPECTION" }
|
||||
]
|
||||
},
|
||||
"endLocation": {
|
||||
"label": "终点",
|
||||
"type": "Select",
|
||||
"value": "AP-5",
|
||||
"required": true,
|
||||
"options": [
|
||||
{ "label": "AP-1", "value": "AP-1" },
|
||||
{ "label": "AP-2", "value": "AP-2" },
|
||||
{ "label": "AP-3", "value": "AP-3" },
|
||||
{ "label": "AP-4", "value": "AP-4" },
|
||||
{ "label": "AP-5", "value": "AP-5" },
|
||||
{ "label": "炉前缓存区", "value": "FURNACE_BUFFER" },
|
||||
{ "label": "热处理上料交接区", "value": "HEAT_TREATMENT_LOADING" },
|
||||
{ "label": "成品存储区", "value": "FINISHED_STORAGE" },
|
||||
{ "label": "原料仓库", "value": "RAW_MATERIAL_WAREHOUSE" },
|
||||
{ "label": "质检区", "value": "QUALITY_INSPECTION" }
|
||||
]
|
||||
},
|
||||
"robotAction": {
|
||||
"label": "机器人动作",
|
||||
"type": "Select",
|
||||
"value": "WAIT",
|
||||
"required": true,
|
||||
"options": [
|
||||
{ "label": "运输", "value": "TRANSPORT" },
|
||||
{ "label": "取货", "value": "PICKUP" },
|
||||
{ "label": "卸货", "value": "DROPOFF" },
|
||||
{ "label": "等待", "value": "WAIT" },
|
||||
{ "label": "充电", "value": "CHARGE" },
|
||||
{ "label": "清洁", "value": "CLEAN" }
|
||||
]
|
||||
},
|
||||
"payload": {
|
||||
"label": "载荷",
|
||||
"type": "Select",
|
||||
"value": "空料架-B1",
|
||||
"required": false,
|
||||
"options": [
|
||||
{ "label": "满料架-A1", "value": "满料架-A1" },
|
||||
{ "label": "满料架-A2", "value": "满料架-A2" },
|
||||
{ "label": "空料架-B1", "value": "空料架-B1" },
|
||||
{ "label": "空料架-B2", "value": "空料架-B2" },
|
||||
{ "label": "空车", "value": "空车" },
|
||||
{ "label": "钢材-Q235", "value": "钢材-Q235" },
|
||||
{ "label": "铝合金-6061", "value": "铝合金-6061" }
|
||||
]
|
||||
}
|
||||
}
|
||||
}
|
||||
],
|
||||
"serverUrl": "http://localhost:3000/api"
|
||||
}
|
11
index.js
Normal file
@ -0,0 +1,11 @@
|
||||
import 'react-native-get-random-values';
|
||||
import 'react-native-gesture-handler';
|
||||
/**
|
||||
* @format
|
||||
*/
|
||||
|
||||
import { AppRegistry } from 'react-native';
|
||||
import App from './App';
|
||||
import { name as appName } from './app.json';
|
||||
|
||||
AppRegistry.registerComponent(appName, () => App);
|
11
ios/.xcode.env
Normal file
@ -0,0 +1,11 @@
|
||||
# This `.xcode.env` file is versioned and is used to source the environment
|
||||
# used when running script phases inside Xcode.
|
||||
# To customize your local environment, you can create an `.xcode.env.local`
|
||||
# file that is not versioned.
|
||||
|
||||
# NODE_BINARY variable contains the PATH to the node executable.
|
||||
#
|
||||
# Customize the NODE_BINARY variable here.
|
||||
# For example, to use nvm with brew, add the following line
|
||||
# . "$(brew --prefix nvm)/nvm.sh" --no-use
|
||||
export NODE_BINARY=$(command -v node)
|
505
ios/MyReactNativeApp.xcodeproj/project.pbxproj
Normal file
@ -0,0 +1,505 @@
|
||||
// !$*UTF8*$!
|
||||
{
|
||||
archiveVersion = 1;
|
||||
classes = {
|
||||
};
|
||||
objectVersion = 54;
|
||||
objects = {
|
||||
|
||||
/* Begin PBXBuildFile section */
|
||||
0C80B921A6F3F58F76C31292 /* libPods-MyReactNativeApp.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 5DCACB8F33CDC322A6C60F78 /* libPods-MyReactNativeApp.a */; };
|
||||
13B07FBF1A68108700A75B9A /* Images.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 13B07FB51A68108700A75B9A /* Images.xcassets */; };
|
||||
761780ED2CA45674006654EE /* AppDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = 761780EC2CA45674006654EE /* AppDelegate.swift */; };
|
||||
81AB9BB82411601600AC10FF /* LaunchScreen.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 81AB9BB72411601600AC10FF /* LaunchScreen.storyboard */; };
|
||||
/* End PBXBuildFile section */
|
||||
|
||||
/* Begin PBXContainerItemProxy section */
|
||||
00E356F41AD99517003FC87E /* PBXContainerItemProxy */ = {
|
||||
isa = PBXContainerItemProxy;
|
||||
containerPortal = 83CBB9F71A601CBA00E9B192 /* Project object */;
|
||||
proxyType = 1;
|
||||
remoteGlobalIDString = 13B07F861A680F5B00A75B9A;
|
||||
remoteInfo = MyReactNativeApp;
|
||||
};
|
||||
/* End PBXContainerItemProxy section */
|
||||
|
||||
/* Begin PBXFileReference section */
|
||||
00E356F11AD99517003FC87E /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = "<group>"; };
|
||||
13B07F961A680F5B00A75B9A /* MyReactNativeApp.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = MyReactNativeApp.app; sourceTree = BUILT_PRODUCTS_DIR; };
|
||||
13B07FB51A68108700A75B9A /* Images.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; name = Images.xcassets; path = MyReactNativeApp/Images.xcassets; sourceTree = "<group>"; };
|
||||
13B07FB61A68108700A75B9A /* Info.plist */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.xml; name = Info.plist; path = MyReactNativeApp/Info.plist; sourceTree = "<group>"; };
|
||||
13B07FB81A68108700A75B9A /* PrivacyInfo.xcprivacy */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.plist.xml; name = PrivacyInfo.xcprivacy; path = MyReactNativeApp/PrivacyInfo.xcprivacy; sourceTree = "<group>"; };
|
||||
3B4392A12AC88292D35C810B /* Pods-MyReactNativeApp.debug.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-MyReactNativeApp.debug.xcconfig"; path = "Target Support Files/Pods-MyReactNativeApp/Pods-MyReactNativeApp.debug.xcconfig"; sourceTree = "<group>"; };
|
||||
5709B34CF0A7D63546082F79 /* Pods-MyReactNativeApp.release.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-MyReactNativeApp.release.xcconfig"; path = "Target Support Files/Pods-MyReactNativeApp/Pods-MyReactNativeApp.release.xcconfig"; sourceTree = "<group>"; };
|
||||
5DCACB8F33CDC322A6C60F78 /* libPods-MyReactNativeApp.a */ = {isa = PBXFileReference; explicitFileType = archive.ar; includeInIndex = 0; path = "libPods-MyReactNativeApp.a"; sourceTree = BUILT_PRODUCTS_DIR; };
|
||||
761780EC2CA45674006654EE /* AppDelegate.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; name = AppDelegate.swift; path = MyReactNativeApp/AppDelegate.swift; sourceTree = "<group>"; };
|
||||
81AB9BB72411601600AC10FF /* LaunchScreen.storyboard */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = file.storyboard; name = LaunchScreen.storyboard; path = MyReactNativeApp/LaunchScreen.storyboard; sourceTree = "<group>"; };
|
||||
ED297162215061F000B7C4FE /* JavaScriptCore.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = JavaScriptCore.framework; path = System/Library/Frameworks/JavaScriptCore.framework; sourceTree = SDKROOT; };
|
||||
/* End PBXFileReference section */
|
||||
|
||||
/* Begin PBXFrameworksBuildPhase section */
|
||||
13B07F8C1A680F5B00A75B9A /* Frameworks */ = {
|
||||
isa = PBXFrameworksBuildPhase;
|
||||
buildActionMask = 2147483647;
|
||||
files = (
|
||||
0C80B921A6F3F58F76C31292 /* libPods-MyReactNativeApp.a in Frameworks */,
|
||||
);
|
||||
runOnlyForDeploymentPostprocessing = 0;
|
||||
};
|
||||
/* End PBXFrameworksBuildPhase section */
|
||||
|
||||
/* Begin PBXGroup section */
|
||||
00E356F01AD99517003FC87E /* Supporting Files */ = {
|
||||
isa = PBXGroup;
|
||||
children = (
|
||||
00E356F11AD99517003FC87E /* Info.plist */,
|
||||
);
|
||||
name = "Supporting Files";
|
||||
sourceTree = "<group>";
|
||||
};
|
||||
13B07FAE1A68108700A75B9A /* MyReactNativeApp */ = {
|
||||
isa = PBXGroup;
|
||||
children = (
|
||||
13B07FB51A68108700A75B9A /* Images.xcassets */,
|
||||
761780EC2CA45674006654EE /* AppDelegate.swift */,
|
||||
13B07FB61A68108700A75B9A /* Info.plist */,
|
||||
81AB9BB72411601600AC10FF /* LaunchScreen.storyboard */,
|
||||
13B07FB81A68108700A75B9A /* PrivacyInfo.xcprivacy */,
|
||||
);
|
||||
name = MyReactNativeApp;
|
||||
sourceTree = "<group>";
|
||||
};
|
||||
2D16E6871FA4F8E400B85C8A /* Frameworks */ = {
|
||||
isa = PBXGroup;
|
||||
children = (
|
||||
ED297162215061F000B7C4FE /* JavaScriptCore.framework */,
|
||||
5DCACB8F33CDC322A6C60F78 /* libPods-MyReactNativeApp.a */,
|
||||
);
|
||||
name = Frameworks;
|
||||
sourceTree = "<group>";
|
||||
};
|
||||
832341AE1AAA6A7D00B99B32 /* Libraries */ = {
|
||||
isa = PBXGroup;
|
||||
children = (
|
||||
);
|
||||
name = Libraries;
|
||||
sourceTree = "<group>";
|
||||
};
|
||||
83CBB9F61A601CBA00E9B192 = {
|
||||
isa = PBXGroup;
|
||||
children = (
|
||||
13B07FAE1A68108700A75B9A /* MyReactNativeApp */,
|
||||
832341AE1AAA6A7D00B99B32 /* Libraries */,
|
||||
83CBBA001A601CBA00E9B192 /* Products */,
|
||||
2D16E6871FA4F8E400B85C8A /* Frameworks */,
|
||||
BBD78D7AC51CEA395F1C20DB /* Pods */,
|
||||
);
|
||||
indentWidth = 2;
|
||||
sourceTree = "<group>";
|
||||
tabWidth = 2;
|
||||
usesTabs = 0;
|
||||
};
|
||||
83CBBA001A601CBA00E9B192 /* Products */ = {
|
||||
isa = PBXGroup;
|
||||
children = (
|
||||
13B07F961A680F5B00A75B9A /* MyReactNativeApp.app */,
|
||||
);
|
||||
name = Products;
|
||||
sourceTree = "<group>";
|
||||
};
|
||||
BBD78D7AC51CEA395F1C20DB /* Pods */ = {
|
||||
isa = PBXGroup;
|
||||
children = (
|
||||
3B4392A12AC88292D35C810B /* Pods-MyReactNativeApp.debug.xcconfig */,
|
||||
5709B34CF0A7D63546082F79 /* Pods-MyReactNativeApp.release.xcconfig */,
|
||||
);
|
||||
path = Pods;
|
||||
sourceTree = "<group>";
|
||||
};
|
||||
/* End PBXGroup section */
|
||||
|
||||
/* Begin PBXNativeTarget section */
|
||||
13B07F861A680F5B00A75B9A /* MyReactNativeApp */ = {
|
||||
isa = PBXNativeTarget;
|
||||
buildConfigurationList = 13B07F931A680F5B00A75B9A /* Build configuration list for PBXNativeTarget "MyReactNativeApp" */;
|
||||
buildPhases = (
|
||||
C38B50BA6285516D6DCD4F65 /* [CP] Check Pods Manifest.lock */,
|
||||
13B07F871A680F5B00A75B9A /* Sources */,
|
||||
13B07F8C1A680F5B00A75B9A /* Frameworks */,
|
||||
13B07F8E1A680F5B00A75B9A /* Resources */,
|
||||
00DD1BFF1BD5951E006B06BC /* Bundle React Native code and images */,
|
||||
00EEFC60759A1932668264C0 /* [CP] Embed Pods Frameworks */,
|
||||
E235C05ADACE081382539298 /* [CP] Copy Pods Resources */,
|
||||
);
|
||||
buildRules = (
|
||||
);
|
||||
dependencies = (
|
||||
);
|
||||
name = MyReactNativeApp;
|
||||
productName = MyReactNativeApp;
|
||||
productReference = 13B07F961A680F5B00A75B9A /* MyReactNativeApp.app */;
|
||||
productType = "com.apple.product-type.application";
|
||||
};
|
||||
/* End PBXNativeTarget section */
|
||||
|
||||
/* Begin PBXProject section */
|
||||
83CBB9F71A601CBA00E9B192 /* Project object */ = {
|
||||
isa = PBXProject;
|
||||
attributes = {
|
||||
LastUpgradeCheck = 1210;
|
||||
TargetAttributes = {
|
||||
13B07F861A680F5B00A75B9A = {
|
||||
LastSwiftMigration = 1120;
|
||||
};
|
||||
};
|
||||
};
|
||||
buildConfigurationList = 83CBB9FA1A601CBA00E9B192 /* Build configuration list for PBXProject "MyReactNativeApp" */;
|
||||
compatibilityVersion = "Xcode 12.0";
|
||||
developmentRegion = en;
|
||||
hasScannedForEncodings = 0;
|
||||
knownRegions = (
|
||||
en,
|
||||
Base,
|
||||
);
|
||||
mainGroup = 83CBB9F61A601CBA00E9B192;
|
||||
productRefGroup = 83CBBA001A601CBA00E9B192 /* Products */;
|
||||
projectDirPath = "";
|
||||
projectRoot = "";
|
||||
targets = (
|
||||
13B07F861A680F5B00A75B9A /* MyReactNativeApp */,
|
||||
);
|
||||
};
|
||||
/* End PBXProject section */
|
||||
|
||||
/* Begin PBXResourcesBuildPhase section */
|
||||
00E356EC1AD99517003FC87E /* Resources */ = {
|
||||
isa = PBXResourcesBuildPhase;
|
||||
buildActionMask = 2147483647;
|
||||
files = (
|
||||
);
|
||||
runOnlyForDeploymentPostprocessing = 0;
|
||||
};
|
||||
13B07F8E1A680F5B00A75B9A /* Resources */ = {
|
||||
isa = PBXResourcesBuildPhase;
|
||||
buildActionMask = 2147483647;
|
||||
files = (
|
||||
81AB9BB82411601600AC10FF /* LaunchScreen.storyboard in Resources */,
|
||||
13B07FBF1A68108700A75B9A /* Images.xcassets in Resources */,
|
||||
);
|
||||
runOnlyForDeploymentPostprocessing = 0;
|
||||
};
|
||||
/* End PBXResourcesBuildPhase section */
|
||||
|
||||
/* Begin PBXShellScriptBuildPhase section */
|
||||
00DD1BFF1BD5951E006B06BC /* Bundle React Native code and images */ = {
|
||||
isa = PBXShellScriptBuildPhase;
|
||||
buildActionMask = 2147483647;
|
||||
files = (
|
||||
);
|
||||
inputPaths = (
|
||||
"$(SRCROOT)/.xcode.env.local",
|
||||
"$(SRCROOT)/.xcode.env",
|
||||
);
|
||||
name = "Bundle React Native code and images";
|
||||
outputPaths = (
|
||||
);
|
||||
runOnlyForDeploymentPostprocessing = 0;
|
||||
shellPath = /bin/sh;
|
||||
shellScript = "set -e\n\nWITH_ENVIRONMENT=\"$REACT_NATIVE_PATH/scripts/xcode/with-environment.sh\"\nREACT_NATIVE_XCODE=\"$REACT_NATIVE_PATH/scripts/react-native-xcode.sh\"\n\n/bin/sh -c \"$WITH_ENVIRONMENT $REACT_NATIVE_XCODE\"\n";
|
||||
};
|
||||
00EEFC60759A1932668264C0 /* [CP] Embed Pods Frameworks */ = {
|
||||
isa = PBXShellScriptBuildPhase;
|
||||
buildActionMask = 2147483647;
|
||||
files = (
|
||||
);
|
||||
inputFileListPaths = (
|
||||
"${PODS_ROOT}/Target Support Files/Pods-MyReactNativeApp/Pods-MyReactNativeApp-frameworks-${CONFIGURATION}-input-files.xcfilelist",
|
||||
);
|
||||
name = "[CP] Embed Pods Frameworks";
|
||||
outputFileListPaths = (
|
||||
"${PODS_ROOT}/Target Support Files/Pods-MyReactNativeApp/Pods-MyReactNativeApp-frameworks-${CONFIGURATION}-output-files.xcfilelist",
|
||||
);
|
||||
runOnlyForDeploymentPostprocessing = 0;
|
||||
shellPath = /bin/sh;
|
||||
shellScript = "\"${PODS_ROOT}/Target Support Files/Pods-MyReactNativeApp/Pods-MyReactNativeApp-frameworks.sh\"\n";
|
||||
showEnvVarsInLog = 0;
|
||||
};
|
||||
C38B50BA6285516D6DCD4F65 /* [CP] Check Pods Manifest.lock */ = {
|
||||
isa = PBXShellScriptBuildPhase;
|
||||
buildActionMask = 2147483647;
|
||||
files = (
|
||||
);
|
||||
inputFileListPaths = (
|
||||
);
|
||||
inputPaths = (
|
||||
"${PODS_PODFILE_DIR_PATH}/Podfile.lock",
|
||||
"${PODS_ROOT}/Manifest.lock",
|
||||
);
|
||||
name = "[CP] Check Pods Manifest.lock";
|
||||
outputFileListPaths = (
|
||||
);
|
||||
outputPaths = (
|
||||
"$(DERIVED_FILE_DIR)/Pods-MyReactNativeApp-checkManifestLockResult.txt",
|
||||
);
|
||||
runOnlyForDeploymentPostprocessing = 0;
|
||||
shellPath = /bin/sh;
|
||||
shellScript = "diff \"${PODS_PODFILE_DIR_PATH}/Podfile.lock\" \"${PODS_ROOT}/Manifest.lock\" > /dev/null\nif [ $? != 0 ] ; then\n # print error to STDERR\n echo \"error: The sandbox is not in sync with the Podfile.lock. Run 'pod install' or update your CocoaPods installation.\" >&2\n exit 1\nfi\n# This output is used by Xcode 'outputs' to avoid re-running this script phase.\necho \"SUCCESS\" > \"${SCRIPT_OUTPUT_FILE_0}\"\n";
|
||||
showEnvVarsInLog = 0;
|
||||
};
|
||||
E235C05ADACE081382539298 /* [CP] Copy Pods Resources */ = {
|
||||
isa = PBXShellScriptBuildPhase;
|
||||
buildActionMask = 2147483647;
|
||||
files = (
|
||||
);
|
||||
inputFileListPaths = (
|
||||
"${PODS_ROOT}/Target Support Files/Pods-MyReactNativeApp/Pods-MyReactNativeApp-resources-${CONFIGURATION}-input-files.xcfilelist",
|
||||
);
|
||||
name = "[CP] Copy Pods Resources";
|
||||
outputFileListPaths = (
|
||||
"${PODS_ROOT}/Target Support Files/Pods-MyReactNativeApp/Pods-MyReactNativeApp-resources-${CONFIGURATION}-output-files.xcfilelist",
|
||||
);
|
||||
runOnlyForDeploymentPostprocessing = 0;
|
||||
shellPath = /bin/sh;
|
||||
shellScript = "\"${PODS_ROOT}/Target Support Files/Pods-MyReactNativeApp/Pods-MyReactNativeApp-resources.sh\"\n";
|
||||
showEnvVarsInLog = 0;
|
||||
};
|
||||
/* End PBXShellScriptBuildPhase section */
|
||||
|
||||
/* Begin PBXSourcesBuildPhase section */
|
||||
13B07F871A680F5B00A75B9A /* Sources */ = {
|
||||
isa = PBXSourcesBuildPhase;
|
||||
buildActionMask = 2147483647;
|
||||
files = (
|
||||
761780ED2CA45674006654EE /* AppDelegate.swift in Sources */,
|
||||
);
|
||||
runOnlyForDeploymentPostprocessing = 0;
|
||||
};
|
||||
/* End PBXSourcesBuildPhase section */
|
||||
|
||||
/* Begin PBXTargetDependency section */
|
||||
00E356F51AD99517003FC87E /* PBXTargetDependency */ = {
|
||||
isa = PBXTargetDependency;
|
||||
target = 13B07F861A680F5B00A75B9A /* MyReactNativeApp */;
|
||||
targetProxy = 00E356F41AD99517003FC87E /* PBXContainerItemProxy */;
|
||||
};
|
||||
/* End PBXTargetDependency section */
|
||||
|
||||
/* Begin XCBuildConfiguration section */
|
||||
13B07F941A680F5B00A75B9A /* Debug */ = {
|
||||
isa = XCBuildConfiguration;
|
||||
baseConfigurationReference = 3B4392A12AC88292D35C810B /* Pods-MyReactNativeApp.debug.xcconfig */;
|
||||
buildSettings = {
|
||||
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
|
||||
CLANG_ENABLE_MODULES = YES;
|
||||
CURRENT_PROJECT_VERSION = 1;
|
||||
ENABLE_BITCODE = NO;
|
||||
INFOPLIST_FILE = MyReactNativeApp/Info.plist;
|
||||
IPHONEOS_DEPLOYMENT_TARGET = 15.1;
|
||||
LD_RUNPATH_SEARCH_PATHS = (
|
||||
"$(inherited)",
|
||||
"@executable_path/Frameworks",
|
||||
);
|
||||
MARKETING_VERSION = 1.0;
|
||||
OTHER_LDFLAGS = (
|
||||
"$(inherited)",
|
||||
"-ObjC",
|
||||
"-lc++",
|
||||
);
|
||||
PRODUCT_BUNDLE_IDENTIFIER = "org.reactjs.native.example.$(PRODUCT_NAME:rfc1034identifier)";
|
||||
PRODUCT_NAME = MyReactNativeApp;
|
||||
SWIFT_OPTIMIZATION_LEVEL = "-Onone";
|
||||
SWIFT_VERSION = 5.0;
|
||||
VERSIONING_SYSTEM = "apple-generic";
|
||||
};
|
||||
name = Debug;
|
||||
};
|
||||
13B07F951A680F5B00A75B9A /* Release */ = {
|
||||
isa = XCBuildConfiguration;
|
||||
baseConfigurationReference = 5709B34CF0A7D63546082F79 /* Pods-MyReactNativeApp.release.xcconfig */;
|
||||
buildSettings = {
|
||||
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;
|
||||
CLANG_ENABLE_MODULES = YES;
|
||||
CURRENT_PROJECT_VERSION = 1;
|
||||
INFOPLIST_FILE = MyReactNativeApp/Info.plist;
|
||||
IPHONEOS_DEPLOYMENT_TARGET = 15.1;
|
||||
LD_RUNPATH_SEARCH_PATHS = (
|
||||
"$(inherited)",
|
||||
"@executable_path/Frameworks",
|
||||
);
|
||||
MARKETING_VERSION = 1.0;
|
||||
OTHER_LDFLAGS = (
|
||||
"$(inherited)",
|
||||
"-ObjC",
|
||||
"-lc++",
|
||||
);
|
||||
PRODUCT_BUNDLE_IDENTIFIER = "org.reactjs.native.example.$(PRODUCT_NAME:rfc1034identifier)";
|
||||
PRODUCT_NAME = MyReactNativeApp;
|
||||
SWIFT_VERSION = 5.0;
|
||||
VERSIONING_SYSTEM = "apple-generic";
|
||||
};
|
||||
name = Release;
|
||||
};
|
||||
83CBBA201A601CBA00E9B192 /* Debug */ = {
|
||||
isa = XCBuildConfiguration;
|
||||
buildSettings = {
|
||||
ALWAYS_SEARCH_USER_PATHS = NO;
|
||||
CLANG_ANALYZER_LOCALIZABILITY_NONLOCALIZED = YES;
|
||||
CLANG_CXX_LANGUAGE_STANDARD = "c++20";
|
||||
CLANG_CXX_LIBRARY = "libc++";
|
||||
CLANG_ENABLE_MODULES = YES;
|
||||
CLANG_ENABLE_OBJC_ARC = YES;
|
||||
CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES;
|
||||
CLANG_WARN_BOOL_CONVERSION = YES;
|
||||
CLANG_WARN_COMMA = YES;
|
||||
CLANG_WARN_CONSTANT_CONVERSION = YES;
|
||||
CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES;
|
||||
CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;
|
||||
CLANG_WARN_EMPTY_BODY = YES;
|
||||
CLANG_WARN_ENUM_CONVERSION = YES;
|
||||
CLANG_WARN_INFINITE_RECURSION = YES;
|
||||
CLANG_WARN_INT_CONVERSION = YES;
|
||||
CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES;
|
||||
CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES;
|
||||
CLANG_WARN_OBJC_LITERAL_CONVERSION = YES;
|
||||
CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;
|
||||
CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES;
|
||||
CLANG_WARN_RANGE_LOOP_ANALYSIS = YES;
|
||||
CLANG_WARN_STRICT_PROTOTYPES = YES;
|
||||
CLANG_WARN_SUSPICIOUS_MOVE = YES;
|
||||
CLANG_WARN_UNREACHABLE_CODE = YES;
|
||||
CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
|
||||
"CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer";
|
||||
COPY_PHASE_STRIP = NO;
|
||||
ENABLE_STRICT_OBJC_MSGSEND = YES;
|
||||
ENABLE_TESTABILITY = YES;
|
||||
"EXCLUDED_ARCHS[sdk=iphonesimulator*]" = "";
|
||||
GCC_C_LANGUAGE_STANDARD = gnu99;
|
||||
GCC_DYNAMIC_NO_PIC = NO;
|
||||
GCC_NO_COMMON_BLOCKS = YES;
|
||||
GCC_OPTIMIZATION_LEVEL = 0;
|
||||
GCC_PREPROCESSOR_DEFINITIONS = (
|
||||
"DEBUG=1",
|
||||
"$(inherited)",
|
||||
);
|
||||
GCC_SYMBOLS_PRIVATE_EXTERN = NO;
|
||||
GCC_WARN_64_TO_32_BIT_CONVERSION = YES;
|
||||
GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR;
|
||||
GCC_WARN_UNDECLARED_SELECTOR = YES;
|
||||
GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;
|
||||
GCC_WARN_UNUSED_FUNCTION = YES;
|
||||
GCC_WARN_UNUSED_VARIABLE = YES;
|
||||
IPHONEOS_DEPLOYMENT_TARGET = 15.1;
|
||||
LD_RUNPATH_SEARCH_PATHS = (
|
||||
/usr/lib/swift,
|
||||
"$(inherited)",
|
||||
);
|
||||
LIBRARY_SEARCH_PATHS = (
|
||||
"\"$(SDKROOT)/usr/lib/swift\"",
|
||||
"\"$(TOOLCHAIN_DIR)/usr/lib/swift/$(PLATFORM_NAME)\"",
|
||||
"\"$(inherited)\"",
|
||||
);
|
||||
MTL_ENABLE_DEBUG_INFO = YES;
|
||||
ONLY_ACTIVE_ARCH = YES;
|
||||
OTHER_CPLUSPLUSFLAGS = (
|
||||
"$(OTHER_CFLAGS)",
|
||||
"-DFOLLY_NO_CONFIG",
|
||||
"-DFOLLY_MOBILE=1",
|
||||
"-DFOLLY_USE_LIBCPP=1",
|
||||
"-DFOLLY_CFG_NO_COROUTINES=1",
|
||||
"-DFOLLY_HAVE_CLOCK_GETTIME=1",
|
||||
);
|
||||
SDKROOT = iphoneos;
|
||||
};
|
||||
name = Debug;
|
||||
};
|
||||
83CBBA211A601CBA00E9B192 /* Release */ = {
|
||||
isa = XCBuildConfiguration;
|
||||
buildSettings = {
|
||||
ALWAYS_SEARCH_USER_PATHS = NO;
|
||||
CLANG_ANALYZER_LOCALIZABILITY_NONLOCALIZED = YES;
|
||||
CLANG_CXX_LANGUAGE_STANDARD = "c++20";
|
||||
CLANG_CXX_LIBRARY = "libc++";
|
||||
CLANG_ENABLE_MODULES = YES;
|
||||
CLANG_ENABLE_OBJC_ARC = YES;
|
||||
CLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES;
|
||||
CLANG_WARN_BOOL_CONVERSION = YES;
|
||||
CLANG_WARN_COMMA = YES;
|
||||
CLANG_WARN_CONSTANT_CONVERSION = YES;
|
||||
CLANG_WARN_DEPRECATED_OBJC_IMPLEMENTATIONS = YES;
|
||||
CLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;
|
||||
CLANG_WARN_EMPTY_BODY = YES;
|
||||
CLANG_WARN_ENUM_CONVERSION = YES;
|
||||
CLANG_WARN_INFINITE_RECURSION = YES;
|
||||
CLANG_WARN_INT_CONVERSION = YES;
|
||||
CLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES;
|
||||
CLANG_WARN_OBJC_IMPLICIT_RETAIN_SELF = YES;
|
||||
CLANG_WARN_OBJC_LITERAL_CONVERSION = YES;
|
||||
CLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;
|
||||
CLANG_WARN_QUOTED_INCLUDE_IN_FRAMEWORK_HEADER = YES;
|
||||
CLANG_WARN_RANGE_LOOP_ANALYSIS = YES;
|
||||
CLANG_WARN_STRICT_PROTOTYPES = YES;
|
||||
CLANG_WARN_SUSPICIOUS_MOVE = YES;
|
||||
CLANG_WARN_UNREACHABLE_CODE = YES;
|
||||
CLANG_WARN__DUPLICATE_METHOD_MATCH = YES;
|
||||
"CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer";
|
||||
COPY_PHASE_STRIP = YES;
|
||||
ENABLE_NS_ASSERTIONS = NO;
|
||||
ENABLE_STRICT_OBJC_MSGSEND = YES;
|
||||
"EXCLUDED_ARCHS[sdk=iphonesimulator*]" = "";
|
||||
GCC_C_LANGUAGE_STANDARD = gnu99;
|
||||
GCC_NO_COMMON_BLOCKS = YES;
|
||||
GCC_WARN_64_TO_32_BIT_CONVERSION = YES;
|
||||
GCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR;
|
||||
GCC_WARN_UNDECLARED_SELECTOR = YES;
|
||||
GCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;
|
||||
GCC_WARN_UNUSED_FUNCTION = YES;
|
||||
GCC_WARN_UNUSED_VARIABLE = YES;
|
||||
IPHONEOS_DEPLOYMENT_TARGET = 15.1;
|
||||
LD_RUNPATH_SEARCH_PATHS = (
|
||||
/usr/lib/swift,
|
||||
"$(inherited)",
|
||||
);
|
||||
LIBRARY_SEARCH_PATHS = (
|
||||
"\"$(SDKROOT)/usr/lib/swift\"",
|
||||
"\"$(TOOLCHAIN_DIR)/usr/lib/swift/$(PLATFORM_NAME)\"",
|
||||
"\"$(inherited)\"",
|
||||
);
|
||||
MTL_ENABLE_DEBUG_INFO = NO;
|
||||
OTHER_CPLUSPLUSFLAGS = (
|
||||
"$(OTHER_CFLAGS)",
|
||||
"-DFOLLY_NO_CONFIG",
|
||||
"-DFOLLY_MOBILE=1",
|
||||
"-DFOLLY_USE_LIBCPP=1",
|
||||
"-DFOLLY_CFG_NO_COROUTINES=1",
|
||||
"-DFOLLY_HAVE_CLOCK_GETTIME=1",
|
||||
);
|
||||
SDKROOT = iphoneos;
|
||||
VALIDATE_PRODUCT = YES;
|
||||
};
|
||||
name = Release;
|
||||
};
|
||||
/* End XCBuildConfiguration section */
|
||||
|
||||
/* Begin XCConfigurationList section */
|
||||
13B07F931A680F5B00A75B9A /* Build configuration list for PBXNativeTarget "MyReactNativeApp" */ = {
|
||||
isa = XCConfigurationList;
|
||||
buildConfigurations = (
|
||||
13B07F941A680F5B00A75B9A /* Debug */,
|
||||
13B07F951A680F5B00A75B9A /* Release */,
|
||||
);
|
||||
defaultConfigurationIsVisible = 0;
|
||||
defaultConfigurationName = Release;
|
||||
};
|
||||
83CBB9FA1A601CBA00E9B192 /* Build configuration list for PBXProject "MyReactNativeApp" */ = {
|
||||
isa = XCConfigurationList;
|
||||
buildConfigurations = (
|
||||
83CBBA201A601CBA00E9B192 /* Debug */,
|
||||
83CBBA211A601CBA00E9B192 /* Release */,
|
||||
);
|
||||
defaultConfigurationIsVisible = 0;
|
||||
defaultConfigurationName = Release;
|
||||
};
|
||||
/* End XCConfigurationList section */
|
||||
};
|
||||
rootObject = 83CBB9F71A601CBA00E9B192 /* Project object */;
|
||||
}
|
@ -0,0 +1,88 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<Scheme
|
||||
LastUpgradeVersion = "1210"
|
||||
version = "1.3">
|
||||
<BuildAction
|
||||
parallelizeBuildables = "YES"
|
||||
buildImplicitDependencies = "YES">
|
||||
<BuildActionEntries>
|
||||
<BuildActionEntry
|
||||
buildForTesting = "YES"
|
||||
buildForRunning = "YES"
|
||||
buildForProfiling = "YES"
|
||||
buildForArchiving = "YES"
|
||||
buildForAnalyzing = "YES">
|
||||
<BuildableReference
|
||||
BuildableIdentifier = "primary"
|
||||
BlueprintIdentifier = "13B07F861A680F5B00A75B9A"
|
||||
BuildableName = "MyReactNativeApp.app"
|
||||
BlueprintName = "MyReactNativeApp"
|
||||
ReferencedContainer = "container:MyReactNativeApp.xcodeproj">
|
||||
</BuildableReference>
|
||||
</BuildActionEntry>
|
||||
</BuildActionEntries>
|
||||
</BuildAction>
|
||||
<TestAction
|
||||
buildConfiguration = "Debug"
|
||||
selectedDebuggerIdentifier = "Xcode.DebuggerFoundation.Debugger.LLDB"
|
||||
selectedLauncherIdentifier = "Xcode.DebuggerFoundation.Launcher.LLDB"
|
||||
shouldUseLaunchSchemeArgsEnv = "YES">
|
||||
<Testables>
|
||||
<TestableReference
|
||||
skipped = "NO">
|
||||
<BuildableReference
|
||||
BuildableIdentifier = "primary"
|
||||
BlueprintIdentifier = "00E356ED1AD99517003FC87E"
|
||||
BuildableName = "MyReactNativeAppTests.xctest"
|
||||
BlueprintName = "MyReactNativeAppTests"
|
||||
ReferencedContainer = "container:MyReactNativeApp.xcodeproj">
|
||||
</BuildableReference>
|
||||
</TestableReference>
|
||||
</Testables>
|
||||
</TestAction>
|
||||
<LaunchAction
|
||||
buildConfiguration = "Debug"
|
||||
selectedDebuggerIdentifier = "Xcode.DebuggerFoundation.Debugger.LLDB"
|
||||
selectedLauncherIdentifier = "Xcode.DebuggerFoundation.Launcher.LLDB"
|
||||
launchStyle = "0"
|
||||
useCustomWorkingDirectory = "NO"
|
||||
ignoresPersistentStateOnLaunch = "NO"
|
||||
debugDocumentVersioning = "YES"
|
||||
debugServiceExtension = "internal"
|
||||
allowLocationSimulation = "YES">
|
||||
<BuildableProductRunnable
|
||||
runnableDebuggingMode = "0">
|
||||
<BuildableReference
|
||||
BuildableIdentifier = "primary"
|
||||
BlueprintIdentifier = "13B07F861A680F5B00A75B9A"
|
||||
BuildableName = "MyReactNativeApp.app"
|
||||
BlueprintName = "MyReactNativeApp"
|
||||
ReferencedContainer = "container:MyReactNativeApp.xcodeproj">
|
||||
</BuildableReference>
|
||||
</BuildableProductRunnable>
|
||||
</LaunchAction>
|
||||
<ProfileAction
|
||||
buildConfiguration = "Release"
|
||||
shouldUseLaunchSchemeArgsEnv = "YES"
|
||||
savedToolIdentifier = ""
|
||||
useCustomWorkingDirectory = "NO"
|
||||
debugDocumentVersioning = "YES">
|
||||
<BuildableProductRunnable
|
||||
runnableDebuggingMode = "0">
|
||||
<BuildableReference
|
||||
BuildableIdentifier = "primary"
|
||||
BlueprintIdentifier = "13B07F861A680F5B00A75B9A"
|
||||
BuildableName = "MyReactNativeApp.app"
|
||||
BlueprintName = "MyReactNativeApp"
|
||||
ReferencedContainer = "container:MyReactNativeApp.xcodeproj">
|
||||
</BuildableReference>
|
||||
</BuildableProductRunnable>
|
||||
</ProfileAction>
|
||||
<AnalyzeAction
|
||||
buildConfiguration = "Debug">
|
||||
</AnalyzeAction>
|
||||
<ArchiveAction
|
||||
buildConfiguration = "Release"
|
||||
revealArchiveInOrganizer = "YES">
|
||||
</ArchiveAction>
|
||||
</Scheme>
|
48
ios/MyReactNativeApp/AppDelegate.swift
Normal file
@ -0,0 +1,48 @@
|
||||
import UIKit
|
||||
import React
|
||||
import React_RCTAppDelegate
|
||||
import ReactAppDependencyProvider
|
||||
|
||||
@main
|
||||
class AppDelegate: UIResponder, UIApplicationDelegate {
|
||||
var window: UIWindow?
|
||||
|
||||
var reactNativeDelegate: ReactNativeDelegate?
|
||||
var reactNativeFactory: RCTReactNativeFactory?
|
||||
|
||||
func application(
|
||||
_ application: UIApplication,
|
||||
didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]? = nil
|
||||
) -> Bool {
|
||||
let delegate = ReactNativeDelegate()
|
||||
let factory = RCTReactNativeFactory(delegate: delegate)
|
||||
delegate.dependencyProvider = RCTAppDependencyProvider()
|
||||
|
||||
reactNativeDelegate = delegate
|
||||
reactNativeFactory = factory
|
||||
|
||||
window = UIWindow(frame: UIScreen.main.bounds)
|
||||
|
||||
factory.startReactNative(
|
||||
withModuleName: "MyReactNativeApp",
|
||||
in: window,
|
||||
launchOptions: launchOptions
|
||||
)
|
||||
|
||||
return true
|
||||
}
|
||||
}
|
||||
|
||||
class ReactNativeDelegate: RCTDefaultReactNativeFactoryDelegate {
|
||||
override func sourceURL(for bridge: RCTBridge) -> URL? {
|
||||
self.bundleURL()
|
||||
}
|
||||
|
||||
override func bundleURL() -> URL? {
|
||||
#if DEBUG
|
||||
RCTBundleURLProvider.sharedSettings().jsBundleURL(forBundleRoot: "index")
|
||||
#else
|
||||
Bundle.main.url(forResource: "main", withExtension: "jsbundle")
|
||||
#endif
|
||||
}
|
||||
}
|
@ -0,0 +1,53 @@
|
||||
{
|
||||
"images" : [
|
||||
{
|
||||
"idiom" : "iphone",
|
||||
"scale" : "2x",
|
||||
"size" : "20x20"
|
||||
},
|
||||
{
|
||||
"idiom" : "iphone",
|
||||
"scale" : "3x",
|
||||
"size" : "20x20"
|
||||
},
|
||||
{
|
||||
"idiom" : "iphone",
|
||||
"scale" : "2x",
|
||||
"size" : "29x29"
|
||||
},
|
||||
{
|
||||
"idiom" : "iphone",
|
||||
"scale" : "3x",
|
||||
"size" : "29x29"
|
||||
},
|
||||
{
|
||||
"idiom" : "iphone",
|
||||
"scale" : "2x",
|
||||
"size" : "40x40"
|
||||
},
|
||||
{
|
||||
"idiom" : "iphone",
|
||||
"scale" : "3x",
|
||||
"size" : "40x40"
|
||||
},
|
||||
{
|
||||
"idiom" : "iphone",
|
||||
"scale" : "2x",
|
||||
"size" : "60x60"
|
||||
},
|
||||
{
|
||||
"idiom" : "iphone",
|
||||
"scale" : "3x",
|
||||
"size" : "60x60"
|
||||
},
|
||||
{
|
||||
"idiom" : "ios-marketing",
|
||||
"scale" : "1x",
|
||||
"size" : "1024x1024"
|
||||
}
|
||||
],
|
||||
"info" : {
|
||||
"author" : "xcode",
|
||||
"version" : 1
|
||||
}
|
||||
}
|
6
ios/MyReactNativeApp/Images.xcassets/Contents.json
Normal file
@ -0,0 +1,6 @@
|
||||
{
|
||||
"info" : {
|
||||
"version" : 1,
|
||||
"author" : "xcode"
|
||||
}
|
||||
}
|
52
ios/MyReactNativeApp/Info.plist
Normal file
@ -0,0 +1,52 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||
<plist version="1.0">
|
||||
<dict>
|
||||
<key>CFBundleDevelopmentRegion</key>
|
||||
<string>en</string>
|
||||
<key>CFBundleDisplayName</key>
|
||||
<string>MyReactNativeApp</string>
|
||||
<key>CFBundleExecutable</key>
|
||||
<string>$(EXECUTABLE_NAME)</string>
|
||||
<key>CFBundleIdentifier</key>
|
||||
<string>$(PRODUCT_BUNDLE_IDENTIFIER)</string>
|
||||
<key>CFBundleInfoDictionaryVersion</key>
|
||||
<string>6.0</string>
|
||||
<key>CFBundleName</key>
|
||||
<string>$(PRODUCT_NAME)</string>
|
||||
<key>CFBundlePackageType</key>
|
||||
<string>APPL</string>
|
||||
<key>CFBundleShortVersionString</key>
|
||||
<string>$(MARKETING_VERSION)</string>
|
||||
<key>CFBundleSignature</key>
|
||||
<string>????</string>
|
||||
<key>CFBundleVersion</key>
|
||||
<string>$(CURRENT_PROJECT_VERSION)</string>
|
||||
<key>LSRequiresIPhoneOS</key>
|
||||
<true/>
|
||||
<key>NSAppTransportSecurity</key>
|
||||
<dict>
|
||||
<!-- Do not change NSAllowsArbitraryLoads to true, or you will risk app rejection! -->
|
||||
<key>NSAllowsArbitraryLoads</key>
|
||||
<false/>
|
||||
<key>NSAllowsLocalNetworking</key>
|
||||
<true/>
|
||||
</dict>
|
||||
<key>NSLocationWhenInUseUsageDescription</key>
|
||||
<string></string>
|
||||
<key>UILaunchStoryboardName</key>
|
||||
<string>LaunchScreen</string>
|
||||
<key>UIRequiredDeviceCapabilities</key>
|
||||
<array>
|
||||
<string>arm64</string>
|
||||
</array>
|
||||
<key>UISupportedInterfaceOrientations</key>
|
||||
<array>
|
||||
<string>UIInterfaceOrientationPortrait</string>
|
||||
<string>UIInterfaceOrientationLandscapeLeft</string>
|
||||
<string>UIInterfaceOrientationLandscapeRight</string>
|
||||
</array>
|
||||
<key>UIViewControllerBasedStatusBarAppearance</key>
|
||||
<false/>
|
||||
</dict>
|
||||
</plist>
|
47
ios/MyReactNativeApp/LaunchScreen.storyboard
Normal file
@ -0,0 +1,47 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<document type="com.apple.InterfaceBuilder3.CocoaTouch.Storyboard.XIB" version="3.0" toolsVersion="15702" targetRuntime="iOS.CocoaTouch" propertyAccessControl="none" useAutolayout="YES" launchScreen="YES" useTraitCollections="YES" useSafeAreas="YES" colorMatched="YES" initialViewController="01J-lp-oVM">
|
||||
<device id="retina4_7" orientation="portrait" appearance="light"/>
|
||||
<dependencies>
|
||||
<deployment identifier="iOS"/>
|
||||
<plugIn identifier="com.apple.InterfaceBuilder.IBCocoaTouchPlugin" version="15704"/>
|
||||
<capability name="Safe area layout guides" minToolsVersion="9.0"/>
|
||||
<capability name="documents saved in the Xcode 8 format" minToolsVersion="8.0"/>
|
||||
</dependencies>
|
||||
<scenes>
|
||||
<!--View Controller-->
|
||||
<scene sceneID="EHf-IW-A2E">
|
||||
<objects>
|
||||
<viewController id="01J-lp-oVM" sceneMemberID="viewController">
|
||||
<view key="view" contentMode="scaleToFill" id="Ze5-6b-2t3">
|
||||
<rect key="frame" x="0.0" y="0.0" width="375" height="667"/>
|
||||
<autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/>
|
||||
<subviews>
|
||||
<label opaque="NO" clipsSubviews="YES" userInteractionEnabled="NO" contentMode="left" horizontalHuggingPriority="251" verticalHuggingPriority="251" text="MyReactNativeApp" textAlignment="center" lineBreakMode="middleTruncation" baselineAdjustment="alignBaselines" minimumFontSize="18" translatesAutoresizingMaskIntoConstraints="NO" id="GJd-Yh-RWb">
|
||||
<rect key="frame" x="0.0" y="202" width="375" height="43"/>
|
||||
<fontDescription key="fontDescription" type="boldSystem" pointSize="36"/>
|
||||
<nil key="highlightedColor"/>
|
||||
</label>
|
||||
<label opaque="NO" clipsSubviews="YES" userInteractionEnabled="NO" contentMode="left" horizontalHuggingPriority="251" verticalHuggingPriority="251" text="Powered by React Native" textAlignment="center" lineBreakMode="tailTruncation" baselineAdjustment="alignBaselines" minimumFontSize="9" translatesAutoresizingMaskIntoConstraints="NO" id="MN2-I3-ftu">
|
||||
<rect key="frame" x="0.0" y="626" width="375" height="21"/>
|
||||
<fontDescription key="fontDescription" type="system" pointSize="17"/>
|
||||
<nil key="highlightedColor"/>
|
||||
</label>
|
||||
</subviews>
|
||||
<color key="backgroundColor" systemColor="systemBackgroundColor" cocoaTouchSystemColor="whiteColor"/>
|
||||
<constraints>
|
||||
<constraint firstItem="Bcu-3y-fUS" firstAttribute="bottom" secondItem="MN2-I3-ftu" secondAttribute="bottom" constant="20" id="OZV-Vh-mqD"/>
|
||||
<constraint firstItem="Bcu-3y-fUS" firstAttribute="centerX" secondItem="GJd-Yh-RWb" secondAttribute="centerX" id="Q3B-4B-g5h"/>
|
||||
<constraint firstItem="MN2-I3-ftu" firstAttribute="centerX" secondItem="Bcu-3y-fUS" secondAttribute="centerX" id="akx-eg-2ui"/>
|
||||
<constraint firstItem="MN2-I3-ftu" firstAttribute="leading" secondItem="Bcu-3y-fUS" secondAttribute="leading" id="i1E-0Y-4RG"/>
|
||||
<constraint firstItem="GJd-Yh-RWb" firstAttribute="centerY" secondItem="Ze5-6b-2t3" secondAttribute="bottom" multiplier="1/3" constant="1" id="moa-c2-u7t"/>
|
||||
<constraint firstItem="GJd-Yh-RWb" firstAttribute="leading" secondItem="Bcu-3y-fUS" secondAttribute="leading" symbolic="YES" id="x7j-FC-K8j"/>
|
||||
</constraints>
|
||||
<viewLayoutGuide key="safeArea" id="Bcu-3y-fUS"/>
|
||||
</view>
|
||||
</viewController>
|
||||
<placeholder placeholderIdentifier="IBFirstResponder" id="iYj-Kq-Ea1" userLabel="First Responder" sceneMemberID="firstResponder"/>
|
||||
</objects>
|
||||
<point key="canvasLocation" x="52.173913043478265" y="375"/>
|
||||
</scene>
|
||||
</scenes>
|
||||
</document>
|
37
ios/MyReactNativeApp/PrivacyInfo.xcprivacy
Normal file
@ -0,0 +1,37 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
||||
<plist version="1.0">
|
||||
<dict>
|
||||
<key>NSPrivacyAccessedAPITypes</key>
|
||||
<array>
|
||||
<dict>
|
||||
<key>NSPrivacyAccessedAPIType</key>
|
||||
<string>NSPrivacyAccessedAPICategoryFileTimestamp</string>
|
||||
<key>NSPrivacyAccessedAPITypeReasons</key>
|
||||
<array>
|
||||
<string>C617.1</string>
|
||||
</array>
|
||||
</dict>
|
||||
<dict>
|
||||
<key>NSPrivacyAccessedAPIType</key>
|
||||
<string>NSPrivacyAccessedAPICategoryUserDefaults</string>
|
||||
<key>NSPrivacyAccessedAPITypeReasons</key>
|
||||
<array>
|
||||
<string>CA92.1</string>
|
||||
</array>
|
||||
</dict>
|
||||
<dict>
|
||||
<key>NSPrivacyAccessedAPIType</key>
|
||||
<string>NSPrivacyAccessedAPICategorySystemBootTime</string>
|
||||
<key>NSPrivacyAccessedAPITypeReasons</key>
|
||||
<array>
|
||||
<string>35F9.1</string>
|
||||
</array>
|
||||
</dict>
|
||||
</array>
|
||||
<key>NSPrivacyCollectedDataTypes</key>
|
||||
<array/>
|
||||
<key>NSPrivacyTracking</key>
|
||||
<false/>
|
||||
</dict>
|
||||
</plist>
|
35
ios/Podfile
Normal file
@ -0,0 +1,35 @@
|
||||
# Resolve react_native_pods.rb with node to allow for hoisting
|
||||
require Pod::Executable.execute_command('node', ['-p',
|
||||
'require.resolve(
|
||||
"react-native/scripts/react_native_pods.rb",
|
||||
{paths: [process.argv[1]]},
|
||||
)', __dir__]).strip
|
||||
|
||||
platform :ios, min_ios_version_supported
|
||||
prepare_react_native_project!
|
||||
|
||||
linkage = ENV['USE_FRAMEWORKS']
|
||||
if linkage != nil
|
||||
Pod::UI.puts "Configuring Pod with #{linkage}ally linked Frameworks".green
|
||||
use_frameworks! :linkage => linkage.to_sym
|
||||
end
|
||||
|
||||
target 'MyReactNativeApp' do
|
||||
config = use_native_modules!
|
||||
|
||||
use_react_native!(
|
||||
:path => config[:reactNativePath],
|
||||
# An absolute path to your application root.
|
||||
:app_path => "#{Pod::Config.instance.installation_root}/.."
|
||||
)
|
||||
|
||||
post_install do |installer|
|
||||
# https://github.com/facebook/react-native/blob/main/packages/react-native/scripts/react_native_pods.rb#L197-L202
|
||||
react_native_post_install(
|
||||
installer,
|
||||
config[:reactNativePath],
|
||||
:mac_catalyst_enabled => false,
|
||||
# :ccache_enabled => true
|
||||
)
|
||||
end
|
||||
end
|
3
jest.config.js
Normal file
@ -0,0 +1,3 @@
|
||||
module.exports = {
|
||||
preset: 'react-native',
|
||||
};
|
11
metro.config.js
Normal file
@ -0,0 +1,11 @@
|
||||
const { getDefaultConfig, mergeConfig } = require('@react-native/metro-config');
|
||||
|
||||
/**
|
||||
* Metro configuration
|
||||
* https://reactnative.dev/docs/metro
|
||||
*
|
||||
* @type {import('@react-native/metro-config').MetroConfig}
|
||||
*/
|
||||
const config = {};
|
||||
|
||||
module.exports = mergeConfig(getDefaultConfig(__dirname), config);
|
12918
package-lock.json
generated
Normal file
53
package.json
Normal file
@ -0,0 +1,53 @@
|
||||
{
|
||||
"name": "MyReactNativeApp",
|
||||
"version": "0.0.1",
|
||||
"private": true,
|
||||
"scripts": {
|
||||
"android": "react-native run-android",
|
||||
"ios": "react-native run-ios",
|
||||
"lint": "eslint .",
|
||||
"start": "react-native start",
|
||||
"test": "jest"
|
||||
},
|
||||
"dependencies": {
|
||||
"@react-native-async-storage/async-storage": "^2.2.0",
|
||||
"@react-native/new-app-screen": "0.80.1",
|
||||
"@react-navigation/bottom-tabs": "^7.4.2",
|
||||
"@react-navigation/native": "^7.1.14",
|
||||
"@react-navigation/stack": "^7.4.2",
|
||||
"@rneui/base": "^4.0.0-rc.8",
|
||||
"@rneui/themed": "^4.0.0-rc.8",
|
||||
"react": "19.1.0",
|
||||
"react-native": "0.80.1",
|
||||
"react-native-gesture-handler": "^2.27.1",
|
||||
"react-native-get-random-values": "^1.11.0",
|
||||
"react-native-safe-area-context": "^5.5.2",
|
||||
"react-native-screens": "^4.13.1",
|
||||
"react-native-vector-icons": "^10.2.0",
|
||||
"uuid": "^11.1.0"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@babel/core": "^7.25.2",
|
||||
"@babel/preset-env": "^7.25.3",
|
||||
"@babel/runtime": "^7.25.0",
|
||||
"@react-native-community/cli": "19.0.0",
|
||||
"@react-native-community/cli-platform-android": "19.0.0",
|
||||
"@react-native-community/cli-platform-ios": "19.0.0",
|
||||
"@react-native/babel-preset": "0.80.1",
|
||||
"@react-native/eslint-config": "0.80.1",
|
||||
"@react-native/metro-config": "0.80.1",
|
||||
"@react-native/typescript-config": "0.80.1",
|
||||
"@types/jest": "^29.5.13",
|
||||
"@types/react": "^19.1.0",
|
||||
"@types/react-test-renderer": "^19.1.0",
|
||||
"@types/uuid": "^10.0.0",
|
||||
"eslint": "^8.19.0",
|
||||
"jest": "^29.6.3",
|
||||
"prettier": "2.8.8",
|
||||
"react-test-renderer": "19.1.0",
|
||||
"typescript": "5.0.4"
|
||||
},
|
||||
"engines": {
|
||||
"node": ">=18"
|
||||
}
|
||||
}
|
50
src/components/BottomActionBar.tsx
Normal file
@ -0,0 +1,50 @@
|
||||
import React from 'react';
|
||||
import { View, StyleSheet } from 'react-native';
|
||||
import { Button, useTheme } from '@rneui/themed';
|
||||
import Icon from 'react-native-vector-icons/MaterialCommunityIcons';
|
||||
|
||||
interface BottomActionBarProps {
|
||||
onRun: () => void;
|
||||
onSave: () => void;
|
||||
onUndo: () => void;
|
||||
onRestore: () => void;
|
||||
onBack: () => void;
|
||||
isSaveDisabled?: boolean;
|
||||
}
|
||||
|
||||
const BottomActionBar: React.FC<BottomActionBarProps> = ({
|
||||
onRun,
|
||||
onSave,
|
||||
onUndo,
|
||||
onRestore,
|
||||
onBack,
|
||||
isSaveDisabled = true,
|
||||
}) => {
|
||||
const { theme } = useTheme();
|
||||
|
||||
return (
|
||||
<View style={[styles.bottom, { backgroundColor: theme.colors.background }]}>
|
||||
<Button type="clear" onPress={onBack} icon={<Icon name="arrow-left" size={24} color={theme.colors.primary} />} />
|
||||
<Button type="clear" onPress={onRun} icon={<Icon name="play" size={24} color={theme.colors.primary} />} />
|
||||
<Button type="clear" onPress={onSave} disabled={isSaveDisabled} icon={<Icon name="content-save" size={24} color={isSaveDisabled ? theme.colors.disabled : theme.colors.primary} />} />
|
||||
<Button type="clear" onPress={onUndo} icon={<Icon name="undo" size={24} color={theme.colors.primary} />} />
|
||||
<Button type="clear" onPress={onRestore} icon={<Icon name="restore" size={24} color={theme.colors.primary} />} />
|
||||
</View>
|
||||
);
|
||||
};
|
||||
|
||||
const styles = StyleSheet.create({
|
||||
bottom: {
|
||||
position: 'absolute',
|
||||
left: 0,
|
||||
right: 0,
|
||||
bottom: 0,
|
||||
flexDirection: 'row',
|
||||
justifyContent: 'space-around',
|
||||
paddingVertical: 8,
|
||||
borderTopWidth: 1,
|
||||
borderTopColor: '#e0e0e0',
|
||||
},
|
||||
});
|
||||
|
||||
export default BottomActionBar;
|
68
src/components/TaskCard.tsx
Normal file
@ -0,0 +1,68 @@
|
||||
import React from 'react';
|
||||
import { StyleSheet, TouchableOpacity, Dimensions } from 'react-native';
|
||||
import { Card, Chip } from '@rneui/themed';
|
||||
import { Task, TaskStatus } from '../types/task';
|
||||
|
||||
interface TaskCardProps {
|
||||
task: Task;
|
||||
onPress: (task: Task) => void;
|
||||
}
|
||||
|
||||
const statusColors: Record<TaskStatus, string> = {
|
||||
IDLE: 'grey',
|
||||
RUNNING: 'blue',
|
||||
COMPLETED: 'green',
|
||||
ERROR: 'red',
|
||||
};
|
||||
|
||||
const { width } = Dimensions.get('window');
|
||||
const cardWidth = (width - 32) / 2; // 减去padding,除以2得到每个卡片的宽度
|
||||
|
||||
const TaskCard: React.FC<TaskCardProps> = ({ task, onPress }) => {
|
||||
return (
|
||||
<TouchableOpacity onPress={() => onPress(task)} style={styles.touchable}>
|
||||
<Card containerStyle={styles.card}>
|
||||
<Card.Title style={styles.title}>{task.name}</Card.Title>
|
||||
<Card.Divider />
|
||||
<Chip
|
||||
title={task.status}
|
||||
icon={{
|
||||
name: 'information',
|
||||
type: 'material-community',
|
||||
size: 20,
|
||||
color: statusColors[task.status],
|
||||
}}
|
||||
type="outline"
|
||||
containerStyle={styles.chip}
|
||||
titleStyle={[styles.chipTitle, { color: statusColors[task.status] }]}
|
||||
buttonStyle={{ borderColor: statusColors[task.status] }}
|
||||
/>
|
||||
</Card>
|
||||
</TouchableOpacity>
|
||||
);
|
||||
};
|
||||
|
||||
const styles = StyleSheet.create({
|
||||
touchable: {
|
||||
width: cardWidth,
|
||||
marginBottom: 8,
|
||||
},
|
||||
card: {
|
||||
margin: 4,
|
||||
borderRadius: 8,
|
||||
width: cardWidth - 8, // 减去margin
|
||||
},
|
||||
title: {
|
||||
marginBottom: 12,
|
||||
minHeight: 50, // Ensure cards have similar height
|
||||
textAlign: 'left',
|
||||
},
|
||||
chip: {
|
||||
alignSelf: 'flex-start',
|
||||
},
|
||||
chipTitle: {
|
||||
fontSize: 12,
|
||||
},
|
||||
});
|
||||
|
||||
export default TaskCard;
|
318
src/components/TaskForm.tsx
Normal file
@ -0,0 +1,318 @@
|
||||
import React, { useState, useRef, useEffect } from 'react';
|
||||
import { StyleSheet, ScrollView, TouchableOpacity, View } from 'react-native';
|
||||
import { Input, BottomSheet, ListItem, Button } from '@rneui/themed';
|
||||
import { Task, RobotAction } from '../types/task';
|
||||
import { useTasks } from '../context/TasksContext';
|
||||
|
||||
interface TaskFormProps {
|
||||
task: Task;
|
||||
onTaskChange: (updatedTask: Task) => void;
|
||||
}
|
||||
|
||||
const TaskForm: React.FC<TaskFormProps> = ({ task, onTaskChange }) => {
|
||||
const { locations, payloads, robotActions } = useTasks();
|
||||
const [isVisible, setIsVisible] = useState(false);
|
||||
const [currentField, setCurrentField] = useState('');
|
||||
const [currentItems, setCurrentItems] = useState<
|
||||
{ label: string; value: string }[]
|
||||
>([]);
|
||||
const [isQrInputFocused, setIsQrInputFocused] = useState(true);
|
||||
|
||||
// 创建隐藏的二维码扫描输入框的ref
|
||||
const qrScanInputRef = useRef<any>(null);
|
||||
|
||||
// 用于防止循环更新的标志
|
||||
const isUpdatingFromQrRef = useRef(false);
|
||||
const isUpdatingFromFormRef = useRef(false);
|
||||
|
||||
// 组件挂载时自动聚焦到隐藏的扫描输入框
|
||||
useEffect(() => {
|
||||
const timer = setTimeout(() => {
|
||||
if (qrScanInputRef.current) {
|
||||
qrScanInputRef.current.focus();
|
||||
}
|
||||
}, 100);
|
||||
|
||||
return () => clearTimeout(timer);
|
||||
}, []);
|
||||
|
||||
// 解析二维码信息对象字面量格式
|
||||
const parseQrCodeInfo = (infoString: string): Partial<Task['parameters']> => {
|
||||
const result: Partial<Task['parameters']> = {};
|
||||
|
||||
try {
|
||||
// 移除花括号和换行符,准备解析
|
||||
let cleanString = infoString.trim();
|
||||
if (cleanString.startsWith('{')) {
|
||||
cleanString = cleanString.substring(1);
|
||||
}
|
||||
if (cleanString.endsWith('}')) {
|
||||
cleanString = cleanString.substring(0, cleanString.length - 1);
|
||||
}
|
||||
|
||||
// 按逗号分割各个属性
|
||||
const items = cleanString.split(',');
|
||||
|
||||
for (const item of items) {
|
||||
const trimmedItem = item.trim();
|
||||
if (!trimmedItem) continue;
|
||||
|
||||
// 解析 "key: value" 格式
|
||||
const colonIndex = trimmedItem.indexOf(':');
|
||||
if (colonIndex === -1) continue;
|
||||
|
||||
const key = trimmedItem.substring(0, colonIndex).trim();
|
||||
const value = trimmedItem.substring(colonIndex + 1).trim();
|
||||
|
||||
// 映射字段名
|
||||
switch (key) {
|
||||
case 'startLocation':
|
||||
result.startLocation = value;
|
||||
break;
|
||||
case 'endLocation':
|
||||
result.endLocation = value;
|
||||
break;
|
||||
case 'waypoint':
|
||||
result.waypoint = value;
|
||||
break;
|
||||
case 'robotAction':
|
||||
result.robotAction = value as RobotAction;
|
||||
break;
|
||||
case 'payload':
|
||||
result.payload = value;
|
||||
break;
|
||||
case 'locationBay':
|
||||
result.locationBay = value;
|
||||
break;
|
||||
}
|
||||
}
|
||||
} catch (error) {
|
||||
// 解析失败时不报错,返回空对象
|
||||
console.log('解析二维码信息失败:', error);
|
||||
}
|
||||
|
||||
return result;
|
||||
};
|
||||
|
||||
// 处理二维码扫描输入,更新表单字段
|
||||
const handleQrCodeScan = (scanData: string) => {
|
||||
if (isUpdatingFromFormRef.current || !scanData.trim()) return;
|
||||
|
||||
isUpdatingFromQrRef.current = true;
|
||||
const parsedParams = parseQrCodeInfo(scanData);
|
||||
|
||||
const updatedTask = {
|
||||
...task,
|
||||
parameters: {
|
||||
...task.parameters,
|
||||
...parsedParams,
|
||||
},
|
||||
};
|
||||
|
||||
onTaskChange(updatedTask);
|
||||
|
||||
// 清空扫描输入框
|
||||
if (qrScanInputRef.current) {
|
||||
qrScanInputRef.current.clear();
|
||||
}
|
||||
|
||||
// 重置标志
|
||||
setTimeout(() => {
|
||||
isUpdatingFromQrRef.current = false;
|
||||
}, 0);
|
||||
};
|
||||
|
||||
// 重新扫描按钮点击处理
|
||||
const handleRescan = () => {
|
||||
if (qrScanInputRef.current) {
|
||||
qrScanInputRef.current.focus();
|
||||
}
|
||||
};
|
||||
|
||||
// 处理其他输入框获得焦点
|
||||
const handleOtherInputFocus = () => {
|
||||
setIsQrInputFocused(false);
|
||||
};
|
||||
|
||||
// 处理二维码输入框焦点事件
|
||||
const handleQrInputFocus = () => {
|
||||
setIsQrInputFocused(true);
|
||||
};
|
||||
|
||||
const handleQrInputBlur = () => {
|
||||
setIsQrInputFocused(false);
|
||||
};
|
||||
|
||||
const handleParamChange = (field: string, value: string | RobotAction) => {
|
||||
if (isUpdatingFromQrRef.current) return;
|
||||
|
||||
isUpdatingFromFormRef.current = true;
|
||||
|
||||
const updatedTask = {
|
||||
...task,
|
||||
parameters: {
|
||||
...task.parameters,
|
||||
[field]: value,
|
||||
},
|
||||
};
|
||||
onTaskChange(updatedTask);
|
||||
|
||||
// 重置标志
|
||||
setTimeout(() => {
|
||||
isUpdatingFromFormRef.current = false;
|
||||
}, 0);
|
||||
};
|
||||
|
||||
const openBottomSheet = (
|
||||
field: string,
|
||||
items: { label: string; value: string }[],
|
||||
) => {
|
||||
setCurrentField(field);
|
||||
setCurrentItems(items);
|
||||
setIsVisible(true);
|
||||
};
|
||||
|
||||
const renderDropdown = (
|
||||
name: string,
|
||||
label: string,
|
||||
value: string,
|
||||
items: { label: string; value: string }[],
|
||||
) => (
|
||||
<TouchableOpacity onPress={() => openBottomSheet(name, items)}>
|
||||
<Input
|
||||
label={label}
|
||||
value={value}
|
||||
editable={false}
|
||||
rightIcon={{ name: 'arrow-drop-down' }}
|
||||
/>
|
||||
</TouchableOpacity>
|
||||
);
|
||||
|
||||
return (
|
||||
<ScrollView contentContainerStyle={styles.container}>
|
||||
{/* 隐藏的二维码扫描输入框 */}
|
||||
<Input
|
||||
ref={qrScanInputRef}
|
||||
style={styles.hiddenInput}
|
||||
containerStyle={styles.hiddenContainer}
|
||||
inputContainerStyle={styles.hiddenContainer}
|
||||
onChangeText={handleQrCodeScan}
|
||||
onFocus={handleQrInputFocus}
|
||||
onBlur={handleQrInputBlur}
|
||||
autoFocus={false}
|
||||
showSoftInputOnFocus={false}
|
||||
/>
|
||||
|
||||
{/* 扫描二维码按钮 */}
|
||||
<View style={styles.scanButtonContainer}>
|
||||
<Button
|
||||
title={
|
||||
isQrInputFocused ? '正在等待二维码扫描...' : '扫描二维码填充表单'
|
||||
}
|
||||
onPress={handleRescan}
|
||||
icon={{
|
||||
name: isQrInputFocused ? 'hourglass-empty' : 'qr-code-scanner',
|
||||
type: 'material',
|
||||
}}
|
||||
buttonStyle={[
|
||||
styles.scanButton,
|
||||
isQrInputFocused && styles.waitingButton,
|
||||
]}
|
||||
disabled={isQrInputFocused}
|
||||
/>
|
||||
</View>
|
||||
|
||||
<Input
|
||||
label="任务名称"
|
||||
value={task.name}
|
||||
onChangeText={text => onTaskChange({ ...task, name: text })}
|
||||
onFocus={handleOtherInputFocus}
|
||||
/>
|
||||
|
||||
{renderDropdown(
|
||||
'startLocation',
|
||||
'起点',
|
||||
task.parameters.startLocation,
|
||||
locations,
|
||||
)}
|
||||
{renderDropdown(
|
||||
'endLocation',
|
||||
'终点',
|
||||
task.parameters.endLocation,
|
||||
locations,
|
||||
)}
|
||||
|
||||
<Input
|
||||
label="途经点 (可选)"
|
||||
value={task.parameters.waypoint || ''}
|
||||
onChangeText={text => handleParamChange('waypoint', text)}
|
||||
onFocus={handleOtherInputFocus}
|
||||
/>
|
||||
|
||||
{renderDropdown(
|
||||
'robotAction',
|
||||
'机器人动作',
|
||||
task.parameters.robotAction,
|
||||
robotActions,
|
||||
)}
|
||||
{renderDropdown('payload', '载荷', task.parameters.payload, payloads)}
|
||||
|
||||
{/* 库位字段 */}
|
||||
<Input
|
||||
label="库位"
|
||||
value={task.parameters.locationBay || ''}
|
||||
onChangeText={text => handleParamChange('locationBay', text)}
|
||||
placeholder="请输入库位"
|
||||
onFocus={handleOtherInputFocus}
|
||||
/>
|
||||
|
||||
<BottomSheet
|
||||
isVisible={isVisible}
|
||||
onBackdropPress={() => setIsVisible(false)}
|
||||
>
|
||||
<ScrollView>
|
||||
{currentItems.map((item, index) => (
|
||||
<ListItem
|
||||
key={index}
|
||||
onPress={() => {
|
||||
handleParamChange(currentField, item.value);
|
||||
setIsVisible(false);
|
||||
}}
|
||||
>
|
||||
<ListItem.Content>
|
||||
<ListItem.Title>{item.label}</ListItem.Title>
|
||||
</ListItem.Content>
|
||||
</ListItem>
|
||||
))}
|
||||
</ScrollView>
|
||||
</BottomSheet>
|
||||
</ScrollView>
|
||||
);
|
||||
};
|
||||
|
||||
const styles = StyleSheet.create({
|
||||
container: {
|
||||
padding: 16,
|
||||
},
|
||||
hiddenInput: {
|
||||
height: 0,
|
||||
opacity: 0,
|
||||
},
|
||||
hiddenContainer: {
|
||||
height: 0,
|
||||
margin: 0,
|
||||
padding: 0,
|
||||
},
|
||||
scanButtonContainer: {
|
||||
marginBottom: 16,
|
||||
},
|
||||
scanButton: {
|
||||
backgroundColor: '#2196F3',
|
||||
borderRadius: 8,
|
||||
},
|
||||
waitingButton: {
|
||||
backgroundColor: '#9E9E9E', // 浅灰色,表示等待状态
|
||||
},
|
||||
});
|
||||
|
||||
export default TaskForm;
|
184
src/context/TasksContext.tsx
Normal file
@ -0,0 +1,184 @@
|
||||
import React, {
|
||||
createContext,
|
||||
useState,
|
||||
useContext,
|
||||
ReactNode,
|
||||
useEffect,
|
||||
} from 'react';
|
||||
import { Task } from '../types/task';
|
||||
import {
|
||||
AppConfig,
|
||||
LocationOption,
|
||||
PayloadOption,
|
||||
RobotActionOption,
|
||||
} from '../types/config';
|
||||
import {
|
||||
getConfig,
|
||||
getSettings,
|
||||
executeTask,
|
||||
clearCachedConfig,
|
||||
} from '../services/configService';
|
||||
|
||||
interface TasksContextData {
|
||||
tasks: Task[];
|
||||
locations: LocationOption[];
|
||||
locationsBays: LocationOption[];
|
||||
payloads: PayloadOption[];
|
||||
robotActions: RobotActionOption[];
|
||||
getTaskById: (id: string) => Task | undefined;
|
||||
updateTask: (updatedTask: Task) => void;
|
||||
runTask: (id: string) => void;
|
||||
refreshConfig: () => Promise<void>;
|
||||
isConfigLoaded: boolean;
|
||||
}
|
||||
|
||||
const TasksContext = createContext<TasksContextData>({} as TasksContextData);
|
||||
|
||||
export const TasksProvider: React.FC<{ children: ReactNode }> = ({
|
||||
children,
|
||||
}) => {
|
||||
const [tasks, setTasks] = useState<Task[]>([]);
|
||||
const [locations, setLocations] = useState<LocationOption[]>([]);
|
||||
const [locationsBays, setLocationsBays] = useState<LocationOption[]>([]);
|
||||
const [payloads, setPayloads] = useState<PayloadOption[]>([]);
|
||||
const [robotActions, setRobotActions] = useState<RobotActionOption[]>([]);
|
||||
const [isConfigLoaded, setIsConfigLoaded] = useState(false);
|
||||
|
||||
// 组件初始化时加载配置
|
||||
useEffect(() => {
|
||||
const loadConfig = async () => {
|
||||
try {
|
||||
// 清除缓存以确保加载最新的配置
|
||||
await clearCachedConfig();
|
||||
|
||||
const config = await getConfig();
|
||||
if (config) {
|
||||
applyConfig(config);
|
||||
setIsConfigLoaded(true);
|
||||
console.log('成功加载配置,任务数量:', config.tasks?.length || 0);
|
||||
} else {
|
||||
console.log('没有找到配置文件,使用空数据');
|
||||
// 使用空数据而不是mock数据
|
||||
applyConfig({
|
||||
version: '0.0.0',
|
||||
locations: [],
|
||||
locationsBays: [],
|
||||
payloads: [],
|
||||
robotActions: [],
|
||||
tasks: [],
|
||||
});
|
||||
setIsConfigLoaded(false);
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('加载配置失败:', error);
|
||||
// 发生错误时也使用空数据
|
||||
applyConfig({
|
||||
version: '0.0.0',
|
||||
locations: [],
|
||||
locationsBays: [],
|
||||
payloads: [],
|
||||
robotActions: [],
|
||||
tasks: [],
|
||||
});
|
||||
setIsConfigLoaded(false);
|
||||
}
|
||||
};
|
||||
|
||||
loadConfig();
|
||||
}, []);
|
||||
|
||||
const applyConfig = (config: AppConfig) => {
|
||||
setTasks(config.tasks || []);
|
||||
setLocations(config.locations || []);
|
||||
setLocationsBays(config.locationsBays || []);
|
||||
setPayloads(config.payloads || []);
|
||||
setRobotActions(config.robotActions || []);
|
||||
};
|
||||
|
||||
const refreshConfig = async () => {
|
||||
try {
|
||||
// 刷新时也清除缓存
|
||||
await clearCachedConfig();
|
||||
|
||||
const config = await getConfig();
|
||||
if (config) {
|
||||
applyConfig(config);
|
||||
setIsConfigLoaded(true);
|
||||
console.log('成功刷新配置,任务数量:', config.tasks?.length || 0);
|
||||
} else {
|
||||
console.log('刷新配置时没有找到配置文件');
|
||||
setIsConfigLoaded(false);
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('刷新配置失败:', error);
|
||||
setIsConfigLoaded(false);
|
||||
}
|
||||
};
|
||||
|
||||
const getTaskById = (id: string) => {
|
||||
return tasks.find(task => task.id === id);
|
||||
};
|
||||
|
||||
const updateTask = (updatedTask: Task) => {
|
||||
setTasks(prevTasks =>
|
||||
prevTasks.map(task => (task.id === updatedTask.id ? updatedTask : task)),
|
||||
);
|
||||
};
|
||||
|
||||
const runTask = async (id: string) => {
|
||||
const task = getTaskById(id);
|
||||
if (!task) return;
|
||||
|
||||
// 更新任务状态为运行中
|
||||
setTasks(prevTasks =>
|
||||
prevTasks.map(t => (t.id === id ? { ...t, status: 'RUNNING' } : t)),
|
||||
);
|
||||
|
||||
try {
|
||||
// 获取服务器设置并发送任务执行请求
|
||||
const settings = await getSettings();
|
||||
if (settings.serverUrl) {
|
||||
await executeTask(settings.serverUrl, task.id, {
|
||||
name: task.name,
|
||||
parameters: task.parameters,
|
||||
});
|
||||
}
|
||||
|
||||
// 模拟任务完成(实际项目中应该通过WebSocket或轮询获取任务状态)
|
||||
setTimeout(() => {
|
||||
setTasks(prevTasks =>
|
||||
prevTasks.map(t => (t.id === id ? { ...t, status: 'COMPLETED' } : t)),
|
||||
);
|
||||
}, 5000);
|
||||
} catch (error) {
|
||||
console.error('任务执行失败:', error);
|
||||
// 任务执行失败,更新状态为错误
|
||||
setTasks(prevTasks =>
|
||||
prevTasks.map(t => (t.id === id ? { ...t, status: 'ERROR' } : t)),
|
||||
);
|
||||
}
|
||||
};
|
||||
|
||||
return (
|
||||
<TasksContext.Provider
|
||||
value={{
|
||||
tasks,
|
||||
locations,
|
||||
locationsBays,
|
||||
payloads,
|
||||
robotActions,
|
||||
getTaskById,
|
||||
updateTask,
|
||||
runTask,
|
||||
refreshConfig,
|
||||
isConfigLoaded,
|
||||
}}
|
||||
>
|
||||
{children}
|
||||
</TasksContext.Provider>
|
||||
);
|
||||
};
|
||||
|
||||
export function useTasks() {
|
||||
return useContext(TasksContext);
|
||||
}
|
66
src/navigation/AppNavigator.tsx
Normal file
@ -0,0 +1,66 @@
|
||||
import React from 'react';
|
||||
import { createStackNavigator } from '@react-navigation/stack';
|
||||
import { createBottomTabNavigator } from '@react-navigation/bottom-tabs';
|
||||
import MaterialIcons from 'react-native-vector-icons/MaterialIcons';
|
||||
import TaskListScreen from '../screens/TaskListScreen';
|
||||
import TaskEditScreen from '../screens/TaskEditScreen';
|
||||
import RunScreen from '../screens/RunScreen';
|
||||
import EditScreen from '../screens/EditScreen';
|
||||
import SettingsScreen from '../screens/SettingsScreen';
|
||||
|
||||
const HomeStack = createStackNavigator();
|
||||
const Tab = createBottomTabNavigator();
|
||||
|
||||
function HomeStackNavigator() {
|
||||
return (
|
||||
<HomeStack.Navigator>
|
||||
<HomeStack.Screen
|
||||
name="TaskList"
|
||||
component={TaskListScreen}
|
||||
options={{ title: '任务列表' }}
|
||||
/>
|
||||
<HomeStack.Screen
|
||||
name="TaskEdit"
|
||||
component={TaskEditScreen}
|
||||
options={{ title: '编辑任务' }}
|
||||
/>
|
||||
</HomeStack.Navigator>
|
||||
);
|
||||
}
|
||||
|
||||
export default function AppNavigator() {
|
||||
return (
|
||||
<Tab.Navigator
|
||||
screenOptions={({ route }) => ({
|
||||
tabBarIcon: ({ color, size }) => {
|
||||
let iconName: string;
|
||||
|
||||
if (route.name === '主页') {
|
||||
iconName = 'home';
|
||||
} else if (route.name === '运行') {
|
||||
iconName = 'play-arrow';
|
||||
} else if (route.name === '编辑') {
|
||||
iconName = 'edit';
|
||||
} else if (route.name === '设置') {
|
||||
iconName = 'settings';
|
||||
} else {
|
||||
iconName = 'help';
|
||||
}
|
||||
|
||||
return <MaterialIcons name={iconName} size={size} color={color} />;
|
||||
},
|
||||
tabBarActiveTintColor: '#2196F3',
|
||||
tabBarInactiveTintColor: 'gray',
|
||||
})}
|
||||
>
|
||||
<Tab.Screen
|
||||
name="主页"
|
||||
component={HomeStackNavigator}
|
||||
options={{ headerShown: false }}
|
||||
/>
|
||||
<Tab.Screen name="运行" component={RunScreen} />
|
||||
<Tab.Screen name="编辑" component={EditScreen} />
|
||||
<Tab.Screen name="设置" component={SettingsScreen} />
|
||||
</Tab.Navigator>
|
||||
);
|
||||
}
|
18
src/screens/EditScreen.tsx
Normal file
@ -0,0 +1,18 @@
|
||||
import React from 'react';
|
||||
import { StyleSheet, Text, View } from 'react-native';
|
||||
|
||||
export default function EditScreen() {
|
||||
return (
|
||||
<View style={styles.screenContainer}>
|
||||
<Text>编辑!</Text>
|
||||
</View>
|
||||
);
|
||||
}
|
||||
|
||||
const styles = StyleSheet.create({
|
||||
screenContainer: {
|
||||
flex: 1,
|
||||
justifyContent: 'center',
|
||||
alignItems: 'center',
|
||||
},
|
||||
});
|
32
src/screens/HomeScreen.tsx
Normal file
@ -0,0 +1,32 @@
|
||||
import React from 'react';
|
||||
import { StyleSheet, View, Button, SafeAreaView } from 'react-native';
|
||||
|
||||
export default function HomeScreen() {
|
||||
return (
|
||||
<SafeAreaView style={styles.container}>
|
||||
<View style={styles.grid}>
|
||||
{[...Array(8)].map((_, i) => (
|
||||
<View key={i} style={styles.buttonContainer}>
|
||||
<Button title={`测试 ${i + 1}`} onPress={() => {}} />
|
||||
</View>
|
||||
))}
|
||||
</View>
|
||||
</SafeAreaView>
|
||||
);
|
||||
}
|
||||
|
||||
const styles = StyleSheet.create({
|
||||
container: {
|
||||
flex: 1,
|
||||
marginTop: 20,
|
||||
},
|
||||
grid: {
|
||||
flexDirection: 'row',
|
||||
flexWrap: 'wrap',
|
||||
justifyContent: 'center',
|
||||
},
|
||||
buttonContainer: {
|
||||
width: '40%',
|
||||
margin: 10,
|
||||
},
|
||||
});
|
18
src/screens/RunScreen.tsx
Normal file
@ -0,0 +1,18 @@
|
||||
import React from 'react';
|
||||
import { StyleSheet, Text, View } from 'react-native';
|
||||
|
||||
export default function RunScreen() {
|
||||
return (
|
||||
<View style={styles.screenContainer}>
|
||||
<Text>运行!</Text>
|
||||
</View>
|
||||
);
|
||||
}
|
||||
|
||||
const styles = StyleSheet.create({
|
||||
screenContainer: {
|
||||
flex: 1,
|
||||
justifyContent: 'center',
|
||||
alignItems: 'center',
|
||||
},
|
||||
});
|
467
src/screens/SettingsScreen.tsx
Normal file
@ -0,0 +1,467 @@
|
||||
import React, { useState, useEffect } from 'react';
|
||||
import {
|
||||
StyleSheet,
|
||||
Text,
|
||||
View,
|
||||
TextInput,
|
||||
TouchableOpacity,
|
||||
Alert,
|
||||
ActivityIndicator,
|
||||
ScrollView,
|
||||
} from 'react-native';
|
||||
import { AppSettings } from '../types/config';
|
||||
import {
|
||||
getSettings,
|
||||
saveSettings,
|
||||
downloadConfig,
|
||||
getConfig,
|
||||
clearCachedConfig,
|
||||
} from '../services/configService';
|
||||
import { useTasks } from '../context/TasksContext';
|
||||
|
||||
export default function SettingsScreen() {
|
||||
const { refreshConfig } = useTasks();
|
||||
const [settings, setSettings] = useState<AppSettings>({
|
||||
configFileName: '',
|
||||
serverUrl: '',
|
||||
});
|
||||
const [loading, setLoading] = useState(false);
|
||||
const [configStatus, setConfigStatus] = useState<string>('未加载');
|
||||
|
||||
// 组件加载时获取设置
|
||||
useEffect(() => {
|
||||
loadSettings();
|
||||
checkConfigStatus();
|
||||
}, []);
|
||||
|
||||
const loadSettings = async () => {
|
||||
try {
|
||||
const currentSettings = await getSettings();
|
||||
setSettings(currentSettings);
|
||||
} catch (error) {
|
||||
Alert.alert('错误', '加载设置失败');
|
||||
}
|
||||
};
|
||||
|
||||
const checkConfigStatus = async () => {
|
||||
const config = await getConfig();
|
||||
if (config) {
|
||||
setConfigStatus(
|
||||
`已加载 (版本: ${config.version}, 任务数: ${
|
||||
config.tasks?.length || 0
|
||||
})`,
|
||||
);
|
||||
} else {
|
||||
setConfigStatus('未加载');
|
||||
}
|
||||
};
|
||||
|
||||
const handleSaveSettings = async () => {
|
||||
if (!settings.configFileName.trim()) {
|
||||
Alert.alert('错误', '请输入配置文件名');
|
||||
return;
|
||||
}
|
||||
if (!settings.serverUrl.trim()) {
|
||||
Alert.alert('错误', '请输入服务器地址');
|
||||
return;
|
||||
}
|
||||
|
||||
try {
|
||||
await saveSettings(settings);
|
||||
Alert.alert('成功', '设置已保存');
|
||||
} catch (error) {
|
||||
Alert.alert('错误', '保存设置失败');
|
||||
}
|
||||
};
|
||||
|
||||
const handleRefreshLocalConfig = async () => {
|
||||
setLoading(true);
|
||||
try {
|
||||
// 清除缓存并重新加载本地配置
|
||||
await clearCachedConfig();
|
||||
await refreshConfig();
|
||||
await checkConfigStatus();
|
||||
|
||||
Alert.alert('成功', '本地配置已刷新!');
|
||||
} catch (error) {
|
||||
console.error('刷新本地配置失败:', error);
|
||||
Alert.alert('错误', '刷新本地配置失败');
|
||||
} finally {
|
||||
setLoading(false);
|
||||
}
|
||||
};
|
||||
|
||||
const handleDownloadConfig = async () => {
|
||||
if (!settings.configFileName.trim()) {
|
||||
Alert.alert('错误', '请先输入配置文件名');
|
||||
return;
|
||||
}
|
||||
if (!settings.serverUrl.trim()) {
|
||||
Alert.alert('错误', '请先输入服务器地址');
|
||||
return;
|
||||
}
|
||||
|
||||
setLoading(true);
|
||||
try {
|
||||
// 先保存设置
|
||||
await saveSettings(settings);
|
||||
|
||||
// 下载配置文件
|
||||
const config = await downloadConfig(
|
||||
settings.serverUrl,
|
||||
settings.configFileName,
|
||||
);
|
||||
|
||||
Alert.alert(
|
||||
'成功',
|
||||
`配置文件下载成功!\n版本: ${config.version}\n任务数量: ${
|
||||
config.tasks?.length || 0
|
||||
}`,
|
||||
[
|
||||
{
|
||||
text: '确定',
|
||||
onPress: async () => {
|
||||
await refreshConfig();
|
||||
checkConfigStatus();
|
||||
},
|
||||
},
|
||||
],
|
||||
);
|
||||
} catch (error) {
|
||||
Alert.alert(
|
||||
'下载失败',
|
||||
error instanceof Error ? error.message : '未知错误',
|
||||
);
|
||||
} finally {
|
||||
setLoading(false);
|
||||
}
|
||||
};
|
||||
|
||||
const handleClearCache = async () => {
|
||||
Alert.alert(
|
||||
'确认清除',
|
||||
'确定要清除缓存的配置文件吗?系统将重新加载本地 config.json 文件。',
|
||||
[
|
||||
{ text: '取消', style: 'cancel' },
|
||||
{
|
||||
text: '确定',
|
||||
onPress: async () => {
|
||||
setLoading(true);
|
||||
try {
|
||||
await clearCachedConfig();
|
||||
await refreshConfig();
|
||||
checkConfigStatus();
|
||||
Alert.alert('成功', '缓存已清除,已重新加载配置');
|
||||
} catch (error) {
|
||||
Alert.alert('错误', '清除缓存失败');
|
||||
} finally {
|
||||
setLoading(false);
|
||||
}
|
||||
},
|
||||
},
|
||||
],
|
||||
);
|
||||
};
|
||||
|
||||
const handleTestConnection = async () => {
|
||||
if (!settings.serverUrl.trim()) {
|
||||
Alert.alert('错误', '请先输入服务器地址');
|
||||
return;
|
||||
}
|
||||
|
||||
setLoading(true);
|
||||
try {
|
||||
const controller = new AbortController();
|
||||
const timeoutId = setTimeout(() => controller.abort(), 5000);
|
||||
|
||||
const response = await fetch(`${settings.serverUrl}/health`, {
|
||||
method: 'GET',
|
||||
signal: controller.signal,
|
||||
});
|
||||
|
||||
clearTimeout(timeoutId);
|
||||
|
||||
if (response.ok) {
|
||||
Alert.alert('连接成功', '服务器连接正常');
|
||||
} else {
|
||||
Alert.alert('连接失败', `HTTP ${response.status}`);
|
||||
}
|
||||
} catch (error) {
|
||||
Alert.alert('连接失败', '无法连接到服务器');
|
||||
} finally {
|
||||
setLoading(false);
|
||||
}
|
||||
};
|
||||
|
||||
return (
|
||||
<ScrollView style={styles.container}>
|
||||
<View style={styles.content}>
|
||||
{loading && (
|
||||
<View style={styles.loadingOverlay}>
|
||||
<ActivityIndicator size="large" color="#007AFF" />
|
||||
<Text style={styles.loadingText}>处理中...</Text>
|
||||
</View>
|
||||
)}
|
||||
|
||||
{/* 配置状态 */}
|
||||
<View style={styles.section}>
|
||||
<Text style={styles.sectionTitle}>配置状态</Text>
|
||||
<Text style={styles.statusText}>{configStatus}</Text>
|
||||
</View>
|
||||
|
||||
{/* 配置文件设置 */}
|
||||
<View style={styles.section}>
|
||||
<Text style={styles.sectionTitle}>配置文件</Text>
|
||||
|
||||
<View style={styles.inputGroup}>
|
||||
<Text style={styles.label}>配置文件名:</Text>
|
||||
<TextInput
|
||||
style={styles.textInput}
|
||||
value={settings.configFileName}
|
||||
onChangeText={text =>
|
||||
setSettings(prev => ({ ...prev, configFileName: text }))
|
||||
}
|
||||
placeholder="config"
|
||||
placeholderTextColor="#999"
|
||||
autoCapitalize="none"
|
||||
/>
|
||||
</View>
|
||||
|
||||
<View style={styles.buttonRow}>
|
||||
<TouchableOpacity
|
||||
style={[styles.button, styles.downloadButton]}
|
||||
onPress={handleDownloadConfig}
|
||||
disabled={loading}
|
||||
>
|
||||
<Text style={styles.buttonText}>下载配置文件</Text>
|
||||
</TouchableOpacity>
|
||||
|
||||
<TouchableOpacity
|
||||
style={[styles.button, styles.refreshButton]}
|
||||
onPress={handleRefreshLocalConfig}
|
||||
disabled={loading}
|
||||
>
|
||||
<Text style={styles.buttonText}>刷新本地配置</Text>
|
||||
</TouchableOpacity>
|
||||
</View>
|
||||
|
||||
<View style={styles.buttonRow}>
|
||||
<TouchableOpacity
|
||||
style={[styles.button, styles.clearButton]}
|
||||
onPress={handleClearCache}
|
||||
disabled={loading}
|
||||
>
|
||||
<Text style={styles.buttonText}>清除缓存</Text>
|
||||
</TouchableOpacity>
|
||||
</View>
|
||||
</View>
|
||||
|
||||
{/* 服务器设置 */}
|
||||
<View style={styles.section}>
|
||||
<Text style={styles.sectionTitle}>服务器设置</Text>
|
||||
|
||||
<View style={styles.inputGroup}>
|
||||
<Text style={styles.label}>服务器地址:</Text>
|
||||
<TextInput
|
||||
style={styles.textInput}
|
||||
value={settings.serverUrl}
|
||||
onChangeText={text =>
|
||||
setSettings(prev => ({ ...prev, serverUrl: text }))
|
||||
}
|
||||
placeholder="http://localhost:3000/api"
|
||||
placeholderTextColor="#999"
|
||||
autoCapitalize="none"
|
||||
keyboardType="url"
|
||||
/>
|
||||
</View>
|
||||
|
||||
<View style={styles.buttonRow}>
|
||||
<TouchableOpacity
|
||||
style={[styles.button, styles.testButton]}
|
||||
onPress={handleTestConnection}
|
||||
disabled={loading}
|
||||
>
|
||||
<Text style={styles.buttonText}>测试连接</Text>
|
||||
</TouchableOpacity>
|
||||
|
||||
<TouchableOpacity
|
||||
style={[styles.button, styles.saveButton]}
|
||||
onPress={handleSaveSettings}
|
||||
disabled={loading}
|
||||
>
|
||||
<Text style={styles.buttonText}>保存设置</Text>
|
||||
</TouchableOpacity>
|
||||
</View>
|
||||
</View>
|
||||
|
||||
{/* 说明信息 */}
|
||||
<View style={styles.infoSection}>
|
||||
<Text style={styles.infoTitle}>使用说明</Text>
|
||||
<Text style={styles.infoText}>配置文件加载优先级:</Text>
|
||||
<Text style={styles.infoText}>
|
||||
1. 缓存的服务器配置(从服务器下载的配置)
|
||||
</Text>
|
||||
<Text style={styles.infoText}>2. 本地 config.json 文件</Text>
|
||||
<Text style={styles.infoText}>3. 空数据(如果都没有找到)</Text>
|
||||
<Text style={styles.infoText}>
|
||||
• 点击"下载配置文件"从服务器获取最新配置
|
||||
</Text>
|
||||
<Text style={styles.infoText}>
|
||||
• 点击"清除缓存"强制重新加载本地配置文件
|
||||
</Text>
|
||||
</View>
|
||||
</View>
|
||||
</ScrollView>
|
||||
);
|
||||
}
|
||||
|
||||
const styles = StyleSheet.create({
|
||||
container: {
|
||||
flex: 1,
|
||||
backgroundColor: '#f5f5f5',
|
||||
},
|
||||
content: {
|
||||
padding: 20,
|
||||
},
|
||||
title: {
|
||||
fontSize: 24,
|
||||
fontWeight: 'bold',
|
||||
textAlign: 'center',
|
||||
marginBottom: 30,
|
||||
color: '#333',
|
||||
},
|
||||
section: {
|
||||
backgroundColor: 'white',
|
||||
borderRadius: 10,
|
||||
padding: 20,
|
||||
marginBottom: 20,
|
||||
shadowColor: '#000',
|
||||
shadowOffset: {
|
||||
width: 0,
|
||||
height: 2,
|
||||
},
|
||||
shadowOpacity: 0.1,
|
||||
shadowRadius: 3.84,
|
||||
elevation: 5,
|
||||
},
|
||||
sectionTitle: {
|
||||
fontSize: 18,
|
||||
fontWeight: 'bold',
|
||||
marginBottom: 15,
|
||||
color: '#333',
|
||||
},
|
||||
inputGroup: {
|
||||
marginBottom: 15,
|
||||
},
|
||||
label: {
|
||||
fontSize: 16,
|
||||
marginBottom: 8,
|
||||
color: '#333',
|
||||
fontWeight: '500',
|
||||
},
|
||||
inputRow: {
|
||||
flexDirection: 'row',
|
||||
alignItems: 'center',
|
||||
},
|
||||
textInput: {
|
||||
flex: 1,
|
||||
borderWidth: 1,
|
||||
borderColor: '#ddd',
|
||||
borderRadius: 8,
|
||||
padding: 12,
|
||||
fontSize: 16,
|
||||
backgroundColor: '#fafafa',
|
||||
},
|
||||
extension: {
|
||||
marginLeft: 8,
|
||||
fontSize: 16,
|
||||
color: '#666',
|
||||
fontWeight: '500',
|
||||
},
|
||||
statusRow: {
|
||||
flexDirection: 'row',
|
||||
alignItems: 'center',
|
||||
marginBottom: 15,
|
||||
},
|
||||
statusLabel: {
|
||||
fontSize: 16,
|
||||
color: '#333',
|
||||
fontWeight: '500',
|
||||
},
|
||||
statusText: {
|
||||
fontSize: 16,
|
||||
color: '#007AFF',
|
||||
marginLeft: 8,
|
||||
},
|
||||
button: {
|
||||
borderRadius: 8,
|
||||
padding: 12,
|
||||
alignItems: 'center',
|
||||
minHeight: 48,
|
||||
justifyContent: 'center',
|
||||
},
|
||||
downloadButton: {
|
||||
backgroundColor: '#007AFF',
|
||||
flex: 1,
|
||||
marginRight: 10,
|
||||
},
|
||||
refreshButton: {
|
||||
backgroundColor: '#32D74B',
|
||||
flex: 1,
|
||||
},
|
||||
testButton: {
|
||||
backgroundColor: '#34C759',
|
||||
flex: 1,
|
||||
marginRight: 10,
|
||||
},
|
||||
saveButton: {
|
||||
backgroundColor: '#FF9500',
|
||||
flex: 1,
|
||||
},
|
||||
buttonText: {
|
||||
color: 'white',
|
||||
fontSize: 16,
|
||||
fontWeight: '600',
|
||||
},
|
||||
buttonRow: {
|
||||
flexDirection: 'row',
|
||||
marginTop: 10,
|
||||
},
|
||||
infoSection: {
|
||||
backgroundColor: 'white',
|
||||
borderRadius: 10,
|
||||
padding: 20,
|
||||
marginBottom: 20,
|
||||
},
|
||||
infoTitle: {
|
||||
fontSize: 18,
|
||||
fontWeight: 'bold',
|
||||
marginBottom: 15,
|
||||
color: '#333',
|
||||
},
|
||||
infoText: {
|
||||
fontSize: 14,
|
||||
lineHeight: 20,
|
||||
color: '#666',
|
||||
marginBottom: 5,
|
||||
},
|
||||
loadingOverlay: {
|
||||
position: 'absolute',
|
||||
top: 0,
|
||||
left: 0,
|
||||
right: 0,
|
||||
bottom: 0,
|
||||
backgroundColor: 'rgba(255, 255, 255, 0.8)',
|
||||
justifyContent: 'center',
|
||||
alignItems: 'center',
|
||||
zIndex: 1,
|
||||
},
|
||||
loadingText: {
|
||||
marginTop: 10,
|
||||
fontSize: 16,
|
||||
color: '#007AFF',
|
||||
},
|
||||
clearButton: {
|
||||
backgroundColor: '#FF3B30',
|
||||
},
|
||||
});
|
97
src/screens/TaskEditScreen.tsx
Normal file
@ -0,0 +1,97 @@
|
||||
import React, { useState, useEffect } from 'react';
|
||||
import { View, StyleSheet, Alert } from 'react-native';
|
||||
import { useRoute, useNavigation } from '@react-navigation/native';
|
||||
import { RouteProp } from '@react-navigation/native';
|
||||
import { useTasks } from '../context/TasksContext';
|
||||
import TaskForm from '../components/TaskForm';
|
||||
import BottomActionBar from '../components/BottomActionBar';
|
||||
import { Task } from '../types/task';
|
||||
import { Dialog } from '@rneui/themed';
|
||||
|
||||
type RootStackParamList = {
|
||||
TaskEdit: { task: Task };
|
||||
};
|
||||
|
||||
type TaskEditRouteProp = RouteProp<RootStackParamList, 'TaskEdit'>;
|
||||
|
||||
export default function TaskEditScreen() {
|
||||
const route = useRoute<TaskEditRouteProp>();
|
||||
const navigation = useNavigation();
|
||||
const { task: initialTask } = route.params;
|
||||
|
||||
const { updateTask, runTask } = useTasks();
|
||||
|
||||
const [task, setTask] = useState<Task | null>(initialTask);
|
||||
const [originalTask, setOriginalTask] = useState<Task | null>(initialTask);
|
||||
const [isModified, setIsModified] = useState(false);
|
||||
|
||||
useEffect(() => {
|
||||
if (initialTask) {
|
||||
setTask(initialTask);
|
||||
setOriginalTask(initialTask);
|
||||
}
|
||||
}, [initialTask]);
|
||||
|
||||
const handleTaskChange = (updatedTask: Task) => {
|
||||
setTask(updatedTask);
|
||||
if (!isModified) {
|
||||
setIsModified(true);
|
||||
}
|
||||
};
|
||||
|
||||
const handleSave = () => {
|
||||
if (task) {
|
||||
updateTask(task);
|
||||
setOriginalTask(task);
|
||||
setIsModified(false);
|
||||
Alert.alert('已保存', `任务 "${task.name}" 已被保存。`);
|
||||
}
|
||||
};
|
||||
|
||||
const handleRun = () => {
|
||||
if (task) {
|
||||
runTask(task.id);
|
||||
navigation.goBack();
|
||||
}
|
||||
};
|
||||
|
||||
const handleUndo = () => {
|
||||
setTask(originalTask);
|
||||
setIsModified(false);
|
||||
};
|
||||
|
||||
const handleRestore = () => {
|
||||
setTask(originalTask);
|
||||
setIsModified(false);
|
||||
};
|
||||
|
||||
if (!task) {
|
||||
return (
|
||||
<Dialog isVisible={true}>
|
||||
<Dialog.Loading />
|
||||
</Dialog>
|
||||
);
|
||||
}
|
||||
|
||||
return (
|
||||
<View style={styles.container}>
|
||||
<TaskForm task={task} onTaskChange={handleTaskChange} />
|
||||
<BottomActionBar
|
||||
onRun={handleRun}
|
||||
onSave={handleSave}
|
||||
onUndo={handleUndo}
|
||||
onRestore={handleRestore}
|
||||
onBack={() => navigation.goBack()}
|
||||
isSaveDisabled={!isModified}
|
||||
/>
|
||||
</View>
|
||||
);
|
||||
}
|
||||
|
||||
const styles = StyleSheet.create({
|
||||
container: {
|
||||
flex: 1,
|
||||
backgroundColor: '#fff',
|
||||
paddingBottom: 60,
|
||||
},
|
||||
});
|
56
src/screens/TaskListScreen.tsx
Normal file
@ -0,0 +1,56 @@
|
||||
import React from 'react';
|
||||
import { View, ScrollView, StyleSheet } from 'react-native';
|
||||
import { useNavigation } from '@react-navigation/native';
|
||||
import { StackNavigationProp } from '@react-navigation/stack';
|
||||
import { useTasks } from '../context/TasksContext';
|
||||
import TaskCard from '../components/TaskCard';
|
||||
import { Task } from '../types/task';
|
||||
|
||||
type RootStackParamList = {
|
||||
TaskList: undefined;
|
||||
TaskEdit: { task: Task };
|
||||
};
|
||||
|
||||
type TaskListNavigationProp = StackNavigationProp<
|
||||
RootStackParamList,
|
||||
'TaskList'
|
||||
>;
|
||||
|
||||
export default function TaskListScreen() {
|
||||
const { tasks } = useTasks();
|
||||
const navigation = useNavigation<TaskListNavigationProp>();
|
||||
|
||||
const handlePressTask = (task: Task) => {
|
||||
navigation.navigate('TaskEdit', { task });
|
||||
};
|
||||
|
||||
return (
|
||||
<View style={styles.container}>
|
||||
<ScrollView contentContainerStyle={styles.scrollContainer}>
|
||||
<View style={styles.tasksContainer}>
|
||||
{tasks.map(task => (
|
||||
<TaskCard
|
||||
key={task.id}
|
||||
task={task}
|
||||
onPress={() => handlePressTask(task)}
|
||||
/>
|
||||
))}
|
||||
</View>
|
||||
</ScrollView>
|
||||
</View>
|
||||
);
|
||||
}
|
||||
|
||||
const styles = StyleSheet.create({
|
||||
container: {
|
||||
flex: 1,
|
||||
},
|
||||
scrollContainer: {
|
||||
padding: 16,
|
||||
},
|
||||
tasksContainer: {
|
||||
flexDirection: 'row',
|
||||
flexWrap: 'wrap',
|
||||
justifyContent: 'space-between',
|
||||
},
|
||||
});
|
175
src/services/configService.ts
Normal file
@ -0,0 +1,175 @@
|
||||
import AsyncStorage from '@react-native-async-storage/async-storage';
|
||||
import { AppConfig, AppSettings } from '../types/config';
|
||||
|
||||
const SETTINGS_KEY = 'app_settings';
|
||||
const CONFIG_CACHE_KEY = 'cached_config';
|
||||
|
||||
// 默认设置
|
||||
const DEFAULT_SETTINGS: AppSettings = {
|
||||
configFileName: 'config',
|
||||
serverUrl: 'http://localhost:3000/api',
|
||||
};
|
||||
|
||||
// 获取设置
|
||||
export const getSettings = async (): Promise<AppSettings> => {
|
||||
try {
|
||||
const stored = await AsyncStorage.getItem(SETTINGS_KEY);
|
||||
if (stored) {
|
||||
return { ...DEFAULT_SETTINGS, ...JSON.parse(stored) };
|
||||
}
|
||||
return DEFAULT_SETTINGS;
|
||||
} catch (error) {
|
||||
console.error('获取设置失败:', error);
|
||||
return DEFAULT_SETTINGS;
|
||||
}
|
||||
};
|
||||
|
||||
// 保存设置
|
||||
export const saveSettings = async (settings: AppSettings): Promise<void> => {
|
||||
try {
|
||||
await AsyncStorage.setItem(SETTINGS_KEY, JSON.stringify(settings));
|
||||
} catch (error) {
|
||||
console.error('保存设置失败:', error);
|
||||
throw error;
|
||||
}
|
||||
};
|
||||
|
||||
// 加载本地配置文件
|
||||
export const loadLocalConfig = async (): Promise<AppConfig | null> => {
|
||||
try {
|
||||
// 尝试加载项目根目录下的 config.json 文件
|
||||
const localConfig = require('../../config.json');
|
||||
console.log('成功加载本地配置文件:', localConfig);
|
||||
|
||||
return localConfig as AppConfig;
|
||||
} catch (error) {
|
||||
console.log('本地配置文件不存在或加载失败:', error);
|
||||
return null;
|
||||
}
|
||||
};
|
||||
|
||||
// 从服务器下载配置文件
|
||||
export const downloadConfig = async (
|
||||
serverUrl: string,
|
||||
configFileName: string,
|
||||
): Promise<AppConfig> => {
|
||||
try {
|
||||
const url = `${serverUrl.replace(/\/$/, '')}/${configFileName}.json`;
|
||||
console.log('下载配置文件:', url);
|
||||
|
||||
const response = await fetch(url, {
|
||||
method: 'GET',
|
||||
headers: {
|
||||
'Content-Type': 'application/json',
|
||||
},
|
||||
});
|
||||
|
||||
if (!response.ok) {
|
||||
throw new Error(`HTTP ${response.status}: ${response.statusText}`);
|
||||
}
|
||||
|
||||
const config: AppConfig = await response.json();
|
||||
|
||||
// 缓存服务器配置文件(标记为服务器来源)
|
||||
const configWithSource = { ...config, _source: 'server' };
|
||||
await AsyncStorage.setItem(
|
||||
CONFIG_CACHE_KEY,
|
||||
JSON.stringify(configWithSource),
|
||||
);
|
||||
|
||||
return config;
|
||||
} catch (error) {
|
||||
console.error('下载配置文件失败:', error);
|
||||
throw error;
|
||||
}
|
||||
};
|
||||
|
||||
// 获取缓存的配置文件
|
||||
export const getCachedConfig = async (): Promise<AppConfig | null> => {
|
||||
try {
|
||||
const cached = await AsyncStorage.getItem(CONFIG_CACHE_KEY);
|
||||
if (cached) {
|
||||
const config = JSON.parse(cached);
|
||||
// 移除内部标记字段
|
||||
delete config._source;
|
||||
return config;
|
||||
}
|
||||
return null;
|
||||
} catch (error) {
|
||||
console.error('获取缓存配置失败:', error);
|
||||
return null;
|
||||
}
|
||||
};
|
||||
|
||||
// 获取配置文件(优先级:缓存的服务器配置 > 本地config.json > 空数据)
|
||||
export const getConfig = async (): Promise<AppConfig | null> => {
|
||||
try {
|
||||
// 1. 先尝试获取缓存的配置(通常是从服务器下载的)
|
||||
const cachedConfig = await getCachedConfig();
|
||||
if (cachedConfig) {
|
||||
console.log('使用缓存的配置文件');
|
||||
return cachedConfig;
|
||||
}
|
||||
|
||||
// 2. 如果没有缓存,尝试加载本地配置文件
|
||||
const localConfig = await loadLocalConfig();
|
||||
if (localConfig) {
|
||||
console.log('使用本地 config.json 文件');
|
||||
// 将本地配置也缓存起来,但标记为本地来源
|
||||
const configWithSource = { ...localConfig, _source: 'local' };
|
||||
await AsyncStorage.setItem(
|
||||
CONFIG_CACHE_KEY,
|
||||
JSON.stringify(configWithSource),
|
||||
);
|
||||
return localConfig;
|
||||
}
|
||||
|
||||
console.log('没有找到任何配置文件');
|
||||
return null;
|
||||
} catch (error) {
|
||||
console.error('获取配置文件失败:', error);
|
||||
return null;
|
||||
}
|
||||
};
|
||||
|
||||
// 清除缓存配置(用于强制重新加载)
|
||||
export const clearCachedConfig = async (): Promise<void> => {
|
||||
try {
|
||||
await AsyncStorage.removeItem(CONFIG_CACHE_KEY);
|
||||
console.log('已清除缓存配置');
|
||||
} catch (error) {
|
||||
console.error('清除缓存配置失败:', error);
|
||||
}
|
||||
};
|
||||
|
||||
// 向服务器发送任务执行请求
|
||||
export const executeTask = async (
|
||||
serverUrl: string,
|
||||
taskId: string,
|
||||
taskData: any,
|
||||
): Promise<any> => {
|
||||
try {
|
||||
const url = `${serverUrl.replace(/\/$/, '')}/execute-task`;
|
||||
console.log('执行任务请求:', url, taskData);
|
||||
|
||||
const response = await fetch(url, {
|
||||
method: 'POST',
|
||||
headers: {
|
||||
'Content-Type': 'application/json',
|
||||
},
|
||||
body: JSON.stringify({
|
||||
taskId,
|
||||
...taskData,
|
||||
}),
|
||||
});
|
||||
|
||||
if (!response.ok) {
|
||||
throw new Error(`HTTP ${response.status}: ${response.statusText}`);
|
||||
}
|
||||
|
||||
return await response.json();
|
||||
} catch (error) {
|
||||
console.error('执行任务失败:', error);
|
||||
throw error;
|
||||
}
|
||||
};
|
36
src/types/config.ts
Normal file
@ -0,0 +1,36 @@
|
||||
import { Task, RobotAction } from './task';
|
||||
|
||||
// 配置文件中的位置选项
|
||||
export interface LocationOption {
|
||||
label: string;
|
||||
value: string;
|
||||
}
|
||||
|
||||
// 配置文件中的载荷选项
|
||||
export interface PayloadOption {
|
||||
label: string;
|
||||
value: string;
|
||||
}
|
||||
|
||||
// 配置文件中的机器人动作选项
|
||||
export interface RobotActionOption {
|
||||
label: string;
|
||||
value: RobotAction;
|
||||
}
|
||||
|
||||
// 完整的配置文件结构
|
||||
export interface AppConfig {
|
||||
version: string;
|
||||
locations: LocationOption[];
|
||||
locationsBays: LocationOption[];
|
||||
payloads: PayloadOption[];
|
||||
robotActions: RobotActionOption[];
|
||||
tasks: Task[];
|
||||
serverUrl?: string; // 服务器地址
|
||||
}
|
||||
|
||||
// 设置存储接口
|
||||
export interface AppSettings {
|
||||
configFileName: string;
|
||||
serverUrl: string;
|
||||
}
|
49
src/types/task.ts
Normal file
@ -0,0 +1,49 @@
|
||||
// 机器人的具体动作,可以定义为枚举或联合类型
|
||||
export type RobotAction =
|
||||
| 'PICKUP'
|
||||
| 'DROPOFF'
|
||||
| 'TRANSPORT'
|
||||
| 'WAIT'
|
||||
| 'CHARGE'
|
||||
| 'CLEAN';
|
||||
|
||||
// 任务的状态
|
||||
export type TaskStatus = 'IDLE' | 'RUNNING' | 'COMPLETED' | 'ERROR';
|
||||
|
||||
// 参数选项接口
|
||||
export interface ParameterOption {
|
||||
label: string;
|
||||
value: string;
|
||||
}
|
||||
|
||||
// 动态参数配置接口
|
||||
export interface DynamicParameter {
|
||||
label: string;
|
||||
type: 'Simple' | 'Select' | 'MultiSelect' | 'Text' | 'Number';
|
||||
value: string | string[] | number;
|
||||
required: boolean;
|
||||
options?: ParameterOption[];
|
||||
placeholder?: string;
|
||||
min?: number;
|
||||
max?: number;
|
||||
}
|
||||
|
||||
// 任务参数 - 支持动态配置
|
||||
export interface TaskParameters {
|
||||
startLocation: DynamicParameter; // 起点
|
||||
endLocation: DynamicParameter; // 终点
|
||||
waypoint?: DynamicParameter; // 途经点 (可选)
|
||||
robotAction: DynamicParameter; // 机器人动作
|
||||
payload: DynamicParameter; // 载荷,比如 '空料架' 或具体的物料ID
|
||||
locationBay?: DynamicParameter; // 库位 (可选)
|
||||
[key: string]: DynamicParameter | undefined; // 支持扩展参数
|
||||
}
|
||||
|
||||
// 核心任务对象
|
||||
export interface Task {
|
||||
id: string; // 唯一ID,例如使用 uuid
|
||||
name: string; // 任务名称, e.g., "炉前缓存区到热处理上料交接区运输"
|
||||
status: TaskStatus; // 任务当前状态
|
||||
parameters: TaskParameters; // 任务的具体执行参数
|
||||
createdAt: string; // 创建时间
|
||||
}
|
3
tsconfig.json
Normal file
@ -0,0 +1,3 @@
|
||||
{
|
||||
"extends": "@react-native/typescript-config"
|
||||
}
|