update interface, diary

This commit is contained in:
2025-12-04 15:54:49 +07:00
parent 4d60ce279e
commit 0672f8adf9
10 changed files with 291 additions and 90 deletions

View File

@@ -60,7 +60,7 @@ export default function TabLayout() {
options={{ options={{
title: t("navigation.manager"), title: t("navigation.manager"),
tabBarIcon: ({ color }) => ( tabBarIcon: ({ color }) => (
<IconSymbol size={28} name="square.stack.3d.up" color={color} /> <IconSymbol size={28} name="square.stack.3d.up.fill" color={color} />
), ),
}} }}
/> />

View File

@@ -1,5 +1,12 @@
import { useState } from "react"; import { useEffect, useState } from "react";
import { Platform, ScrollView, StyleSheet, Text, TouchableOpacity, View } from "react-native"; import {
Platform,
ScrollView,
StyleSheet,
Text,
TouchableOpacity,
View,
} from "react-native";
import { SafeAreaView } from "react-native-safe-area-context"; import { SafeAreaView } from "react-native-safe-area-context";
import { Ionicons } from "@expo/vector-icons"; import { Ionicons } from "@expo/vector-icons";
import SearchBar from "@/components/diary/SearchBar"; import SearchBar from "@/components/diary/SearchBar";
@@ -7,6 +14,7 @@ import FilterButton from "@/components/diary/FilterButton";
import TripCard from "@/components/diary/TripCard"; import TripCard from "@/components/diary/TripCard";
import FilterModal, { FilterValues } from "@/components/diary/FilterModal"; import FilterModal, { FilterValues } from "@/components/diary/FilterModal";
import { MOCK_TRIPS } from "@/components/diary/mockData"; import { MOCK_TRIPS } from "@/components/diary/mockData";
import { useThings } from "@/state/use-thing";
export default function diary() { export default function diary() {
const [searchText, setSearchText] = useState(""); const [searchText, setSearchText] = useState("");
@@ -16,6 +24,23 @@ export default function diary() {
startDate: null, startDate: null,
endDate: null, endDate: null,
}); });
// Body things (đang fix cứng)
const payloadThings: Model.SearchThingBody = {
offset: 0,
limit: 200,
order: "name",
dir: "asc",
metadata: {
not_empty: "ship_name, ship_reg_number",
},
};
// Gọi API things
const { things, getThings } = useThings();
useEffect(() => {
getThings(payloadThings);
}, []);
console.log(things);
// Filter trips based on search text and filters // Filter trips based on search text and filters
const filteredTrips = MOCK_TRIPS.filter((trip) => { const filteredTrips = MOCK_TRIPS.filter((trip) => {
@@ -73,6 +98,31 @@ export default function diary() {
console.log("Trip pressed:", tripId); console.log("Trip pressed:", tripId);
}; };
const handleViewTrip = (tripId: string) => {
console.log("View trip:", tripId);
// TODO: Navigate to trip detail view
};
const handleEditTrip = (tripId: string) => {
console.log("Edit trip:", tripId);
// TODO: Navigate to trip edit screen
};
const handleViewTeam = (tripId: string) => {
console.log("View team:", tripId);
// TODO: Navigate to team management
};
const handleSendTrip = (tripId: string) => {
console.log("Send trip:", tripId);
// TODO: Send trip for approval
};
const handleDeleteTrip = (tripId: string) => {
console.log("Delete trip:", tripId);
// TODO: Show confirmation dialog and delete trip
};
return ( return (
<SafeAreaView style={styles.safeArea}> <SafeAreaView style={styles.safeArea}>
<View style={styles.container}> <View style={styles.container}>
@@ -111,6 +161,11 @@ export default function diary() {
key={trip.id} key={trip.id}
trip={trip} trip={trip}
onPress={() => handleTripPress(trip.id)} onPress={() => handleTripPress(trip.id)}
onView={() => handleViewTrip(trip.id)}
onEdit={() => handleEditTrip(trip.id)}
onTeam={() => handleViewTeam(trip.id)}
onSend={() => handleSendTrip(trip.id)}
onDelete={() => handleDeleteTrip(trip.id)}
/> />
))} ))}
@@ -141,13 +196,13 @@ const styles = StyleSheet.create({
}, },
container: { container: {
flex: 1, flex: 1,
padding: 16, padding: 10,
}, },
titleText: { titleText: {
fontSize: 28, fontSize: 28,
fontWeight: "700", fontWeight: "700",
lineHeight: 36, lineHeight: 36,
marginBottom: 20, marginBottom: 10,
color: "#111827", color: "#111827",
fontFamily: Platform.select({ fontFamily: Platform.select({
ios: "System", ios: "System",

View File

@@ -73,7 +73,7 @@ export default function HomeScreen() {
offset: 0, offset: 0,
limit: 50, limit: 50,
order: "name", order: "name",
sort: "asc", dir: "asc",
metadata: { metadata: {
not_empty: "ship_id", not_empty: "ship_id",
}, },
@@ -384,7 +384,7 @@ export default function HomeScreen() {
offset: 0, offset: 0,
limit: 50, limit: 50,
order: "name", order: "name",
sort: "asc", dir: "asc",
metadata: { metadata: {
...metaFormQuery, ...metaFormQuery,
...metaStateQuery, ...metaStateQuery,

View File

@@ -18,9 +18,11 @@ interface StatusDropdownProps {
const STATUS_OPTIONS: Array<{ value: TripStatus | null; label: string }> = [ const STATUS_OPTIONS: Array<{ value: TripStatus | null; label: string }> = [
{ value: null, label: "Vui lòng chọn" }, { value: null, label: "Vui lòng chọn" },
{ value: "completed", label: "Hoàn thành" }, { value: "created", label: "Đã khởi tạo" },
{ value: "pending", label: "Chờ duyệt" },
{ value: "approved", label: "Đã duyệt" },
{ value: "in-progress", label: "Đang hoạt động" }, { value: "in-progress", label: "Đang hoạt động" },
{ value: "quality-check", label: "Đã khởi tạo" }, { value: "completed", label: "Hoàn thành" },
{ value: "cancelled", label: "Đã hủy" }, { value: "cancelled", label: "Đã hủy" },
]; ];
@@ -162,7 +164,6 @@ const styles = StyleSheet.create({
paddingVertical: 16, paddingVertical: 16,
borderBottomWidth: 1, borderBottomWidth: 1,
borderBottomColor: "#F3F4F6", borderBottomColor: "#F3F4F6",
}, },
selectedOption: { selectedOption: {
backgroundColor: "#EFF6FF", backgroundColor: "#EFF6FF",

View File

@@ -12,13 +12,24 @@ import { Trip, TRIP_STATUS_CONFIG } from "./types";
interface TripCardProps { interface TripCardProps {
trip: Trip; trip: Trip;
onPress?: () => void; onPress?: () => void;
onView?: () => void;
onEdit?: () => void;
onTeam?: () => void;
onSend?: () => void;
onDelete?: () => void;
} }
export default function TripCard({ trip, onPress }: TripCardProps) { export default function TripCard({ trip, onPress, onView, onEdit, onTeam, onSend, onDelete }: TripCardProps) {
const statusConfig = TRIP_STATUS_CONFIG[trip.status]; const statusConfig = TRIP_STATUS_CONFIG[trip.status];
// Determine which actions to show based on status
const showEdit = trip.status === 'created' || trip.status === 'pending';
const showSend = trip.status === 'created';
const showDelete = trip.status === 'pending';
return ( return (
<TouchableOpacity style={styles.card} onPress={onPress} activeOpacity={0.7}> <View style={styles.card}>
<TouchableOpacity onPress={onPress} activeOpacity={0.7}>
{/* Header */} {/* Header */}
<View style={styles.header}> <View style={styles.header}>
<View style={styles.headerLeft}> <View style={styles.headerLeft}>
@@ -78,6 +89,42 @@ export default function TripCard({ trip, onPress }: TripCardProps) {
</View> </View>
</View> </View>
</TouchableOpacity> </TouchableOpacity>
{/* Action Buttons */}
<View style={styles.divider} />
<View style={styles.actionsContainer}>
<TouchableOpacity style={styles.actionButton} onPress={onView} activeOpacity={0.7}>
<Ionicons name="eye-outline" size={20} color="#6B7280" />
<Text style={styles.actionText}>View</Text>
</TouchableOpacity>
{showEdit && (
<TouchableOpacity style={styles.actionButton} onPress={onEdit} activeOpacity={0.7}>
<Ionicons name="create-outline" size={20} color="#6B7280" />
<Text style={styles.actionText}>Edit</Text>
</TouchableOpacity>
)}
<TouchableOpacity style={styles.actionButton} onPress={onTeam} activeOpacity={0.7}>
<Ionicons name="people-outline" size={20} color="#6B7280" />
<Text style={styles.actionText}>Team</Text>
</TouchableOpacity>
{showSend && (
<TouchableOpacity style={styles.actionButton} onPress={onSend} activeOpacity={0.7}>
<Ionicons name="send-outline" size={20} color="#6B7280" />
<Text style={styles.actionText}>Send</Text>
</TouchableOpacity>
)}
{showDelete && (
<TouchableOpacity style={styles.actionButton} onPress={onDelete} activeOpacity={0.7}>
<Ionicons name="trash-outline" size={20} color="#EF4444" />
<Text style={[styles.actionText, styles.deleteText]}>Delete</Text>
</TouchableOpacity>
)}
</View>
</View>
); );
} }
@@ -178,4 +225,34 @@ const styles = StyleSheet.create({
duration: { duration: {
color: "#3B82F6", color: "#3B82F6",
}, },
divider: {
height: 1,
backgroundColor: "#F3F4F6",
marginTop: 16,
},
actionsContainer: {
flexDirection: "row",
justifyContent: "space-around",
paddingTop: 12,
},
actionButton: {
flexDirection: "row",
alignItems: "center",
gap: 6,
paddingVertical: 8,
paddingHorizontal: 12,
},
actionText: {
fontSize: 14,
color: "#6B7280",
fontWeight: "500",
fontFamily: Platform.select({
ios: "System",
android: "Roboto",
default: "System",
}),
},
deleteText: {
color: "#EF4444",
},
}); });

View File

@@ -54,7 +54,7 @@ export const MOCK_TRIPS: Trip[] = [
departureDate: "2025-11-18 07:00", departureDate: "2025-11-18 07:00",
returnDate: "2025-11-23 14:00", returnDate: "2025-11-23 14:00",
duration: "5 ngày 7 giờ", duration: "5 ngày 7 giờ",
status: "quality-check", status: "created",
}, },
{ {
id: "T006", id: "T006",
@@ -67,4 +67,26 @@ export const MOCK_TRIPS: Trip[] = [
duration: "6 giờ", duration: "6 giờ",
status: "in-progress", status: "in-progress",
}, },
{
id: "T007",
title: "Khảo sát vùng biển",
code: "T007",
vessel: "Ngọc Lan",
vesselCode: "V002",
departureDate: "2025-12-01 07:00",
returnDate: null,
duration: "-",
status: "pending",
},
{
id: "T008",
title: "Đánh cá xa bờ",
code: "T008",
vessel: "Việt Thắng",
vesselCode: "V003",
departureDate: "2025-12-05 05:00",
returnDate: null,
duration: "-",
status: "approved",
},
]; ];

View File

@@ -1,8 +1,10 @@
export type TripStatus = export type TripStatus =
| "completed" | "created" // Đã khởi tạo
| "in-progress" | "pending" // Chờ duyệt
| "cancelled" | "approved" // Đã duyệt
| "quality-check"; | "in-progress" // Đang hoạt động
| "completed" // Hoàn thành
| "cancelled"; // Đã hủy
export interface Trip { export interface Trip {
id: string; id: string;
@@ -17,28 +19,40 @@ export interface Trip {
} }
export const TRIP_STATUS_CONFIG = { export const TRIP_STATUS_CONFIG = {
completed: { created: {
label: "Hoàn thành", label: "Đã khởi tạo",
bgColor: "#D1FAE5", bgColor: "#F3F4F6", // Gray background
textColor: "#065F46", textColor: "#4B5563", // Gray text
icon: "checkmark-circle", icon: "document-text",
},
pending: {
label: "Chờ duyệt",
bgColor: "#FEF3C7", // Yellow background
textColor: "#92400E", // Dark yellow text
icon: "hourglass",
},
approved: {
label: "Đã duyệt",
bgColor: "#E0E7FF", // Indigo background
textColor: "#3730A3", // Dark indigo text
icon: "checkmark-done",
}, },
"in-progress": { "in-progress": {
label: "Đang diễn ra", label: "Đang hoạt động",
bgColor: "#DBEAFE", bgColor: "#DBEAFE", // Blue background
textColor: "#1E40AF", textColor: "#1E40AF", // Dark blue text
icon: "time", icon: "sync",
},
completed: {
label: "Hoàn thành",
bgColor: "#D1FAE5", // Green background
textColor: "#065F46", // Dark green text
icon: "checkmark-circle",
}, },
cancelled: { cancelled: {
label: "Đã hủy", label: "Đã hủy",
bgColor: "#FEE2E2", bgColor: "#FEE2E2", // Red background
textColor: "#991B1B", textColor: "#991B1B", // Dark red text
icon: "close-circle", icon: "close-circle",
}, },
"quality-check": {
label: "Khảo sát địa chất",
bgColor: "#D1FAE5",
textColor: "#065F46",
icon: "checkmark-circle",
},
} as const; } as const;

View File

@@ -31,7 +31,7 @@ const MAPPING = {
xmark: "close", xmark: "close",
pencil: "edit", pencil: "edit",
trash: "delete", trash: "delete",
"square.stack.3d.up": "layers", "square.stack.3d.up.fill": "layers",
"bell.fill": "notifications", "bell.fill": "notifications",
} as IconMapping; } as IconMapping;

View File

@@ -213,12 +213,13 @@ declare namespace Model {
vn_law: boolean; vn_law: boolean;
} }
// Seagateway Owner Appp // Seagateway Owner App
// Thing
interface SearchThingBody { interface SearchThingBody {
offset?: number; offset?: number;
limit?: number; limit?: number;
order?: string; order?: string;
sort?: "asc" | "desc"; dir?: "asc" | "desc";
name?: string; name?: string;
metadata?: any; metadata?: any;
} }

31
state/use-thing.ts Normal file
View File

@@ -0,0 +1,31 @@
import { querySearchThings } from "@/controller/DeviceController";
import { create } from "zustand";
type ThingState = {
things: Model.Thing[] | null;
getThings: (body: Model.SearchThingBody) => Promise<void>;
error: string | null;
loading?: boolean;
};
export const useThings = create<ThingState>((set) => ({
things: null,
getThings: async (body: Model.SearchThingBody) => {
set({ loading: true, error: null });
try {
const response = await querySearchThings(body);
console.log("Things fetching API: ", response.data.things?.length);
set({ things: response.data.things ?? [], loading: false });
} catch (error) {
console.error("Error when fetch things: ", error);
set({
error: "Failed to fetch things data",
loading: false,
things: null,
});
}
},
error: null,
loading: false,
}));