111
This commit is contained in:
parent
c2e52df560
commit
a9055522cf
3
package-lock.json
generated
3
package-lock.json
generated
|
@ -13,7 +13,7 @@
|
|||
"@react-native-camera-roll/camera-roll": "^7.9.0",
|
||||
"@react-native-community/geolocation": "^3.4.0",
|
||||
"@react-native-community/slider": "^4.5.5",
|
||||
"@react-native-picker/picker": "^2.9.0",
|
||||
"@react-native-picker/picker": "^2.10.2",
|
||||
"@react-navigation/bottom-tabs": "^6.4.0",
|
||||
"@react-navigation/native": "^6.1.18",
|
||||
"@react-navigation/stack": "^6.3.8",
|
||||
|
@ -4829,7 +4829,6 @@
|
|||
"version": "2.10.2",
|
||||
"resolved": "https://registry.npmmirror.com/@react-native-picker/picker/-/picker-2.10.2.tgz",
|
||||
"integrity": "sha512-kr3OvCRwTYjR/OKlb52k4xmQVU7dPRIALqpyiihexdJxEgvc1smnepgqCeM9oXmNSG4YaV5/RSxFlLC5Z/T/Eg==",
|
||||
"license": "MIT",
|
||||
"workspaces": [
|
||||
"example"
|
||||
],
|
||||
|
|
|
@ -15,7 +15,7 @@
|
|||
"@react-native-camera-roll/camera-roll": "^7.9.0",
|
||||
"@react-native-community/geolocation": "^3.4.0",
|
||||
"@react-native-community/slider": "^4.5.5",
|
||||
"@react-native-picker/picker": "^2.9.0",
|
||||
"@react-native-picker/picker": "^2.10.2",
|
||||
"@react-navigation/bottom-tabs": "^6.4.0",
|
||||
"@react-navigation/native": "^6.1.18",
|
||||
"@react-navigation/stack": "^6.3.8",
|
||||
|
|
|
@ -17,6 +17,7 @@ import { HomeStackParamList } from './types';
|
|||
import TestBule from '../views/device/test_bule';
|
||||
import ShareDetailScreen from '../views/device/KeyDetail';
|
||||
import ExpiredKeysScreen from '../views/device/ExpiredKeysScreen';
|
||||
import deviceDetailSet from '../views/device/deviceDetailSet';
|
||||
const Stack = createStackNavigator<HomeStackParamList>();
|
||||
|
||||
const createScreenOptions = (title: string): StackNavigationOptions => {
|
||||
|
@ -52,6 +53,13 @@ export default function HomeStackNavigator({ navigation, route }: Props) {
|
|||
headerShown: false,
|
||||
}}
|
||||
/>
|
||||
<Stack.Screen
|
||||
name="deviceDetailSet"
|
||||
component={deviceDetailSet}
|
||||
options={{
|
||||
headerShown: false,
|
||||
}}
|
||||
/>
|
||||
<Stack.Screen
|
||||
name="ShareDetailScreen"
|
||||
component={ShareDetailScreen}
|
||||
|
|
|
@ -30,6 +30,7 @@ export type HomeStackParamList = {
|
|||
TestBule: undefined;
|
||||
ShareDetailScreen: undefined;
|
||||
ExpiredKeysScreen: undefined;
|
||||
deviceDetailSet: undefined;
|
||||
};
|
||||
|
||||
// 导航属性类型
|
||||
|
|
|
@ -113,5 +113,9 @@ export const apiService = {
|
|||
getExpiredKeys: () => api.get('/appVerify/getExpiredKeyListByOwnerId'),
|
||||
getKeyInfo: (keyId: string) => api.get('/appVerify/key/'+keyId),
|
||||
bindKey: (keyId: string) => api.post('/appVerify/claimKey?keyId='+keyId),
|
||||
updateDevice: ( data: any) => api.put('/appVerify/device/edit', data),
|
||||
untieDevice: (deviceId: string) => api.post('/appVerify/untie/'+deviceId),
|
||||
getQiniuToken: () => api.get('/common/qiniu/uploadInfo'),
|
||||
getModelList: () => api.get('/appVerify/modelList'),
|
||||
// updateKeyExpiration: (keyId: string, expirationTime: string) => api.put('/appVerify/updateKeyExpiration', { keyId, expirationTime }),
|
||||
};
|
183
src/utils/uploadImg.ts
Normal file
183
src/utils/uploadImg.ts
Normal file
|
@ -0,0 +1,183 @@
|
|||
import { Platform } from 'react-native';
|
||||
import { request, PERMISSIONS, RESULTS } from 'react-native-permissions';
|
||||
import { launchImageLibrary } from 'react-native-image-picker';
|
||||
import { apiService } from './api';
|
||||
import Toast from 'react-native-toast-message';
|
||||
|
||||
interface UploadResponse {
|
||||
success: boolean;
|
||||
imageUrl?: string;
|
||||
error?: string;
|
||||
}
|
||||
|
||||
interface QiniuUploadInfo {
|
||||
token: string;
|
||||
domain: string;
|
||||
}
|
||||
|
||||
class UploadImageService {
|
||||
private static instance: UploadImageService;
|
||||
private uploadInfo: QiniuUploadInfo | null = null;
|
||||
|
||||
private constructor() {}
|
||||
|
||||
public static getInstance(): UploadImageService {
|
||||
if (!UploadImageService.instance) {
|
||||
UploadImageService.instance = new UploadImageService();
|
||||
}
|
||||
return UploadImageService.instance;
|
||||
}
|
||||
|
||||
private showToast(message: string, type: 'success' | 'error' | 'info' = 'info') {
|
||||
Toast.show({
|
||||
type,
|
||||
text1: message,
|
||||
position: 'top',
|
||||
topOffset: Platform.OS === 'ios' ? 60 : 20,
|
||||
visibilityTime: 2000,
|
||||
});
|
||||
}
|
||||
|
||||
private async checkPermission(): Promise<boolean> {
|
||||
try {
|
||||
const permission = Platform.select({
|
||||
android: PERMISSIONS.ANDROID.READ_EXTERNAL_STORAGE,
|
||||
ios: PERMISSIONS.IOS.PHOTO_LIBRARY,
|
||||
default: PERMISSIONS.ANDROID.READ_EXTERNAL_STORAGE,
|
||||
});
|
||||
|
||||
const result = await request(permission);
|
||||
|
||||
if (result !== RESULTS.GRANTED) {
|
||||
this.showToast('请在设置中授予相册访问权限', 'error');
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
} catch (error) {
|
||||
console.error('Permission check error:', error);
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
private async getQiniuToken(): Promise<boolean> {
|
||||
try {
|
||||
const response = await apiService.getQiniuToken();
|
||||
if (response.code == 200) {
|
||||
this.uploadInfo = {
|
||||
token: response.token,
|
||||
domain: response.domain
|
||||
};
|
||||
return true;
|
||||
}
|
||||
this.showToast('获取上传凭证失败', 'error');
|
||||
return false;
|
||||
} catch (error) {
|
||||
console.error('Get qiniu token error:', error);
|
||||
this.showToast('获取上传凭证失败', 'error');
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
private generateKey(): string {
|
||||
const timestamp = Date.now();
|
||||
const random = Math.random().toString(36).substring(2, 15);
|
||||
return `bike/img/static/${timestamp}_${random}`;
|
||||
}
|
||||
|
||||
private async uploadToQiniu(uri: string, key: string): Promise<UploadResponse> {
|
||||
if (!this.uploadInfo?.token) {
|
||||
return { success: false, error: '上传凭证无效' };
|
||||
}
|
||||
|
||||
const formData = new FormData();
|
||||
formData.append('token', this.uploadInfo.token);
|
||||
formData.append('key', key);
|
||||
formData.append('file', {
|
||||
uri: uri,
|
||||
type: 'image/jpeg',
|
||||
name: 'image.jpg',
|
||||
});
|
||||
|
||||
try {
|
||||
const response = await fetch('https://up-z2.qiniup.com', {
|
||||
method: 'POST',
|
||||
body: formData,
|
||||
headers: {
|
||||
'Content-Type': 'multipart/form-data',
|
||||
},
|
||||
});
|
||||
|
||||
const result = await response.json();
|
||||
|
||||
if (result.key) {
|
||||
return {
|
||||
success: true,
|
||||
imageUrl: `${this.uploadInfo.domain}/${result.key}`
|
||||
};
|
||||
}
|
||||
|
||||
return { success: false, error: '上传失败' };
|
||||
} catch (error) {
|
||||
console.error('Upload error:', error);
|
||||
return { success: false, error: '上传过程中出错' };
|
||||
}
|
||||
}
|
||||
|
||||
public async uploadImage(options = { multiple: false }): Promise<string[]> {
|
||||
try {
|
||||
// 检查权限
|
||||
const hasPermission = await this.checkPermission();
|
||||
if (!hasPermission) {
|
||||
return [];
|
||||
}
|
||||
|
||||
// 获取七牛云上传凭证
|
||||
const hasToken = await this.getQiniuToken();
|
||||
if (!hasToken) {
|
||||
return [];
|
||||
}
|
||||
|
||||
// 选择图片
|
||||
const result = await launchImageLibrary({
|
||||
mediaType: 'photo',
|
||||
quality: 1,
|
||||
selectionLimit: options.multiple ? 0 : 1,
|
||||
});
|
||||
|
||||
if (result.didCancel || !result.assets) {
|
||||
return [];
|
||||
}
|
||||
|
||||
// 上传所有选中的图片
|
||||
const uploadPromises = result.assets.map(async (asset) => {
|
||||
if (!asset.uri) return null;
|
||||
|
||||
const key = this.generateKey();
|
||||
const uploadResult = await this.uploadToQiniu(asset.uri, key);
|
||||
|
||||
if (uploadResult.success && uploadResult.imageUrl) {
|
||||
return uploadResult.imageUrl;
|
||||
}
|
||||
return null;
|
||||
});
|
||||
|
||||
const results = await Promise.all(uploadPromises);
|
||||
const successUrls = results.filter((url): url is string => url !== null);
|
||||
|
||||
if (successUrls.length > 0) {
|
||||
this.showToast('上传成功', 'success');
|
||||
} else {
|
||||
this.showToast('上传失败', 'error');
|
||||
}
|
||||
|
||||
return successUrls;
|
||||
} catch (error) {
|
||||
console.error('Upload process error:', error);
|
||||
this.showToast('上传过程中出错', 'error');
|
||||
return [];
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
export const uploadImageService = UploadImageService.getInstance();
|
|
@ -40,8 +40,11 @@ type DeviceType = {
|
|||
remainingMileage: number;
|
||||
isDefault: number;
|
||||
lockStatus: number;
|
||||
type: number;
|
||||
expirationTime: string;
|
||||
mac: string;
|
||||
remark: string;
|
||||
};
|
||||
|
||||
// 定义导航类型
|
||||
type NavigationProp = StackNavigationProp<RootStackParamList>;
|
||||
|
||||
|
@ -56,7 +59,7 @@ const NormaIndex: React.FC = () => {
|
|||
navigation.navigate('DeviceList');
|
||||
};
|
||||
const toSet = () => {
|
||||
navigation.navigate('DeviceSet');
|
||||
navigation.navigate('deviceDetailSet');
|
||||
};
|
||||
|
||||
const toMap = () => {
|
||||
|
@ -186,7 +189,7 @@ const NormaIndex: React.FC = () => {
|
|||
numberOfLines={2}
|
||||
ellipsizeMode="tail"
|
||||
>
|
||||
{defaultDevice?.model || '未选择车辆'}
|
||||
{defaultDevice?.remark || defaultDevice?.model}
|
||||
</Text>
|
||||
<Image
|
||||
source={{ uri: 'https://api.ccttiot.com/smartmeter/img/static/uJRIitv0Yn5K7CEVe9qd' }}
|
||||
|
|
|
@ -0,0 +1,541 @@
|
|||
import React, { useState, useEffect } from 'react';
|
||||
import { View, StyleSheet, ImageBackground, Text, TouchableOpacity, Modal, TextInput, ActivityIndicator, Button, Image, ScrollView } from 'react-native';
|
||||
import { TopNavigation, TopNavigationAction, Icon, IconElement, Select, SelectItem, IndexPath } from '@ui-kitten/components';
|
||||
import { useNavigation } from '@react-navigation/native';
|
||||
import { rpx } from '../../utils/rpx';
|
||||
import { apiService } from '../../utils/api';
|
||||
import AsyncStorage from '@react-native-async-storage/async-storage';
|
||||
import Toast from 'react-native-toast-message';
|
||||
import { uploadImageService } from '../../utils/uploadImg';
|
||||
|
||||
const BackIcon = (props): IconElement => (
|
||||
<Icon
|
||||
{...props}
|
||||
name='arrow-back'
|
||||
fill='#000000'
|
||||
/>
|
||||
);
|
||||
|
||||
interface EditModalProps {
|
||||
visible: boolean;
|
||||
onClose: () => void;
|
||||
title: string;
|
||||
value: string;
|
||||
onSubmit: (value: string) => void;
|
||||
placeholder?: string;
|
||||
}
|
||||
|
||||
const EditModal: React.FC<EditModalProps> = ({
|
||||
visible,
|
||||
onClose,
|
||||
title,
|
||||
value,
|
||||
onSubmit,
|
||||
placeholder
|
||||
}) => {
|
||||
const [inputValue, setInputValue] = useState(value);
|
||||
|
||||
useEffect(() => {
|
||||
setInputValue(value);
|
||||
}, [value]);
|
||||
|
||||
const handleSubmit = () => {
|
||||
onSubmit(inputValue);
|
||||
};
|
||||
|
||||
return (
|
||||
<Modal
|
||||
visible={visible}
|
||||
transparent={true}
|
||||
animationType="fade"
|
||||
onRequestClose={onClose}
|
||||
>
|
||||
<View style={styles.modalOverlay}>
|
||||
<View style={styles.modalContent}>
|
||||
<Text style={styles.modalTitle}>{title}</Text>
|
||||
<TextInput
|
||||
style={styles.modalInput}
|
||||
value={inputValue}
|
||||
onChangeText={setInputValue}
|
||||
placeholder={placeholder}
|
||||
placeholderTextColor="#999999"
|
||||
/>
|
||||
<View style={styles.modalButtons}>
|
||||
<TouchableOpacity style={styles.modalButton} onPress={onClose}>
|
||||
<Text style={styles.modalButtonText}>取消</Text>
|
||||
</TouchableOpacity>
|
||||
<TouchableOpacity style={[styles.modalButton, styles.modalButtonPrimary]} onPress={handleSubmit}>
|
||||
<Text style={[styles.modalButtonText, styles.modalButtonTextPrimary]}>确定</Text>
|
||||
</TouchableOpacity>
|
||||
</View>
|
||||
</View>
|
||||
</View>
|
||||
</Modal>
|
||||
);
|
||||
};
|
||||
|
||||
const deviceDetailSet = () => {
|
||||
const navigation = useNavigation();
|
||||
const [modalVisible, setModalVisible] = useState(false);
|
||||
const [isLoading, setIsLoading] = useState(true);
|
||||
const [deviceInfo, setDeviceInfo] = useState<any>(null);
|
||||
const [currentEdit, setCurrentEdit] = useState<{
|
||||
title: string;
|
||||
value: string;
|
||||
type: 'remark' | 'fullVoltage' | 'vehicleNum' | 'lowVoltage' | 'fullEndurance';
|
||||
} | null>(null);
|
||||
|
||||
const [modelData, setModelData] = useState([]);
|
||||
const [selectedBrandIndex, setSelectedBrandIndex] = useState<IndexPath | null>(null);
|
||||
const [selectedModelIndex, setSelectedModelIndex] = useState<IndexPath | null>(null);
|
||||
const [brandList, setBrandList] = useState([]);
|
||||
const [modelList, setModelList] = useState([]);
|
||||
const [modelModalVisible, setModelModalVisible] = useState(false);
|
||||
const [previewImage, setPreviewImage] = useState<string | null>(null);
|
||||
|
||||
const upload = async () => {
|
||||
try {
|
||||
const result = await uploadImageService.uploadImage();
|
||||
if (result) {
|
||||
const response = await apiService.updateDevice({
|
||||
sn: deviceInfo.sn,
|
||||
pricture: result
|
||||
});
|
||||
|
||||
if (response.code === 200) {
|
||||
setDeviceInfo(prevInfo => ({
|
||||
...prevInfo,
|
||||
pricture: result
|
||||
}));
|
||||
Toast.show({
|
||||
type: 'success',
|
||||
text1: '图片上传成功',
|
||||
});
|
||||
fetchDeviceInfo();
|
||||
} else {
|
||||
Toast.show({
|
||||
type: 'error',
|
||||
text1: response.msg || '图片上传失败',
|
||||
});
|
||||
}
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Upload failed:', error);
|
||||
Toast.show({
|
||||
type: 'error',
|
||||
text1: '上传失败',
|
||||
text2: '请检查网络连接',
|
||||
});
|
||||
}
|
||||
};
|
||||
|
||||
const fetchDeviceInfo = async () => {
|
||||
setIsLoading(true);
|
||||
try {
|
||||
const sn = await AsyncStorage.getItem('defaultDeviceSN');
|
||||
if (!sn) {
|
||||
Toast.show({
|
||||
type: 'error',
|
||||
text1: '未找到设备信息',
|
||||
});
|
||||
return;
|
||||
}
|
||||
|
||||
const response = await apiService.getDeviceInfo(sn);
|
||||
if (response.code === 200 && response.data) {
|
||||
setDeviceInfo(response.data);
|
||||
} else {
|
||||
Toast.show({
|
||||
type: 'error',
|
||||
text1: response.msg || '获取设备信息失败',
|
||||
});
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('获取设备信息错误:', error);
|
||||
Toast.show({
|
||||
type: 'error',
|
||||
text1: '获取设备信息失败',
|
||||
text2: '请检查网络连接',
|
||||
});
|
||||
} finally {
|
||||
setIsLoading(false);
|
||||
}
|
||||
};
|
||||
|
||||
const fetchModelList = async () => {
|
||||
const response = await apiService.getModelList();
|
||||
if (response.code === 200) {
|
||||
setModelData(response.data);
|
||||
const brands = Array.from(new Set(response.data.map(item => item.brandName)));
|
||||
setBrandList(brands);
|
||||
setSelectedBrandIndex(new IndexPath(0));
|
||||
filterModels(brands[0]);
|
||||
}
|
||||
};
|
||||
|
||||
const filterModels = (brand) => {
|
||||
const models = modelData.filter(item => item.brandName === brand).map(item => item.model);
|
||||
setModelList(models);
|
||||
setSelectedModelIndex(new IndexPath(0));
|
||||
const initialModel = models[0];
|
||||
const initialImageUrl = getImageUrl(brand, initialModel);
|
||||
setPreviewImage(initialImageUrl);
|
||||
};
|
||||
|
||||
const getImageUrl = (brand, model) => {
|
||||
const selectedModel = modelData.find(item => item.brandName === brand && item.model === model);
|
||||
return selectedModel ? selectedModel.picture : null;
|
||||
};
|
||||
|
||||
const handleBrandChange = (index) => {
|
||||
setSelectedBrandIndex(index);
|
||||
const brand = brandList[index.row];
|
||||
filterModels(brand);
|
||||
};
|
||||
|
||||
const handleModelChange = (index) => {
|
||||
setSelectedModelIndex(index);
|
||||
const model = modelList[index.row];
|
||||
const brand = brandList[selectedBrandIndex.row];
|
||||
const imageUrl = getImageUrl(brand, model);
|
||||
setPreviewImage(imageUrl);
|
||||
};
|
||||
|
||||
const handleModelSelection = async () => {
|
||||
if (!selectedBrandIndex || !selectedModelIndex) return;
|
||||
|
||||
const brand = brandList[selectedBrandIndex.row];
|
||||
const model = modelList[selectedModelIndex.row];
|
||||
const imageUrl = getImageUrl(brand, model);
|
||||
|
||||
try {
|
||||
const response = await apiService.updateDevice({
|
||||
sn: deviceInfo.sn,
|
||||
pricture: imageUrl
|
||||
});
|
||||
|
||||
if (response.code === 200) {
|
||||
setDeviceInfo(prevInfo => ({
|
||||
...prevInfo,
|
||||
pricture: imageUrl
|
||||
}));
|
||||
Toast.show({
|
||||
type: 'success',
|
||||
text1: '车辆信息更新成功',
|
||||
});
|
||||
fetchDeviceInfo();
|
||||
} else {
|
||||
Toast.show({
|
||||
type: 'error',
|
||||
text1: response.msg || '车辆信息更新失败',
|
||||
});
|
||||
}
|
||||
} catch (error) {
|
||||
console.error('Update failed:', error);
|
||||
Toast.show({
|
||||
type: 'error',
|
||||
text1: '更新失败',
|
||||
text2: '请检查网络连接',
|
||||
});
|
||||
}
|
||||
|
||||
setModelModalVisible(false);
|
||||
};
|
||||
|
||||
useEffect(() => {
|
||||
fetchDeviceInfo();
|
||||
fetchModelList();
|
||||
}, []);
|
||||
|
||||
const navigateBack = () => {
|
||||
navigation.goBack();
|
||||
};
|
||||
|
||||
const BackAction = () => (
|
||||
<TopNavigationAction
|
||||
icon={BackIcon}
|
||||
onPress={navigateBack}
|
||||
/>
|
||||
);
|
||||
|
||||
return (
|
||||
<View style={styles.container}>
|
||||
<TopNavigation
|
||||
title='设备设置'
|
||||
alignment='center'
|
||||
accessoryLeft={BackAction}
|
||||
style={styles.header}
|
||||
/>
|
||||
<ScrollView
|
||||
contentContainerStyle={styles.scrollContent}
|
||||
scrollEventThrottle={16}
|
||||
nestedScrollEnabled={true}
|
||||
>
|
||||
<ImageBackground
|
||||
source={{ uri: 'https://lxnapi.ccttiot.com/bike/img/static/uYRs7Cv2Pbp95w3KjGO3' }}
|
||||
style={styles.imageBackground}
|
||||
>
|
||||
<ImageBackground
|
||||
source={{ uri: 'https://lxnapi.ccttiot.com/FqAu65KgcylSRv1pD4008a_Ctpz-' }}
|
||||
style={styles.carInfo}
|
||||
>
|
||||
<Image source={{ uri: deviceInfo?.pricture || 'https://lxnapi.ccttiot.com/bike/img/static/uVnIDwcwQP7oo12PeYVJ' }} style={{ width: rpx(440), height: rpx(340) }} />
|
||||
<View style={styles.carInfoButtons}>
|
||||
<TouchableOpacity style={styles.carInfoButton} onPress={upload} >
|
||||
<Text style={styles.carInfoButtonText}>上传图片</Text>
|
||||
</TouchableOpacity>
|
||||
<TouchableOpacity style={[styles.carInfoButton, styles.carInfoButtonPrimary]} onPress={() => setModelModalVisible(true)}>
|
||||
<Text style={[styles.carInfoButtonText, styles.carInfoButtonTextPrimary]}>匹配车辆</Text>
|
||||
</TouchableOpacity>
|
||||
</View>
|
||||
</ImageBackground>
|
||||
<View style={styles.infoContainer}>
|
||||
{/* ... existing info items ... */}
|
||||
</View>
|
||||
|
||||
<TouchableOpacity style={styles.unbindButton} onPress={handleUnbind}>
|
||||
<Text style={styles.unbindText}>车辆解绑</Text>
|
||||
</TouchableOpacity>
|
||||
|
||||
{currentEdit && (
|
||||
<EditModal
|
||||
visible={modalVisible}
|
||||
onClose={() => setModalVisible(false)}
|
||||
title={currentEdit.title}
|
||||
value={currentEdit.value}
|
||||
onSubmit={handleSubmit}
|
||||
placeholder={`请输入${currentEdit.title.slice(2)}`}
|
||||
/>
|
||||
)}
|
||||
|
||||
<Modal
|
||||
visible={modelModalVisible}
|
||||
transparent={true}
|
||||
animationType="slide"
|
||||
onRequestClose={() => setModelModalVisible(false)}
|
||||
>
|
||||
<View style={styles.modalOverlay}>
|
||||
<View style={styles.modalContent}>
|
||||
<Text style={styles.modalTitle}>选择车型</Text>
|
||||
<Image
|
||||
source={{ uri: previewImage || 'https://lxnapi.ccttiot.com/bike/img/static/uVnIDwcwQP7oo12PeYVJ' }}
|
||||
style={styles.modalImage}
|
||||
/>
|
||||
<Select
|
||||
selectedIndex={selectedBrandIndex}
|
||||
onSelect={handleBrandChange}
|
||||
style={styles.select}
|
||||
value={brandList[selectedBrandIndex?.row]}
|
||||
>
|
||||
{brandList.map((brand, index) => (
|
||||
<SelectItem key={index} title={brand} />
|
||||
))}
|
||||
</Select>
|
||||
<Select
|
||||
selectedIndex={selectedModelIndex}
|
||||
onSelect={handleModelChange}
|
||||
style={styles.select}
|
||||
value={modelList[selectedModelIndex?.row]}
|
||||
>
|
||||
{modelList.map((model, index) => (
|
||||
<SelectItem key={index} title={model} />
|
||||
))}
|
||||
</Select>
|
||||
<View style={styles.modalButtons}>
|
||||
<TouchableOpacity style={styles.modalButton} onPress={handleModelSelection}>
|
||||
<Text style={styles.modalButtonText}>确定</Text>
|
||||
</TouchableOpacity>
|
||||
<TouchableOpacity style={styles.modalButton} onPress={() => setModelModalVisible(false)}>
|
||||
<Text style={styles.modalButtonText}>关闭</Text>
|
||||
</TouchableOpacity>
|
||||
</View>
|
||||
</View>
|
||||
</View>
|
||||
</Modal>
|
||||
</ImageBackground>
|
||||
</ScrollView>
|
||||
</View>
|
||||
);
|
||||
};
|
||||
|
||||
const styles = StyleSheet.create({
|
||||
scrollContent: {
|
||||
flexGrow: 1,
|
||||
},
|
||||
container: {
|
||||
flex: 1,
|
||||
},
|
||||
imageBackground: {
|
||||
paddingTop: rpx(40),
|
||||
paddingBottom: rpx(60),
|
||||
flex: 1,
|
||||
width: '100%',
|
||||
height: '100%',
|
||||
},
|
||||
carInfo: {
|
||||
paddingTop: rpx(180),
|
||||
width: '100%',
|
||||
height: rpx(828),
|
||||
alignItems: 'center',
|
||||
},
|
||||
carInfoButtons: {
|
||||
marginTop: rpx(160),
|
||||
flexDirection: 'row',
|
||||
justifyContent: 'space-between',
|
||||
alignItems: 'center',
|
||||
width: '80%',
|
||||
},
|
||||
carInfoButton: {
|
||||
display: 'flex',
|
||||
justifyContent: 'center',
|
||||
alignItems: 'center',
|
||||
width: rpx(250),
|
||||
height: rpx(80),
|
||||
backgroundColor: '#ffffff',
|
||||
borderRadius: rpx(20),
|
||||
borderWidth: rpx(2),
|
||||
borderColor: '#808080',
|
||||
},
|
||||
carInfoButtonPrimary: {
|
||||
borderRadius: rpx(20),
|
||||
backgroundColor: '#333333',
|
||||
},
|
||||
carInfoButtonText: {
|
||||
color: '#333333',
|
||||
fontSize: rpx(32),
|
||||
fontWeight: '500',
|
||||
},
|
||||
carInfoButtonTextPrimary: {
|
||||
color: '#FFFFFF',
|
||||
},
|
||||
loadingContainer: {
|
||||
flex: 1,
|
||||
justifyContent: 'center',
|
||||
alignItems: 'center',
|
||||
backgroundColor: '#FFFFFF',
|
||||
},
|
||||
header: {
|
||||
backgroundColor: 'transparent',
|
||||
elevation: 0,
|
||||
shadowOpacity: 0,
|
||||
},
|
||||
titleStyle: {
|
||||
color: '#000000',
|
||||
fontSize: rpx(36),
|
||||
fontWeight: 'bold',
|
||||
},
|
||||
infoContainer: {
|
||||
marginTop: rpx(40),
|
||||
marginHorizontal: rpx(32),
|
||||
backgroundColor: '#FFFFFF',
|
||||
borderRadius: rpx(20),
|
||||
padding: rpx(20),
|
||||
},
|
||||
infoItem: {
|
||||
flexDirection: 'row',
|
||||
justifyContent: 'space-between',
|
||||
alignItems: 'center',
|
||||
paddingVertical: rpx(24),
|
||||
borderBottomWidth: 1,
|
||||
borderBottomColor: '#F5F5F5',
|
||||
},
|
||||
label: {
|
||||
fontSize: rpx(28),
|
||||
color: '#333333',
|
||||
},
|
||||
valueContainer: {
|
||||
flexDirection: 'row',
|
||||
alignItems: 'center',
|
||||
},
|
||||
value: {
|
||||
fontSize: rpx(28),
|
||||
color: '#999999',
|
||||
marginRight: rpx(8),
|
||||
},
|
||||
arrow: {
|
||||
width: rpx(32),
|
||||
height: rpx(32),
|
||||
},
|
||||
unbindButton: {
|
||||
marginHorizontal: rpx(32),
|
||||
marginTop: rpx(40),
|
||||
height: rpx(88),
|
||||
backgroundColor: '#333333',
|
||||
borderRadius: rpx(20),
|
||||
justifyContent: 'center',
|
||||
alignItems: 'center',
|
||||
},
|
||||
unbindText: {
|
||||
color: '#FFFFFF',
|
||||
fontSize: rpx(32),
|
||||
fontWeight: '500',
|
||||
},
|
||||
modalOverlay: {
|
||||
flex: 1,
|
||||
backgroundColor: 'rgba(0, 0, 0, 0.5)',
|
||||
justifyContent: 'center',
|
||||
alignItems: 'center',
|
||||
},
|
||||
modalContent: {
|
||||
width: rpx(650),
|
||||
backgroundColor: '#FFFFFF',
|
||||
borderRadius: rpx(20),
|
||||
padding: rpx(40),
|
||||
shadowColor: '#000',
|
||||
shadowOffset: { width: 0, height: 2 },
|
||||
shadowOpacity: 0.25,
|
||||
shadowRadius: 3.84,
|
||||
elevation: 5,
|
||||
},
|
||||
modalTitle: {
|
||||
fontSize: rpx(36),
|
||||
color: '#333333',
|
||||
fontWeight: 'bold',
|
||||
marginBottom: rpx(40),
|
||||
textAlign: 'center',
|
||||
},
|
||||
modalInput: {
|
||||
height: rpx(88),
|
||||
borderWidth: 1,
|
||||
borderColor: '#E5E5E5',
|
||||
borderRadius: rpx(12),
|
||||
paddingHorizontal: rpx(24),
|
||||
fontSize: rpx(28),
|
||||
color: '#333333',
|
||||
marginBottom: rpx(32),
|
||||
},
|
||||
modalButtons: {
|
||||
flexDirection: 'row',
|
||||
justifyContent: 'space-between',
|
||||
marginTop: rpx(20),
|
||||
},
|
||||
modalButton: {
|
||||
flex: 1,
|
||||
height: rpx(88),
|
||||
justifyContent: 'center',
|
||||
alignItems: 'center',
|
||||
borderRadius: rpx(12),
|
||||
marginHorizontal: rpx(8),
|
||||
backgroundColor: '#F5F5F5',
|
||||
},
|
||||
modalButtonPrimary: {
|
||||
backgroundColor: '#333333',
|
||||
},
|
||||
modalButtonText: {
|
||||
fontSize: rpx(28),
|
||||
color: '#333333',
|
||||
},
|
||||
modalButtonTextPrimary: {
|
||||
color: '#FFFFFF',
|
||||
},
|
||||
select: {
|
||||
marginBottom: rpx(20),
|
||||
},
|
||||
modalImage: {
|
||||
width: rpx(440),
|
||||
height: rpx(340),
|
||||
marginBottom: rpx(20),
|
||||
},
|
||||
});
|
||||
|
||||
export default deviceDetailSet;
|
|
@ -2119,7 +2119,7 @@
|
|||
resolved "https://registry.npmmirror.com/@react-native-community/slider/-/slider-4.5.5.tgz"
|
||||
integrity sha512-x2N415pg4ZxIltArOKczPwn7JEYh+1OxQ4+hTnafomnMsqs65HZuEWcX+Ch8c5r8V83DiunuQUf5hWGWlw8hQQ==
|
||||
|
||||
"@react-native-picker/picker@^2.9.0":
|
||||
"@react-native-picker/picker@^2.10.2":
|
||||
version "2.10.2"
|
||||
resolved "https://registry.npmmirror.com/@react-native-picker/picker/-/picker-2.10.2.tgz"
|
||||
integrity sha512-kr3OvCRwTYjR/OKlb52k4xmQVU7dPRIALqpyiihexdJxEgvc1smnepgqCeM9oXmNSG4YaV5/RSxFlLC5Z/T/Eg==
|
||||
|
|
Loading…
Reference in New Issue
Block a user