Compare commits
1 Commits
amalia/10-
...
v1
| Author | SHA1 | Date | |
|---|---|---|---|
| 54a12669e9 |
86
GEMINI.md
@@ -1,86 +0,0 @@
|
|||||||
# Project Overview: Desa+
|
|
||||||
|
|
||||||
Desa+ is a mobile application built with React Native and Expo, designed to facilitate management and communication within villages/communities. It aims to streamline village administration, inter-community communication, and the management of essential information.
|
|
||||||
|
|
||||||
## Key Features:
|
|
||||||
- Village announcements and information
|
|
||||||
- Community discussion forum
|
|
||||||
- Village activity calendar
|
|
||||||
- Village documentation and archives
|
|
||||||
- Project and task management
|
|
||||||
- Member and organizational structure management
|
|
||||||
- Push notifications for important updates
|
|
||||||
- Verification and authentication features
|
|
||||||
|
|
||||||
## Technologies Used:
|
|
||||||
- **React Native**: Cross-platform mobile development framework.
|
|
||||||
- **Expo**: Platform for React Native application development.
|
|
||||||
- **Firebase**: Backend services including Authentication, Realtime Database, and Cloud Messaging.
|
|
||||||
- **Redux Toolkit**: State management.
|
|
||||||
- **React Navigation**: Application navigation.
|
|
||||||
- **TypeScript**: For type safety.
|
|
||||||
|
|
||||||
## Building and Running:
|
|
||||||
|
|
||||||
### Installation
|
|
||||||
1. **Clone the repository:**
|
|
||||||
```bash
|
|
||||||
git clone <repository-url>
|
|
||||||
cd mobile-darmasaba
|
|
||||||
```
|
|
||||||
2. **Install dependencies:**
|
|
||||||
```bash
|
|
||||||
npm install
|
|
||||||
```
|
|
||||||
3. **Configure environment variables:**
|
|
||||||
Create a `.env` file in the root directory and add the following variables:
|
|
||||||
```
|
|
||||||
URL_API=<api-endpoint>
|
|
||||||
URL_OTP=<otp-service-endpoint>
|
|
||||||
URL_STORAGE=<storage-endpoint>
|
|
||||||
URL_FIREBASE_DB=<firebase-database-url>
|
|
||||||
PASS_ENC=<encryption-password>
|
|
||||||
WA_SERVER_TOKEN=<whatsapp-server-token>
|
|
||||||
IOS_GOOGLE_SERVICES_FILE=<path-to-ios-google-services>
|
|
||||||
```
|
|
||||||
|
|
||||||
### Running the Application
|
|
||||||
- **Start development server:**
|
|
||||||
```bash
|
|
||||||
npx expo start
|
|
||||||
```
|
|
||||||
- **Run on Android emulator/device:**
|
|
||||||
```bash
|
|
||||||
npm run android
|
|
||||||
```
|
|
||||||
- **Run on iOS simulator/device:**
|
|
||||||
```bash
|
|
||||||
npm run ios
|
|
||||||
```
|
|
||||||
|
|
||||||
### Build Production
|
|
||||||
- **Build Android production package:**
|
|
||||||
```bash
|
|
||||||
npm run build:android
|
|
||||||
```
|
|
||||||
|
|
||||||
## Development Conventions:
|
|
||||||
|
|
||||||
### Project Structure:
|
|
||||||
- `app/`: Main page files.
|
|
||||||
- `components/`: Reusable UI components, categorized by feature (e.g., `announcement/`, `auth/`, `discussion/`).
|
|
||||||
- `assets/`: Images and static assets.
|
|
||||||
- `constants/`: Global constants.
|
|
||||||
- `lib/`: Libraries and utilities.
|
|
||||||
|
|
||||||
### Contribution Guidelines:
|
|
||||||
1. Fork the repository.
|
|
||||||
2. Create a new feature branch (`git checkout -b feature/FeatureName`).
|
|
||||||
3. Commit your changes (`git commit -m 'Add FeatureName feature'`).
|
|
||||||
4. Push to the branch (`git push origin feature/FeatureName`).
|
|
||||||
5. Create a pull request.
|
|
||||||
|
|
||||||
## Platform Support:
|
|
||||||
- ✅ Android
|
|
||||||
- ✅ iOS
|
|
||||||
- ❌ Web (not yet optimized)
|
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
# Desa+
|
# Desa+
|
||||||
|
|
||||||
Desa+ (Desa Plus) adalah aplikasi mobile berbasis React Native yang dikembangkan dengan Expo untuk membantu pengelolaan dan komunikasi di lingkungan desa/kelurahan. Aplikasi ini menyediakan berbagai fitur untuk memudahkan administrasi desa, komunikasi antar warga, dan pengelolaan informasi penting.
|
Desa+ adalah aplikasi mobile berbasis React Native yang dikembangkan dengan Expo untuk membantu pengelolaan dan komunikasi di lingkungan desa/kelurahan. Aplikasi ini menyediakan berbagai fitur untuk memudahkan administrasi desa, komunikasi antar warga, dan pengelolaan informasi penting.
|
||||||
|
|
||||||
## Fitur Utama
|
## Fitur Utama
|
||||||
|
|
||||||
|
|||||||
@@ -92,8 +92,8 @@ android {
|
|||||||
applicationId 'mobiledarmasaba.app'
|
applicationId 'mobiledarmasaba.app'
|
||||||
minSdkVersion rootProject.ext.minSdkVersion
|
minSdkVersion rootProject.ext.minSdkVersion
|
||||||
targetSdkVersion rootProject.ext.targetSdkVersion
|
targetSdkVersion rootProject.ext.targetSdkVersion
|
||||||
versionCode 16
|
versionCode 6
|
||||||
versionName "2.1.0"
|
versionName "1.0.2"
|
||||||
}
|
}
|
||||||
signingConfigs {
|
signingConfigs {
|
||||||
debug {
|
debug {
|
||||||
|
|||||||
|
Before Width: | Height: | Size: 6.2 KiB After Width: | Height: | Size: 7.1 KiB |
|
Before Width: | Height: | Size: 3.9 KiB After Width: | Height: | Size: 4.6 KiB |
|
Before Width: | Height: | Size: 9.0 KiB After Width: | Height: | Size: 10 KiB |
|
Before Width: | Height: | Size: 16 KiB After Width: | Height: | Size: 18 KiB |
|
Before Width: | Height: | Size: 23 KiB After Width: | Height: | Size: 28 KiB |
|
Before Width: | Height: | Size: 1.3 KiB After Width: | Height: | Size: 1.6 KiB |
|
Before Width: | Height: | Size: 2.7 KiB After Width: | Height: | Size: 3.5 KiB |
|
Before Width: | Height: | Size: 1.9 KiB After Width: | Height: | Size: 2.2 KiB |
|
Before Width: | Height: | Size: 904 B After Width: | Height: | Size: 1.1 KiB |
|
Before Width: | Height: | Size: 1.8 KiB After Width: | Height: | Size: 2.4 KiB |
|
Before Width: | Height: | Size: 1.3 KiB After Width: | Height: | Size: 1.6 KiB |
|
Before Width: | Height: | Size: 1.7 KiB After Width: | Height: | Size: 2.1 KiB |
|
Before Width: | Height: | Size: 3.7 KiB After Width: | Height: | Size: 4.9 KiB |
|
Before Width: | Height: | Size: 2.6 KiB After Width: | Height: | Size: 3.0 KiB |
|
Before Width: | Height: | Size: 2.4 KiB After Width: | Height: | Size: 3.1 KiB |
|
Before Width: | Height: | Size: 5.9 KiB After Width: | Height: | Size: 8.1 KiB |
|
Before Width: | Height: | Size: 3.8 KiB After Width: | Height: | Size: 4.5 KiB |
|
Before Width: | Height: | Size: 3.3 KiB After Width: | Height: | Size: 4.4 KiB |
|
Before Width: | Height: | Size: 8.6 KiB After Width: | Height: | Size: 12 KiB |
|
Before Width: | Height: | Size: 5.2 KiB After Width: | Height: | Size: 6.3 KiB |
@@ -1,9 +1,9 @@
|
|||||||
<resources xmlns:tools="http://schemas.android.com/tools">
|
<resources xmlns:tools="http://schemas.android.com/tools">
|
||||||
<style name="AppTheme" parent="Theme.AppCompat.DayNight.NoActionBar">
|
<style name="AppTheme" parent="Theme.AppCompat.DayNight.NoActionBar">
|
||||||
<item name="android:editTextBackground">@drawable/rn_edit_text_material</item>
|
<item name="android:editTextBackground">@drawable/rn_edit_text_material</item>
|
||||||
|
<item name="android:windowOptOutEdgeToEdgeEnforcement" tools:targetApi="35">true</item>
|
||||||
<item name="colorPrimary">@color/colorPrimary</item>
|
<item name="colorPrimary">@color/colorPrimary</item>
|
||||||
<item name="android:statusBarColor">#ffffff</item>
|
<item name="android:statusBarColor">#ffffff</item>
|
||||||
<item name="android:windowOptOutEdgeToEdgeEnforcement" tools:targetApi="35">true</item>
|
|
||||||
</style>
|
</style>
|
||||||
<style name="Theme.App.SplashScreen" parent="Theme.SplashScreen">
|
<style name="Theme.App.SplashScreen" parent="Theme.SplashScreen">
|
||||||
<item name="windowSplashScreenBackground">@color/splashscreen_background</item>
|
<item name="windowSplashScreenBackground">@color/splashscreen_background</item>
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ export default {
|
|||||||
expo: {
|
expo: {
|
||||||
name: "Desa+",
|
name: "Desa+",
|
||||||
slug: "mobile-darmasaba",
|
slug: "mobile-darmasaba",
|
||||||
version: "2.1.0", // Versi aplikasi (App Store)
|
version: "2.0.5", // Versi aplikasi (App Store)
|
||||||
jsEngine: "jsc",
|
jsEngine: "jsc",
|
||||||
orientation: "portrait",
|
orientation: "portrait",
|
||||||
icon: "./assets/images/logo-icon-small.png",
|
icon: "./assets/images/logo-icon-small.png",
|
||||||
@@ -14,7 +14,7 @@ export default {
|
|||||||
ios: {
|
ios: {
|
||||||
supportsTablet: true,
|
supportsTablet: true,
|
||||||
bundleIdentifier: "mobiledarmasaba.app",
|
bundleIdentifier: "mobiledarmasaba.app",
|
||||||
buildNumber: "9",
|
buildNumber: "7",
|
||||||
infoPlist: {
|
infoPlist: {
|
||||||
ITSAppUsesNonExemptEncryption: false,
|
ITSAppUsesNonExemptEncryption: false,
|
||||||
CFBundleDisplayName: "Desa+"
|
CFBundleDisplayName: "Desa+"
|
||||||
@@ -23,7 +23,7 @@ export default {
|
|||||||
},
|
},
|
||||||
android: {
|
android: {
|
||||||
package: "mobiledarmasaba.app",
|
package: "mobiledarmasaba.app",
|
||||||
versionCode: 16,
|
versionCode: 15,
|
||||||
adaptiveIcon: {
|
adaptiveIcon: {
|
||||||
foregroundImage: "./assets/images/logo-icon-small.png",
|
foregroundImage: "./assets/images/logo-icon-small.png",
|
||||||
backgroundColor: "#ffffff"
|
backgroundColor: "#ffffff"
|
||||||
@@ -79,12 +79,6 @@ export default {
|
|||||||
URL_FIREBASE_DB: process.env.URL_FIREBASE_DB,
|
URL_FIREBASE_DB: process.env.URL_FIREBASE_DB,
|
||||||
PASS_ENC: process.env.PASS_ENC,
|
PASS_ENC: process.env.PASS_ENC,
|
||||||
WA_SERVER_TOKEN: process.env.WA_SERVER_TOKEN,
|
WA_SERVER_TOKEN: process.env.WA_SERVER_TOKEN,
|
||||||
FIREBASE_API_KEY: process.env.FIREBASE_API_KEY,
|
|
||||||
FIREBASE_AUTH_DOMAIN: process.env.FIREBASE_AUTH_DOMAIN,
|
|
||||||
FIREBASE_PROJECT_ID: process.env.FIREBASE_PROJECT_ID,
|
|
||||||
FIREBASE_STORAGE_BUCKET: process.env.FIREBASE_STORAGE_BUCKET,
|
|
||||||
FIREBASE_MESSAGING_SENDER_ID: process.env.FIREBASE_MESSAGING_SENDER_ID,
|
|
||||||
FIREBASE_APP_ID: process.env.FIREBASE_APP_ID,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
import HeaderRightAnnouncementList from "@/components/announcement/headerAnnouncementList";
|
import HeaderRightAnnouncementList from "@/components/announcement/headerAnnouncementList";
|
||||||
import AppHeader from "@/components/AppHeader";
|
import AppHeader from "@/components/AppHeader";
|
||||||
import Styles from "@/constants/Styles";
|
|
||||||
import HeaderDiscussionGeneral from "@/components/discussion_general/headerDiscussionGeneral";
|
import HeaderDiscussionGeneral from "@/components/discussion_general/headerDiscussionGeneral";
|
||||||
import HeaderRightDivisionList from "@/components/division/headerDivisionList";
|
import HeaderRightDivisionList from "@/components/division/headerDivisionList";
|
||||||
import HeaderRightGroupList from "@/components/group/headerGroupList";
|
import HeaderRightGroupList from "@/components/group/headerGroupList";
|
||||||
@@ -9,100 +8,22 @@ import HeaderRightPositionList from "@/components/position/headerRightPositionLi
|
|||||||
import HeaderRightProjectList from "@/components/project/headerProjectList";
|
import HeaderRightProjectList from "@/components/project/headerProjectList";
|
||||||
import Text from "@/components/Text";
|
import Text from "@/components/Text";
|
||||||
import ToastCustom from "@/components/toastCustom";
|
import ToastCustom from "@/components/toastCustom";
|
||||||
import ModalUpdateMaintenance from "@/components/ModalUpdateMaintenance";
|
import { apiReadOneNotification } from "@/lib/api";
|
||||||
import { apiGetVersion, apiReadOneNotification } from "@/lib/api";
|
|
||||||
import { pushToPage } from "@/lib/pushToPage";
|
import { pushToPage } from "@/lib/pushToPage";
|
||||||
import store from "@/lib/store";
|
import store from "@/lib/store";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import AsyncStorage from "@react-native-async-storage/async-storage";
|
import AsyncStorage from "@react-native-async-storage/async-storage";
|
||||||
import Constants from "expo-constants";
|
|
||||||
import { getApp } from "@react-native-firebase/app";
|
import { getApp } from "@react-native-firebase/app";
|
||||||
import { getMessaging, onMessage } from "@react-native-firebase/messaging";
|
import { getMessaging, onMessage } from "@react-native-firebase/messaging";
|
||||||
import { Redirect, router, Stack, usePathname } from "expo-router";
|
import { Redirect, router, Stack, usePathname } from "expo-router";
|
||||||
import { StatusBar } from 'expo-status-bar';
|
import { StatusBar } from 'expo-status-bar';
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect } from "react";
|
||||||
import { Easing, Notifier, NotifierComponents } from 'react-native-notifier';
|
import { Easing, Notifier } from 'react-native-notifier';
|
||||||
import { Provider } from "react-redux";
|
import { Provider } from "react-redux";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
|
|
||||||
export default function RootLayout() {
|
export default function RootLayout() {
|
||||||
const { token, decryptToken, isLoading } = useAuthSession()
|
const { token, decryptToken, isLoading } = useAuthSession()
|
||||||
const pathname = usePathname()
|
const pathname = usePathname()
|
||||||
const { colors } = useTheme()
|
|
||||||
const [modalUpdateMaintenance, setModalUpdateMaintenance] = useState(false)
|
|
||||||
const [modalType, setModalType] = useState<'update' | 'maintenance'>('update')
|
|
||||||
const [isForceUpdate, setIsForceUpdate] = useState(false)
|
|
||||||
const [updateMessage, setUpdateMessage] = useState('')
|
|
||||||
|
|
||||||
const currentVersion = Constants.expoConfig?.version ?? '0.0.0'
|
|
||||||
|
|
||||||
const compareVersions = (v1: string, v2: string) => {
|
|
||||||
const parts1 = v1.split('.').map(Number);
|
|
||||||
const parts2 = v2.split('.').map(Number);
|
|
||||||
for (let i = 0; i < Math.max(parts1.length, parts2.length); i++) {
|
|
||||||
const p1 = parts1[i] || 0;
|
|
||||||
const p2 = parts2[i] || 0;
|
|
||||||
if (p1 < p2) return -1;
|
|
||||||
if (p1 > p2) return 1;
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
};
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
const checkVersion = async () => {
|
|
||||||
try {
|
|
||||||
const response = await apiGetVersion();
|
|
||||||
if (response.success && response.data) {
|
|
||||||
const maintenance = response.data.find((item: any) => item.id === 'mobile_maintenance')?.value === 'true';
|
|
||||||
const latestVersion = response.data.find((item: any) => item.id === 'mobile_latest_version')?.value || '0.0.0';
|
|
||||||
const minVersion = response.data.find((item: any) => item.id === 'mobile_minimum_version')?.value || '0.0.0';
|
|
||||||
const message = response.data.find((item: any) => item.id === 'mobile_message_update')?.value || '';
|
|
||||||
|
|
||||||
if (maintenance) {
|
|
||||||
setModalType('maintenance');
|
|
||||||
setModalUpdateMaintenance(true);
|
|
||||||
setIsForceUpdate(true);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (compareVersions(currentVersion, minVersion) === -1) {
|
|
||||||
setModalType('update');
|
|
||||||
setIsForceUpdate(true);
|
|
||||||
setUpdateMessage(message);
|
|
||||||
setModalUpdateMaintenance(true);
|
|
||||||
} else if (compareVersions(currentVersion, latestVersion) === -1) {
|
|
||||||
// Check if this soft update version was already dismissed
|
|
||||||
const dismissedVersion = await AsyncStorage.getItem('dismissed_update_version');
|
|
||||||
if (dismissedVersion !== latestVersion) {
|
|
||||||
setModalType('update');
|
|
||||||
setIsForceUpdate(false);
|
|
||||||
setUpdateMessage(message);
|
|
||||||
setModalUpdateMaintenance(true);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
console.error('Failed to check version:', error);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
checkVersion();
|
|
||||||
}, [currentVersion]);
|
|
||||||
|
|
||||||
const handleDismissUpdate = async () => {
|
|
||||||
if (!isForceUpdate) {
|
|
||||||
try {
|
|
||||||
const response = await apiGetVersion();
|
|
||||||
const latestVersion = response.data.find((item: any) => item.id === 'mobile_latest_version')?.value;
|
|
||||||
if (latestVersion) {
|
|
||||||
await AsyncStorage.setItem('dismissed_update_version', latestVersion);
|
|
||||||
}
|
|
||||||
} catch (e) {
|
|
||||||
console.error(e);
|
|
||||||
}
|
|
||||||
setModalUpdateMaintenance(false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async function handleReadNotification(id: string, category: string, idContent: string, title: string) {
|
async function handleReadNotification(id: string, category: string, idContent: string, title: string) {
|
||||||
try {
|
try {
|
||||||
@@ -144,34 +65,12 @@ export default function RootLayout() {
|
|||||||
} else if (pathname !== `/${category}/${content}`) {
|
} else if (pathname !== `/${category}/${content}`) {
|
||||||
Notifier.showNotification({
|
Notifier.showNotification({
|
||||||
title: title,
|
title: title,
|
||||||
description: String(remoteMessage.notification?.body),
|
description: remoteMessage.notification?.body,
|
||||||
duration: 3000,
|
duration: 3000,
|
||||||
animationDuration: 300,
|
animationDuration: 300,
|
||||||
showEasing: Easing.ease,
|
showEasing: Easing.ease,
|
||||||
onPress: () => handleReadNotification(String(id), String(category), String(content), String(title)),
|
onPress: () => handleReadNotification(String(id), String(category), String(content), String(title)),
|
||||||
hideOnPress: true,
|
hideOnPress: true,
|
||||||
Component: NotifierComponents.Notification,
|
|
||||||
componentProps: {
|
|
||||||
containerStyle: [
|
|
||||||
Styles.shadowBox,
|
|
||||||
{
|
|
||||||
backgroundColor: colors.modalBackground,
|
|
||||||
borderRadius: 5,
|
|
||||||
marginHorizontal: 15,
|
|
||||||
marginTop: 10,
|
|
||||||
padding: 15,
|
|
||||||
}
|
|
||||||
],
|
|
||||||
titleStyle: {
|
|
||||||
color: colors.text,
|
|
||||||
fontWeight: 'bold',
|
|
||||||
fontSize: 16,
|
|
||||||
},
|
|
||||||
descriptionStyle: {
|
|
||||||
color: colors.dimmed,
|
|
||||||
fontSize: 14,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -194,10 +93,11 @@ export default function RootLayout() {
|
|||||||
<Stack screenOptions={{
|
<Stack screenOptions={{
|
||||||
headerShown: true,
|
headerShown: true,
|
||||||
animation: "slide_from_right",
|
animation: "slide_from_right",
|
||||||
|
|
||||||
|
// ⬇️ PENTING BANGET
|
||||||
animationTypeForReplace: "pop",
|
animationTypeForReplace: "pop",
|
||||||
fullScreenGestureEnabled: true,
|
fullScreenGestureEnabled: true,
|
||||||
gestureEnabled: true,
|
gestureEnabled: true,
|
||||||
contentStyle: { backgroundColor: colors.header },
|
|
||||||
}} >
|
}} >
|
||||||
<Stack.Screen name="home" options={{ title: 'Home' }} />
|
<Stack.Screen name="home" options={{ title: 'Home' }} />
|
||||||
<Stack.Screen name="feature" options={{ title: 'Fitur' }} />
|
<Stack.Screen name="feature" options={{ title: 'Fitur' }} />
|
||||||
@@ -212,18 +112,6 @@ export default function RootLayout() {
|
|||||||
)
|
)
|
||||||
}} />
|
}} />
|
||||||
<Stack.Screen name="profile" options={{ title: 'Profile' }} />
|
<Stack.Screen name="profile" options={{ title: 'Profile' }} />
|
||||||
<Stack.Screen name="setting/index" options={{
|
|
||||||
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
|
||||||
title: 'Pengaturan',
|
|
||||||
headerTitleAlign: 'center',
|
|
||||||
// headerRight: () => <HeaderRightProjectList />
|
|
||||||
header: () => (
|
|
||||||
<AppHeader title="Pengaturan"
|
|
||||||
showBack={true}
|
|
||||||
onPressLeft={() => router.back()}
|
|
||||||
/>
|
|
||||||
)
|
|
||||||
}} />
|
|
||||||
<Stack.Screen name="member/index" options={{
|
<Stack.Screen name="member/index" options={{
|
||||||
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
||||||
title: 'Anggota',
|
title: 'Anggota',
|
||||||
@@ -322,13 +210,6 @@ export default function RootLayout() {
|
|||||||
</Stack>
|
</Stack>
|
||||||
<StatusBar style={'light'} translucent={false} backgroundColor="black" />
|
<StatusBar style={'light'} translucent={false} backgroundColor="black" />
|
||||||
<ToastCustom />
|
<ToastCustom />
|
||||||
<ModalUpdateMaintenance
|
|
||||||
visible={modalUpdateMaintenance}
|
|
||||||
type={modalType}
|
|
||||||
isForceUpdate={isForceUpdate}
|
|
||||||
customDescription={updateMessage}
|
|
||||||
onDismiss={handleDismissUpdate}
|
|
||||||
/>
|
|
||||||
</Provider>
|
</Provider>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,7 +8,6 @@ import { isImageFile } from "@/constants/FileExtensions";
|
|||||||
import Styles from "@/constants/Styles";
|
import Styles from "@/constants/Styles";
|
||||||
import { apiGetAnnouncementOne } from "@/lib/api";
|
import { apiGetAnnouncementOne } from "@/lib/api";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { Entypo, MaterialCommunityIcons, MaterialIcons } from "@expo/vector-icons";
|
import { Entypo, MaterialCommunityIcons, MaterialIcons } from "@expo/vector-icons";
|
||||||
import * as FileSystem from 'expo-file-system';
|
import * as FileSystem from 'expo-file-system';
|
||||||
import { startActivityAsync } from 'expo-intent-launcher';
|
import { startActivityAsync } from 'expo-intent-launcher';
|
||||||
@@ -52,7 +51,6 @@ interface ApiResponse {
|
|||||||
export default function DetailAnnouncement() {
|
export default function DetailAnnouncement() {
|
||||||
const { id } = useLocalSearchParams<{ id: string }>();
|
const { id } = useLocalSearchParams<{ id: string }>();
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
const { colors } = useTheme();
|
|
||||||
const [data, setData] = useState<AnnouncementData>({ id: '', title: '', desc: '' })
|
const [data, setData] = useState<AnnouncementData>({ id: '', title: '', desc: '' })
|
||||||
const [dataMember, setDataMember] = useState<Record<string, MemberData[]>>({})
|
const [dataMember, setDataMember] = useState<Record<string, MemberData[]>>({})
|
||||||
const [dataFile, setDataFile] = useState<FileData[]>([])
|
const [dataFile, setDataFile] = useState<FileData[]>([])
|
||||||
@@ -89,11 +87,9 @@ export default function DetailAnnouncement() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message })
|
Toast.show({ type: 'small', text1: response.message })
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal mengambil data"
|
Toast.show({ type: 'small', text1: 'Gagal mengambil data' })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
@@ -179,7 +175,7 @@ export default function DetailAnnouncement() {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={[Styles.flex1, { backgroundColor: colors.background }]}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
||||||
@@ -197,23 +193,22 @@ export default function DetailAnnouncement() {
|
|||||||
/>
|
/>
|
||||||
<ScrollView
|
<ScrollView
|
||||||
showsVerticalScrollIndicator={false}
|
showsVerticalScrollIndicator={false}
|
||||||
style={[Styles.h100, { backgroundColor: colors.background }]}
|
style={[Styles.h100]}
|
||||||
refreshControl={
|
refreshControl={
|
||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={() => handleRefresh()}
|
onRefresh={() => handleRefresh()}
|
||||||
tintColor={colors.icon}
|
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
<View style={[Styles.p15, Styles.mb50]}>
|
<View style={[Styles.p15, Styles.mb50]}>
|
||||||
<View style={[Styles.wrapPaper, Styles.borderAll, { backgroundColor: colors.card, borderColor: colors.icon + '20' }]}>
|
<View style={[Styles.wrapPaper]}>
|
||||||
{
|
{
|
||||||
loading ?
|
loading ?
|
||||||
<View>
|
<View>
|
||||||
<View style={[Styles.rowOnly]}>
|
<View style={[Styles.rowOnly]}>
|
||||||
<Skeleton width={30} height={30} borderRadius={10} />
|
<Skeleton width={30} height={30} borderRadius={10} />
|
||||||
<View style={[Styles.flex1, Styles.ph05]}>
|
<View style={[{ flex: 1 }, Styles.ph05]}>
|
||||||
<Skeleton width={100} widthType="percent" height={30} borderRadius={10} />
|
<Skeleton width={100} widthType="percent" height={30} borderRadius={10} />
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
@@ -223,8 +218,8 @@ export default function DetailAnnouncement() {
|
|||||||
</View>
|
</View>
|
||||||
:
|
:
|
||||||
<>
|
<>
|
||||||
<View style={[Styles.rowOnly, Styles.alignStart]}>
|
<View style={[Styles.rowItemsCenter, { alignItems: 'flex-start' }]}>
|
||||||
<MaterialIcons name="campaign" size={25} color={colors.text} style={[Styles.mr05]} />
|
<MaterialIcons name="campaign" size={25} color="black" style={[Styles.mr05]} />
|
||||||
<Text style={[Styles.textDefaultSemiBold, Styles.w90, Styles.mt02]}>{data?.title}</Text>
|
<Text style={[Styles.textDefaultSemiBold, Styles.w90, Styles.mt02]}>{data?.title}</Text>
|
||||||
</View>
|
</View>
|
||||||
<View style={[Styles.mt10]}>
|
<View style={[Styles.mt10]}>
|
||||||
@@ -233,7 +228,7 @@ export default function DetailAnnouncement() {
|
|||||||
<RenderHTML
|
<RenderHTML
|
||||||
contentWidth={contentWidth}
|
contentWidth={contentWidth}
|
||||||
source={{ html: data?.desc }}
|
source={{ html: data?.desc }}
|
||||||
baseStyle={{ color: colors.text }}
|
baseStyle={{ color: 'black' }}
|
||||||
/>
|
/>
|
||||||
:
|
:
|
||||||
<Text>{data?.desc}</Text>
|
<Text>{data?.desc}</Text>
|
||||||
@@ -245,18 +240,18 @@ export default function DetailAnnouncement() {
|
|||||||
</View>
|
</View>
|
||||||
{
|
{
|
||||||
dataFile.length > 0 && (
|
dataFile.length > 0 && (
|
||||||
<View style={[Styles.wrapPaper, Styles.borderAll, Styles.mt10, { backgroundColor: colors.card, borderColor: colors.icon + '20' }]}>
|
<View style={[Styles.wrapPaper, Styles.mt10]}>
|
||||||
<View style={[Styles.mb05]}>
|
<View style={[Styles.mb05]}>
|
||||||
<Text style={[Styles.textDefaultSemiBold]}>File</Text>
|
<Text style={[Styles.textDefaultSemiBold]}>File</Text>
|
||||||
</View>
|
</View>
|
||||||
{dataFile.map((item, index) => (
|
{dataFile.map((item, index) => (
|
||||||
<BorderBottomItem
|
<BorderBottomItem
|
||||||
key={`${item.id}-${index}`}
|
key={`${item.id}-${index}`}
|
||||||
borderType={index === dataFile.length - 1 ? 'none' : 'bottom'}
|
borderType="bottom"
|
||||||
icon={<MaterialCommunityIcons
|
icon={<MaterialCommunityIcons
|
||||||
name={isImageFile(item.extension) ? "file-image-outline" : "file-document-outline"}
|
name={isImageFile(item.extension) ? "file-image-outline" : "file-document-outline"}
|
||||||
size={25}
|
size={25}
|
||||||
color={colors.text}
|
color="black"
|
||||||
/>}
|
/>}
|
||||||
title={item.name + '.' + item.extension}
|
title={item.name + '.' + item.extension}
|
||||||
titleWeight="normal"
|
titleWeight="normal"
|
||||||
@@ -270,7 +265,7 @@ export default function DetailAnnouncement() {
|
|||||||
</View>
|
</View>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
<View style={[Styles.wrapPaper, Styles.borderAll, Styles.mt10, { backgroundColor: colors.card, borderColor: colors.icon + '20' }]}>
|
<View style={[Styles.wrapPaper, Styles.mt10]}>
|
||||||
{
|
{
|
||||||
loading ?
|
loading ?
|
||||||
arrSkeleton.map((item, index) => {
|
arrSkeleton.map((item, index) => {
|
||||||
@@ -291,7 +286,7 @@ export default function DetailAnnouncement() {
|
|||||||
dataMember[v].map((item: any, x: any) => {
|
dataMember[v].map((item: any, x: any) => {
|
||||||
return (
|
return (
|
||||||
<View key={x} style={[Styles.rowItemsCenter, Styles.w90]}>
|
<View key={x} style={[Styles.rowItemsCenter, Styles.w90]}>
|
||||||
<Entypo name="dot-single" size={24} color={colors.text} />
|
<Entypo name="dot-single" size={24} color="black" />
|
||||||
<Text style={[Styles.textDefault]} numberOfLines={1} ellipsizeMode='tail'>{item.division}</Text>
|
<Text style={[Styles.textDefault]} numberOfLines={1} ellipsizeMode='tail'>{item.division}</Text>
|
||||||
</View>
|
</View>
|
||||||
)
|
)
|
||||||
@@ -320,7 +315,7 @@ export default function DetailAnnouncement() {
|
|||||||
accessibilityRole="button"
|
accessibilityRole="button"
|
||||||
accessibilityLabel="Close image viewer"
|
accessibilityLabel="Close image viewer"
|
||||||
>
|
>
|
||||||
<Text style={[Styles.textWhite, Styles.font26]}>✕</Text>
|
<Text style={{ color: 'white', fontSize: 26 }}>✕</Text>
|
||||||
</Pressable>
|
</Pressable>
|
||||||
|
|
||||||
{/* MENU */}
|
{/* MENU */}
|
||||||
@@ -330,17 +325,17 @@ export default function DetailAnnouncement() {
|
|||||||
accessibilityLabel="Download or share image"
|
accessibilityLabel="Download or share image"
|
||||||
disabled={loadingOpen}
|
disabled={loadingOpen}
|
||||||
>
|
>
|
||||||
<Text style={[{ color: loadingOpen ? 'gray' : 'white' }, Styles.font26]}>⋯</Text>
|
<Text style={{ color: loadingOpen ? 'gray' : 'white', fontSize: 22 }}>⋯</Text>
|
||||||
</Pressable>
|
</Pressable>
|
||||||
</View>
|
</View>
|
||||||
)}
|
)}
|
||||||
FooterComponent={({ imageIndex }) => (
|
FooterComponent={({ imageIndex }) => (
|
||||||
<View style={[
|
<View style={{
|
||||||
Styles.pb20,
|
paddingBottom: 20,
|
||||||
Styles.ph16,
|
paddingHorizontal: 16,
|
||||||
Styles.alignCenter,
|
alignItems: 'center',
|
||||||
]}>
|
}}>
|
||||||
<Text style={[Styles.textWhite, Styles.font16]}>{chooseFile?.name}.{chooseFile?.extension}</Text>
|
<Text style={{ color: 'white', fontSize: 16 }}>{chooseFile?.name}.{chooseFile?.extension}</Text>
|
||||||
</View>
|
</View>
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ import ButtonSaveHeader from "@/components/buttonSaveHeader";
|
|||||||
import ButtonSelect from "@/components/buttonSelect";
|
import ButtonSelect from "@/components/buttonSelect";
|
||||||
import DrawerBottom from "@/components/drawerBottom";
|
import DrawerBottom from "@/components/drawerBottom";
|
||||||
import { InputForm } from "@/components/inputForm";
|
import { InputForm } from "@/components/inputForm";
|
||||||
import LoadingCenter from "@/components/loadingCenter";
|
import LoadingOverlay from "@/components/loadingOverlay";
|
||||||
import MenuItemRow from "@/components/menuItemRow";
|
import MenuItemRow from "@/components/menuItemRow";
|
||||||
import ModalSelectMultiple from "@/components/modalSelectMultiple";
|
import ModalSelectMultiple from "@/components/modalSelectMultiple";
|
||||||
import Text from "@/components/Text";
|
import Text from "@/components/Text";
|
||||||
@@ -12,12 +12,11 @@ import Styles from "@/constants/Styles";
|
|||||||
import { setUpdateAnnouncement } from "@/lib/announcementUpdate";
|
import { setUpdateAnnouncement } from "@/lib/announcementUpdate";
|
||||||
import { apiCreateAnnouncement } from "@/lib/api";
|
import { apiCreateAnnouncement } from "@/lib/api";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { Entypo, Ionicons, MaterialCommunityIcons } from "@expo/vector-icons";
|
import { Entypo, Ionicons, MaterialCommunityIcons } from "@expo/vector-icons";
|
||||||
import * as DocumentPicker from "expo-document-picker";
|
import * as DocumentPicker from "expo-document-picker";
|
||||||
import { router, Stack } from "expo-router";
|
import { router, Stack } from "expo-router";
|
||||||
import React, { useEffect, useState } from "react";
|
import React, { useEffect, useState } from "react";
|
||||||
import { SafeAreaView, ScrollView, View } from "react-native";
|
import { SafeAreaView, ScrollView, StyleSheet, View } from "react-native";
|
||||||
import Toast from "react-native-toast-message";
|
import Toast from "react-native-toast-message";
|
||||||
import { useDispatch, useSelector } from "react-redux";
|
import { useDispatch, useSelector } from "react-redux";
|
||||||
|
|
||||||
@@ -25,7 +24,6 @@ export default function CreateAnnouncement() {
|
|||||||
const dispatch = useDispatch()
|
const dispatch = useDispatch()
|
||||||
const update = useSelector((state: any) => state.announcementUpdate)
|
const update = useSelector((state: any) => state.announcementUpdate)
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
const { colors } = useTheme();
|
|
||||||
const [disableBtn, setDisableBtn] = useState(true);
|
const [disableBtn, setDisableBtn] = useState(true);
|
||||||
const [modalDivisi, setModalDivisi] = useState(false);
|
const [modalDivisi, setModalDivisi] = useState(false);
|
||||||
const [divisionMember, setDivisionMember] = useState<any>([])
|
const [divisionMember, setDivisionMember] = useState<any>([])
|
||||||
@@ -95,21 +93,17 @@ export default function CreateAnnouncement() {
|
|||||||
|
|
||||||
const response = await apiCreateAnnouncement(fd)
|
const response = await apiCreateAnnouncement(fd)
|
||||||
|
|
||||||
|
// const response = await apiCreateAnnouncement({
|
||||||
|
// data: { ...dataForm, user: hasil, groups: divisionMember },
|
||||||
|
// });
|
||||||
|
|
||||||
if (response.success) {
|
if (response.success) {
|
||||||
dispatch(setUpdateAnnouncement(!update))
|
dispatch(setUpdateAnnouncement(!update))
|
||||||
Toast.show({ type: 'small', text1: 'Berhasil menambahkan data', })
|
Toast.show({ type: 'small', text1: 'Berhasil menambahkan data', })
|
||||||
router.back();
|
router.back();
|
||||||
} else {
|
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Tidak dapat terhubung ke server"
|
|
||||||
|
|
||||||
Toast.show({
|
|
||||||
type: 'small',
|
|
||||||
text1: message
|
|
||||||
})
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
@@ -135,7 +129,7 @@ export default function CreateAnnouncement() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={[Styles.flex1, { backgroundColor: colors.background }]}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => (
|
// headerLeft: () => (
|
||||||
@@ -164,24 +158,24 @@ export default function CreateAnnouncement() {
|
|||||||
showBack={true}
|
showBack={true}
|
||||||
onPressLeft={() => router.back()}
|
onPressLeft={() => router.back()}
|
||||||
right={
|
right={
|
||||||
<ButtonSaveHeader
|
<ButtonSaveHeader
|
||||||
disable={disableBtn || divisionMember.length == 0 || loading ? true : false}
|
disable={disableBtn || divisionMember.length == 0 || loading ? true : false}
|
||||||
category="create"
|
category="create"
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
divisionMember.length == 0
|
divisionMember.length == 0
|
||||||
? Toast.show({ type: 'small', text1: "Anda belum memilih divisi", })
|
? Toast.show({ type: 'small', text1: "Anda belum memilih divisi", })
|
||||||
: handleCreate();
|
: handleCreate();
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
{loading && <LoadingCenter />}
|
<LoadingOverlay visible={loading} />
|
||||||
<ScrollView
|
<ScrollView
|
||||||
showsVerticalScrollIndicator={false}
|
showsVerticalScrollIndicator={false}
|
||||||
style={[Styles.h100, { backgroundColor: colors.background }]}
|
style={[Styles.h100]}
|
||||||
>
|
>
|
||||||
<View style={[Styles.p15]}>
|
<View style={[Styles.p15]}>
|
||||||
<InputForm
|
<InputForm
|
||||||
@@ -190,7 +184,6 @@ export default function CreateAnnouncement() {
|
|||||||
placeholder="Judul Pengumuman"
|
placeholder="Judul Pengumuman"
|
||||||
required
|
required
|
||||||
error={error.title}
|
error={error.title}
|
||||||
bg={colors.card}
|
|
||||||
errorText="Judul harus diisi"
|
errorText="Judul harus diisi"
|
||||||
onChange={(val) => validationForm("title", val)}
|
onChange={(val) => validationForm("title", val)}
|
||||||
/>
|
/>
|
||||||
@@ -200,7 +193,6 @@ export default function CreateAnnouncement() {
|
|||||||
placeholder="Deskripsi Pengumuman"
|
placeholder="Deskripsi Pengumuman"
|
||||||
required
|
required
|
||||||
error={error.desc}
|
error={error.desc}
|
||||||
bg={colors.card}
|
|
||||||
errorText="Pengumuman harus diisi"
|
errorText="Pengumuman harus diisi"
|
||||||
onChange={(val) => validationForm("desc", val)}
|
onChange={(val) => validationForm("desc", val)}
|
||||||
multiline
|
multiline
|
||||||
@@ -209,27 +201,21 @@ export default function CreateAnnouncement() {
|
|||||||
{
|
{
|
||||||
fileForm.length > 0
|
fileForm.length > 0
|
||||||
&&
|
&&
|
||||||
<>
|
<View style={[Styles.borderAll, Styles.round10, Styles.p10, Styles.mb10]}>
|
||||||
<View style={[Styles.rowSpaceBetween, Styles.mv05]}>
|
<Text style={[Styles.textDefaultSemiBold]}>File</Text>
|
||||||
<Text style={[Styles.textDefaultSemiBold]}>File</Text>
|
{
|
||||||
<Text style={[Styles.textDefault]}>{fileForm.length} file</Text>
|
fileForm.map((item, index) => (
|
||||||
</View>
|
<BorderBottomItem
|
||||||
<View style={[Styles.borderAll, Styles.round05, Styles.p10, Styles.mb10, { backgroundColor: colors.card, borderColor: colors.icon + '20' }]}>
|
key={index}
|
||||||
{
|
borderType={fileForm.length > 1 ? "bottom" : "none"}
|
||||||
fileForm.map((item, index) => (
|
icon={<MaterialCommunityIcons name="file-outline" size={25} color="black" />}
|
||||||
<BorderBottomItem
|
title={item.name}
|
||||||
key={index}
|
titleWeight="normal"
|
||||||
borderType={fileForm.length - 1 == index ? "none" : "bottom"}
|
onPress={() => { setIndexDelFile(index); setModalFile(true) }}
|
||||||
icon={<MaterialCommunityIcons name="file-outline" size={25} color={colors.text} />}
|
/>
|
||||||
title={item.name}
|
))
|
||||||
bgColor="transparent"
|
}
|
||||||
titleWeight="normal"
|
</View>
|
||||||
onPress={() => { setIndexDelFile(index); setModalFile(true) }}
|
|
||||||
/>
|
|
||||||
))
|
|
||||||
}
|
|
||||||
</View>
|
|
||||||
</>
|
|
||||||
}
|
}
|
||||||
|
|
||||||
<ButtonSelect
|
<ButtonSelect
|
||||||
@@ -242,30 +228,25 @@ export default function CreateAnnouncement() {
|
|||||||
{
|
{
|
||||||
divisionMember.length > 0
|
divisionMember.length > 0
|
||||||
&&
|
&&
|
||||||
<>
|
<View style={[Styles.borderAll, Styles.round10, Styles.p10]}>
|
||||||
<View style={[Styles.rowSpaceBetween, Styles.mv05]}>
|
{
|
||||||
<Text style={[Styles.textDefaultSemiBold]}>Divisi</Text>
|
divisionMember.map((item: { name: any; Division: any }, index: any) => {
|
||||||
</View>
|
return (
|
||||||
<View style={[Styles.borderAll, Styles.round05, Styles.p10, { backgroundColor: colors.card, borderColor: colors.icon + '20' }]}>
|
<View key={index}>
|
||||||
{
|
<Text style={[Styles.textDefaultSemiBold]}>{item.name}</Text>
|
||||||
divisionMember.map((item: { name: any; Division: any }, index: any) => {
|
{
|
||||||
return (
|
item.Division.map((division: any, i: any) => (
|
||||||
<View key={index}>
|
<View key={i} style={[Styles.rowItemsCenter, Styles.w90]}>
|
||||||
<Text style={[Styles.textDefaultSemiBold]}>{item.name}</Text>
|
<Entypo name="dot-single" size={24} color="black" />
|
||||||
{
|
<Text style={[Styles.textDefault]} numberOfLines={1} ellipsizeMode='tail'>{division.name}</Text>
|
||||||
item.Division.map((division: any, i: any) => (
|
</View>
|
||||||
<View key={i} style={[Styles.rowItemsCenter, Styles.w90]}>
|
))
|
||||||
<Entypo name="dot-single" size={24} color={colors.text} />
|
}
|
||||||
<Text style={[Styles.textDefault]} numberOfLines={1} ellipsizeMode='tail'>{division.name}</Text>
|
</View>
|
||||||
</View>
|
)
|
||||||
))
|
})
|
||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
</View>
|
|
||||||
</>
|
|
||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
</ScrollView>
|
</ScrollView>
|
||||||
@@ -285,7 +266,7 @@ export default function CreateAnnouncement() {
|
|||||||
<DrawerBottom animation="slide" isVisible={isModalFile} setVisible={setModalFile} title="Menu">
|
<DrawerBottom animation="slide" isVisible={isModalFile} setVisible={setModalFile} title="Menu">
|
||||||
<View style={Styles.rowItemsCenter}>
|
<View style={Styles.rowItemsCenter}>
|
||||||
<MenuItemRow
|
<MenuItemRow
|
||||||
icon={<Ionicons name="trash-outline" color={colors.text} size={25} />}
|
icon={<Ionicons name="trash" color="black" size={25} />}
|
||||||
title="Hapus"
|
title="Hapus"
|
||||||
onPress={() => { deleteFile(indexDelFile) }}
|
onPress={() => { deleteFile(indexDelFile) }}
|
||||||
/>
|
/>
|
||||||
@@ -295,4 +276,15 @@ export default function CreateAnnouncement() {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const styles = StyleSheet.create({
|
||||||
|
container: {
|
||||||
|
padding: 20,
|
||||||
|
},
|
||||||
|
textArea: {
|
||||||
|
height: 100, // Or use flex-based sizing
|
||||||
|
borderColor: 'gray',
|
||||||
|
borderWidth: 1,
|
||||||
|
padding: 10,
|
||||||
|
textAlignVertical: 'top', // Important for Android to align text at the top
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ import ButtonSaveHeader from "@/components/buttonSaveHeader";
|
|||||||
import ButtonSelect from "@/components/buttonSelect";
|
import ButtonSelect from "@/components/buttonSelect";
|
||||||
import DrawerBottom from "@/components/drawerBottom";
|
import DrawerBottom from "@/components/drawerBottom";
|
||||||
import { InputForm } from "@/components/inputForm";
|
import { InputForm } from "@/components/inputForm";
|
||||||
import LoadingCenter from "@/components/loadingCenter";
|
import LoadingOverlay from "@/components/loadingOverlay";
|
||||||
import MenuItemRow from "@/components/menuItemRow";
|
import MenuItemRow from "@/components/menuItemRow";
|
||||||
import ModalSelectMultiple from "@/components/modalSelectMultiple";
|
import ModalSelectMultiple from "@/components/modalSelectMultiple";
|
||||||
import Text from '@/components/Text';
|
import Text from '@/components/Text';
|
||||||
@@ -12,7 +12,6 @@ import Styles from "@/constants/Styles";
|
|||||||
import { setUpdateAnnouncement } from "@/lib/announcementUpdate";
|
import { setUpdateAnnouncement } from "@/lib/announcementUpdate";
|
||||||
import { apiEditAnnouncement, apiGetAnnouncementOne } from "@/lib/api";
|
import { apiEditAnnouncement, apiGetAnnouncementOne } from "@/lib/api";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { Entypo, Ionicons, MaterialCommunityIcons } from "@expo/vector-icons";
|
import { Entypo, Ionicons, MaterialCommunityIcons } from "@expo/vector-icons";
|
||||||
import * as DocumentPicker from "expo-document-picker";
|
import * as DocumentPicker from "expo-document-picker";
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
@@ -36,7 +35,6 @@ export default function EditAnnouncement() {
|
|||||||
const dispatch = useDispatch()
|
const dispatch = useDispatch()
|
||||||
const update = useSelector((state: any) => state.announcementUpdate)
|
const update = useSelector((state: any) => state.announcementUpdate)
|
||||||
const { token, decryptToken } = useAuthSession();
|
const { token, decryptToken } = useAuthSession();
|
||||||
const { colors } = useTheme();
|
|
||||||
const [modalDivisi, setModalDivisi] = useState(false);
|
const [modalDivisi, setModalDivisi] = useState(false);
|
||||||
const [disableBtn, setDisableBtn] = useState(true);
|
const [disableBtn, setDisableBtn] = useState(true);
|
||||||
const [dataMember, setDataMember] = useState<any>([]);
|
const [dataMember, setDataMember] = useState<any>([]);
|
||||||
@@ -144,14 +142,9 @@ export default function EditAnnouncement() {
|
|||||||
dispatch(setUpdateAnnouncement(!update))
|
dispatch(setUpdateAnnouncement(!update))
|
||||||
Toast.show({ type: 'small', text1: 'Berhasil mengubah data', })
|
Toast.show({ type: 'small', text1: 'Berhasil mengubah data', })
|
||||||
router.back();
|
router.back();
|
||||||
} else {
|
|
||||||
Toast.show({ type: 'small', text1: 'Gagal mengubah data', })
|
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Gagal mengubah data"
|
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
@@ -187,7 +180,7 @@ export default function EditAnnouncement() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={{ flex: 1, backgroundColor: colors.background }}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => (
|
// headerLeft: () => (
|
||||||
@@ -230,10 +223,10 @@ export default function EditAnnouncement() {
|
|||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
{loading && <LoadingCenter />}
|
<LoadingOverlay visible={loading} />
|
||||||
<ScrollView
|
<ScrollView
|
||||||
showsVerticalScrollIndicator={false}
|
showsVerticalScrollIndicator={false}
|
||||||
style={[Styles.h100, { backgroundColor: colors.background }]}
|
style={[Styles.h100]}
|
||||||
>
|
>
|
||||||
<View style={[Styles.p15]}>
|
<View style={[Styles.p15]}>
|
||||||
<InputForm
|
<InputForm
|
||||||
@@ -242,7 +235,6 @@ export default function EditAnnouncement() {
|
|||||||
placeholder="Judul Pengumuman"
|
placeholder="Judul Pengumuman"
|
||||||
required
|
required
|
||||||
error={error.title}
|
error={error.title}
|
||||||
bg={colors.card}
|
|
||||||
errorText="Judul harus diisi"
|
errorText="Judul harus diisi"
|
||||||
onChange={(val) => validationForm("title", val)}
|
onChange={(val) => validationForm("title", val)}
|
||||||
value={dataForm.title}
|
value={dataForm.title}
|
||||||
@@ -253,7 +245,6 @@ export default function EditAnnouncement() {
|
|||||||
placeholder="Deskripsi Pengumuman"
|
placeholder="Deskripsi Pengumuman"
|
||||||
required
|
required
|
||||||
error={error.desc}
|
error={error.desc}
|
||||||
bg={colors.card}
|
|
||||||
errorText="Pengumuman harus diisi"
|
errorText="Pengumuman harus diisi"
|
||||||
onChange={(val) => validationForm("desc", val)}
|
onChange={(val) => validationForm("desc", val)}
|
||||||
value={dataForm.desc}
|
value={dataForm.desc}
|
||||||
@@ -263,40 +254,33 @@ export default function EditAnnouncement() {
|
|||||||
{
|
{
|
||||||
(fileForm.length > 0 || dataFile.filter((val) => !val.delete).length > 0)
|
(fileForm.length > 0 || dataFile.filter((val) => !val.delete).length > 0)
|
||||||
&&
|
&&
|
||||||
<>
|
<View style={[Styles.borderAll, Styles.round10, Styles.p10, Styles.mb10]}>
|
||||||
<View style={[Styles.rowSpaceBetween, Styles.mv05]}>
|
<Text style={[Styles.textDefaultSemiBold]}>File</Text>
|
||||||
<Text style={[Styles.textDefaultSemiBold]}>File</Text>
|
{
|
||||||
<Text style={[Styles.textDefault]}>{fileForm.length + dataFile.filter((val) => !val.delete).length} file</Text>
|
dataFile.filter((val) => !val.delete).map((item, index) => (
|
||||||
</View>
|
<BorderBottomItem
|
||||||
<View style={[Styles.borderAll, Styles.round05, Styles.p10, Styles.mb10, { backgroundColor: colors.card, borderColor: colors.icon + '20' }]}>
|
key={index}
|
||||||
{
|
borderType={(fileForm.length + dataFile.length) > 1 ? "bottom" : "none"}
|
||||||
dataFile.filter((val) => !val.delete).map((item, index) => (
|
icon={<MaterialCommunityIcons name="file-outline" size={25} color="black" />}
|
||||||
<BorderBottomItem
|
title={item.name + '.' + item.extension}
|
||||||
key={index}
|
titleWeight="normal"
|
||||||
borderType={dataFile.filter((val) => !val.delete).length - 1 == index && fileForm.length == 0 ? "none" : "bottom"}
|
onPress={() => { setIndexDelFile({ id: item.id, cat: "oldFile" }); setModalFile(true) }}
|
||||||
icon={<MaterialCommunityIcons name="file-outline" size={25} color={colors.text} />}
|
/>
|
||||||
title={item.name + '.' + item.extension}
|
))
|
||||||
titleWeight="normal"
|
}
|
||||||
bgColor="transparent"
|
{
|
||||||
onPress={() => { setIndexDelFile({ id: item.id, cat: "oldFile" }); setModalFile(true) }}
|
fileForm.map((item, index) => (
|
||||||
/>
|
<BorderBottomItem
|
||||||
))
|
key={index}
|
||||||
}
|
borderType={(fileForm.length + dataFile.length) > 1 ? "bottom" : "none"}
|
||||||
{
|
icon={<MaterialCommunityIcons name="file-outline" size={25} color="black" />}
|
||||||
fileForm.map((item, index) => (
|
title={item.name}
|
||||||
<BorderBottomItem
|
titleWeight="normal"
|
||||||
key={index}
|
onPress={() => { setIndexDelFile({ id: index, cat: "newFile" }); setModalFile(true) }}
|
||||||
borderType={fileForm.length - 1 == index ? "none" : "bottom"}
|
/>
|
||||||
icon={<MaterialCommunityIcons name="file-outline" size={25} color={colors.text} />}
|
))
|
||||||
title={item.name}
|
}
|
||||||
titleWeight="normal"
|
</View>
|
||||||
bgColor="transparent"
|
|
||||||
onPress={() => { setIndexDelFile({ id: index, cat: "newFile" }); setModalFile(true) }}
|
|
||||||
/>
|
|
||||||
))
|
|
||||||
}
|
|
||||||
</View>
|
|
||||||
</>
|
|
||||||
}
|
}
|
||||||
<ButtonSelect
|
<ButtonSelect
|
||||||
value="Pilih divisi penerima pengumuman"
|
value="Pilih divisi penerima pengumuman"
|
||||||
@@ -307,30 +291,25 @@ export default function EditAnnouncement() {
|
|||||||
{
|
{
|
||||||
dataMember.length > 0
|
dataMember.length > 0
|
||||||
&&
|
&&
|
||||||
<>
|
<View style={[Styles.borderAll, Styles.round10, Styles.p10]}>
|
||||||
<View style={[Styles.rowSpaceBetween, Styles.mv05]}>
|
{
|
||||||
<Text style={[Styles.textDefaultSemiBold]}>Divisi</Text>
|
dataMember.map((item: { name: any; Division: any }, index: any) => {
|
||||||
</View>
|
return (
|
||||||
<View style={[Styles.borderAll, Styles.round05, Styles.p10, { backgroundColor: colors.card, borderColor: colors.icon + '20' }]}>
|
<View key={index}>
|
||||||
{
|
<Text style={[Styles.textDefaultSemiBold]}>{item.name}</Text>
|
||||||
dataMember.map((item: { name: any; Division: any }, index: any) => {
|
{
|
||||||
return (
|
item.Division.map((division: any, i: any) => (
|
||||||
<View key={index}>
|
<View key={i} style={[Styles.rowItemsCenter, Styles.w90]}>
|
||||||
<Text style={[Styles.textDefaultSemiBold]}>{item.name}</Text>
|
<Entypo name="dot-single" size={24} color="black" />
|
||||||
{
|
<Text style={[Styles.textDefault]} numberOfLines={1} ellipsizeMode='tail'>{division.name}</Text>
|
||||||
item.Division.map((division: any, i: any) => (
|
</View>
|
||||||
<View key={i} style={[Styles.rowItemsCenter, Styles.w90]}>
|
))
|
||||||
<Entypo name="dot-single" size={24} color={colors.text} />
|
}
|
||||||
<Text style={[Styles.textDefault]} numberOfLines={1} ellipsizeMode='tail'>{division.name}</Text>
|
</View>
|
||||||
</View>
|
)
|
||||||
))
|
})
|
||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
</View>
|
|
||||||
</>
|
|
||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
</ScrollView>
|
</ScrollView>
|
||||||
@@ -351,7 +330,7 @@ export default function EditAnnouncement() {
|
|||||||
<DrawerBottom animation="slide" isVisible={isModalFile} setVisible={setModalFile} title="Menu">
|
<DrawerBottom animation="slide" isVisible={isModalFile} setVisible={setModalFile} title="Menu">
|
||||||
<View style={Styles.rowItemsCenter}>
|
<View style={Styles.rowItemsCenter}>
|
||||||
<MenuItemRow
|
<MenuItemRow
|
||||||
icon={<Ionicons name="trash-outline" color={colors.text} size={25} />}
|
icon={<Ionicons name="trash" color="black" size={25} />}
|
||||||
title="Hapus"
|
title="Hapus"
|
||||||
onPress={() => { deleteFile(indexDelFile.id, indexDelFile.cat) }}
|
onPress={() => { deleteFile(indexDelFile.id, indexDelFile.cat) }}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -6,7 +6,6 @@ import { ColorsStatus } from "@/constants/ColorsStatus";
|
|||||||
import Styles from "@/constants/Styles";
|
import Styles from "@/constants/Styles";
|
||||||
import { apiGetAnnouncement } from "@/lib/api";
|
import { apiGetAnnouncement } from "@/lib/api";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { MaterialIcons } from "@expo/vector-icons";
|
import { MaterialIcons } from "@expo/vector-icons";
|
||||||
import { router } from "expo-router";
|
import { router } from "expo-router";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
@@ -23,7 +22,6 @@ type Props = {
|
|||||||
|
|
||||||
export default function Announcement() {
|
export default function Announcement() {
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
const { colors } = useTheme();
|
|
||||||
const [data, setData] = useState<Props[]>([])
|
const [data, setData] = useState<Props[]>([])
|
||||||
const [search, setSearch] = useState('')
|
const [search, setSearch] = useState('')
|
||||||
const update = useSelector((state: any) => state.announcementUpdate)
|
const update = useSelector((state: any) => state.announcementUpdate)
|
||||||
@@ -85,11 +83,11 @@ export default function Announcement() {
|
|||||||
})
|
})
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<View style={[Styles.p15, Styles.flex1, { backgroundColor: colors.background }]}>
|
<View style={[Styles.p15, { flex: 1 }]}>
|
||||||
<View>
|
<View>
|
||||||
<InputSearch onChange={setSearch} />
|
<InputSearch onChange={setSearch} />
|
||||||
</View>
|
</View>
|
||||||
<View style={[Styles.flex2, Styles.mt05]}>
|
<View style={[{ flex: 2 }, Styles.mt05]}>
|
||||||
{
|
{
|
||||||
loading ?
|
loading ?
|
||||||
arrSkeleton.map((item, index) => {
|
arrSkeleton.map((item, index) => {
|
||||||
@@ -110,11 +108,10 @@ export default function Announcement() {
|
|||||||
key={index}
|
key={index}
|
||||||
onPress={() => { router.push(`/announcement/${item.id}`) }}
|
onPress={() => { router.push(`/announcement/${item.id}`) }}
|
||||||
borderType="bottom"
|
borderType="bottom"
|
||||||
bgColor="transparent"
|
|
||||||
icon={
|
icon={
|
||||||
// <View style={[Styles.iconContent]}>
|
<View style={[Styles.iconContent, ColorsStatus.lightGreen]}>
|
||||||
<MaterialIcons name="campaign" size={25} color={colors.text} />
|
<MaterialIcons name="campaign" size={25} color={'#384288'} />
|
||||||
// </View>
|
</View>
|
||||||
}
|
}
|
||||||
title={item.title}
|
title={item.title}
|
||||||
desc={item.desc.replace(/<[^>]*>?/gm, '').replace(/\r?\n|\r/g, ' ')}
|
desc={item.desc.replace(/<[^>]*>?/gm, '').replace(/\r?\n|\r/g, ' ')}
|
||||||
@@ -130,12 +127,11 @@ export default function Announcement() {
|
|||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={handleRefresh}
|
onRefresh={handleRefresh}
|
||||||
tintColor={colors.icon}
|
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
:
|
:
|
||||||
<Text style={[Styles.textDefault, Styles.textCenter, { color: colors.dimmed }]}>Tidak ada pengumuman</Text>
|
<Text style={[Styles.textDefault, Styles.cGray, { textAlign: 'center' }]}>Tidak ada pengumuman</Text>
|
||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
|
|||||||
@@ -1,14 +1,12 @@
|
|||||||
import AppHeader from "@/components/AppHeader";
|
import AppHeader from "@/components/AppHeader";
|
||||||
import ButtonSaveHeader from "@/components/buttonSaveHeader";
|
import ButtonSaveHeader from "@/components/buttonSaveHeader";
|
||||||
import { InputForm } from "@/components/inputForm";
|
import { InputForm } from "@/components/inputForm";
|
||||||
import LoadingCenter from "@/components/loadingCenter";
|
|
||||||
import Text from "@/components/Text";
|
import Text from "@/components/Text";
|
||||||
import { ConstEnv } from "@/constants/ConstEnv";
|
import { ConstEnv } from "@/constants/ConstEnv";
|
||||||
import Styles from "@/constants/Styles";
|
import Styles from "@/constants/Styles";
|
||||||
import { apiEditBanner, apiGetBanner, apiGetBannerOne } from "@/lib/api";
|
import { apiEditBanner, apiGetBanner, apiGetBannerOne } from "@/lib/api";
|
||||||
import { setEntities } from "@/lib/bannerSlice";
|
import { setEntities } from "@/lib/bannerSlice";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { Entypo } from "@expo/vector-icons";
|
import { Entypo } from "@expo/vector-icons";
|
||||||
import * as ImagePicker from "expo-image-picker";
|
import * as ImagePicker from "expo-image-picker";
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
@@ -26,7 +24,6 @@ import { useDispatch } from "react-redux";
|
|||||||
export default function EditBanner() {
|
export default function EditBanner() {
|
||||||
const dispatch = useDispatch();
|
const dispatch = useDispatch();
|
||||||
const { decryptToken, token } = useAuthSession();
|
const { decryptToken, token } = useAuthSession();
|
||||||
const { colors } = useTheme();
|
|
||||||
const { id } = useLocalSearchParams<{ id: string }>();
|
const { id } = useLocalSearchParams<{ id: string }>();
|
||||||
const [selectedImage, setSelectedImage] = useState<
|
const [selectedImage, setSelectedImage] = useState<
|
||||||
string | undefined | { uri: string }
|
string | undefined | { uri: string }
|
||||||
@@ -106,18 +103,16 @@ export default function EditBanner() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: 'Gagal mengupdate data', })
|
Toast.show({ type: 'small', text1: 'Gagal mengupdate data', })
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Gagal mengupdate data"
|
Toast.show({ type: 'small', text1: 'Gagal mengupdate data', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={[Styles.flex1, { backgroundColor: colors.background }]}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => (
|
// headerLeft: () => (
|
||||||
@@ -148,8 +143,7 @@ export default function EditBanner() {
|
|||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
{loading && <LoadingCenter />}
|
<ScrollView showsVerticalScrollIndicator={false} style={[Styles.h100]}>
|
||||||
<ScrollView showsVerticalScrollIndicator={false} style={[Styles.h100, { backgroundColor: colors.background }]}>
|
|
||||||
<View style={[Styles.p15, Styles.mb100]}>
|
<View style={[Styles.p15, Styles.mb100]}>
|
||||||
<View style={[Styles.mb15]}>
|
<View style={[Styles.mb15]}>
|
||||||
{selectedImage != undefined ? (
|
{selectedImage != undefined ? (
|
||||||
@@ -160,7 +154,7 @@ export default function EditBanner() {
|
|||||||
? selectedImage
|
? selectedImage
|
||||||
: selectedImage.uri
|
: selectedImage.uri
|
||||||
}
|
}
|
||||||
style={[Styles.resizeContain, Styles.w100, { height: 100 }]}
|
style={{ resizeMode: "contain", width: "100%", height: 100 }}
|
||||||
/>
|
/>
|
||||||
</Pressable>
|
</Pressable>
|
||||||
) : (
|
) : (
|
||||||
@@ -169,7 +163,7 @@ export default function EditBanner() {
|
|||||||
style={[Styles.wrapPaper, Styles.contentItemCenter]}
|
style={[Styles.wrapPaper, Styles.contentItemCenter]}
|
||||||
>
|
>
|
||||||
<View
|
<View
|
||||||
style={[Styles.contentItemCenter]}
|
style={{ justifyContent: "center", alignItems: "center" }}
|
||||||
>
|
>
|
||||||
<Entypo name="image" size={50} color={"#aeaeae"} />
|
<Entypo name="image" size={50} color={"#aeaeae"} />
|
||||||
<Text style={[Styles.textInformation, Styles.mt05]}>
|
<Text style={[Styles.textInformation, Styles.mt05]}>
|
||||||
@@ -185,7 +179,7 @@ export default function EditBanner() {
|
|||||||
type="default"
|
type="default"
|
||||||
placeholder="Judul"
|
placeholder="Judul"
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
bg="white"
|
||||||
value={title}
|
value={title}
|
||||||
error={error}
|
error={error}
|
||||||
onChange={onValidate}
|
onChange={onValidate}
|
||||||
|
|||||||
@@ -1,13 +1,11 @@
|
|||||||
import AppHeader from "@/components/AppHeader";
|
import AppHeader from "@/components/AppHeader";
|
||||||
import ButtonSaveHeader from "@/components/buttonSaveHeader";
|
import ButtonSaveHeader from "@/components/buttonSaveHeader";
|
||||||
import { InputForm } from "@/components/inputForm";
|
import { InputForm } from "@/components/inputForm";
|
||||||
import LoadingCenter from "@/components/loadingCenter";
|
|
||||||
import Text from "@/components/Text";
|
import Text from "@/components/Text";
|
||||||
import Styles from "@/constants/Styles";
|
import Styles from "@/constants/Styles";
|
||||||
import { apiCreateBanner, apiGetBanner } from "@/lib/api";
|
import { apiCreateBanner, apiGetBanner } from "@/lib/api";
|
||||||
import { setEntities } from "@/lib/bannerSlice";
|
import { setEntities } from "@/lib/bannerSlice";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { Entypo } from "@expo/vector-icons";
|
import { Entypo } from "@expo/vector-icons";
|
||||||
import * as ImagePicker from "expo-image-picker";
|
import * as ImagePicker from "expo-image-picker";
|
||||||
import { router, Stack } from "expo-router";
|
import { router, Stack } from "expo-router";
|
||||||
@@ -24,7 +22,6 @@ import { useDispatch } from "react-redux";
|
|||||||
|
|
||||||
export default function CreateBanner() {
|
export default function CreateBanner() {
|
||||||
const { decryptToken, token } = useAuthSession();
|
const { decryptToken, token } = useAuthSession();
|
||||||
const { colors } = useTheme();
|
|
||||||
const dispatch = useDispatch();
|
const dispatch = useDispatch();
|
||||||
const [selectedImage, setSelectedImage] = useState<string | undefined>(
|
const [selectedImage, setSelectedImage] = useState<string | undefined>(
|
||||||
undefined
|
undefined
|
||||||
@@ -88,22 +85,36 @@ export default function CreateBanner() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: 'Gagal menambahkan data', })
|
Toast.show({ type: 'small', text1: 'Gagal menambahkan data', })
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Gagal menambahkan data"
|
Toast.show({ type: 'small', text1: 'Gagal menambahkan data', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={[Styles.flex1, { backgroundColor: colors.background }]}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
|
// headerLeft: () => (
|
||||||
|
// <ButtonBackHeader
|
||||||
|
// onPress={() => {
|
||||||
|
// router.back();
|
||||||
|
// }}
|
||||||
|
// />
|
||||||
|
// ),
|
||||||
headerTitle: "Tambah Banner",
|
headerTitle: "Tambah Banner",
|
||||||
headerTitleAlign: "center",
|
headerTitleAlign: "center",
|
||||||
|
// headerRight: () => (
|
||||||
|
// <ButtonSaveHeader
|
||||||
|
// disable={title == "" || selectedImage == undefined || error || loading ? true : false}
|
||||||
|
// category="create"
|
||||||
|
// onPress={() => {
|
||||||
|
// handleCreateEntity();
|
||||||
|
// }}
|
||||||
|
// />
|
||||||
|
// ),
|
||||||
header: () => (
|
header: () => (
|
||||||
<AppHeader
|
<AppHeader
|
||||||
title="Fitur"
|
title="Fitur"
|
||||||
@@ -122,27 +133,26 @@ export default function CreateBanner() {
|
|||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
{loading && <LoadingCenter />}
|
<ScrollView showsVerticalScrollIndicator={false} style={[Styles.h100]}>
|
||||||
<ScrollView showsVerticalScrollIndicator={false} style={[Styles.h100, { backgroundColor: colors.background }]}>
|
|
||||||
<View style={[Styles.p15]}>
|
<View style={[Styles.p15]}>
|
||||||
<View style={[Styles.mb15]}>
|
<View style={[Styles.mb15]}>
|
||||||
{selectedImage != undefined ? (
|
{selectedImage != undefined ? (
|
||||||
<Pressable onPress={pickImageAsync}>
|
<Pressable onPress={pickImageAsync}>
|
||||||
<Image
|
<Image
|
||||||
src={selectedImage}
|
src={selectedImage}
|
||||||
style={[Styles.resizeContain, Styles.w100, { height: 100 }]}
|
style={{ resizeMode: "contain", width: "100%", height: 100 }}
|
||||||
/>
|
/>
|
||||||
</Pressable>
|
</Pressable>
|
||||||
) : (
|
) : (
|
||||||
<Pressable
|
<Pressable
|
||||||
onPress={pickImageAsync}
|
onPress={pickImageAsync}
|
||||||
style={[Styles.wrapPaper, Styles.contentItemCenter, Styles.borderAll, { backgroundColor: colors.card, borderColor: colors.icon + '20' }]}
|
style={[Styles.wrapPaper, Styles.contentItemCenter]}
|
||||||
>
|
>
|
||||||
<View
|
<View
|
||||||
style={[Styles.contentItemCenter]}
|
style={{ justifyContent: "center", alignItems: "center" }}
|
||||||
>
|
>
|
||||||
<Entypo name="image" size={50} color={colors.dimmed} />
|
<Entypo name="image" size={50} color={"#aeaeae"} />
|
||||||
<Text style={[Styles.textInformation, Styles.mt05, { color: colors.dimmed }]}>
|
<Text style={[Styles.textInformation, Styles.mt05]}>
|
||||||
Mohon unggah gambar dalam resolusi 1650 x 720 pixel untuk
|
Mohon unggah gambar dalam resolusi 1650 x 720 pixel untuk
|
||||||
memastikan
|
memastikan
|
||||||
</Text>
|
</Text>
|
||||||
@@ -155,7 +165,7 @@ export default function CreateBanner() {
|
|||||||
type="default"
|
type="default"
|
||||||
placeholder="Judul"
|
placeholder="Judul"
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
bg="white"
|
||||||
onChange={onValidate}
|
onChange={onValidate}
|
||||||
error={error}
|
error={error}
|
||||||
errorText="Judul harus diisi"
|
errorText="Judul harus diisi"
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
|
import AlertKonfirmasi from "@/components/alertKonfirmasi"
|
||||||
import AppHeader from "@/components/AppHeader"
|
import AppHeader from "@/components/AppHeader"
|
||||||
import HeaderRightBannerList from "@/components/banner/headerBannerList"
|
import HeaderRightBannerList from "@/components/banner/headerBannerList"
|
||||||
import BorderBottomItem from "@/components/borderBottomItem"
|
import BorderBottomItem from "@/components/borderBottomItem"
|
||||||
import DrawerBottom from "@/components/drawerBottom"
|
import DrawerBottom from "@/components/drawerBottom"
|
||||||
import MenuItemRow from "@/components/menuItemRow"
|
import MenuItemRow from "@/components/menuItemRow"
|
||||||
import ModalConfirmation from "@/components/ModalConfirmation"
|
|
||||||
import ModalLoading from "@/components/modalLoading"
|
import ModalLoading from "@/components/modalLoading"
|
||||||
import Text from "@/components/Text"
|
import Text from "@/components/Text"
|
||||||
import { ConstEnv } from "@/constants/ConstEnv"
|
import { ConstEnv } from "@/constants/ConstEnv"
|
||||||
@@ -11,7 +11,6 @@ import Styles from "@/constants/Styles"
|
|||||||
import { apiDeleteBanner, apiGetBanner } from "@/lib/api"
|
import { apiDeleteBanner, apiGetBanner } from "@/lib/api"
|
||||||
import { setEntities } from "@/lib/bannerSlice"
|
import { setEntities } from "@/lib/bannerSlice"
|
||||||
import { useAuthSession } from "@/providers/AuthProvider"
|
import { useAuthSession } from "@/providers/AuthProvider"
|
||||||
import { useTheme } from "@/providers/ThemeProvider"
|
|
||||||
import { Ionicons, MaterialCommunityIcons } from "@expo/vector-icons"
|
import { Ionicons, MaterialCommunityIcons } from "@expo/vector-icons"
|
||||||
import * as FileSystem from 'expo-file-system'
|
import * as FileSystem from 'expo-file-system'
|
||||||
import { startActivityAsync } from 'expo-intent-launcher'
|
import { startActivityAsync } from 'expo-intent-launcher'
|
||||||
@@ -33,7 +32,6 @@ type Props = {
|
|||||||
|
|
||||||
export default function BannerList() {
|
export default function BannerList() {
|
||||||
const { decryptToken, token } = useAuthSession()
|
const { decryptToken, token } = useAuthSession()
|
||||||
const { colors } = useTheme()
|
|
||||||
const [isModal, setModal] = useState(false)
|
const [isModal, setModal] = useState(false)
|
||||||
const entities = useSelector((state: any) => state.banner)
|
const entities = useSelector((state: any) => state.banner)
|
||||||
const [dataId, setDataId] = useState('')
|
const [dataId, setDataId] = useState('')
|
||||||
@@ -42,7 +40,6 @@ export default function BannerList() {
|
|||||||
const [refreshing, setRefreshing] = useState(false)
|
const [refreshing, setRefreshing] = useState(false)
|
||||||
const [loadingOpen, setLoadingOpen] = useState(false)
|
const [loadingOpen, setLoadingOpen] = useState(false)
|
||||||
const [viewImg, setViewImg] = useState(false)
|
const [viewImg, setViewImg] = useState(false)
|
||||||
const [showDeleteModal, setShowDeleteModal] = useState(false)
|
|
||||||
|
|
||||||
const handleDeleteEntity = async () => {
|
const handleDeleteEntity = async () => {
|
||||||
try {
|
try {
|
||||||
@@ -56,11 +53,9 @@ export default function BannerList() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: 'Gagal menghapus data', })
|
Toast.show({ type: 'small', text1: 'Gagal menghapus data', })
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal menghapus data"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setModal(false)
|
setModal(false)
|
||||||
}
|
}
|
||||||
@@ -110,7 +105,7 @@ export default function BannerList() {
|
|||||||
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={[Styles.flex1, { backgroundColor: colors.background }]}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
||||||
@@ -135,10 +130,9 @@ export default function BannerList() {
|
|||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={handleRefresh}
|
onRefresh={handleRefresh}
|
||||||
tintColor={colors.icon}
|
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
style={[Styles.h100, { backgroundColor: colors.background }]}
|
style={[Styles.h100]}
|
||||||
>
|
>
|
||||||
{
|
{
|
||||||
entities.length > 0
|
entities.length > 0
|
||||||
@@ -160,12 +154,13 @@ export default function BannerList() {
|
|||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
title={index.title}
|
title={index.title}
|
||||||
|
width={65}
|
||||||
/>
|
/>
|
||||||
))}
|
))}
|
||||||
</View>
|
</View>
|
||||||
:
|
:
|
||||||
<View style={[Styles.p15, Styles.mb100]}>
|
<View style={[Styles.p15, Styles.mb100]}>
|
||||||
<Text style={[Styles.textDefault, Styles.textCenter]}>Tidak ada data</Text>
|
<Text style={[Styles.textDefault, Styles.cGray, { textAlign: 'center' }]}>Tidak ada data</Text>
|
||||||
</View>
|
</View>
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -175,7 +170,7 @@ export default function BannerList() {
|
|||||||
<DrawerBottom animation="slide" isVisible={isModal} setVisible={() => setModal(false)} title="Menu">
|
<DrawerBottom animation="slide" isVisible={isModal} setVisible={() => setModal(false)} title="Menu">
|
||||||
<View style={Styles.rowItemsCenter}>
|
<View style={Styles.rowItemsCenter}>
|
||||||
<MenuItemRow
|
<MenuItemRow
|
||||||
icon={<MaterialCommunityIcons name="pencil-outline" color={colors.text} size={25} />}
|
icon={<MaterialCommunityIcons name="pencil-outline" color="black" size={25} />}
|
||||||
title="Edit"
|
title="Edit"
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
setModal(false)
|
setModal(false)
|
||||||
@@ -183,7 +178,7 @@ export default function BannerList() {
|
|||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<MenuItemRow
|
<MenuItemRow
|
||||||
icon={<MaterialCommunityIcons name="file-eye" color={colors.text} size={25} />}
|
icon={<MaterialCommunityIcons name="file-eye" color="black" size={25} />}
|
||||||
title="Lihat"
|
title="Lihat"
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
setModal(false)
|
setModal(false)
|
||||||
@@ -194,13 +189,15 @@ export default function BannerList() {
|
|||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<MenuItemRow
|
<MenuItemRow
|
||||||
icon={<Ionicons name="trash-outline" color={colors.text} size={25} />}
|
icon={<Ionicons name="trash" color="black" size={25} />}
|
||||||
title="Hapus"
|
title="Hapus"
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
setModal(false)
|
setModal(false)
|
||||||
setTimeout(() => {
|
AlertKonfirmasi({
|
||||||
setShowDeleteModal(true)
|
title: 'Konfirmasi',
|
||||||
}, 600)
|
desc: 'Apakah anda yakin ingin menghapus data?',
|
||||||
|
onPress: () => { handleDeleteEntity() }
|
||||||
|
})
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
</View>
|
</View>
|
||||||
@@ -213,19 +210,6 @@ export default function BannerList() {
|
|||||||
onRequestClose={() => setViewImg(false)}
|
onRequestClose={() => setViewImg(false)}
|
||||||
doubleTapToZoomEnabled
|
doubleTapToZoomEnabled
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<ModalConfirmation
|
|
||||||
visible={showDeleteModal}
|
|
||||||
title="Konfirmasi"
|
|
||||||
message="Apakah anda yakin ingin menghapus data?"
|
|
||||||
onConfirm={() => {
|
|
||||||
setShowDeleteModal(false)
|
|
||||||
handleDeleteEntity()
|
|
||||||
}}
|
|
||||||
onCancel={() => setShowDeleteModal(false)}
|
|
||||||
confirmText="Hapus"
|
|
||||||
cancelText="Batal"
|
|
||||||
/>
|
|
||||||
</SafeAreaView>
|
</SafeAreaView>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@@ -1,3 +1,4 @@
|
|||||||
|
import AlertKonfirmasi from "@/components/alertKonfirmasi";
|
||||||
import AppHeader from "@/components/AppHeader";
|
import AppHeader from "@/components/AppHeader";
|
||||||
import BorderBottomItem from "@/components/borderBottomItem";
|
import BorderBottomItem from "@/components/borderBottomItem";
|
||||||
import BorderBottomItem2 from "@/components/borderBottomItem2";
|
import BorderBottomItem2 from "@/components/borderBottomItem2";
|
||||||
@@ -7,17 +8,16 @@ import ImageUser from "@/components/imageNew";
|
|||||||
import { InputForm } from "@/components/inputForm";
|
import { InputForm } from "@/components/inputForm";
|
||||||
import LabelStatus from "@/components/labelStatus";
|
import LabelStatus from "@/components/labelStatus";
|
||||||
import MenuItemRow from "@/components/menuItemRow";
|
import MenuItemRow from "@/components/menuItemRow";
|
||||||
import ModalConfirmation from "@/components/ModalConfirmation";
|
|
||||||
import Skeleton from "@/components/skeleton";
|
import Skeleton from "@/components/skeleton";
|
||||||
import SkeletonContent from "@/components/skeletonContent";
|
import SkeletonContent from "@/components/skeletonContent";
|
||||||
import Text from '@/components/Text';
|
import Text from '@/components/Text';
|
||||||
|
import { ColorsStatus } from "@/constants/ColorsStatus";
|
||||||
import { ConstEnv } from "@/constants/ConstEnv";
|
import { ConstEnv } from "@/constants/ConstEnv";
|
||||||
import { regexOnlySpacesOrEnter } from "@/constants/OnlySpaceOrEnter";
|
import { regexOnlySpacesOrEnter } from "@/constants/OnlySpaceOrEnter";
|
||||||
import Styles from "@/constants/Styles";
|
import Styles from "@/constants/Styles";
|
||||||
import { apiDeleteDiscussionGeneralCommentar, apiGetDiscussionGeneralOne, apiSendDiscussionGeneralCommentar, apiUpdateDiscussionGeneralCommentar } from "@/lib/api";
|
import { apiDeleteDiscussionGeneralCommentar, apiGetDiscussionGeneralOne, apiSendDiscussionGeneralCommentar, apiUpdateDiscussionGeneralCommentar } from "@/lib/api";
|
||||||
import { getDB } from "@/lib/firebaseDatabase";
|
import { getDB } from "@/lib/firebaseDatabase";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { Feather, Ionicons, MaterialCommunityIcons, MaterialIcons } from "@expo/vector-icons";
|
import { Feather, Ionicons, MaterialCommunityIcons, MaterialIcons } from "@expo/vector-icons";
|
||||||
import { ref } from '@react-native-firebase/database';
|
import { ref } from '@react-native-firebase/database';
|
||||||
import { useHeaderHeight } from '@react-navigation/elements';
|
import { useHeaderHeight } from '@react-navigation/elements';
|
||||||
@@ -56,7 +56,6 @@ type PropsFile = {
|
|||||||
|
|
||||||
export default function DetailDiscussionGeneral() {
|
export default function DetailDiscussionGeneral() {
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
const { colors } = useTheme();
|
|
||||||
const entityUser = useSelector((state: any) => state.user)
|
const entityUser = useSelector((state: any) => state.user)
|
||||||
const entities = useSelector((state: any) => state.entities)
|
const entities = useSelector((state: any) => state.entities)
|
||||||
const { id } = useLocalSearchParams<{ id: string }>();
|
const { id } = useLocalSearchParams<{ id: string }>();
|
||||||
@@ -80,7 +79,6 @@ export default function DetailDiscussionGeneral() {
|
|||||||
comment: ''
|
comment: ''
|
||||||
})
|
})
|
||||||
const [viewEdit, setViewEdit] = useState(false)
|
const [viewEdit, setViewEdit] = useState(false)
|
||||||
const [showDeleteModal, setShowDeleteModal] = useState(false)
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const onValueChange = reference.on('value', snapshot => {
|
const onValueChange = reference.on('value', snapshot => {
|
||||||
@@ -158,11 +156,8 @@ export default function DetailDiscussionGeneral() {
|
|||||||
Toast.show({ type: 'small', text1: response.message })
|
Toast.show({ type: 'small', text1: response.message })
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal menambahkan data"
|
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoadingSendKomentar(false)
|
setLoadingSendKomentar(false)
|
||||||
}
|
}
|
||||||
@@ -178,11 +173,8 @@ export default function DetailDiscussionGeneral() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message })
|
Toast.show({ type: 'small', text1: response.message })
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal mengupdate data"
|
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoadingSendKomentar(false)
|
setLoadingSendKomentar(false)
|
||||||
handleViewEditKomentar()
|
handleViewEditKomentar()
|
||||||
@@ -199,11 +191,8 @@ export default function DetailDiscussionGeneral() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message })
|
Toast.show({ type: 'small', text1: response.message })
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal menghapus data"
|
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoadingSendKomentar(false)
|
setLoadingSendKomentar(false)
|
||||||
setVisible(false)
|
setVisible(false)
|
||||||
@@ -248,15 +237,14 @@ export default function DetailDiscussionGeneral() {
|
|||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<View style={[Styles.flex1, { backgroundColor: colors.background }]}>
|
<View style={{ flex: 1 }}>
|
||||||
<ScrollView
|
<ScrollView
|
||||||
showsVerticalScrollIndicator={false}
|
showsVerticalScrollIndicator={false}
|
||||||
style={[Styles.h100, { backgroundColor: colors.background }]}
|
style={[Styles.h100]}
|
||||||
refreshControl={
|
refreshControl={
|
||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={() => handleRefresh()}
|
onRefresh={() => handleRefresh()}
|
||||||
tintColor={colors.icon}
|
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
@@ -268,11 +256,10 @@ export default function DetailDiscussionGeneral() {
|
|||||||
<BorderBottomItem2
|
<BorderBottomItem2
|
||||||
dataFile={fileDiscussion}
|
dataFile={fileDiscussion}
|
||||||
descEllipsize={false}
|
descEllipsize={false}
|
||||||
borderType="all"
|
borderType="bottom"
|
||||||
bgColor="white"
|
|
||||||
icon={
|
icon={
|
||||||
<View style={[Styles.iconContent]}>
|
<View style={[Styles.iconContent, ColorsStatus.lightGreen]}>
|
||||||
<MaterialIcons name="chat" size={25} color={'black'} />
|
<MaterialIcons name="chat" size={25} color={'#384288'} />
|
||||||
</View>
|
</View>
|
||||||
}
|
}
|
||||||
title={data?.title}
|
title={data?.title}
|
||||||
@@ -286,18 +273,18 @@ export default function DetailDiscussionGeneral() {
|
|||||||
desc={data?.desc}
|
desc={data?.desc}
|
||||||
leftBottomInfo={
|
leftBottomInfo={
|
||||||
<View style={[Styles.rowItemsCenter]}>
|
<View style={[Styles.rowItemsCenter]}>
|
||||||
<Ionicons name="chatbox-ellipses-outline" size={18} color={colors.dimmed} style={Styles.mr05} />
|
<Ionicons name="chatbox-ellipses-outline" size={18} color="grey" style={Styles.mr05} />
|
||||||
<Text style={[Styles.textInformation, { color: colors.dimmed }, Styles.mb05]}>{dataKomentar.length} Komentar</Text>
|
<Text style={[Styles.textInformation, Styles.cGray, Styles.mb05]}>{dataKomentar.length} Komentar</Text>
|
||||||
</View>
|
</View>
|
||||||
}
|
}
|
||||||
rightBottomInfo={
|
rightBottomInfo={
|
||||||
<View style={[Styles.rowItemsCenter]}>
|
<View style={[Styles.rowItemsCenter]}>
|
||||||
<Text style={[Styles.textInformation, { color: colors.dimmed }, Styles.mb05]}>{data?.createdAt}</Text>
|
<Text style={[Styles.textInformation, Styles.cGray, Styles.mb05]}>{data?.createdAt}</Text>
|
||||||
</View>
|
</View>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
<View style={[Styles.mt10]}>
|
<View style={[Styles.p15]}>
|
||||||
{
|
{
|
||||||
loadingKomentar ?
|
loadingKomentar ?
|
||||||
arrSkeleton.map((item: any, i: number) => {
|
arrSkeleton.map((item: any, i: number) => {
|
||||||
@@ -310,7 +297,7 @@ export default function DetailDiscussionGeneral() {
|
|||||||
return (
|
return (
|
||||||
<BorderBottomItem
|
<BorderBottomItem
|
||||||
key={i}
|
key={i}
|
||||||
borderType="all"
|
borderType="bottom"
|
||||||
colorPress
|
colorPress
|
||||||
icon={
|
icon={
|
||||||
<ImageUser src={`${ConstEnv.url_storage}/files/${item.img}`} size="xs" />
|
<ImageUser src={`${ConstEnv.url_storage}/files/${item.img}`} size="xs" />
|
||||||
@@ -320,7 +307,6 @@ export default function DetailDiscussionGeneral() {
|
|||||||
desc={item.comment}
|
desc={item.comment}
|
||||||
rightBottomInfo={item.isEdited ? "Edited" : ""}
|
rightBottomInfo={item.isEdited ? "Edited" : ""}
|
||||||
descEllipsize={detailMore.includes(item.id) ? false : true}
|
descEllipsize={detailMore.includes(item.id) ? false : true}
|
||||||
bgColor="white"
|
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
setDetailMore((prev: any) => {
|
setDetailMore((prev: any) => {
|
||||||
if (prev.includes(item.id)) {
|
if (prev.includes(item.id)) {
|
||||||
@@ -347,7 +333,7 @@ export default function DetailDiscussionGeneral() {
|
|||||||
<View style={[
|
<View style={[
|
||||||
Styles.contentItemCenter,
|
Styles.contentItemCenter,
|
||||||
Styles.w100,
|
Styles.w100,
|
||||||
{ backgroundColor: colors.background },
|
{ backgroundColor: "#f4f4f4" },
|
||||||
viewEdit && Styles.borderTop
|
viewEdit && Styles.borderTop
|
||||||
]}>
|
]}>
|
||||||
{
|
{
|
||||||
@@ -355,11 +341,11 @@ export default function DetailDiscussionGeneral() {
|
|||||||
<>
|
<>
|
||||||
<View style={[Styles.w90, Styles.rowSpaceBetween, Styles.pv05]}>
|
<View style={[Styles.w90, Styles.rowSpaceBetween, Styles.pv05]}>
|
||||||
<View style={[Styles.rowItemsCenter]}>
|
<View style={[Styles.rowItemsCenter]}>
|
||||||
<Feather name="edit-3" color={colors.text} size={22} style={[Styles.mh05]} />
|
<Feather name="edit-3" color="black" size={22} style={[Styles.mh05]} />
|
||||||
<Text style={[Styles.textMediumSemiBold]}>Edit Komentar</Text>
|
<Text style={[Styles.textMediumSemiBold]}>Edit Komentar</Text>
|
||||||
</View>
|
</View>
|
||||||
<Pressable onPress={() => handleViewEditKomentar()}>
|
<Pressable onPress={() => handleViewEditKomentar()}>
|
||||||
<MaterialIcons name="close" color={colors.text} size={22} />
|
<MaterialIcons name="close" color="black" size={22} />
|
||||||
</Pressable>
|
</Pressable>
|
||||||
</View>
|
</View>
|
||||||
<InputForm
|
<InputForm
|
||||||
@@ -367,6 +353,7 @@ export default function DetailDiscussionGeneral() {
|
|||||||
type="default"
|
type="default"
|
||||||
round
|
round
|
||||||
placeholder="Kirim Komentar"
|
placeholder="Kirim Komentar"
|
||||||
|
bg="white"
|
||||||
onChange={(val: string) => setSelectKomentar({ ...selectKomentar, comment: val })}
|
onChange={(val: string) => setSelectKomentar({ ...selectKomentar, comment: val })}
|
||||||
value={selectKomentar.comment}
|
value={selectKomentar.comment}
|
||||||
multiline
|
multiline
|
||||||
@@ -380,7 +367,7 @@ export default function DetailDiscussionGeneral() {
|
|||||||
Platform.OS == 'android' && Styles.mb12,
|
Platform.OS == 'android' && Styles.mb12,
|
||||||
]}
|
]}
|
||||||
>
|
>
|
||||||
<MaterialIcons name="send" size={25} style={(loadingSendKomentar || selectKomentar.comment == '' || regexOnlySpacesOrEnter.test(selectKomentar.comment) || data?.status === 2 || !data?.isActive || (!memberDiscussion && (entityUser.role == "user" || entityUser.role == "coadmin"))) ? { color: colors.dimmed } : { color: colors.tint }} />
|
<MaterialIcons name="send" size={25} style={(loadingSendKomentar || selectKomentar.comment == '' || regexOnlySpacesOrEnter.test(selectKomentar.comment) || data?.status === 2 || !data?.isActive || (!memberDiscussion && (entityUser.role == "user" || entityUser.role == "coadmin"))) ? Styles.cGray : Styles.cDefault} />
|
||||||
</Pressable>
|
</Pressable>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
@@ -393,6 +380,7 @@ export default function DetailDiscussionGeneral() {
|
|||||||
type="default"
|
type="default"
|
||||||
round
|
round
|
||||||
placeholder="Kirim Komentar"
|
placeholder="Kirim Komentar"
|
||||||
|
bg="white"
|
||||||
onChange={setKomentar}
|
onChange={setKomentar}
|
||||||
value={komentar}
|
value={komentar}
|
||||||
multiline
|
multiline
|
||||||
@@ -406,13 +394,13 @@ export default function DetailDiscussionGeneral() {
|
|||||||
Platform.OS == 'android' && Styles.mb12,
|
Platform.OS == 'android' && Styles.mb12,
|
||||||
]}
|
]}
|
||||||
>
|
>
|
||||||
<MaterialIcons name="send" size={25} style={(loadingSendKomentar || komentar == '' || regexOnlySpacesOrEnter.test(komentar) || data?.status === 2 || !data?.isActive || (!memberDiscussion && (entityUser.role == "user" || entityUser.role == "coadmin"))) ? { color: colors.dimmed } : { color: colors.tint }} />
|
<MaterialIcons name="send" size={25} style={(loadingSendKomentar || komentar == '' || regexOnlySpacesOrEnter.test(komentar) || data?.status === 2 || !data?.isActive || (!memberDiscussion && (entityUser.role == "user" || entityUser.role == "coadmin"))) ? Styles.cGray : Styles.cDefault} />
|
||||||
</Pressable>
|
</Pressable>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
:
|
:
|
||||||
<View style={[Styles.pv20, Styles.itemsCenter]}>
|
<View style={[Styles.pv20, { alignItems: 'center' }]}>
|
||||||
<Text style={[Styles.textInformation, { color: colors.dimmed }]}>
|
<Text style={[Styles.textInformation, Styles.cGray]}>
|
||||||
{
|
{
|
||||||
data?.status == 2 ? "Diskusi telah ditutup" : data?.isActive == false ? "Diskusi telah diarsipkan" : "Hanya anggota diskusi yang dapat memberikan komentar"
|
data?.status == 2 ? "Diskusi telah ditutup" : data?.isActive == false ? "Diskusi telah diarsipkan" : "Hanya anggota diskusi yang dapat memberikan komentar"
|
||||||
}
|
}
|
||||||
@@ -427,35 +415,25 @@ export default function DetailDiscussionGeneral() {
|
|||||||
<DrawerBottom animation="slide" isVisible={isVisible} setVisible={setVisible} title="Komentar">
|
<DrawerBottom animation="slide" isVisible={isVisible} setVisible={setVisible} title="Komentar">
|
||||||
<View style={Styles.rowItemsCenter}>
|
<View style={Styles.rowItemsCenter}>
|
||||||
<MenuItemRow
|
<MenuItemRow
|
||||||
icon={<MaterialCommunityIcons name="pencil-outline" color={colors.text} size={25} />}
|
icon={<MaterialCommunityIcons name="pencil-outline" color="black" size={25} />}
|
||||||
title="Edit"
|
title="Edit"
|
||||||
onPress={() => { handleViewEditKomentar() }}
|
onPress={() => { handleViewEditKomentar() }}
|
||||||
/>
|
/>
|
||||||
<MenuItemRow
|
<MenuItemRow
|
||||||
icon={<Ionicons name="trash-outline" color={colors.text} size={25} />}
|
icon={<MaterialIcons name="delete" color="black" size={25} />}
|
||||||
title="Hapus"
|
title="Hapus"
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
setVisible(false)
|
AlertKonfirmasi({
|
||||||
setTimeout(() => {
|
title: 'Konfirmasi',
|
||||||
setShowDeleteModal(true)
|
desc: 'Apakah anda yakin ingin menghapus komentar?',
|
||||||
}, 600)
|
onPress: () => {
|
||||||
|
handleDeleteKomentar()
|
||||||
|
}
|
||||||
|
})
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
</View>
|
</View>
|
||||||
</DrawerBottom>
|
</DrawerBottom>
|
||||||
|
|
||||||
<ModalConfirmation
|
|
||||||
visible={showDeleteModal}
|
|
||||||
title="Konfirmasi"
|
|
||||||
message="Apakah anda yakin ingin menghapus komentar?"
|
|
||||||
onConfirm={() => {
|
|
||||||
setShowDeleteModal(false)
|
|
||||||
handleDeleteKomentar()
|
|
||||||
}}
|
|
||||||
onCancel={() => setShowDeleteModal(false)}
|
|
||||||
confirmText="Hapus"
|
|
||||||
cancelText="Batal"
|
|
||||||
/>
|
|
||||||
</>
|
</>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@@ -9,7 +9,6 @@ import Styles from "@/constants/Styles";
|
|||||||
import { apiAddMemberDiscussionGeneral, apiGetDiscussionGeneralOne, apiGetUser } from "@/lib/api";
|
import { apiAddMemberDiscussionGeneral, apiGetDiscussionGeneralOne, apiGetUser } from "@/lib/api";
|
||||||
import { setUpdateDiscussionGeneralDetail } from "@/lib/discussionGeneralDetail";
|
import { setUpdateDiscussionGeneralDetail } from "@/lib/discussionGeneralDetail";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { AntDesign } from "@expo/vector-icons";
|
import { AntDesign } from "@expo/vector-icons";
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
@@ -27,7 +26,6 @@ export default function AddMemberDiscussionDetail() {
|
|||||||
const dispatch = useDispatch()
|
const dispatch = useDispatch()
|
||||||
const update = useSelector((state: any) => state.discussionGeneralDetailUpdate)
|
const update = useSelector((state: any) => state.discussionGeneralDetailUpdate)
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
const { colors } = useTheme();
|
|
||||||
const { id } = useLocalSearchParams<{ id: string }>()
|
const { id } = useLocalSearchParams<{ id: string }>()
|
||||||
const [dataOld, setDataOld] = useState<Props[]>([])
|
const [dataOld, setDataOld] = useState<Props[]>([])
|
||||||
const [data, setData] = useState<Props[]>([])
|
const [data, setData] = useState<Props[]>([])
|
||||||
@@ -84,11 +82,9 @@ export default function AddMemberDiscussionDetail() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal menambahkan anggota"
|
Toast.show({ type: 'small', text1: 'Gagal menambahkan anggota', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
@@ -96,7 +92,7 @@ export default function AddMemberDiscussionDetail() {
|
|||||||
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
||||||
@@ -129,7 +125,7 @@ export default function AddMemberDiscussionDetail() {
|
|||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<View style={[Styles.p15, Styles.flex1, { backgroundColor: colors.background }]}>
|
<View style={[Styles.p15]}>
|
||||||
<InputSearch onChange={setSearch} value={search} />
|
<InputSearch onChange={setSearch} value={search} />
|
||||||
|
|
||||||
{
|
{
|
||||||
@@ -151,7 +147,7 @@ export default function AddMemberDiscussionDetail() {
|
|||||||
</View>
|
</View>
|
||||||
|
|
||||||
:
|
:
|
||||||
<Text style={[Styles.textDefault, Styles.pv05, Styles.textCenter, { color: colors.dimmed }]}>Tidak ada member yang dipilih</Text>
|
<Text style={[Styles.textDefault, Styles.cGray, Styles.pv05, { textAlign: 'center' }]}>Tidak ada member yang dipilih</Text>
|
||||||
}
|
}
|
||||||
<ScrollView
|
<ScrollView
|
||||||
showsVerticalScrollIndicator={false}
|
showsVerticalScrollIndicator={false}
|
||||||
@@ -164,7 +160,7 @@ export default function AddMemberDiscussionDetail() {
|
|||||||
return (
|
return (
|
||||||
<Pressable
|
<Pressable
|
||||||
key={index}
|
key={index}
|
||||||
style={[Styles.itemSelectModal, { borderColor: colors.icon + '20' }]}
|
style={[Styles.itemSelectModal]}
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
!found && onChoose(item.id, item.name, item.img)
|
!found && onChoose(item.id, item.name, item.img)
|
||||||
}}
|
}}
|
||||||
@@ -174,22 +170,22 @@ export default function AddMemberDiscussionDetail() {
|
|||||||
<View style={[Styles.ml10]}>
|
<View style={[Styles.ml10]}>
|
||||||
<Text style={[Styles.textDefault]}>{item.name}</Text>
|
<Text style={[Styles.textDefault]}>{item.name}</Text>
|
||||||
{
|
{
|
||||||
found && <Text style={[Styles.textInformation, { color: colors.dimmed }]}>sudah menjadi anggota</Text>
|
found && <Text style={[Styles.textInformation, Styles.cGray]}>sudah menjadi anggota</Text>
|
||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
{
|
{
|
||||||
selectMember.some((i: any) => i.idUser == item.id) && <AntDesign name="check" size={20} color={colors.text} />
|
selectMember.some((i: any) => i.idUser == item.id) && <AntDesign name="check" size={20} color={'black'} />
|
||||||
}
|
}
|
||||||
</Pressable>
|
</Pressable>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
:
|
:
|
||||||
<Text style={[Styles.textDefault, Styles.textCenter]}>Tidak ada data</Text>
|
<Text style={[Styles.textDefault, { textAlign: 'center' }]}>Tidak ada data</Text>
|
||||||
}
|
}
|
||||||
</ScrollView>
|
</ScrollView>
|
||||||
</View>
|
</View>
|
||||||
</>
|
</SafeAreaView>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@@ -5,7 +5,7 @@ import ButtonSelect from "@/components/buttonSelect";
|
|||||||
import DrawerBottom from "@/components/drawerBottom";
|
import DrawerBottom from "@/components/drawerBottom";
|
||||||
import ImageUser from "@/components/imageNew";
|
import ImageUser from "@/components/imageNew";
|
||||||
import { InputForm } from "@/components/inputForm";
|
import { InputForm } from "@/components/inputForm";
|
||||||
import LoadingCenter from "@/components/loadingCenter";
|
import LoadingOverlay from "@/components/loadingOverlay";
|
||||||
import MenuItemRow from "@/components/menuItemRow";
|
import MenuItemRow from "@/components/menuItemRow";
|
||||||
import ModalSelect from "@/components/modalSelect";
|
import ModalSelect from "@/components/modalSelect";
|
||||||
import SelectForm from "@/components/selectForm";
|
import SelectForm from "@/components/selectForm";
|
||||||
@@ -16,7 +16,6 @@ import { apiCreateDiscussionGeneral } from "@/lib/api";
|
|||||||
import { setUpdateDiscussionGeneralDetail } from "@/lib/discussionGeneralDetail";
|
import { setUpdateDiscussionGeneralDetail } from "@/lib/discussionGeneralDetail";
|
||||||
import { setMemberChoose } from "@/lib/memberChoose";
|
import { setMemberChoose } from "@/lib/memberChoose";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { Ionicons, MaterialCommunityIcons } from "@expo/vector-icons";
|
import { Ionicons, MaterialCommunityIcons } from "@expo/vector-icons";
|
||||||
import * as DocumentPicker from "expo-document-picker";
|
import * as DocumentPicker from "expo-document-picker";
|
||||||
import { router, Stack } from "expo-router";
|
import { router, Stack } from "expo-router";
|
||||||
@@ -28,7 +27,6 @@ import { useDispatch, useSelector } from "react-redux";
|
|||||||
|
|
||||||
export default function CreateDiscussionGeneral() {
|
export default function CreateDiscussionGeneral() {
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
const { colors } = useTheme();
|
|
||||||
const entityUser = useSelector((state: any) => state.user);
|
const entityUser = useSelector((state: any) => state.user);
|
||||||
const userLogin = useSelector((state: any) => state.entities)
|
const userLogin = useSelector((state: any) => state.entities)
|
||||||
const [chooseGroup, setChooseGroup] = useState({ val: "", label: "" });
|
const [chooseGroup, setChooseGroup] = useState({ val: "", label: "" });
|
||||||
@@ -156,18 +154,16 @@ export default function CreateDiscussionGeneral() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Gagal menambahkan data"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={[Styles.flex1, { backgroundColor: colors.background }]}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => (
|
// headerLeft: () => (
|
||||||
@@ -208,8 +204,8 @@ export default function CreateDiscussionGeneral() {
|
|||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
{loading && <LoadingCenter />}
|
<LoadingOverlay visible={loading} />
|
||||||
<ScrollView showsVerticalScrollIndicator={false} style={[Styles.h100, Styles.flex1, { backgroundColor: colors.background }]}>
|
<ScrollView showsVerticalScrollIndicator={false} style={[Styles.h100]}>
|
||||||
<View style={[Styles.p15, Styles.mb100]}>
|
<View style={[Styles.p15, Styles.mb100]}>
|
||||||
{
|
{
|
||||||
(entityUser.role == "supadmin" ||
|
(entityUser.role == "supadmin" ||
|
||||||
@@ -219,7 +215,6 @@ export default function CreateDiscussionGeneral() {
|
|||||||
placeholder="Pilih Lembaga Desa"
|
placeholder="Pilih Lembaga Desa"
|
||||||
value={chooseGroup.label}
|
value={chooseGroup.label}
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
setValChoose(chooseGroup.val);
|
setValChoose(chooseGroup.val);
|
||||||
setValSelect("group");
|
setValSelect("group");
|
||||||
@@ -236,7 +231,6 @@ export default function CreateDiscussionGeneral() {
|
|||||||
placeholder="Judul"
|
placeholder="Judul"
|
||||||
required
|
required
|
||||||
error={error.title}
|
error={error.title}
|
||||||
bg={colors.card}
|
|
||||||
errorText="Judul tidak boleh kosong"
|
errorText="Judul tidak boleh kosong"
|
||||||
onChange={(val) => { validationForm("title", val) }}
|
onChange={(val) => { validationForm("title", val) }}
|
||||||
/>
|
/>
|
||||||
@@ -246,7 +240,6 @@ export default function CreateDiscussionGeneral() {
|
|||||||
placeholder="Hal yang didiskusikan"
|
placeholder="Hal yang didiskusikan"
|
||||||
required
|
required
|
||||||
error={error.desc}
|
error={error.desc}
|
||||||
bg={colors.card}
|
|
||||||
errorText="Diskusi tidak boleh kosong"
|
errorText="Diskusi tidak boleh kosong"
|
||||||
onChange={(val) => { validationForm("desc", val) }}
|
onChange={(val) => { validationForm("desc", val) }}
|
||||||
multiline
|
multiline
|
||||||
@@ -255,27 +248,21 @@ export default function CreateDiscussionGeneral() {
|
|||||||
{
|
{
|
||||||
fileForm.length > 0
|
fileForm.length > 0
|
||||||
&&
|
&&
|
||||||
<>
|
<View style={[Styles.borderAll, Styles.round10, Styles.p10, Styles.mb10]}>
|
||||||
<View style={[Styles.rowSpaceBetween, Styles.mv05]}>
|
<Text style={[Styles.textDefaultSemiBold]}>File</Text>
|
||||||
<Text style={[Styles.textDefaultSemiBold]}>File</Text>
|
{
|
||||||
<Text style={[Styles.textDefault]}>{fileForm.length} file</Text>
|
fileForm.map((item, index) => (
|
||||||
</View>
|
<BorderBottomItem
|
||||||
<View style={[Styles.borderAll, Styles.round05, Styles.p10, Styles.mb10, { backgroundColor: colors.card, borderColor: colors.icon + '20' }]}>
|
key={index}
|
||||||
{
|
borderType={fileForm.length > 1 ? "bottom" : "none"}
|
||||||
fileForm.map((item, index) => (
|
icon={<MaterialCommunityIcons name="file-outline" size={25} color="black" />}
|
||||||
<BorderBottomItem
|
title={item.name}
|
||||||
key={index}
|
titleWeight="normal"
|
||||||
borderType={fileForm.length - 1 == index ? "none" : "bottom"}
|
onPress={() => { setIndexDelFile(index); setModalFile(true) }}
|
||||||
icon={<MaterialCommunityIcons name="file-outline" size={25} color={colors.text} />}
|
/>
|
||||||
title={item.name}
|
))
|
||||||
bgColor="transparent"
|
}
|
||||||
titleWeight="normal"
|
</View>
|
||||||
onPress={() => { setIndexDelFile(index); setModalFile(true) }}
|
|
||||||
/>
|
|
||||||
))
|
|
||||||
}
|
|
||||||
</View>
|
|
||||||
</>
|
|
||||||
}
|
}
|
||||||
<ButtonSelect
|
<ButtonSelect
|
||||||
value="Pilih Anggota"
|
value="Pilih Anggota"
|
||||||
@@ -300,22 +287,21 @@ export default function CreateDiscussionGeneral() {
|
|||||||
entitiesMember.length > 0 &&
|
entitiesMember.length > 0 &&
|
||||||
<View>
|
<View>
|
||||||
<View style={[Styles.rowSpaceBetween, Styles.mv05]}>
|
<View style={[Styles.rowSpaceBetween, Styles.mv05]}>
|
||||||
<Text style={[Styles.textDefaultSemiBold]}>Anggota</Text>
|
<Text>Anggota</Text>
|
||||||
<Text style={[Styles.textDefault]}>{entitiesMember.length} Anggota</Text>
|
<Text>Total {entitiesMember.length} Anggota</Text>
|
||||||
</View>
|
</View>
|
||||||
|
|
||||||
<View style={[Styles.borderAll, Styles.round05, Styles.p10, { backgroundColor: colors.card, borderColor: colors.icon + '20' }]}>
|
<View style={[Styles.borderAll, Styles.round10, Styles.p10]}>
|
||||||
{
|
{
|
||||||
entitiesMember.map((item: { img: any; name: any; }, index: any) => {
|
entitiesMember.map((item: { img: any; name: any; }, index: any) => {
|
||||||
return (
|
return (
|
||||||
<BorderBottomItem
|
<BorderBottomItem
|
||||||
key={index}
|
key={index}
|
||||||
borderType={entitiesMember.length - 1 == index ? "none" : "bottom"}
|
borderType="bottom"
|
||||||
icon={
|
icon={
|
||||||
<ImageUser src={`${ConstEnv.url_storage}/files/${item.img}`} size="xs" />
|
<ImageUser src={`${ConstEnv.url_storage}/files/${item.img}`} size="sm" />
|
||||||
}
|
}
|
||||||
title={item.name}
|
title={item.name}
|
||||||
bgColor="transparent"
|
|
||||||
/>
|
/>
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
@@ -341,7 +327,7 @@ export default function CreateDiscussionGeneral() {
|
|||||||
<DrawerBottom animation="slide" isVisible={isModalFile} setVisible={setModalFile} title="Menu">
|
<DrawerBottom animation="slide" isVisible={isModalFile} setVisible={setModalFile} title="Menu">
|
||||||
<View style={Styles.rowItemsCenter}>
|
<View style={Styles.rowItemsCenter}>
|
||||||
<MenuItemRow
|
<MenuItemRow
|
||||||
icon={<Ionicons name="trash-outline" color={colors.text} size={25} />}
|
icon={<Ionicons name="trash" color="black" size={25} />}
|
||||||
title="Hapus"
|
title="Hapus"
|
||||||
onPress={() => { deleteFile(indexDelFile) }}
|
onPress={() => { deleteFile(indexDelFile) }}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -5,13 +5,12 @@ import ButtonSaveHeader from "@/components/buttonSaveHeader";
|
|||||||
import ButtonSelect from "@/components/buttonSelect";
|
import ButtonSelect from "@/components/buttonSelect";
|
||||||
import DrawerBottom from "@/components/drawerBottom";
|
import DrawerBottom from "@/components/drawerBottom";
|
||||||
import { InputForm } from "@/components/inputForm";
|
import { InputForm } from "@/components/inputForm";
|
||||||
import LoadingCenter from "@/components/loadingCenter";
|
import LoadingOverlay from "@/components/loadingOverlay";
|
||||||
import MenuItemRow from "@/components/menuItemRow";
|
import MenuItemRow from "@/components/menuItemRow";
|
||||||
import Styles from "@/constants/Styles";
|
import Styles from "@/constants/Styles";
|
||||||
import { apiEditDiscussionGeneral, apiGetDiscussionGeneralOne } from "@/lib/api";
|
import { apiEditDiscussionGeneral, apiGetDiscussionGeneralOne } from "@/lib/api";
|
||||||
import { setUpdateDiscussionGeneralDetail } from "@/lib/discussionGeneralDetail";
|
import { setUpdateDiscussionGeneralDetail } from "@/lib/discussionGeneralDetail";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { Ionicons, MaterialCommunityIcons } from "@expo/vector-icons";
|
import { Ionicons, MaterialCommunityIcons } from "@expo/vector-icons";
|
||||||
import * as DocumentPicker from "expo-document-picker";
|
import * as DocumentPicker from "expo-document-picker";
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
@@ -22,7 +21,6 @@ import { useDispatch, useSelector } from "react-redux";
|
|||||||
|
|
||||||
export default function EditDiscussionGeneral() {
|
export default function EditDiscussionGeneral() {
|
||||||
const { token, decryptToken } = useAuthSession();
|
const { token, decryptToken } = useAuthSession();
|
||||||
const { colors } = useTheme();
|
|
||||||
const { id } = useLocalSearchParams<{ id: string }>();
|
const { id } = useLocalSearchParams<{ id: string }>();
|
||||||
const [disableBtn, setDisableBtn] = useState(false)
|
const [disableBtn, setDisableBtn] = useState(false)
|
||||||
const dispatch = useDispatch()
|
const dispatch = useDispatch()
|
||||||
@@ -154,21 +152,17 @@ export default function EditDiscussionGeneral() {
|
|||||||
dispatch(setUpdateDiscussionGeneralDetail(!update))
|
dispatch(setUpdateDiscussionGeneralDetail(!update))
|
||||||
Toast.show({ type: 'small', text1: 'Berhasil mengubah data', })
|
Toast.show({ type: 'small', text1: 'Berhasil mengubah data', })
|
||||||
router.back();
|
router.back();
|
||||||
} else {
|
|
||||||
Toast.show({ type: 'small', text1: 'Gagal mengubah data', })
|
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Gagal mengubah data"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={[Styles.flex1, { backgroundColor: colors.background }]}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => (
|
// headerLeft: () => (
|
||||||
@@ -203,15 +197,14 @@ export default function EditDiscussionGeneral() {
|
|||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
{loading && <LoadingCenter />}
|
<LoadingOverlay visible={loading} />
|
||||||
<ScrollView showsVerticalScrollIndicator={false} style={[Styles.h100, Styles.flex1, { backgroundColor: colors.background }]}>
|
<ScrollView showsVerticalScrollIndicator={false} style={[Styles.h100]}>
|
||||||
<View style={[Styles.p15]}>
|
<View style={[Styles.p15]}>
|
||||||
<InputForm
|
<InputForm
|
||||||
label="Judul"
|
label="Judul"
|
||||||
type="default"
|
type="default"
|
||||||
placeholder="Judul"
|
placeholder="Judul"
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
|
||||||
error={error.title}
|
error={error.title}
|
||||||
value={dataForm.title}
|
value={dataForm.title}
|
||||||
errorText="Judul tidak boleh kosong"
|
errorText="Judul tidak boleh kosong"
|
||||||
@@ -222,7 +215,6 @@ export default function EditDiscussionGeneral() {
|
|||||||
type="default"
|
type="default"
|
||||||
placeholder="Hal yang didiskusikan"
|
placeholder="Hal yang didiskusikan"
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
|
||||||
error={error.desc}
|
error={error.desc}
|
||||||
value={dataForm.desc}
|
value={dataForm.desc}
|
||||||
errorText="Diskusi tidak boleh kosong"
|
errorText="Diskusi tidak boleh kosong"
|
||||||
@@ -233,40 +225,33 @@ export default function EditDiscussionGeneral() {
|
|||||||
{
|
{
|
||||||
(fileForm.length > 0 || dataFile.filter((val) => !val.delete).length > 0)
|
(fileForm.length > 0 || dataFile.filter((val) => !val.delete).length > 0)
|
||||||
&&
|
&&
|
||||||
<>
|
<View style={[Styles.borderAll, Styles.round10, Styles.p10, Styles.mb10]}>
|
||||||
<View style={[Styles.rowSpaceBetween, Styles.mv05]}>
|
<Text style={[Styles.textDefaultSemiBold]}>File</Text>
|
||||||
<Text style={[Styles.textDefaultSemiBold]}>File</Text>
|
{
|
||||||
<Text style={[Styles.textDefault]}>{fileForm.length + dataFile.filter((val) => !val.delete).length} file</Text>
|
dataFile.filter((val) => !val.delete).map((item, index) => (
|
||||||
</View>
|
<BorderBottomItem
|
||||||
<View style={[Styles.borderAll, Styles.round05, Styles.p10, Styles.mb10, { backgroundColor: colors.card, borderColor: colors.icon + '20' }]}>
|
key={index}
|
||||||
{
|
borderType={(fileForm.length + dataFile.length) > 1 ? "bottom" : "none"}
|
||||||
dataFile.filter((val) => !val.delete).map((item, index) => (
|
icon={<MaterialCommunityIcons name="file-outline" size={25} color="black" />}
|
||||||
<BorderBottomItem
|
title={item.name + '.' + item.extension}
|
||||||
key={index}
|
titleWeight="normal"
|
||||||
borderType={dataFile.filter((val) => !val.delete).length - 1 == index && fileForm.length == 0 ? "none" : "bottom"}
|
onPress={() => { setIndexDelFile({ id: item.id, cat: "oldFile" }); setModalFile(true) }}
|
||||||
icon={<MaterialCommunityIcons name="file-outline" size={25} color={colors.text} />}
|
/>
|
||||||
title={item.name + '.' + item.extension}
|
))
|
||||||
titleWeight="normal"
|
}
|
||||||
bgColor="transparent"
|
{
|
||||||
onPress={() => { setIndexDelFile({ id: item.id, cat: "oldFile" }); setModalFile(true) }}
|
fileForm.map((item, index) => (
|
||||||
/>
|
<BorderBottomItem
|
||||||
))
|
key={index}
|
||||||
}
|
borderType={(fileForm.length + dataFile.length) > 1 ? "bottom" : "none"}
|
||||||
{
|
icon={<MaterialCommunityIcons name="file-outline" size={25} color="black" />}
|
||||||
fileForm.map((item, index) => (
|
title={item.name}
|
||||||
<BorderBottomItem
|
titleWeight="normal"
|
||||||
key={index}
|
onPress={() => { setIndexDelFile({ id: index, cat: "newFile" }); setModalFile(true) }}
|
||||||
borderType={fileForm.length - 1 == index ? "none" : "bottom"}
|
/>
|
||||||
icon={<MaterialCommunityIcons name="file-outline" size={25} color={colors.text} />}
|
))
|
||||||
title={item.name}
|
}
|
||||||
titleWeight="normal"
|
</View>
|
||||||
bgColor="transparent"
|
|
||||||
onPress={() => { setIndexDelFile({ id: index, cat: "newFile" }); setModalFile(true) }}
|
|
||||||
/>
|
|
||||||
))
|
|
||||||
}
|
|
||||||
</View>
|
|
||||||
</>
|
|
||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
</ScrollView>
|
</ScrollView>
|
||||||
@@ -274,7 +259,7 @@ export default function EditDiscussionGeneral() {
|
|||||||
<DrawerBottom animation="slide" isVisible={isModalFile} setVisible={setModalFile} title="Menu">
|
<DrawerBottom animation="slide" isVisible={isModalFile} setVisible={setModalFile} title="Menu">
|
||||||
<View style={Styles.rowItemsCenter}>
|
<View style={Styles.rowItemsCenter}>
|
||||||
<MenuItemRow
|
<MenuItemRow
|
||||||
icon={<Ionicons name="trash-outline" color={colors.text} size={25} />}
|
icon={<Ionicons name="trash" color="black" size={25} />}
|
||||||
title="Hapus"
|
title="Hapus"
|
||||||
onPress={() => { deleteFile(indexDelFile.id, indexDelFile.cat) }}
|
onPress={() => { deleteFile(indexDelFile.id, indexDelFile.cat) }}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -4,12 +4,10 @@ import InputSearch from "@/components/inputSearch";
|
|||||||
import LabelStatus from "@/components/labelStatus";
|
import LabelStatus from "@/components/labelStatus";
|
||||||
import SkeletonContent from "@/components/skeletonContent";
|
import SkeletonContent from "@/components/skeletonContent";
|
||||||
import Text from "@/components/Text";
|
import Text from "@/components/Text";
|
||||||
import WrapTab from "@/components/wrapTab";
|
|
||||||
import { ColorsStatus } from "@/constants/ColorsStatus";
|
import { ColorsStatus } from "@/constants/ColorsStatus";
|
||||||
import Styles from "@/constants/Styles";
|
import Styles from "@/constants/Styles";
|
||||||
import { apiGetDiscussionGeneral } from "@/lib/api";
|
import { apiGetDiscussionGeneral } from "@/lib/api";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { AntDesign, Feather, Ionicons, MaterialIcons } from "@expo/vector-icons";
|
import { AntDesign, Feather, Ionicons, MaterialIcons } from "@expo/vector-icons";
|
||||||
import { router, useLocalSearchParams } from "expo-router";
|
import { router, useLocalSearchParams } from "expo-router";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
@@ -29,7 +27,6 @@ type Props = {
|
|||||||
export default function Discussion() {
|
export default function Discussion() {
|
||||||
const entityUser = useSelector((state: any) => state.user)
|
const entityUser = useSelector((state: any) => state.user)
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
const { colors } = useTheme();
|
|
||||||
const { active, group } = useLocalSearchParams<{ active?: string, group?: string }>()
|
const { active, group } = useLocalSearchParams<{ active?: string, group?: string }>()
|
||||||
const [search, setSearch] = useState('')
|
const [search, setSearch] = useState('')
|
||||||
const [nameGroup, setNameGroup] = useState('')
|
const [nameGroup, setNameGroup] = useState('')
|
||||||
@@ -99,26 +96,26 @@ export default function Discussion() {
|
|||||||
})
|
})
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<View style={[Styles.p15, Styles.flex1, { backgroundColor: colors.background }]}>
|
<View style={[Styles.p15, { flex: 1 }]}>
|
||||||
<View>
|
<View>
|
||||||
{
|
{
|
||||||
entityUser.role != "user" && entityUser.role != "coadmin" &&
|
entityUser.role != "user" && entityUser.role != "coadmin" &&
|
||||||
<WrapTab>
|
<View style={[Styles.wrapBtnTab]}>
|
||||||
<ButtonTab
|
<ButtonTab
|
||||||
active={status == "false" ? "false" : "true"}
|
active={status == "false" ? "false" : "true"}
|
||||||
value="true"
|
value="true"
|
||||||
onPress={() => { setStatus("true") }}
|
onPress={() => { setStatus("true") }}
|
||||||
label="Aktif"
|
label="Aktif"
|
||||||
icon={<Feather name="check-circle" color={status == "false" ? colors.dimmed : 'white'} size={20} />}
|
icon={<Feather name="check-circle" color={status == "false" ? 'black' : 'white'} size={20} />}
|
||||||
n={2} />
|
n={2} />
|
||||||
<ButtonTab
|
<ButtonTab
|
||||||
active={status == "false" ? "false" : "true"}
|
active={status == "false" ? "false" : "true"}
|
||||||
value="false"
|
value="false"
|
||||||
onPress={() => { setStatus("false") }}
|
onPress={() => { setStatus("false") }}
|
||||||
label="Arsip"
|
label="Arsip"
|
||||||
icon={<AntDesign name="closecircleo" color={status == "true" ? colors.dimmed : 'white'} size={20} />}
|
icon={<AntDesign name="closecircleo" color={status == "true" ? 'black' : 'white'} size={20} />}
|
||||||
n={2} />
|
n={2} />
|
||||||
</WrapTab>
|
</View>
|
||||||
}
|
}
|
||||||
|
|
||||||
<InputSearch onChange={setSearch} />
|
<InputSearch onChange={setSearch} />
|
||||||
@@ -130,7 +127,7 @@ export default function Discussion() {
|
|||||||
</View>
|
</View>
|
||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
<View style={[Styles.flex2, Styles.mt05]}>
|
<View style={[{ flex: 2 }, Styles.mt05]}>
|
||||||
{
|
{
|
||||||
loading ?
|
loading ?
|
||||||
arrSkeleton.map((item: any, i: number) => {
|
arrSkeleton.map((item: any, i: number) => {
|
||||||
@@ -148,14 +145,13 @@ export default function Discussion() {
|
|||||||
renderItem={({ item, index }: { item: Props, index: number }) => {
|
renderItem={({ item, index }: { item: Props, index: number }) => {
|
||||||
return (
|
return (
|
||||||
<BorderBottomItem
|
<BorderBottomItem
|
||||||
bgColor="transparent"
|
|
||||||
key={index}
|
key={index}
|
||||||
onPress={() => { router.push(`/discussion/${item.id}`) }}
|
onPress={() => { router.push(`/discussion/${item.id}`) }}
|
||||||
borderType="bottom"
|
borderType="bottom"
|
||||||
icon={
|
icon={
|
||||||
// <View style={[Styles.iconContent]}>
|
<View style={[Styles.iconContent, ColorsStatus.lightGreen]}>
|
||||||
<MaterialIcons name="chat" size={25} color={colors.text} />
|
<MaterialIcons name="chat" size={25} color={'#384288'} />
|
||||||
// </View>
|
</View>
|
||||||
}
|
}
|
||||||
title={item.title}
|
title={item.title}
|
||||||
subtitle={
|
subtitle={
|
||||||
@@ -165,8 +161,8 @@ export default function Discussion() {
|
|||||||
desc={item.desc.replace(/<[^>]*>?/gm, ' ').replace(/\r?\n|\r/g, ' ')}
|
desc={item.desc.replace(/<[^>]*>?/gm, ' ').replace(/\r?\n|\r/g, ' ')}
|
||||||
leftBottomInfo={
|
leftBottomInfo={
|
||||||
<View style={[Styles.rowItemsCenter]}>
|
<View style={[Styles.rowItemsCenter]}>
|
||||||
<Ionicons name="chatbox-ellipses-outline" size={18} color={colors.dimmed} style={Styles.mr05} />
|
<Ionicons name="chatbox-ellipses-outline" size={18} color="grey" style={Styles.mr05} />
|
||||||
<Text style={[Styles.textInformation, { color: colors.dimmed }, Styles.mb05]}>Diskusikan</Text>
|
<Text style={[Styles.textInformation, Styles.cGray, Styles.mb05]}>Diskusikan</Text>
|
||||||
</View>
|
</View>
|
||||||
}
|
}
|
||||||
rightBottomInfo={`${item.total_komentar} Komentar`}
|
rightBottomInfo={`${item.total_komentar} Komentar`}
|
||||||
@@ -182,12 +178,11 @@ export default function Discussion() {
|
|||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={handleRefresh}
|
onRefresh={handleRefresh}
|
||||||
tintColor={colors.icon}
|
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
:
|
:
|
||||||
<Text style={[Styles.textDefault, Styles.textCenter, { color: colors.dimmed }]}>Tidak ada data</Text>
|
<Text style={[Styles.textDefault, Styles.cGray, { textAlign: 'center' }]}>Tidak ada data</Text>
|
||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
|
import AlertKonfirmasi from "@/components/alertKonfirmasi";
|
||||||
import AppHeader from "@/components/AppHeader";
|
import AppHeader from "@/components/AppHeader";
|
||||||
import BorderBottomItem from "@/components/borderBottomItem";
|
import BorderBottomItem from "@/components/borderBottomItem";
|
||||||
import DrawerBottom from "@/components/drawerBottom";
|
import DrawerBottom from "@/components/drawerBottom";
|
||||||
import ImageUser from "@/components/imageNew";
|
import ImageUser from "@/components/imageNew";
|
||||||
import MenuItemRow from "@/components/menuItemRow";
|
import MenuItemRow from "@/components/menuItemRow";
|
||||||
import ModalConfirmation from "@/components/ModalConfirmation";
|
|
||||||
import SkeletonTwoItem from "@/components/skeletonTwoItem";
|
import SkeletonTwoItem from "@/components/skeletonTwoItem";
|
||||||
import Text from '@/components/Text';
|
import Text from '@/components/Text';
|
||||||
import { ColorsStatus } from "@/constants/ColorsStatus";
|
import { ColorsStatus } from "@/constants/ColorsStatus";
|
||||||
@@ -11,7 +11,6 @@ import { ConstEnv } from "@/constants/ConstEnv";
|
|||||||
import Styles from "@/constants/Styles";
|
import Styles from "@/constants/Styles";
|
||||||
import { apiDeleteMemberDiscussionGeneral, apiGetDiscussionGeneralOne } from "@/lib/api";
|
import { apiDeleteMemberDiscussionGeneral, apiGetDiscussionGeneralOne } from "@/lib/api";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { Feather, MaterialCommunityIcons } from "@expo/vector-icons";
|
import { Feather, MaterialCommunityIcons } from "@expo/vector-icons";
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
@@ -27,7 +26,6 @@ type Props = {
|
|||||||
|
|
||||||
export default function MemberDiscussionDetail() {
|
export default function MemberDiscussionDetail() {
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
const { colors } = useTheme();
|
|
||||||
const entityUser = useSelector((state: any) => state.user)
|
const entityUser = useSelector((state: any) => state.user)
|
||||||
const { id } = useLocalSearchParams<{ id: string }>()
|
const { id } = useLocalSearchParams<{ id: string }>()
|
||||||
const [data, setData] = useState<Props[]>([])
|
const [data, setData] = useState<Props[]>([])
|
||||||
@@ -36,8 +34,6 @@ export default function MemberDiscussionDetail() {
|
|||||||
const update = useSelector((state: any) => state.discussionGeneralDetailUpdate)
|
const update = useSelector((state: any) => state.discussionGeneralDetailUpdate)
|
||||||
const [loading, setLoading] = useState(true)
|
const [loading, setLoading] = useState(true)
|
||||||
const arrSkeleton = Array.from({ length: 5 }, (_, index) => index)
|
const arrSkeleton = Array.from({ length: 5 }, (_, index) => index)
|
||||||
const [showDeleteModal, setShowDeleteModal] = useState(false)
|
|
||||||
|
|
||||||
|
|
||||||
async function handleLoad(loading: boolean) {
|
async function handleLoad(loading: boolean) {
|
||||||
try {
|
try {
|
||||||
@@ -67,18 +63,15 @@ export default function MemberDiscussionDetail() {
|
|||||||
await apiDeleteMemberDiscussionGeneral({ user: hasil, idUser: chooseUser.idUser }, id)
|
await apiDeleteMemberDiscussionGeneral({ user: hasil, idUser: chooseUser.idUser }, id)
|
||||||
Toast.show({ type: 'small', text1: 'Berhasil mengeluarkan anggota dari diskusi', })
|
Toast.show({ type: 'small', text1: 'Berhasil mengeluarkan anggota dari diskusi', })
|
||||||
handleLoad(false)
|
handleLoad(false)
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal mengeluarkan anggota"
|
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setModal(false)
|
setModal(false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={[Styles.flex1, { backgroundColor: colors.background }]}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
||||||
@@ -93,10 +86,10 @@ export default function MemberDiscussionDetail() {
|
|||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<ScrollView style={[Styles.h100, Styles.flex1, { backgroundColor: colors.background }]}>
|
<ScrollView>
|
||||||
<View style={[Styles.p15]}>
|
<View style={[Styles.p15]}>
|
||||||
<Text style={[Styles.textDefault, Styles.mv05]}>{data.length} Anggota</Text>
|
<Text style={[Styles.textDefault, Styles.mv05]}>{data.length} Anggota</Text>
|
||||||
<View style={[Styles.wrapPaper, { backgroundColor: colors.card, borderColor: colors.background }]}>
|
<View style={[Styles.wrapPaper, Styles.mb100]}>
|
||||||
{
|
{
|
||||||
entityUser.role != "user" && entityUser.role != "coadmin" &&
|
entityUser.role != "user" && entityUser.role != "coadmin" &&
|
||||||
<BorderBottomItem
|
<BorderBottomItem
|
||||||
@@ -142,7 +135,7 @@ export default function MemberDiscussionDetail() {
|
|||||||
<DrawerBottom animation="slide" isVisible={isModal} setVisible={setModal} title={chooseUser.name}>
|
<DrawerBottom animation="slide" isVisible={isModal} setVisible={setModal} title={chooseUser.name}>
|
||||||
<View style={Styles.rowItemsCenter}>
|
<View style={Styles.rowItemsCenter}>
|
||||||
<MenuItemRow
|
<MenuItemRow
|
||||||
icon={<MaterialCommunityIcons name="account-eye" color={colors.text} size={25} />}
|
icon={<MaterialCommunityIcons name="account-eye" color="black" size={25} />}
|
||||||
title="Lihat Profil"
|
title="Lihat Profil"
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
setModal(false)
|
setModal(false)
|
||||||
@@ -152,32 +145,24 @@ export default function MemberDiscussionDetail() {
|
|||||||
{
|
{
|
||||||
entityUser.role != "user" && entityUser.role != "coadmin" &&
|
entityUser.role != "user" && entityUser.role != "coadmin" &&
|
||||||
<MenuItemRow
|
<MenuItemRow
|
||||||
icon={<MaterialCommunityIcons name="account-remove" color={colors.text} size={25} />}
|
icon={<MaterialCommunityIcons name="account-remove" color="black" size={25} />}
|
||||||
title="Keluarkan"
|
title="Keluarkan"
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
setModal(false)
|
setModal(false)
|
||||||
setTimeout(() => {
|
AlertKonfirmasi({
|
||||||
setShowDeleteModal(true)
|
title: 'Konfirmasi',
|
||||||
}, 600)
|
desc: 'Apakah Anda yakin ingin mengeluarkan anggota?',
|
||||||
|
onPress: () => {
|
||||||
|
handleDeleteUser()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
|
|
||||||
</View>
|
</View>
|
||||||
</DrawerBottom>
|
</DrawerBottom>
|
||||||
|
|
||||||
<ModalConfirmation
|
|
||||||
visible={showDeleteModal}
|
|
||||||
title="Konfirmasi"
|
|
||||||
message="Apakah anda yakin ingin mengeluarkan anggota?"
|
|
||||||
onConfirm={() => {
|
|
||||||
setShowDeleteModal(false)
|
|
||||||
handleDeleteUser()
|
|
||||||
}}
|
|
||||||
onCancel={() => setShowDeleteModal(false)}
|
|
||||||
confirmText="Hapus"
|
|
||||||
cancelText="Batal"
|
|
||||||
/>
|
|
||||||
</SafeAreaView>
|
</SafeAreaView>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@@ -9,7 +9,6 @@ import Styles from "@/constants/Styles";
|
|||||||
import { apiAddMemberCalendar, apiGetCalendarOne, apiGetDivisionMember } from "@/lib/api";
|
import { apiAddMemberCalendar, apiGetCalendarOne, apiGetDivisionMember } from "@/lib/api";
|
||||||
import { setUpdateCalendar } from "@/lib/calendarUpdate";
|
import { setUpdateCalendar } from "@/lib/calendarUpdate";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { AntDesign } from "@expo/vector-icons";
|
import { AntDesign } from "@expo/vector-icons";
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
@@ -24,7 +23,6 @@ type Props = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export default function AddMemberCalendarEvent() {
|
export default function AddMemberCalendarEvent() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const dispatch = useDispatch()
|
const dispatch = useDispatch()
|
||||||
const update = useSelector((state: any) => state.calendarUpdate)
|
const update = useSelector((state: any) => state.calendarUpdate)
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
@@ -92,11 +90,9 @@ export default function AddMemberCalendarEvent() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal menambahkan anggota"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
@@ -104,7 +100,7 @@ export default function AddMemberCalendarEvent() {
|
|||||||
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={{ flex: 1, backgroundColor: colors.background }}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
||||||
@@ -158,7 +154,7 @@ export default function AddMemberCalendarEvent() {
|
|||||||
</View>
|
</View>
|
||||||
|
|
||||||
:
|
:
|
||||||
<Text style={[Styles.textDefault, Styles.pv05, { textAlign: 'center', color: colors.dimmed }]}>Tidak ada member yang dipilih</Text>
|
<Text style={[Styles.textDefault, Styles.cGray, Styles.pv05, { textAlign: 'center' }]}>Tidak ada member yang dipilih</Text>
|
||||||
}
|
}
|
||||||
<ScrollView
|
<ScrollView
|
||||||
showsVerticalScrollIndicator={false}
|
showsVerticalScrollIndicator={false}
|
||||||
@@ -171,7 +167,7 @@ export default function AddMemberCalendarEvent() {
|
|||||||
return (
|
return (
|
||||||
<Pressable
|
<Pressable
|
||||||
key={index}
|
key={index}
|
||||||
style={[Styles.itemSelectModal, {borderColor: colors.icon + '20'}]}
|
style={[Styles.itemSelectModal]}
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
!found && onChoose(item.idUser, item.name, item.img)
|
!found && onChoose(item.idUser, item.name, item.img)
|
||||||
}}
|
}}
|
||||||
@@ -181,12 +177,12 @@ export default function AddMemberCalendarEvent() {
|
|||||||
<View style={[Styles.ml10, { width: '80%' }]}>
|
<View style={[Styles.ml10, { width: '80%' }]}>
|
||||||
<Text numberOfLines={1} ellipsizeMode="tail" style={[Styles.textDefault]}>{item.name}</Text>
|
<Text numberOfLines={1} ellipsizeMode="tail" style={[Styles.textDefault]}>{item.name}</Text>
|
||||||
{
|
{
|
||||||
found && <Text style={[Styles.textInformation, { color: colors.dimmed }]}>sudah menjadi anggota</Text>
|
found && <Text style={[Styles.textInformation, Styles.cGray]}>sudah menjadi anggota</Text>
|
||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
{
|
{
|
||||||
selectMember.some((i: any) => i.idUser == item.idUser) && <AntDesign name="check" size={20} color={colors.text} />
|
selectMember.some((i: any) => i.idUser == item.idUser) && <AntDesign name="check" size={20} color={'black'} />
|
||||||
}
|
}
|
||||||
</Pressable>
|
</Pressable>
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -9,7 +9,6 @@ import { valueTypeEventRepeat } from "@/constants/TypeEventRepeat"
|
|||||||
import { apiGetCalendarOne, apiUpdateCalendar } from "@/lib/api"
|
import { apiGetCalendarOne, apiUpdateCalendar } from "@/lib/api"
|
||||||
import { stringToDateTime } from "@/lib/fun_stringToDate"
|
import { stringToDateTime } from "@/lib/fun_stringToDate"
|
||||||
import { useAuthSession } from "@/providers/AuthProvider"
|
import { useAuthSession } from "@/providers/AuthProvider"
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { useHeaderHeight } from "@react-navigation/elements"
|
import { useHeaderHeight } from "@react-navigation/elements"
|
||||||
import { Stack, router, useLocalSearchParams } from "expo-router"
|
import { Stack, router, useLocalSearchParams } from "expo-router"
|
||||||
import moment from "moment"
|
import moment from "moment"
|
||||||
@@ -18,7 +17,6 @@ import { KeyboardAvoidingView, Platform, SafeAreaView, ScrollView, View } from "
|
|||||||
import Toast from "react-native-toast-message"
|
import Toast from "react-native-toast-message"
|
||||||
|
|
||||||
export default function EditEventCalendar() {
|
export default function EditEventCalendar() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const { token, decryptToken } = useAuthSession();
|
const { token, decryptToken } = useAuthSession();
|
||||||
const [choose, setChoose] = useState({ val: "", label: "" })
|
const [choose, setChoose] = useState({ val: "", label: "" })
|
||||||
const [isSelect, setSelect] = useState(false)
|
const [isSelect, setSelect] = useState(false)
|
||||||
@@ -57,11 +55,9 @@ export default function EditEventCalendar() {
|
|||||||
setData({ ...response.data, dateStart: moment(response.data.dateStartFormat, 'DD-MM-YYYY').format('DD-MM-YYYY') })
|
setData({ ...response.data, dateStart: moment(response.data.dateStartFormat, 'DD-MM-YYYY').format('DD-MM-YYYY') })
|
||||||
setIdCalendar(response.data.idCalendar)
|
setIdCalendar(response.data.idCalendar)
|
||||||
setChoose({ val: response.data.repeatEventTyper, label: valueTypeEventRepeat.find((item) => item.id == response.data.repeatEventTyper)?.name || "" })
|
setChoose({ val: response.data.repeatEventTyper, label: valueTypeEventRepeat.find((item) => item.id == response.data.repeatEventTyper)?.name || "" })
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Gagal mendapatkan data"
|
Toast.show({ type: 'small', text1: 'Gagal mendapatkan data', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -156,11 +152,9 @@ export default function EditEventCalendar() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal mengubah acara"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
@@ -168,7 +162,7 @@ export default function EditEventCalendar() {
|
|||||||
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={{ flex: 1, backgroundColor: colors.background }}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
||||||
@@ -211,7 +205,7 @@ export default function EditEventCalendar() {
|
|||||||
type="default"
|
type="default"
|
||||||
placeholder="Nama Acara"
|
placeholder="Nama Acara"
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
bg="white"
|
||||||
value={data.title}
|
value={data.title}
|
||||||
onChange={(val) => validationForm("title", val)}
|
onChange={(val) => validationForm("title", val)}
|
||||||
error={error.title}
|
error={error.title}
|
||||||
@@ -257,12 +251,12 @@ export default function EditEventCalendar() {
|
|||||||
label="Link Meet"
|
label="Link Meet"
|
||||||
type="default"
|
type="default"
|
||||||
placeholder="Link Meet"
|
placeholder="Link Meet"
|
||||||
bg={colors.card}
|
bg="white"
|
||||||
value={data.linkMeet}
|
value={data.linkMeet}
|
||||||
onChange={(val) => validationForm("linkMeet", val)}
|
onChange={(val) => validationForm("linkMeet", val)}
|
||||||
/>
|
/>
|
||||||
<SelectForm
|
<SelectForm
|
||||||
bg={colors.card}
|
bg="white"
|
||||||
label="Ulangi Acara"
|
label="Ulangi Acara"
|
||||||
placeholder="Ulangi Acara"
|
placeholder="Ulangi Acara"
|
||||||
value={choose.label}
|
value={choose.label}
|
||||||
@@ -274,7 +268,7 @@ export default function EditEventCalendar() {
|
|||||||
type="numeric"
|
type="numeric"
|
||||||
placeholder="Jumlah Pengulangan"
|
placeholder="Jumlah Pengulangan"
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
bg="white"
|
||||||
value={String(data.repeatValue)}
|
value={String(data.repeatValue)}
|
||||||
onChange={(val) => validationForm("repeatValue", val)}
|
onChange={(val) => validationForm("repeatValue", val)}
|
||||||
error={error.repeatValue}
|
error={error.repeatValue}
|
||||||
@@ -285,7 +279,7 @@ export default function EditEventCalendar() {
|
|||||||
label="Deskripsi"
|
label="Deskripsi"
|
||||||
type="default"
|
type="default"
|
||||||
placeholder="Deskripsi"
|
placeholder="Deskripsi"
|
||||||
bg={colors.card}
|
bg="white"
|
||||||
value={data.desc}
|
value={data.desc}
|
||||||
onChange={(val) => validationForm("desc", val)}
|
onChange={(val) => validationForm("desc", val)}
|
||||||
multiline
|
multiline
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import ModalConfirmation from "@/components/ModalConfirmation"
|
import AlertKonfirmasi from "@/components/alertKonfirmasi"
|
||||||
import AppHeader from "@/components/AppHeader"
|
import AppHeader from "@/components/AppHeader"
|
||||||
import BorderBottomItem from "@/components/borderBottomItem"
|
import BorderBottomItem from "@/components/borderBottomItem"
|
||||||
import ButtonBackHeader from "@/components/buttonBackHeader"
|
import ButtonBackHeader from "@/components/buttonBackHeader"
|
||||||
@@ -13,7 +13,6 @@ import Styles from "@/constants/Styles"
|
|||||||
import { apiDeleteCalendarMember, apiGetCalendarOne, apiGetDivisionOneFeature } from "@/lib/api"
|
import { apiDeleteCalendarMember, apiGetCalendarOne, apiGetDivisionOneFeature } from "@/lib/api"
|
||||||
import { setUpdateCalendar } from "@/lib/calendarUpdate"
|
import { setUpdateCalendar } from "@/lib/calendarUpdate"
|
||||||
import { useAuthSession } from "@/providers/AuthProvider"
|
import { useAuthSession } from "@/providers/AuthProvider"
|
||||||
import { useTheme } from "@/providers/ThemeProvider"
|
|
||||||
import { MaterialCommunityIcons } from "@expo/vector-icons"
|
import { MaterialCommunityIcons } from "@expo/vector-icons"
|
||||||
import Clipboard from "@react-native-clipboard/clipboard"
|
import Clipboard from "@react-native-clipboard/clipboard"
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router"
|
import { router, Stack, useLocalSearchParams } from "expo-router"
|
||||||
@@ -46,7 +45,6 @@ type PropsMember = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export default function DetailEventCalendar() {
|
export default function DetailEventCalendar() {
|
||||||
const { colors } = useTheme()
|
|
||||||
const { id, detail } = useLocalSearchParams<{ id: string, detail: string }>();
|
const { id, detail } = useLocalSearchParams<{ id: string, detail: string }>();
|
||||||
const [data, setData] = useState<Props>()
|
const [data, setData] = useState<Props>()
|
||||||
const [member, setMember] = useState<PropsMember[]>([])
|
const [member, setMember] = useState<PropsMember[]>([])
|
||||||
@@ -57,7 +55,6 @@ export default function DetailEventCalendar() {
|
|||||||
const dispatch = useDispatch()
|
const dispatch = useDispatch()
|
||||||
const entityUser = useSelector((state: any) => state.user);
|
const entityUser = useSelector((state: any) => state.user);
|
||||||
const [isMemberDivision, setIsMemberDivision] = useState(false);
|
const [isMemberDivision, setIsMemberDivision] = useState(false);
|
||||||
const [showDeleteModal, setShowDeleteModal] = useState(false)
|
|
||||||
const [loading, setLoading] = useState(true)
|
const [loading, setLoading] = useState(true)
|
||||||
const [refreshing, setRefreshing] = useState(false)
|
const [refreshing, setRefreshing] = useState(false)
|
||||||
|
|
||||||
@@ -137,11 +134,9 @@ export default function DetailEventCalendar() {
|
|||||||
dispatch(setUpdateCalendar({ ...update, member: !update.member }));
|
dispatch(setUpdateCalendar({ ...update, member: !update.member }));
|
||||||
}
|
}
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Gagal menghapus anggota"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setModalMember(false)
|
setModalMember(false)
|
||||||
}
|
}
|
||||||
@@ -157,14 +152,14 @@ export default function DetailEventCalendar() {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={{ flex: 1, backgroundColor: colors.background }}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
||||||
headerTitle: 'Detail Acara',
|
headerTitle: 'Detail Acara',
|
||||||
headerTitleAlign: 'center',
|
headerTitleAlign: 'center',
|
||||||
// headerRight: () => (entityUser.role == "user" || entityUser.role == "coadmin") && !isMemberDivision ? <></> : <HeaderRightCalendarDetail id={String(data?.idCalendar)} idReminder={String(detail)} />
|
// headerRight: () => (entityUser.role == "user" || entityUser.role == "coadmin") && !isMemberDivision ? <></> : <HeaderRightCalendarDetail id={String(data?.idCalendar)} idReminder={String(detail)} />
|
||||||
header: () => (
|
header:()=>(
|
||||||
<AppHeader
|
<AppHeader
|
||||||
title="Detail Acara"
|
title="Detail Acara"
|
||||||
showBack={true}
|
showBack={true}
|
||||||
@@ -182,14 +177,13 @@ export default function DetailEventCalendar() {
|
|||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={handleRefresh}
|
onRefresh={handleRefresh}
|
||||||
tintColor={colors.icon}
|
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
<View style={[Styles.p15]}>
|
<View style={[Styles.p15]}>
|
||||||
<View style={[Styles.wrapPaper, Styles.mb15, { backgroundColor: colors.card, borderColor: colors.background }]}>
|
<View style={[Styles.wrapPaper, Styles.mb15]}>
|
||||||
<View style={[Styles.rowItemsCenter, { alignItems: 'flex-start' }]}>
|
<View style={[Styles.rowItemsCenter, { alignItems: 'flex-start' }]}>
|
||||||
<MaterialCommunityIcons name="calendar-text" size={30} color={colors.text} style={Styles.mr10} />
|
<MaterialCommunityIcons name="calendar-text" size={30} color="black" style={Styles.mr10} />
|
||||||
{
|
{
|
||||||
loading ?
|
loading ?
|
||||||
<Skeleton width={80} height={10} borderRadius={10} widthType="percent" />
|
<Skeleton width={80} height={10} borderRadius={10} widthType="percent" />
|
||||||
@@ -198,7 +192,7 @@ export default function DetailEventCalendar() {
|
|||||||
|
|
||||||
</View>
|
</View>
|
||||||
<View style={[Styles.rowItemsCenter, Styles.mt10]}>
|
<View style={[Styles.rowItemsCenter, Styles.mt10]}>
|
||||||
<MaterialCommunityIcons name="calendar-month-outline" size={30} color={colors.text} style={Styles.mr10} />
|
<MaterialCommunityIcons name="calendar-month-outline" size={30} color="black" style={Styles.mr10} />
|
||||||
{
|
{
|
||||||
loading ?
|
loading ?
|
||||||
<Skeleton width={80} height={10} borderRadius={10} widthType="percent" />
|
<Skeleton width={80} height={10} borderRadius={10} widthType="percent" />
|
||||||
@@ -207,7 +201,7 @@ export default function DetailEventCalendar() {
|
|||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
<View style={[Styles.rowItemsCenter, Styles.mt10]}>
|
<View style={[Styles.rowItemsCenter, Styles.mt10]}>
|
||||||
<MaterialCommunityIcons name="clock-outline" size={30} color={colors.text} style={Styles.mr10} />
|
<MaterialCommunityIcons name="clock-outline" size={30} color="black" style={Styles.mr10} />
|
||||||
{
|
{
|
||||||
loading ?
|
loading ?
|
||||||
<Skeleton width={80} height={10} borderRadius={10} widthType="percent" />
|
<Skeleton width={80} height={10} borderRadius={10} widthType="percent" />
|
||||||
@@ -216,7 +210,7 @@ export default function DetailEventCalendar() {
|
|||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
<View style={[Styles.rowItemsCenter, Styles.mt10]}>
|
<View style={[Styles.rowItemsCenter, Styles.mt10]}>
|
||||||
<MaterialCommunityIcons name="repeat" size={30} color={colors.text} style={Styles.mr10} />
|
<MaterialCommunityIcons name="repeat" size={30} color="black" style={Styles.mr10} />
|
||||||
{
|
{
|
||||||
loading ?
|
loading ?
|
||||||
<Skeleton width={80} height={10} borderRadius={10} widthType="percent" />
|
<Skeleton width={80} height={10} borderRadius={10} widthType="percent" />
|
||||||
@@ -234,7 +228,7 @@ export default function DetailEventCalendar() {
|
|||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
<View style={[Styles.rowItemsCenter, Styles.mt10]}>
|
<View style={[Styles.rowItemsCenter, Styles.mt10]}>
|
||||||
<MaterialCommunityIcons name="link-variant" size={30} color={colors.text} style={Styles.mr10} />
|
<MaterialCommunityIcons name="link-variant" size={30} color="black" style={Styles.mr10} />
|
||||||
{
|
{
|
||||||
loading ?
|
loading ?
|
||||||
<Skeleton width={80} height={10} borderRadius={10} widthType="percent" />
|
<Skeleton width={80} height={10} borderRadius={10} widthType="percent" />
|
||||||
@@ -247,7 +241,7 @@ export default function DetailEventCalendar() {
|
|||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
<View style={[Styles.rowItemsCenter, Styles.mt10, { alignItems: 'flex-start' }]}>
|
<View style={[Styles.rowItemsCenter, Styles.mt10, { alignItems: 'flex-start' }]}>
|
||||||
<MaterialCommunityIcons name="card-text-outline" size={30} color={colors.text} style={Styles.mr10} />
|
<MaterialCommunityIcons name="card-text-outline" size={30} color="black" style={Styles.mr10} />
|
||||||
{
|
{
|
||||||
loading ?
|
loading ?
|
||||||
<Skeleton width={80} height={10} borderRadius={10} widthType="percent" />
|
<Skeleton width={80} height={10} borderRadius={10} widthType="percent" />
|
||||||
@@ -263,7 +257,7 @@ export default function DetailEventCalendar() {
|
|||||||
<Text style={[Styles.textDefault]}>Total {member.length} Anggota</Text>
|
<Text style={[Styles.textDefault]}>Total {member.length} Anggota</Text>
|
||||||
</View>
|
</View>
|
||||||
|
|
||||||
<View style={[Styles.wrapPaper, { backgroundColor: colors.card, borderColor: colors.background }]}>
|
<View style={[Styles.wrapPaper]}>
|
||||||
{
|
{
|
||||||
member.map((item, index) => (
|
member.map((item, index) => (
|
||||||
<BorderBottomItem
|
<BorderBottomItem
|
||||||
@@ -292,7 +286,7 @@ export default function DetailEventCalendar() {
|
|||||||
<DrawerBottom animation="slide" isVisible={isModalMember} setVisible={setModalMember} title={memberChoose.name}>
|
<DrawerBottom animation="slide" isVisible={isModalMember} setVisible={setModalMember} title={memberChoose.name}>
|
||||||
<View style={Styles.rowItemsCenter}>
|
<View style={Styles.rowItemsCenter}>
|
||||||
<MenuItemRow
|
<MenuItemRow
|
||||||
icon={<MaterialCommunityIcons name="account-eye" color={colors.text} size={25} />}
|
icon={<MaterialCommunityIcons name="account-eye" color="black" size={25} />}
|
||||||
title="Lihat Profil"
|
title="Lihat Profil"
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
setModalMember(false)
|
setModalMember(false)
|
||||||
@@ -301,30 +295,22 @@ export default function DetailEventCalendar() {
|
|||||||
/>
|
/>
|
||||||
|
|
||||||
<MenuItemRow
|
<MenuItemRow
|
||||||
icon={<MaterialCommunityIcons name="account-remove" color={colors.text} size={25} />}
|
icon={<MaterialCommunityIcons name="account-remove" color="black" size={25} />}
|
||||||
title="Keluarkan"
|
title="Keluarkan"
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
setModalMember(false)
|
setModalMember(false)
|
||||||
setTimeout(() => {
|
AlertKonfirmasi({
|
||||||
setShowDeleteModal(true)
|
title: 'Konfirmasi',
|
||||||
}, 600)
|
desc: 'Apakah Anda yakin ingin mengeluarkan anggota?',
|
||||||
|
onPress: () => {
|
||||||
|
handleDeleteUser()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
</View>
|
</View>
|
||||||
</DrawerBottom>
|
</DrawerBottom>
|
||||||
|
|
||||||
<ModalConfirmation
|
|
||||||
visible={showDeleteModal}
|
|
||||||
title="Konfirmasi"
|
|
||||||
message="Apakah Anda yakin ingin mengeluarkan anggota?"
|
|
||||||
onConfirm={() => {
|
|
||||||
setShowDeleteModal(false)
|
|
||||||
handleDeleteUser()
|
|
||||||
}}
|
|
||||||
onCancel={() => setShowDeleteModal(false)}
|
|
||||||
confirmText="Keluar"
|
|
||||||
cancelText="Batal"
|
|
||||||
/>
|
|
||||||
</SafeAreaView>
|
</SafeAreaView>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@@ -10,7 +10,6 @@ import { apiCreateCalendar, apiGetDivisionMember } from "@/lib/api";
|
|||||||
import { setFormCreateCalendar } from "@/lib/calendarCreate";
|
import { setFormCreateCalendar } from "@/lib/calendarCreate";
|
||||||
import { setUpdateCalendar } from "@/lib/calendarUpdate";
|
import { setUpdateCalendar } from "@/lib/calendarUpdate";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { AntDesign } from "@expo/vector-icons";
|
import { AntDesign } from "@expo/vector-icons";
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
@@ -25,7 +24,6 @@ type Props = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export default function CreateCalendarAddMember() {
|
export default function CreateCalendarAddMember() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
const { id } = useLocalSearchParams<{ id: string }>()
|
const { id } = useLocalSearchParams<{ id: string }>()
|
||||||
const [data, setData] = useState<Props[]>([])
|
const [data, setData] = useState<Props[]>([])
|
||||||
@@ -83,18 +81,16 @@ export default function CreateCalendarAddMember() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal membuat acara"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={{ flex: 1, backgroundColor: colors.background }}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
||||||
@@ -145,7 +141,7 @@ export default function CreateCalendarAddMember() {
|
|||||||
</View>
|
</View>
|
||||||
|
|
||||||
:
|
:
|
||||||
<Text style={[Styles.textDefault, Styles.pv05, { textAlign: 'center', color: colors.dimmed }]}>Tidak ada member yang dipilih</Text>
|
<Text style={[Styles.textDefault, Styles.cGray, Styles.pv05, { textAlign: 'center' }]}>Tidak ada member yang dipilih</Text>
|
||||||
}
|
}
|
||||||
<ScrollView
|
<ScrollView
|
||||||
showsVerticalScrollIndicator={false}
|
showsVerticalScrollIndicator={false}
|
||||||
@@ -158,7 +154,7 @@ export default function CreateCalendarAddMember() {
|
|||||||
return (
|
return (
|
||||||
<Pressable
|
<Pressable
|
||||||
key={index}
|
key={index}
|
||||||
style={[Styles.itemSelectModal, { borderColor: colors.icon + '20' }]}
|
style={[Styles.itemSelectModal]}
|
||||||
onPress={() => { onChoose(item.idUser, item.name, item.img) }}
|
onPress={() => { onChoose(item.idUser, item.name, item.img) }}
|
||||||
>
|
>
|
||||||
<View style={[Styles.rowItemsCenter, Styles.w70]}>
|
<View style={[Styles.rowItemsCenter, Styles.w70]}>
|
||||||
@@ -168,7 +164,7 @@ export default function CreateCalendarAddMember() {
|
|||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
{
|
{
|
||||||
selectMember.some((i: any) => i.idUser == item.idUser) && <AntDesign name="check" size={20} color={colors.text} />
|
selectMember.some((i: any) => i.idUser == item.idUser) && <AntDesign name="check" size={20} color={'black'} />
|
||||||
}
|
}
|
||||||
</Pressable>
|
</Pressable>
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -9,7 +9,6 @@ import Styles from "@/constants/Styles";
|
|||||||
import { setFormCreateCalendar } from "@/lib/calendarCreate";
|
import { setFormCreateCalendar } from "@/lib/calendarCreate";
|
||||||
import { stringToDateTime } from "@/lib/fun_stringToDate";
|
import { stringToDateTime } from "@/lib/fun_stringToDate";
|
||||||
import { useHeaderHeight } from '@react-navigation/elements';
|
import { useHeaderHeight } from '@react-navigation/elements';
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { Stack, router, useLocalSearchParams } from "expo-router";
|
import { Stack, router, useLocalSearchParams } from "expo-router";
|
||||||
import { useState } from "react";
|
import { useState } from "react";
|
||||||
import {
|
import {
|
||||||
@@ -22,7 +21,6 @@ import {
|
|||||||
import { useDispatch, useSelector } from "react-redux";
|
import { useDispatch, useSelector } from "react-redux";
|
||||||
|
|
||||||
export default function CalendarDivisionCreate() {
|
export default function CalendarDivisionCreate() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const { id } = useLocalSearchParams<{ id: string }>()
|
const { id } = useLocalSearchParams<{ id: string }>()
|
||||||
const [choose, setChoose] = useState({ val: "", label: "" })
|
const [choose, setChoose] = useState({ val: "", label: "" })
|
||||||
const [isSelect, setSelect] = useState(false)
|
const [isSelect, setSelect] = useState(false)
|
||||||
@@ -128,7 +126,7 @@ export default function CalendarDivisionCreate() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={{ flex: 1, backgroundColor: colors.background }}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => (
|
// headerLeft: () => (
|
||||||
@@ -146,7 +144,7 @@ export default function CalendarDivisionCreate() {
|
|||||||
// disable={Object.values(error).some((val) => val == true) || data.title == "" || data.dateStart == "" || data.timeStart == "" || data.timeEnd == "" || data.repeatEventType == ""}
|
// disable={Object.values(error).some((val) => val == true) || data.title == "" || data.dateStart == "" || data.timeStart == "" || data.timeEnd == "" || data.repeatEventType == ""}
|
||||||
// />
|
// />
|
||||||
// ),
|
// ),
|
||||||
header: () => (
|
header:()=>(
|
||||||
<AppHeader
|
<AppHeader
|
||||||
title="Tambah Acara"
|
title="Tambah Acara"
|
||||||
showBack={true}
|
showBack={true}
|
||||||
@@ -175,7 +173,7 @@ export default function CalendarDivisionCreate() {
|
|||||||
type="default"
|
type="default"
|
||||||
placeholder="Nama Acara"
|
placeholder="Nama Acara"
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
bg="white"
|
||||||
value={data.title}
|
value={data.title}
|
||||||
onChange={(val) => validationForm("title", val)}
|
onChange={(val) => validationForm("title", val)}
|
||||||
error={error.title}
|
error={error.title}
|
||||||
@@ -221,12 +219,12 @@ export default function CalendarDivisionCreate() {
|
|||||||
label="Link Meet"
|
label="Link Meet"
|
||||||
type="default"
|
type="default"
|
||||||
placeholder="Link Meet"
|
placeholder="Link Meet"
|
||||||
bg={colors.card}
|
bg="white"
|
||||||
value={data.linkMeet}
|
value={data.linkMeet}
|
||||||
onChange={(val) => validationForm("linkMeet", val)}
|
onChange={(val) => validationForm("linkMeet", val)}
|
||||||
/>
|
/>
|
||||||
<SelectForm
|
<SelectForm
|
||||||
bg={colors.card}
|
bg="white"
|
||||||
label="Ulangi Acara"
|
label="Ulangi Acara"
|
||||||
placeholder="Ulangi Acara"
|
placeholder="Ulangi Acara"
|
||||||
value={choose.label}
|
value={choose.label}
|
||||||
@@ -238,7 +236,7 @@ export default function CalendarDivisionCreate() {
|
|||||||
type="numeric"
|
type="numeric"
|
||||||
placeholder="Jumlah Pengulangan"
|
placeholder="Jumlah Pengulangan"
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
bg="white"
|
||||||
value={String(data.repeatValue)}
|
value={String(data.repeatValue)}
|
||||||
onChange={(val) => validationForm("repeatValue", val)}
|
onChange={(val) => validationForm("repeatValue", val)}
|
||||||
error={error.repeatValue}
|
error={error.repeatValue}
|
||||||
@@ -249,7 +247,7 @@ export default function CalendarDivisionCreate() {
|
|||||||
label="Deskripsi"
|
label="Deskripsi"
|
||||||
type="default"
|
type="default"
|
||||||
placeholder="Deskripsi"
|
placeholder="Deskripsi"
|
||||||
bg={colors.card}
|
bg="white"
|
||||||
value={data.desc}
|
value={data.desc}
|
||||||
onChange={(val) => validationForm("desc", val)}
|
onChange={(val) => validationForm("desc", val)}
|
||||||
multiline
|
multiline
|
||||||
|
|||||||
@@ -1,10 +1,10 @@
|
|||||||
import InputSearch from "@/components/inputSearch";
|
import InputSearch from "@/components/inputSearch";
|
||||||
import Skeleton from "@/components/skeleton";
|
import Skeleton from "@/components/skeleton";
|
||||||
import Text from "@/components/Text";
|
import Text from "@/components/Text";
|
||||||
|
import { ColorsStatus } from "@/constants/ColorsStatus";
|
||||||
import Styles from "@/constants/Styles";
|
import Styles from "@/constants/Styles";
|
||||||
import { apiGetCalendarHistory } from "@/lib/api";
|
import { apiGetCalendarHistory } from "@/lib/api";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { useLocalSearchParams } from "expo-router";
|
import { useLocalSearchParams } from "expo-router";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { FlatList, View, VirtualizedList } from "react-native";
|
import { FlatList, View, VirtualizedList } from "react-native";
|
||||||
@@ -15,7 +15,6 @@ type Props = {
|
|||||||
data: []
|
data: []
|
||||||
}
|
}
|
||||||
export default function CalendarHistory() {
|
export default function CalendarHistory() {
|
||||||
const { colors, activeTheme } = useTheme();
|
|
||||||
const { id } = useLocalSearchParams<{ id: string }>();
|
const { id } = useLocalSearchParams<{ id: string }>();
|
||||||
const { token, decryptToken } = useAuthSession();
|
const { token, decryptToken } = useAuthSession();
|
||||||
const [data, setData] = useState<Props[]>([])
|
const [data, setData] = useState<Props[]>([])
|
||||||
@@ -65,11 +64,11 @@ export default function CalendarHistory() {
|
|||||||
})
|
})
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<View style={[Styles.p15, { flex: 1, backgroundColor: colors.background }]}>
|
<View style={[Styles.p15, { flex: 1 }]}>
|
||||||
<View>
|
<View>
|
||||||
<InputSearch onChange={(val) => setSearch(val)} />
|
<InputSearch onChange={(val) => setSearch(val)} />
|
||||||
</View>
|
</View>
|
||||||
<View style={[{ flex: 2 }, Styles.mt10]}>
|
<View style={[{ flex: 2, }]}>
|
||||||
{
|
{
|
||||||
loading ?
|
loading ?
|
||||||
arrSkeleton.map((item, index) => (
|
arrSkeleton.map((item, index) => (
|
||||||
@@ -82,7 +81,7 @@ export default function CalendarHistory() {
|
|||||||
getItem={getItem}
|
getItem={getItem}
|
||||||
renderItem={({ item, index }: { item: Props, index: number }) => {
|
renderItem={({ item, index }: { item: Props, index: number }) => {
|
||||||
return (
|
return (
|
||||||
<View key={index} style={[{ flexDirection: 'row' }, Styles.mb05, Styles.borderAll, { backgroundColor: colors.card }, Styles.p10, Styles.round05, { borderColor: colors.icon + '20' }]}>
|
<View key={index} style={[{ flexDirection: 'row' }, Styles.mv05, ColorsStatus.lightGreen, Styles.p10, Styles.round10]}>
|
||||||
<View style={[Styles.mr10, Styles.ph05]}>
|
<View style={[Styles.mr10, Styles.ph05]}>
|
||||||
<Text style={[Styles.textSubtitle]}>{String(item.dateStart)}</Text>
|
<Text style={[Styles.textSubtitle]}>{String(item.dateStart)}</Text>
|
||||||
<Text style={[Styles.textDefault, { textAlign: 'center' }]}>{item.year}</Text>
|
<Text style={[Styles.textDefault, { textAlign: 'center' }]}>{item.year}</Text>
|
||||||
@@ -90,7 +89,7 @@ export default function CalendarHistory() {
|
|||||||
<View style={[{ flex: 1 }]}>
|
<View style={[{ flex: 1 }]}>
|
||||||
<FlatList data={item.data}
|
<FlatList data={item.data}
|
||||||
renderItem={({ item, index }: { item: { title: string, timeStart: string, timeEnd: string }, index: number }) => (
|
renderItem={({ item, index }: { item: { title: string, timeStart: string, timeEnd: string }, index: number }) => (
|
||||||
<View key={index} style={[Styles.mb05]}>
|
<View key={index} style={[Styles.mb05, Styles.w80]}>
|
||||||
<Text style={[Styles.textDefaultSemiBold]} numberOfLines={1} ellipsizeMode="tail">{item.title}</Text>
|
<Text style={[Styles.textDefaultSemiBold]} numberOfLines={1} ellipsizeMode="tail">{item.title}</Text>
|
||||||
<Text style={[Styles.textDefault]}>{item.timeStart} | {item.timeEnd}</Text>
|
<Text style={[Styles.textDefault]}>{item.timeStart} | {item.timeEnd}</Text>
|
||||||
</View>
|
</View>
|
||||||
|
|||||||
@@ -7,7 +7,6 @@ import Text from "@/components/Text";
|
|||||||
import Styles from "@/constants/Styles";
|
import Styles from "@/constants/Styles";
|
||||||
import { apiGetCalendarByDateDivision, apiGetIndicatorCalendar } from "@/lib/api";
|
import { apiGetCalendarByDateDivision, apiGetIndicatorCalendar } from "@/lib/api";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { Feather } from "@expo/vector-icons";
|
import { Feather } from "@expo/vector-icons";
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
import 'intl';
|
import 'intl';
|
||||||
@@ -35,7 +34,6 @@ type Props = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
export default function CalendarDivision() {
|
export default function CalendarDivision() {
|
||||||
const { colors, activeTheme } = useTheme();
|
|
||||||
const [selected, setSelected] = useState<any>(new Date())
|
const [selected, setSelected] = useState<any>(new Date())
|
||||||
const [data, setData] = useState<Props[]>([])
|
const [data, setData] = useState<Props[]>([])
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
@@ -119,15 +117,15 @@ export default function CalendarDivision() {
|
|||||||
);
|
);
|
||||||
},
|
},
|
||||||
IconNext: <Pressable onPress={() => !loadingBtn ? setMonth(month + 1) : null}>
|
IconNext: <Pressable onPress={() => !loadingBtn ? setMonth(month + 1) : null}>
|
||||||
<Feather name="chevron-right" size={20} color={loadingBtn ? 'gray' : colors.text} />
|
<Feather name="chevron-right" size={20} color={loadingBtn ? 'gray' : 'black'} />
|
||||||
</Pressable>,
|
</Pressable>,
|
||||||
IconPrev: <Pressable onPress={() => !loadingBtn ? setMonth(month - 1) : null}>
|
IconPrev: <Pressable onPress={() => !loadingBtn ? setMonth(month - 1) : null}>
|
||||||
<Feather name="chevron-left" size={20} color={loadingBtn ? 'gray' : colors.text} />
|
<Feather name="chevron-left" size={20} color={loadingBtn ? 'gray' : 'black'} />
|
||||||
</Pressable>,
|
</Pressable>,
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={{ flex: 1, backgroundColor: colors.background }}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => (
|
// headerLeft: () => (
|
||||||
@@ -155,13 +153,12 @@ export default function CalendarDivision() {
|
|||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={handleRefresh}
|
onRefresh={handleRefresh}
|
||||||
tintColor={colors.icon}
|
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
style={[Styles.h100]}
|
style={[Styles.h100]}
|
||||||
>
|
>
|
||||||
<View style={[Styles.p15]}>
|
<View style={[Styles.p15]}>
|
||||||
<View style={[Styles.wrapPaper, Styles.p10, { backgroundColor: colors.card, borderColor: colors.background }]}>
|
<View style={[Styles.wrapPaper, Styles.p10]}>
|
||||||
<Datepicker
|
<Datepicker
|
||||||
components={components}
|
components={components}
|
||||||
mode="single"
|
mode="single"
|
||||||
@@ -170,19 +167,19 @@ export default function CalendarDivision() {
|
|||||||
onMonthChange={(month) => setMonth(month)}
|
onMonthChange={(month) => setMonth(month)}
|
||||||
styles={{
|
styles={{
|
||||||
selected: Styles.selectedDate,
|
selected: Styles.selectedDate,
|
||||||
month_label: { color: colors.text },
|
month_label: Styles.cBlack,
|
||||||
month_selector_label: { color: colors.text },
|
month_selector_label: Styles.cBlack,
|
||||||
year_label: { color: colors.text },
|
year_label: Styles.cBlack,
|
||||||
year_selector_label: { color: colors.text },
|
year_selector_label: Styles.cBlack,
|
||||||
day_label: { color: colors.text },
|
day_label: Styles.cBlack,
|
||||||
time_label: { color: colors.text },
|
time_label: Styles.cBlack,
|
||||||
weekday_label: { color: colors.text },
|
weekday_label: Styles.cBlack,
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
</View>
|
</View>
|
||||||
<View style={[Styles.mb15, Styles.mt15]}>
|
<View style={[Styles.mb15, Styles.mt15]}>
|
||||||
<Text style={[Styles.textDefaultSemiBold, Styles.mb05]}>Acara</Text>
|
<Text style={[Styles.textDefaultSemiBold, Styles.mb05]}>Acara</Text>
|
||||||
<View style={[Styles.wrapPaper, { backgroundColor: colors.card, borderColor: colors.background }]}>
|
<View style={[Styles.wrapPaper]}>
|
||||||
{
|
{
|
||||||
loading ?
|
loading ?
|
||||||
<>
|
<>
|
||||||
@@ -205,7 +202,7 @@ export default function CalendarDivision() {
|
|||||||
/>
|
/>
|
||||||
))
|
))
|
||||||
) : (
|
) : (
|
||||||
<Text style={[Styles.textDefault, { textAlign: 'center', color: colors.dimmed }]}>Tidak ada acara</Text>
|
<Text style={[Styles.textDefault, Styles.cGray, { textAlign: 'center' }]}>Tidak ada acara</Text>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
|
|||||||
@@ -4,14 +4,13 @@ import ButtonSaveHeader from "@/components/buttonSaveHeader";
|
|||||||
import ButtonSelect from "@/components/buttonSelect";
|
import ButtonSelect from "@/components/buttonSelect";
|
||||||
import DrawerBottom from "@/components/drawerBottom";
|
import DrawerBottom from "@/components/drawerBottom";
|
||||||
import { InputForm } from "@/components/inputForm";
|
import { InputForm } from "@/components/inputForm";
|
||||||
import LoadingCenter from "@/components/loadingCenter";
|
import LoadingOverlay from "@/components/loadingOverlay";
|
||||||
import MenuItemRow from "@/components/menuItemRow";
|
import MenuItemRow from "@/components/menuItemRow";
|
||||||
import Text from "@/components/Text";
|
import Text from "@/components/Text";
|
||||||
import Styles from "@/constants/Styles";
|
import Styles from "@/constants/Styles";
|
||||||
import { apiEditDiscussion, apiGetDiscussionOne } from "@/lib/api";
|
import { apiEditDiscussion, apiGetDiscussionOne } from "@/lib/api";
|
||||||
import { setUpdateDiscussion } from "@/lib/discussionUpdate";
|
import { setUpdateDiscussion } from "@/lib/discussionUpdate";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { Ionicons, MaterialCommunityIcons } from "@expo/vector-icons";
|
import { Ionicons, MaterialCommunityIcons } from "@expo/vector-icons";
|
||||||
import * as DocumentPicker from "expo-document-picker";
|
import * as DocumentPicker from "expo-document-picker";
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
@@ -21,7 +20,6 @@ import Toast from "react-native-toast-message";
|
|||||||
import { useDispatch, useSelector } from "react-redux";
|
import { useDispatch, useSelector } from "react-redux";
|
||||||
|
|
||||||
export default function DiscussionDivisionEdit() {
|
export default function DiscussionDivisionEdit() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const { id, detail } = useLocalSearchParams<{ id: string; detail: string }>();
|
const { id, detail } = useLocalSearchParams<{ id: string; detail: string }>();
|
||||||
const { token, decryptToken } = useAuthSession();
|
const { token, decryptToken } = useAuthSession();
|
||||||
const [data, setData] = useState("");
|
const [data, setData] = useState("");
|
||||||
@@ -89,11 +87,9 @@ export default function DiscussionDivisionEdit() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Gagal mengubah data"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
@@ -131,7 +127,7 @@ export default function DiscussionDivisionEdit() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={{ flex: 1, backgroundColor: colors.background }}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => (
|
// headerLeft: () => (
|
||||||
@@ -170,7 +166,7 @@ export default function DiscussionDivisionEdit() {
|
|||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
{loading && <LoadingCenter />}
|
<LoadingOverlay visible={loading} />
|
||||||
<ScrollView showsVerticalScrollIndicator={false} style={[Styles.h100]}>
|
<ScrollView showsVerticalScrollIndicator={false} style={[Styles.h100]}>
|
||||||
<View style={[Styles.p15]}>
|
<View style={[Styles.p15]}>
|
||||||
<InputForm
|
<InputForm
|
||||||
@@ -181,45 +177,39 @@ export default function DiscussionDivisionEdit() {
|
|||||||
value={data}
|
value={data}
|
||||||
onChange={setData}
|
onChange={setData}
|
||||||
multiline
|
multiline
|
||||||
bg={colors.card}
|
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<ButtonSelect value="Upload File" onPress={pickDocumentAsync} />
|
<ButtonSelect value="Upload File" onPress={pickDocumentAsync} />
|
||||||
{
|
{
|
||||||
(fileForm.length > 0 || dataFile.filter((val) => !val.delete).length > 0)
|
(fileForm.length > 0 || dataFile.filter((val) => !val.delete).length > 0)
|
||||||
&&
|
&&
|
||||||
<>
|
<View style={[Styles.borderAll, Styles.round10, Styles.p10, Styles.mb10]}>
|
||||||
<View style={[Styles.rowSpaceBetween, Styles.mv05]}>
|
<Text style={[Styles.textDefaultSemiBold]}>File</Text>
|
||||||
<Text style={[Styles.textDefaultSemiBold]}>File</Text>
|
{
|
||||||
<Text style={[Styles.textDefault]}>{fileForm.length + dataFile.filter((val) => !val.delete).length} file</Text>
|
dataFile.filter((val) => !val.delete).map((item, index) => (
|
||||||
</View>
|
<BorderBottomItem
|
||||||
<View style={[Styles.borderAll, Styles.round05, Styles.p10, Styles.mb10, { backgroundColor: colors.card, borderColor: colors.icon + '20' }]}>
|
key={index}
|
||||||
{
|
borderType={(fileForm.length + dataFile.length) > 1 ? "bottom" : "none"}
|
||||||
dataFile.filter((val) => !val.delete).map((item, index) => (
|
icon={<MaterialCommunityIcons name="file-outline" size={25} color="black" />}
|
||||||
<BorderBottomItem
|
title={item.name + '.' + item.extension}
|
||||||
key={index}
|
titleWeight="normal"
|
||||||
borderType={dataFile.filter((val) => !val.delete).length - 1 == index && fileForm.length == 0 ? "none" : "bottom"}
|
onPress={() => { setIndexDelFile({ id: item.id, cat: "oldFile" }); setModalFile(true) }}
|
||||||
icon={<MaterialCommunityIcons name="file-outline" size={25} color={colors.text} />}
|
/>
|
||||||
title={item.name + '.' + item.extension}
|
))
|
||||||
titleWeight="normal"
|
}
|
||||||
onPress={() => { setIndexDelFile({ id: item.id, cat: "oldFile" }); setModalFile(true) }}
|
{
|
||||||
/>
|
fileForm.map((item, index) => (
|
||||||
))
|
<BorderBottomItem
|
||||||
}
|
key={index}
|
||||||
{
|
borderType={fileForm.length > 1 ? "bottom" : "none"}
|
||||||
fileForm.map((item, index) => (
|
icon={<MaterialCommunityIcons name="file-outline" size={25} color="black" />}
|
||||||
<BorderBottomItem
|
title={item.name}
|
||||||
key={index}
|
titleWeight="normal"
|
||||||
borderType={fileForm.length - 1 == index ? "none" : "bottom"}
|
onPress={() => { setIndexDelFile({ id: index, cat: "newFile" }); setModalFile(true) }}
|
||||||
icon={<MaterialCommunityIcons name="file-outline" size={25} color={colors.text} />}
|
/>
|
||||||
title={item.name}
|
))
|
||||||
titleWeight="normal"
|
}
|
||||||
onPress={() => { setIndexDelFile({ id: index, cat: "newFile" }); setModalFile(true) }}
|
</View>
|
||||||
/>
|
|
||||||
))
|
|
||||||
}
|
|
||||||
</View>
|
|
||||||
</>
|
|
||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
</ScrollView>
|
</ScrollView>
|
||||||
@@ -228,7 +218,7 @@ export default function DiscussionDivisionEdit() {
|
|||||||
<DrawerBottom animation="slide" isVisible={isModalFile} setVisible={setModalFile} title="Menu">
|
<DrawerBottom animation="slide" isVisible={isModalFile} setVisible={setModalFile} title="Menu">
|
||||||
<View style={Styles.rowItemsCenter}>
|
<View style={Styles.rowItemsCenter}>
|
||||||
<MenuItemRow
|
<MenuItemRow
|
||||||
icon={<Ionicons name="trash-outline" color={colors.text} size={25} />}
|
icon={<Ionicons name="trash" color="black" size={25} />}
|
||||||
title="Hapus"
|
title="Hapus"
|
||||||
onPress={() => { deleteFile(indexDelFile.id, indexDelFile.cat) }}
|
onPress={() => { deleteFile(indexDelFile.id, indexDelFile.cat) }}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
import AlertKonfirmasi from "@/components/alertKonfirmasi";
|
||||||
import AppHeader from "@/components/AppHeader";
|
import AppHeader from "@/components/AppHeader";
|
||||||
import BorderBottomItem from "@/components/borderBottomItem";
|
import BorderBottomItem from "@/components/borderBottomItem";
|
||||||
import BorderBottomItem2 from "@/components/borderBottomItem2";
|
import BorderBottomItem2 from "@/components/borderBottomItem2";
|
||||||
@@ -7,7 +8,6 @@ import ImageUser from "@/components/imageNew";
|
|||||||
import { InputForm } from "@/components/inputForm";
|
import { InputForm } from "@/components/inputForm";
|
||||||
import LabelStatus from "@/components/labelStatus";
|
import LabelStatus from "@/components/labelStatus";
|
||||||
import MenuItemRow from "@/components/menuItemRow";
|
import MenuItemRow from "@/components/menuItemRow";
|
||||||
import ModalConfirmation from "@/components/ModalConfirmation";
|
|
||||||
import Skeleton from "@/components/skeleton";
|
import Skeleton from "@/components/skeleton";
|
||||||
import SkeletonContent from "@/components/skeletonContent";
|
import SkeletonContent from "@/components/skeletonContent";
|
||||||
import Text from "@/components/Text";
|
import Text from "@/components/Text";
|
||||||
@@ -23,7 +23,6 @@ import {
|
|||||||
} from "@/lib/api";
|
} from "@/lib/api";
|
||||||
import { getDB } from "@/lib/firebaseDatabase";
|
import { getDB } from "@/lib/firebaseDatabase";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { Feather, Ionicons, MaterialCommunityIcons, MaterialIcons } from "@expo/vector-icons";
|
import { Feather, Ionicons, MaterialCommunityIcons, MaterialIcons } from "@expo/vector-icons";
|
||||||
import { ref } from "@react-native-firebase/database";
|
import { ref } from "@react-native-firebase/database";
|
||||||
import { useHeaderHeight } from '@react-navigation/elements';
|
import { useHeaderHeight } from '@react-navigation/elements';
|
||||||
@@ -65,7 +64,6 @@ type PropsFile = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export default function DiscussionDetail() {
|
export default function DiscussionDetail() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const { id, detail } = useLocalSearchParams<{ id: string; detail: string }>();
|
const { id, detail } = useLocalSearchParams<{ id: string; detail: string }>();
|
||||||
const [data, setData] = useState<Props>();
|
const [data, setData] = useState<Props>();
|
||||||
const [dataComment, setDataComment] = useState<PropsComment[]>([]);
|
const [dataComment, setDataComment] = useState<PropsComment[]>([]);
|
||||||
@@ -92,7 +90,6 @@ export default function DiscussionDetail() {
|
|||||||
comment: ''
|
comment: ''
|
||||||
})
|
})
|
||||||
const [viewEdit, setViewEdit] = useState(false)
|
const [viewEdit, setViewEdit] = useState(false)
|
||||||
const [showDeleteModal, setShowDeleteModal] = useState(false)
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -202,11 +199,8 @@ export default function DiscussionDetail() {
|
|||||||
setKomentar("")
|
setKomentar("")
|
||||||
updateTrigger()
|
updateTrigger()
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Gagal menambahkan komentar"
|
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoadingSend(false);
|
setLoadingSend(false);
|
||||||
}
|
}
|
||||||
@@ -225,11 +219,8 @@ export default function DiscussionDetail() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message })
|
Toast.show({ type: 'small', text1: response.message })
|
||||||
}
|
}
|
||||||
} catch (error : any ) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Gagal mengedit komentar"
|
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoadingSend(false);
|
setLoadingSend(false);
|
||||||
handleViewEditKomentar()
|
handleViewEditKomentar()
|
||||||
@@ -249,11 +240,8 @@ export default function DiscussionDetail() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message })
|
Toast.show({ type: 'small', text1: response.message })
|
||||||
}
|
}
|
||||||
} catch (error : any ) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Gagal menghapus komentar"
|
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoadingSend(false)
|
setLoadingSend(false)
|
||||||
setVisible(false)
|
setVisible(false)
|
||||||
@@ -307,7 +295,7 @@ export default function DiscussionDetail() {
|
|||||||
showBack={true}
|
showBack={true}
|
||||||
onPressLeft={() => router.back()}
|
onPressLeft={() => router.back()}
|
||||||
right={
|
right={
|
||||||
((entityUser.role != "user" && entityUser.role != "coadmin") || isAdminDivision || isCreator) ?
|
(entityUser.role != "user" && entityUser.role != "coadmin") || isAdminDivision || isCreator ?
|
||||||
<HeaderRightDiscussionDetail
|
<HeaderRightDiscussionDetail
|
||||||
id={detail}
|
id={detail}
|
||||||
status={data?.status}
|
status={data?.status}
|
||||||
@@ -318,13 +306,12 @@ export default function DiscussionDetail() {
|
|||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<View style={{ flex: 1, backgroundColor: colors.background }}>
|
<View style={{ flex: 1 }}>
|
||||||
<ScrollView
|
<ScrollView
|
||||||
refreshControl={
|
refreshControl={
|
||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={handleRefresh}
|
onRefresh={handleRefresh}
|
||||||
tintColor={colors.icon}
|
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
@@ -336,8 +323,7 @@ export default function DiscussionDetail() {
|
|||||||
<BorderBottomItem2
|
<BorderBottomItem2
|
||||||
dataFile={fileDiscussion}
|
dataFile={fileDiscussion}
|
||||||
descEllipsize={false}
|
descEllipsize={false}
|
||||||
bgColor="white"
|
borderType="bottom"
|
||||||
borderType="all"
|
|
||||||
icon={
|
icon={
|
||||||
<ImageUser
|
<ImageUser
|
||||||
src={`${ConstEnv.url_storage}/files/${data?.user_img}`}
|
src={`${ConstEnv.url_storage}/files/${data?.user_img}`}
|
||||||
@@ -366,7 +352,7 @@ export default function DiscussionDetail() {
|
|||||||
color="grey"
|
color="grey"
|
||||||
style={Styles.mr05}
|
style={Styles.mr05}
|
||||||
/>
|
/>
|
||||||
<Text style={[Styles.textInformation, { color: colors.dimmed }, Styles.mb05]} >
|
<Text style={[Styles.textInformation, Styles.cGray, Styles.mb05]} >
|
||||||
{dataComment.length} Komentar
|
{dataComment.length} Komentar
|
||||||
</Text>
|
</Text>
|
||||||
</View>
|
</View>
|
||||||
@@ -374,7 +360,7 @@ export default function DiscussionDetail() {
|
|||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
|
|
||||||
<View style={[Styles.mt10]}>
|
<View style={[Styles.p15]}>
|
||||||
{
|
{
|
||||||
loadingKomentar ?
|
loadingKomentar ?
|
||||||
arrSkeleton.map((item, index) => (
|
arrSkeleton.map((item, index) => (
|
||||||
@@ -384,7 +370,7 @@ export default function DiscussionDetail() {
|
|||||||
dataComment.map((item, index) => (
|
dataComment.map((item, index) => (
|
||||||
<BorderBottomItem
|
<BorderBottomItem
|
||||||
key={index}
|
key={index}
|
||||||
borderType="all"
|
borderType="bottom"
|
||||||
colorPress
|
colorPress
|
||||||
icon={
|
icon={
|
||||||
<ImageUser
|
<ImageUser
|
||||||
@@ -397,7 +383,6 @@ export default function DiscussionDetail() {
|
|||||||
desc={item.comment}
|
desc={item.comment}
|
||||||
rightBottomInfo={item.isEdited ? "Edited" : ""}
|
rightBottomInfo={item.isEdited ? "Edited" : ""}
|
||||||
descEllipsize={detailMore.includes(item.id) ? false : true}
|
descEllipsize={detailMore.includes(item.id) ? false : true}
|
||||||
bgColor="white"
|
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
setDetailMore((prev: any) => {
|
setDetailMore((prev: any) => {
|
||||||
if (prev.includes(item.id)) {
|
if (prev.includes(item.id)) {
|
||||||
@@ -425,7 +410,7 @@ export default function DiscussionDetail() {
|
|||||||
style={[
|
style={[
|
||||||
Styles.contentItemCenter,
|
Styles.contentItemCenter,
|
||||||
Styles.w100,
|
Styles.w100,
|
||||||
{ backgroundColor: colors.background },
|
{ backgroundColor: "#f4f4f4" },
|
||||||
viewEdit && Styles.borderTop
|
viewEdit && Styles.borderTop
|
||||||
]}
|
]}
|
||||||
>
|
>
|
||||||
@@ -434,15 +419,15 @@ export default function DiscussionDetail() {
|
|||||||
<>
|
<>
|
||||||
<View style={[Styles.w90, Styles.rowSpaceBetween, Styles.pv05]}>
|
<View style={[Styles.w90, Styles.rowSpaceBetween, Styles.pv05]}>
|
||||||
<View style={[Styles.rowItemsCenter]}>
|
<View style={[Styles.rowItemsCenter]}>
|
||||||
<Feather name="edit-3" color={colors.text} size={22} style={[Styles.mh05]} />
|
<Feather name="edit-3" color="black" size={22} style={[Styles.mh05]} />
|
||||||
<Text style={[Styles.textMediumSemiBold]}>Edit Komentar</Text>
|
<Text style={[Styles.textMediumSemiBold]}>Edit Komentar</Text>
|
||||||
</View>
|
</View>
|
||||||
<Pressable onPress={() => handleViewEditKomentar()}>
|
<Pressable onPress={() => handleViewEditKomentar()}>
|
||||||
<MaterialIcons name="close" color={colors.text} size={22} />
|
<MaterialIcons name="close" color="black" size={22} />
|
||||||
</Pressable>
|
</Pressable>
|
||||||
</View>
|
</View>
|
||||||
<InputForm
|
<InputForm
|
||||||
bg={colors.card}
|
bg="white"
|
||||||
type="default"
|
type="default"
|
||||||
round
|
round
|
||||||
multiline
|
multiline
|
||||||
@@ -475,8 +460,8 @@ export default function DiscussionDetail() {
|
|||||||
size={25}
|
size={25}
|
||||||
style={
|
style={
|
||||||
[selectKomentar.comment == "" || regexOnlySpacesOrEnter.test(selectKomentar.comment) || loadingSend || ((entityUser.role == "user" || entityUser.role == "coadmin") && !isMemberDivision)
|
[selectKomentar.comment == "" || regexOnlySpacesOrEnter.test(selectKomentar.comment) || loadingSend || ((entityUser.role == "user" || entityUser.role == "coadmin") && !isMemberDivision)
|
||||||
? { color: colors.dimmed }
|
? Styles.cGray
|
||||||
: { color: colors.tint },
|
: Styles.cDefault,
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
@@ -489,6 +474,7 @@ export default function DiscussionDetail() {
|
|||||||
isMemberDivision)
|
isMemberDivision)
|
||||||
?
|
?
|
||||||
<InputForm
|
<InputForm
|
||||||
|
bg="white"
|
||||||
type="default"
|
type="default"
|
||||||
round
|
round
|
||||||
multiline
|
multiline
|
||||||
@@ -521,8 +507,8 @@ export default function DiscussionDetail() {
|
|||||||
size={25}
|
size={25}
|
||||||
style={
|
style={
|
||||||
[komentar == "" || regexOnlySpacesOrEnter.test(komentar) || loadingSend || ((entityUser.role == "user" || entityUser.role == "coadmin") && !isMemberDivision)
|
[komentar == "" || regexOnlySpacesOrEnter.test(komentar) || loadingSend || ((entityUser.role == "user" || entityUser.role == "coadmin") && !isMemberDivision)
|
||||||
? { color: colors.dimmed }
|
? Styles.cGray
|
||||||
: { color: colors.tint }
|
: Styles.cDefault,
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
@@ -531,7 +517,7 @@ export default function DiscussionDetail() {
|
|||||||
/>
|
/>
|
||||||
:
|
:
|
||||||
<View style={[Styles.pv20, { alignItems: 'center' }]}>
|
<View style={[Styles.pv20, { alignItems: 'center' }]}>
|
||||||
<Text style={[Styles.textInformation, { color: colors.dimmed }]}>
|
<Text style={[Styles.textInformation, Styles.cGray]}>
|
||||||
{
|
{
|
||||||
data?.status == 2 ? "Diskusi telah ditutup" : data?.isActive == false ? "Diskusi telah diarsipkan" : "Hanya anggota divisi yang dapat memberikan komentar"
|
data?.status == 2 ? "Diskusi telah ditutup" : data?.isActive == false ? "Diskusi telah diarsipkan" : "Hanya anggota divisi yang dapat memberikan komentar"
|
||||||
}
|
}
|
||||||
@@ -545,35 +531,25 @@ export default function DiscussionDetail() {
|
|||||||
<DrawerBottom animation="slide" isVisible={isVisible} setVisible={setVisible} title="Komentar">
|
<DrawerBottom animation="slide" isVisible={isVisible} setVisible={setVisible} title="Komentar">
|
||||||
<View style={Styles.rowItemsCenter}>
|
<View style={Styles.rowItemsCenter}>
|
||||||
<MenuItemRow
|
<MenuItemRow
|
||||||
icon={<MaterialCommunityIcons name="pencil-outline" color={colors.text} size={25} />}
|
icon={<MaterialCommunityIcons name="pencil-outline" color="black" size={25} />}
|
||||||
title="Edit"
|
title="Edit"
|
||||||
onPress={() => { handleViewEditKomentar() }}
|
onPress={() => { handleViewEditKomentar() }}
|
||||||
/>
|
/>
|
||||||
<MenuItemRow
|
<MenuItemRow
|
||||||
icon={<Ionicons name="trash-outline" color={colors.text} size={25} />}
|
icon={<MaterialIcons name="delete" color="black" size={25} />}
|
||||||
title="Hapus"
|
title="Hapus"
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
setVisible(false)
|
AlertKonfirmasi({
|
||||||
setTimeout(() => {
|
title: 'Konfirmasi',
|
||||||
setShowDeleteModal(true)
|
desc: 'Apakah anda yakin ingin menghapus komentar?',
|
||||||
}, 600)
|
onPress: () => {
|
||||||
|
handleDeleteKomentar()
|
||||||
|
}
|
||||||
|
})
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
</View>
|
</View>
|
||||||
</DrawerBottom>
|
</DrawerBottom>
|
||||||
|
|
||||||
<ModalConfirmation
|
|
||||||
visible={showDeleteModal}
|
|
||||||
title="Konfirmasi"
|
|
||||||
message="Apakah anda yakin ingin menghapus komentar?"
|
|
||||||
onConfirm={() => {
|
|
||||||
setShowDeleteModal(false)
|
|
||||||
handleDeleteKomentar()
|
|
||||||
}}
|
|
||||||
onCancel={() => setShowDeleteModal(false)}
|
|
||||||
confirmText="Hapus"
|
|
||||||
cancelText="Batal"
|
|
||||||
/>
|
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,14 +4,13 @@ import ButtonSaveHeader from "@/components/buttonSaveHeader"
|
|||||||
import ButtonSelect from "@/components/buttonSelect"
|
import ButtonSelect from "@/components/buttonSelect"
|
||||||
import DrawerBottom from "@/components/drawerBottom"
|
import DrawerBottom from "@/components/drawerBottom"
|
||||||
import { InputForm } from "@/components/inputForm"
|
import { InputForm } from "@/components/inputForm"
|
||||||
import LoadingCenter from "@/components/loadingCenter"
|
import LoadingOverlay from "@/components/loadingOverlay"
|
||||||
import MenuItemRow from "@/components/menuItemRow"
|
import MenuItemRow from "@/components/menuItemRow"
|
||||||
import Text from "@/components/Text"
|
import Text from "@/components/Text"
|
||||||
import Styles from "@/constants/Styles"
|
import Styles from "@/constants/Styles"
|
||||||
import { apiCreateDiscussion } from "@/lib/api"
|
import { apiCreateDiscussion } from "@/lib/api"
|
||||||
import { setUpdateDiscussion } from "@/lib/discussionUpdate"
|
import { setUpdateDiscussion } from "@/lib/discussionUpdate"
|
||||||
import { useAuthSession } from "@/providers/AuthProvider"
|
import { useAuthSession } from "@/providers/AuthProvider"
|
||||||
import { useTheme } from "@/providers/ThemeProvider"
|
|
||||||
import { Ionicons, MaterialCommunityIcons } from "@expo/vector-icons"
|
import { Ionicons, MaterialCommunityIcons } from "@expo/vector-icons"
|
||||||
import * as DocumentPicker from "expo-document-picker"
|
import * as DocumentPicker from "expo-document-picker"
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router"
|
import { router, Stack, useLocalSearchParams } from "expo-router"
|
||||||
@@ -22,7 +21,6 @@ import { useDispatch, useSelector } from "react-redux"
|
|||||||
|
|
||||||
|
|
||||||
export default function CreateDiscussionDivision() {
|
export default function CreateDiscussionDivision() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const { id } = useLocalSearchParams<{ id: string }>()
|
const { id } = useLocalSearchParams<{ id: string }>()
|
||||||
const [desc, setDesc] = useState('')
|
const [desc, setDesc] = useState('')
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
@@ -81,18 +79,16 @@ export default function CreateDiscussionDivision() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal menambahkan data"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={{ flex: 1, backgroundColor: colors.background }}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
||||||
@@ -121,7 +117,7 @@ export default function CreateDiscussionDivision() {
|
|||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
{loading && <LoadingCenter />}
|
<LoadingOverlay visible={loading} />
|
||||||
<ScrollView showsVerticalScrollIndicator={false} style={[Styles.h100]}>
|
<ScrollView showsVerticalScrollIndicator={false} style={[Styles.h100]}>
|
||||||
<View style={[Styles.p15, Styles.mb100]}>
|
<View style={[Styles.p15, Styles.mb100]}>
|
||||||
<InputForm
|
<InputForm
|
||||||
@@ -131,33 +127,26 @@ export default function CreateDiscussionDivision() {
|
|||||||
required
|
required
|
||||||
onChange={setDesc}
|
onChange={setDesc}
|
||||||
multiline
|
multiline
|
||||||
bg={colors.card}
|
|
||||||
/>
|
/>
|
||||||
<ButtonSelect value="Upload File" onPress={pickDocumentAsync} />
|
<ButtonSelect value="Upload File" onPress={pickDocumentAsync} />
|
||||||
{
|
{
|
||||||
fileForm.length > 0
|
fileForm.length > 0
|
||||||
&&
|
&&
|
||||||
<>
|
<View style={[Styles.borderAll, Styles.round10, Styles.p10, Styles.mb10]}>
|
||||||
<View style={[Styles.rowSpaceBetween, Styles.mv05]}>
|
<Text style={[Styles.textDefaultSemiBold]}>File</Text>
|
||||||
<Text style={[Styles.textDefaultSemiBold]}>File</Text>
|
{
|
||||||
<Text style={[Styles.textDefault]}>{fileForm.length} file</Text>
|
fileForm.map((item, index) => (
|
||||||
</View>
|
<BorderBottomItem
|
||||||
<View style={[Styles.borderAll, Styles.round05, Styles.p10, Styles.mb10, { backgroundColor: colors.card, borderColor: colors.icon + '20' }]}>
|
key={index}
|
||||||
{
|
borderType={fileForm.length > 1 ? "bottom" : "none"}
|
||||||
fileForm.map((item, index) => (
|
icon={<MaterialCommunityIcons name="file-outline" size={25} color="black" />}
|
||||||
<BorderBottomItem
|
title={item.name}
|
||||||
key={index}
|
titleWeight="normal"
|
||||||
borderType={fileForm.length - 1 == index ? "none" : "bottom"}
|
onPress={() => { setIndexDelFile(index); setModalFile(true) }}
|
||||||
icon={<MaterialCommunityIcons name="file-outline" size={25} color={colors.text} />}
|
/>
|
||||||
title={item.name}
|
))
|
||||||
titleWeight="normal"
|
}
|
||||||
onPress={() => { setIndexDelFile(index); setModalFile(true) }}
|
</View>
|
||||||
/>
|
|
||||||
))
|
|
||||||
}
|
|
||||||
</View>
|
|
||||||
</>
|
|
||||||
|
|
||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
</ScrollView>
|
</ScrollView>
|
||||||
@@ -165,7 +154,7 @@ export default function CreateDiscussionDivision() {
|
|||||||
<DrawerBottom animation="slide" isVisible={isModalFile} setVisible={setModalFile} title="Menu">
|
<DrawerBottom animation="slide" isVisible={isModalFile} setVisible={setModalFile} title="Menu">
|
||||||
<View style={Styles.rowItemsCenter}>
|
<View style={Styles.rowItemsCenter}>
|
||||||
<MenuItemRow
|
<MenuItemRow
|
||||||
icon={<Ionicons name="trash-outline" color={colors.text} size={25} />}
|
icon={<Ionicons name="trash" color="black" size={25} />}
|
||||||
title="Hapus"
|
title="Hapus"
|
||||||
onPress={() => { deleteFile(indexDelFile) }}
|
onPress={() => { deleteFile(indexDelFile) }}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -5,12 +5,10 @@ import InputSearch from "@/components/inputSearch";
|
|||||||
import LabelStatus from "@/components/labelStatus";
|
import LabelStatus from "@/components/labelStatus";
|
||||||
import SkeletonContent from "@/components/skeletonContent";
|
import SkeletonContent from "@/components/skeletonContent";
|
||||||
import Text from "@/components/Text";
|
import Text from "@/components/Text";
|
||||||
import WrapTab from "@/components/wrapTab";
|
|
||||||
import { ConstEnv } from "@/constants/ConstEnv";
|
import { ConstEnv } from "@/constants/ConstEnv";
|
||||||
import Styles from "@/constants/Styles";
|
import Styles from "@/constants/Styles";
|
||||||
import { apiGetDiscussion, apiGetDivisionOneFeature } from "@/lib/api";
|
import { apiGetDiscussion, apiGetDivisionOneFeature } from "@/lib/api";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { AntDesign, Feather, Ionicons } from "@expo/vector-icons";
|
import { AntDesign, Feather, Ionicons } from "@expo/vector-icons";
|
||||||
import { router, useLocalSearchParams } from "expo-router";
|
import { router, useLocalSearchParams } from "expo-router";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
@@ -32,7 +30,6 @@ type Props = {
|
|||||||
|
|
||||||
|
|
||||||
export default function DiscussionDivision() {
|
export default function DiscussionDivision() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const { id, active } = useLocalSearchParams<{ id: string, active?: string }>()
|
const { id, active } = useLocalSearchParams<{ id: string, active?: string }>()
|
||||||
const [data, setData] = useState<Props[]>([])
|
const [data, setData] = useState<Props[]>([])
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
@@ -131,26 +128,26 @@ export default function DiscussionDivision() {
|
|||||||
})
|
})
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<View style={[Styles.p15, { flex: 1, backgroundColor: colors.background }]}>
|
<View style={[Styles.p15, { flex: 1 }]}>
|
||||||
{
|
{
|
||||||
((entityUser.role != "user" && entityUser.role != "coadmin") || isAdminDivision) &&
|
((entityUser.role != "user" && entityUser.role != "coadmin") || isAdminDivision) &&
|
||||||
<View>
|
<View>
|
||||||
<WrapTab>
|
<View style={[Styles.wrapBtnTab]}>
|
||||||
<ButtonTab
|
<ButtonTab
|
||||||
active={status == "false" ? "false" : "true"}
|
active={status == "false" ? "false" : "true"}
|
||||||
value="true"
|
value="true"
|
||||||
onPress={() => { setStatus("true") }}
|
onPress={() => { setStatus("true") }}
|
||||||
label="Aktif"
|
label="Aktif"
|
||||||
icon={<Feather name="check-circle" color={status == "false" ? colors.dimmed : 'white'} size={20} />}
|
icon={<Feather name="check-circle" color={status == "false" ? 'black' : 'white'} size={20} />}
|
||||||
n={2} />
|
n={2} />
|
||||||
<ButtonTab
|
<ButtonTab
|
||||||
active={status == "false" ? "false" : "true"}
|
active={status == "false" ? "false" : "true"}
|
||||||
value="false"
|
value="false"
|
||||||
onPress={() => { setStatus("false") }}
|
onPress={() => { setStatus("false") }}
|
||||||
label="Arsip"
|
label="Arsip"
|
||||||
icon={<AntDesign name="closecircleo" color={status == "true" ? colors.dimmed : 'white'} size={20} />}
|
icon={<AntDesign name="closecircleo" color={status == "true" ? 'black' : 'white'} size={20} />}
|
||||||
n={2} />
|
n={2} />
|
||||||
</WrapTab>
|
</View>
|
||||||
<InputSearch onChange={setSearch} />
|
<InputSearch onChange={setSearch} />
|
||||||
</View>
|
</View>
|
||||||
}
|
}
|
||||||
@@ -177,7 +174,7 @@ export default function DiscussionDivision() {
|
|||||||
onPress={() => { router.push(`./discussion/${item.id}`) }}
|
onPress={() => { router.push(`./discussion/${item.id}`) }}
|
||||||
borderType="bottom"
|
borderType="bottom"
|
||||||
icon={
|
icon={
|
||||||
<ImageUser src={`${ConstEnv.url_storage}/files/${item.img}`} size="xs" />
|
<ImageUser src={`${ConstEnv.url_storage}/files/${item.img}`} size="sm" />
|
||||||
}
|
}
|
||||||
title={item.user_name}
|
title={item.user_name}
|
||||||
subtitle={
|
subtitle={
|
||||||
@@ -187,12 +184,11 @@ export default function DiscussionDivision() {
|
|||||||
desc={item.desc}
|
desc={item.desc}
|
||||||
leftBottomInfo={
|
leftBottomInfo={
|
||||||
<View style={[Styles.rowItemsCenter]}>
|
<View style={[Styles.rowItemsCenter]}>
|
||||||
<Ionicons name="chatbox-ellipses-outline" size={18} color={colors.dimmed} style={Styles.mr05} />
|
<Ionicons name="chatbox-ellipses-outline" size={18} color="grey" style={Styles.mr05} />
|
||||||
<Text style={[Styles.textInformation, { color: colors.dimmed }, Styles.mb05]}>Diskusikan</Text>
|
<Text style={[Styles.textInformation, Styles.cGray, Styles.mb05]}>Diskusikan</Text>
|
||||||
</View>
|
</View>
|
||||||
}
|
}
|
||||||
rightBottomInfo={item.total_komentar + ' Komentar'}
|
rightBottomInfo={item.total_komentar + ' Komentar'}
|
||||||
bgColor="transparent"
|
|
||||||
/>
|
/>
|
||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
@@ -204,12 +200,11 @@ export default function DiscussionDivision() {
|
|||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={handleRefresh}
|
onRefresh={handleRefresh}
|
||||||
tintColor={colors.icon}
|
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
:
|
:
|
||||||
(<Text style={[Styles.textDefault, Styles.mv10, { textAlign: "center", color: colors.dimmed }]}>Tidak ada diskusi</Text>)
|
(<Text style={[Styles.textDefault, Styles.cGray, Styles.mv10, { textAlign: "center" }]}>Tidak ada diskusi</Text>)
|
||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import ModalConfirmation from "@/components/ModalConfirmation";
|
import AlertKonfirmasi from "@/components/alertKonfirmasi";
|
||||||
import AppHeader from "@/components/AppHeader";
|
import AppHeader from "@/components/AppHeader";
|
||||||
import { ButtonHeader } from "@/components/buttonHeader";
|
import { ButtonHeader } from "@/components/buttonHeader";
|
||||||
import HeaderRightDocument from "@/components/document/headerDocument";
|
import HeaderRightDocument from "@/components/document/headerDocument";
|
||||||
@@ -12,6 +12,7 @@ import ModalLoading from "@/components/modalLoading";
|
|||||||
import ModalSelectMultiple from "@/components/modalSelectMultiple";
|
import ModalSelectMultiple from "@/components/modalSelectMultiple";
|
||||||
import Skeleton from "@/components/skeleton";
|
import Skeleton from "@/components/skeleton";
|
||||||
import Text from "@/components/Text";
|
import Text from "@/components/Text";
|
||||||
|
import { ColorsStatus } from "@/constants/ColorsStatus";
|
||||||
import { ConstEnv } from "@/constants/ConstEnv";
|
import { ConstEnv } from "@/constants/ConstEnv";
|
||||||
import Styles from "@/constants/Styles";
|
import Styles from "@/constants/Styles";
|
||||||
import {
|
import {
|
||||||
@@ -23,7 +24,6 @@ import {
|
|||||||
} from "@/lib/api";
|
} from "@/lib/api";
|
||||||
import { setUpdateDokumen } from "@/lib/dokumenUpdate";
|
import { setUpdateDokumen } from "@/lib/dokumenUpdate";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import {
|
import {
|
||||||
AntDesign,
|
AntDesign,
|
||||||
MaterialCommunityIcons,
|
MaterialCommunityIcons,
|
||||||
@@ -66,7 +66,6 @@ type PropsPath = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
export default function DocumentDivision() {
|
export default function DocumentDivision() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const [loadingRename, setLoadingRename] = useState(false)
|
const [loadingRename, setLoadingRename] = useState(false)
|
||||||
const [isShare, setShare] = useState(false)
|
const [isShare, setShare] = useState(false)
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
@@ -89,7 +88,6 @@ export default function DocumentDivision() {
|
|||||||
const [loadingOpen, setLoadingOpen] = useState(false)
|
const [loadingOpen, setLoadingOpen] = useState(false)
|
||||||
const [isMemberDivision, setIsMemberDivision] = useState(false)
|
const [isMemberDivision, setIsMemberDivision] = useState(false)
|
||||||
const entityUser = useSelector((state: any) => state.user)
|
const entityUser = useSelector((state: any) => state.user)
|
||||||
const [showDeleteModal, setShowDeleteModal] = useState(false)
|
|
||||||
const [bodyRename, setBodyRename] = useState({
|
const [bodyRename, setBodyRename] = useState({
|
||||||
id: "",
|
id: "",
|
||||||
name: "",
|
name: "",
|
||||||
@@ -235,11 +233,9 @@ export default function DocumentDivision() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error : any ) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Gagal mengubah nama"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoadingRename(false)
|
setLoadingRename(false)
|
||||||
setRename(false)
|
setRename(false)
|
||||||
@@ -260,11 +256,9 @@ export default function DocumentDivision() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error : any ) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Gagal menghapus"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -288,11 +282,9 @@ export default function DocumentDivision() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error : any ) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Gagal membagikan"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setShare(false);
|
setShare(false);
|
||||||
}
|
}
|
||||||
@@ -342,7 +334,7 @@ export default function DocumentDivision() {
|
|||||||
}, [path]);
|
}, [path]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={{ flex: 1, backgroundColor: colors.background }}>
|
<SafeAreaView style={{ flex: 1 }}>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () =>
|
// headerLeft: () =>
|
||||||
@@ -388,7 +380,7 @@ export default function DocumentDivision() {
|
|||||||
showBack={(selectedFiles.length > 0 || dariSelectAll) ? false : true}
|
showBack={(selectedFiles.length > 0 || dariSelectAll) ? false : true}
|
||||||
left={
|
left={
|
||||||
<ButtonHeader
|
<ButtonHeader
|
||||||
item={<MaterialIcons name="close" size={25} color="white" />}
|
item={<MaterialIcons name="close" size={20} color="white" />}
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
handleBatal();
|
handleBatal();
|
||||||
}}
|
}}
|
||||||
@@ -401,7 +393,7 @@ export default function DocumentDivision() {
|
|||||||
selectedFiles.length > 0 || dariSelectAll ? (
|
selectedFiles.length > 0 || dariSelectAll ? (
|
||||||
<ButtonHeader
|
<ButtonHeader
|
||||||
item={
|
item={
|
||||||
<MaterialIcons name="checklist-rtl" size={25} color="white" />
|
<MaterialIcons name="checklist-rtl" size={20} color="white" />
|
||||||
}
|
}
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
handleSelectAll();
|
handleSelectAll();
|
||||||
@@ -421,7 +413,6 @@ export default function DocumentDivision() {
|
|||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={handleRefresh}
|
onRefresh={handleRefresh}
|
||||||
tintColor={colors.icon}
|
|
||||||
/>
|
/>
|
||||||
}>
|
}>
|
||||||
<View style={[Styles.p15, Styles.mb100]}>
|
<View style={[Styles.p15, Styles.mb100]}>
|
||||||
@@ -436,9 +427,9 @@ export default function DocumentDivision() {
|
|||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
{item.id != "home" && (
|
{item.id != "home" && (
|
||||||
<AntDesign name="right" style={[Styles.mh05, Styles.mt02]} color={colors.text} />
|
<AntDesign name="right" style={[Styles.mh05, Styles.mt02]} color="black" />
|
||||||
)}
|
)}
|
||||||
<Text style={{ color: colors.text }}> {item.name} </Text>
|
<Text> {item.name} </Text>
|
||||||
</Pressable>
|
</Pressable>
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
@@ -487,7 +478,14 @@ export default function DocumentDivision() {
|
|||||||
);
|
);
|
||||||
})
|
})
|
||||||
) : (
|
) : (
|
||||||
<Text style={[Styles.textDefault, Styles.mt15, { textAlign: "center", color: colors.dimmed }]} >
|
<Text
|
||||||
|
style={[
|
||||||
|
Styles.textDefault,
|
||||||
|
Styles.cGray,
|
||||||
|
Styles.mt15,
|
||||||
|
{ textAlign: "center" },
|
||||||
|
]}
|
||||||
|
>
|
||||||
Tidak ada dokumen
|
Tidak ada dokumen
|
||||||
</Text>
|
</Text>
|
||||||
)}
|
)}
|
||||||
@@ -495,7 +493,7 @@ export default function DocumentDivision() {
|
|||||||
</View>
|
</View>
|
||||||
</ScrollView>
|
</ScrollView>
|
||||||
{(selectedFiles.length > 0 || dariSelectAll) && (
|
{(selectedFiles.length > 0 || dariSelectAll) && (
|
||||||
<View style={[Styles.bottomMenuSelectDocument, { backgroundColor: colors.header }]}>
|
<View style={[ColorsStatus.primary, Styles.bottomMenuSelectDocument]}>
|
||||||
<View style={[Styles.rowItemsCenter, { justifyContent: "center" }]}>
|
<View style={[Styles.rowItemsCenter, { justifyContent: "center" }]}>
|
||||||
<MenuItemRow
|
<MenuItemRow
|
||||||
icon={
|
icon={
|
||||||
@@ -507,7 +505,13 @@ export default function DocumentDivision() {
|
|||||||
}
|
}
|
||||||
title="Hapus"
|
title="Hapus"
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
setShowDeleteModal(true)
|
AlertKonfirmasi({
|
||||||
|
title: "Konfirmasi",
|
||||||
|
desc: "Apakah anda yakin ingin menghapus dokumen?",
|
||||||
|
onPress: () => {
|
||||||
|
handleDelete();
|
||||||
|
},
|
||||||
|
});
|
||||||
}}
|
}}
|
||||||
column="many"
|
column="many"
|
||||||
color="white"
|
color="white"
|
||||||
@@ -614,19 +618,6 @@ export default function DocumentDivision() {
|
|||||||
value={id}
|
value={id}
|
||||||
item={selectedFiles[0]?.id}
|
item={selectedFiles[0]?.id}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<ModalConfirmation
|
|
||||||
visible={showDeleteModal}
|
|
||||||
title="Konfirmasi"
|
|
||||||
message="Apakah anda yakin ingin menghapus dokumen?"
|
|
||||||
onConfirm={() => {
|
|
||||||
setShowDeleteModal(false)
|
|
||||||
handleDelete()
|
|
||||||
}}
|
|
||||||
onCancel={() => setShowDeleteModal(false)}
|
|
||||||
confirmText="Hapus"
|
|
||||||
cancelText="Batal"
|
|
||||||
/>
|
|
||||||
</SafeAreaView>
|
</SafeAreaView>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -9,7 +9,6 @@ import Styles from "@/constants/Styles";
|
|||||||
import { apiAddFileTask, apiCheckFileTask } from "@/lib/api";
|
import { apiAddFileTask, apiCheckFileTask } from "@/lib/api";
|
||||||
import { setUpdateTask } from "@/lib/taskUpdate";
|
import { setUpdateTask } from "@/lib/taskUpdate";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { Ionicons, MaterialCommunityIcons } from "@expo/vector-icons";
|
import { Ionicons, MaterialCommunityIcons } from "@expo/vector-icons";
|
||||||
import * as DocumentPicker from "expo-document-picker";
|
import * as DocumentPicker from "expo-document-picker";
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
@@ -24,7 +23,6 @@ import Toast from "react-native-toast-message";
|
|||||||
import { useDispatch, useSelector } from "react-redux";
|
import { useDispatch, useSelector } from "react-redux";
|
||||||
|
|
||||||
export default function TaskDivisionAddFile() {
|
export default function TaskDivisionAddFile() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const { id, detail } = useLocalSearchParams<{ id: string; detail: string }>();
|
const { id, detail } = useLocalSearchParams<{ id: string; detail: string }>();
|
||||||
const [fileForm, setFileForm] = useState<any[]>([]);
|
const [fileForm, setFileForm] = useState<any[]>([]);
|
||||||
const [listFile, setListFile] = useState<any[]>([]);
|
const [listFile, setListFile] = useState<any[]>([]);
|
||||||
@@ -120,18 +118,16 @@ export default function TaskDivisionAddFile() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error : any ) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Gagal menambahkan file"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={{ flex: 1, backgroundColor: colors.background }}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => (
|
// headerLeft: () => (
|
||||||
@@ -173,13 +169,13 @@ export default function TaskDivisionAddFile() {
|
|||||||
listFile.length > 0 && (
|
listFile.length > 0 && (
|
||||||
<View style={[Styles.mb15]}>
|
<View style={[Styles.mb15]}>
|
||||||
<Text style={[Styles.textDefaultSemiBold, Styles.mv05]}>File</Text>
|
<Text style={[Styles.textDefaultSemiBold, Styles.mv05]}>File</Text>
|
||||||
<View style={[Styles.wrapPaper, { backgroundColor: colors.card, borderColor: colors.background }]}>
|
<View style={[Styles.wrapPaper]}>
|
||||||
{
|
{
|
||||||
listFile.map((item, index) => (
|
listFile.map((item, index) => (
|
||||||
<BorderBottomItem
|
<BorderBottomItem
|
||||||
key={index}
|
key={index}
|
||||||
borderType="all"
|
borderType="all"
|
||||||
icon={<MaterialCommunityIcons name="file-outline" size={25} color={colors.text} />}
|
icon={<MaterialCommunityIcons name="file-outline" size={25} color="black" />}
|
||||||
title={item}
|
title={item}
|
||||||
titleWeight="normal"
|
titleWeight="normal"
|
||||||
onPress={() => { setIndexDelFile(index); setModal(true) }}
|
onPress={() => { setIndexDelFile(index); setModal(true) }}
|
||||||
@@ -201,7 +197,7 @@ export default function TaskDivisionAddFile() {
|
|||||||
<DrawerBottom animation="slide" isVisible={isModal} setVisible={setModal} title="Menu">
|
<DrawerBottom animation="slide" isVisible={isModal} setVisible={setModal} title="Menu">
|
||||||
<View style={Styles.rowItemsCenter}>
|
<View style={Styles.rowItemsCenter}>
|
||||||
<MenuItemRow
|
<MenuItemRow
|
||||||
icon={<Ionicons name="trash-outline" color={colors.text} size={25} />}
|
icon={<Ionicons name="trash" color="black" size={25} />}
|
||||||
title="Hapus"
|
title="Hapus"
|
||||||
onPress={() => { deleteFile(indexDelFile) }}
|
onPress={() => { deleteFile(indexDelFile) }}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -9,7 +9,6 @@ import Styles from "@/constants/Styles";
|
|||||||
import { apiAddMemberTask, apiGetDivisionMember, apiGetTaskOne } from "@/lib/api";
|
import { apiAddMemberTask, apiGetDivisionMember, apiGetTaskOne } from "@/lib/api";
|
||||||
import { setUpdateTask } from "@/lib/taskUpdate";
|
import { setUpdateTask } from "@/lib/taskUpdate";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { AntDesign } from "@expo/vector-icons";
|
import { AntDesign } from "@expo/vector-icons";
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
@@ -24,7 +23,6 @@ type Props = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export default function AddMemberTask() {
|
export default function AddMemberTask() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const dispatch = useDispatch()
|
const dispatch = useDispatch()
|
||||||
const update = useSelector((state: any) => state.projectUpdate)
|
const update = useSelector((state: any) => state.projectUpdate)
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
@@ -86,11 +84,9 @@ export default function AddMemberTask() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error : any ) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal menambahkan anggota"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
@@ -131,7 +127,7 @@ export default function AddMemberTask() {
|
|||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<View style={[Styles.p15, { flex: 1, backgroundColor: colors.background }]}>
|
<View style={[Styles.p15, { flex: 1 }]}>
|
||||||
<InputSearch onChange={(val) => setSearch(val)} value={search} />
|
<InputSearch onChange={(val) => setSearch(val)} value={search} />
|
||||||
|
|
||||||
{
|
{
|
||||||
@@ -153,7 +149,7 @@ export default function AddMemberTask() {
|
|||||||
</View>
|
</View>
|
||||||
|
|
||||||
:
|
:
|
||||||
<Text style={[Styles.textDefault, Styles.pv05, { textAlign: 'center', color: colors.dimmed }]}>Tidak ada member yang dipilih</Text>
|
<Text style={[Styles.textDefault, Styles.cGray, Styles.pv05, { textAlign: 'center' }]}>Tidak ada member yang dipilih</Text>
|
||||||
}
|
}
|
||||||
<ScrollView
|
<ScrollView
|
||||||
showsVerticalScrollIndicator={false}
|
showsVerticalScrollIndicator={false}
|
||||||
@@ -166,22 +162,22 @@ export default function AddMemberTask() {
|
|||||||
return (
|
return (
|
||||||
<Pressable
|
<Pressable
|
||||||
key={index}
|
key={index}
|
||||||
style={[Styles.itemSelectModal, { borderColor: colors.icon + '20' }]}
|
style={[Styles.itemSelectModal]}
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
!found && onChoose(item.idUser, item.name, item.img)
|
!found && onChoose(item.idUser, item.name, item.img)
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<View style={[Styles.rowItemsCenter, Styles.w80,]}>
|
<View style={[Styles.rowItemsCenter, Styles.w80]}>
|
||||||
<ImageUser src={`${ConstEnv.url_storage}/files/${item.img}`} border />
|
<ImageUser src={`${ConstEnv.url_storage}/files/${item.img}`} border />
|
||||||
<View style={[Styles.ml10]}>
|
<View style={[Styles.ml10]}>
|
||||||
<Text style={[Styles.textDefault]} numberOfLines={1}>{item.name}</Text>
|
<Text style={[Styles.textDefault]} numberOfLines={1}>{item.name}</Text>
|
||||||
{
|
{
|
||||||
found && <Text style={[Styles.textInformation, { color: colors.dimmed }]}>sudah menjadi anggota</Text>
|
found && <Text style={[Styles.textInformation, Styles.cGray]}>sudah menjadi anggota</Text>
|
||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
{
|
{
|
||||||
selectMember.some((i: any) => i.idUser == item.idUser) && <AntDesign name="check" size={20} color={colors.text} />
|
selectMember.some((i: any) => i.idUser == item.idUser) && <AntDesign name="check" size={20} color={'black'} />
|
||||||
}
|
}
|
||||||
</Pressable>
|
</Pressable>
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
import AppHeader from "@/components/AppHeader";
|
import AppHeader from "@/components/AppHeader";
|
||||||
import ButtonSaveHeader from "@/components/buttonSaveHeader";
|
import ButtonSaveHeader from "@/components/buttonSaveHeader";
|
||||||
import ButtonSelect from "@/components/buttonSelect";
|
|
||||||
import { InputForm } from "@/components/inputForm";
|
import { InputForm } from "@/components/inputForm";
|
||||||
import ModalAddDetailTugasTask from "@/components/task/modalAddDetailTugasTask";
|
import ModalAddDetailTugasTask from "@/components/task/modalAddDetailTugasTask";
|
||||||
import Text from "@/components/Text";
|
import Text from "@/components/Text";
|
||||||
@@ -10,7 +9,6 @@ import { formatDateOnly } from "@/lib/fun_formatDateOnly";
|
|||||||
import { getDatesInRange } from "@/lib/fun_getDatesInRange";
|
import { getDatesInRange } from "@/lib/fun_getDatesInRange";
|
||||||
import { setUpdateTask } from "@/lib/taskUpdate";
|
import { setUpdateTask } from "@/lib/taskUpdate";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { useHeaderHeight } from '@react-navigation/elements';
|
import { useHeaderHeight } from '@react-navigation/elements';
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
import 'intl';
|
import 'intl';
|
||||||
@@ -18,8 +16,7 @@ import 'intl/locale-data/jsonp/id';
|
|||||||
import moment from "moment";
|
import moment from "moment";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import {
|
import {
|
||||||
KeyboardAvoidingView, Platform,
|
KeyboardAvoidingView, Platform, Pressable, SafeAreaView,
|
||||||
SafeAreaView,
|
|
||||||
ScrollView,
|
ScrollView,
|
||||||
View
|
View
|
||||||
} from "react-native";
|
} from "react-native";
|
||||||
@@ -28,7 +25,6 @@ import DateTimePicker, { DateType } from "react-native-ui-datepicker";
|
|||||||
import { useDispatch, useSelector } from "react-redux";
|
import { useDispatch, useSelector } from "react-redux";
|
||||||
|
|
||||||
export default function TaskDivisionAddTask() {
|
export default function TaskDivisionAddTask() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const { token, decryptToken } = useAuthSession();
|
const { token, decryptToken } = useAuthSession();
|
||||||
const dispatch = useDispatch();
|
const dispatch = useDispatch();
|
||||||
const update = useSelector((state: any) => state.taskUpdate);
|
const update = useSelector((state: any) => state.taskUpdate);
|
||||||
@@ -133,18 +129,16 @@ export default function TaskDivisionAddTask() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error : any ) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Gagal menambahkan data"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={{ flex: 1, backgroundColor: colors.background }}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => (
|
// headerLeft: () => (
|
||||||
@@ -189,7 +183,7 @@ export default function TaskDivisionAddTask() {
|
|||||||
>
|
>
|
||||||
<ScrollView>
|
<ScrollView>
|
||||||
<View style={[Styles.p15, Styles.mb100]}>
|
<View style={[Styles.p15, Styles.mb100]}>
|
||||||
<View style={[Styles.wrapPaper, Styles.p10, { backgroundColor: colors.card, borderColor: colors.background }]}>
|
<View style={[Styles.wrapPaper, Styles.p10]}>
|
||||||
<DateTimePicker
|
<DateTimePicker
|
||||||
mode="range"
|
mode="range"
|
||||||
startDate={range.startDate}
|
startDate={range.startDate}
|
||||||
@@ -199,13 +193,13 @@ export default function TaskDivisionAddTask() {
|
|||||||
selected: Styles.selectedDate,
|
selected: Styles.selectedDate,
|
||||||
selected_label: Styles.cWhite,
|
selected_label: Styles.cWhite,
|
||||||
range_fill: Styles.selectRangeDate,
|
range_fill: Styles.selectRangeDate,
|
||||||
month_label: { color: colors.text },
|
month_label: Styles.cBlack,
|
||||||
month_selector_label: { color: colors.text },
|
month_selector_label: Styles.cBlack,
|
||||||
year_label: { color: colors.text },
|
year_label: Styles.cBlack,
|
||||||
year_selector_label: { color: colors.text },
|
year_selector_label: Styles.cBlack,
|
||||||
day_label: { color: colors.text },
|
day_label: Styles.cBlack,
|
||||||
time_label: { color: colors.text },
|
time_label: Styles.cBlack,
|
||||||
weekday_label: { color: colors.text },
|
weekday_label: Styles.cBlack,
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
</View>
|
</View>
|
||||||
@@ -213,39 +207,38 @@ export default function TaskDivisionAddTask() {
|
|||||||
<View style={[Styles.rowSpaceBetween]}>
|
<View style={[Styles.rowSpaceBetween]}>
|
||||||
<View style={[{ width: "48%" }]}>
|
<View style={[{ width: "48%" }]}>
|
||||||
<Text style={[Styles.mb05]}>
|
<Text style={[Styles.mb05]}>
|
||||||
Tanggal Mulai <Text style={{ color: colors.error }}>*</Text>
|
Tanggal Mulai <Text style={Styles.cError}>*</Text>
|
||||||
</Text>
|
</Text>
|
||||||
<View style={[Styles.wrapPaper, Styles.noShadow, Styles.borderAll, Styles.p10, { backgroundColor: colors.card, borderColor: colors.icon + '20' }]}>
|
<View style={[Styles.wrapPaper, Styles.p10]}>
|
||||||
<Text style={{ textAlign: "center" }}>{from}</Text>
|
<Text style={{ textAlign: "center" }}>{from}</Text>
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
<View style={[{ width: "48%" }]}>
|
<View style={[{ width: "48%" }]}>
|
||||||
<Text style={[Styles.mb05]}>
|
<Text style={[Styles.mb05]}>
|
||||||
Tanggal Berakhir <Text style={{ color: colors.error }}>*</Text>
|
Tanggal Berakhir <Text style={Styles.cError}>*</Text>
|
||||||
</Text>
|
</Text>
|
||||||
<View style={[Styles.wrapPaper, Styles.noShadow, Styles.borderAll, Styles.p10, { backgroundColor: colors.card, borderColor: colors.icon + '20' }]}>
|
<View style={[Styles.wrapPaper, Styles.p10]}>
|
||||||
<Text style={{ textAlign: "center" }}>{to}</Text>
|
<Text style={{ textAlign: "center" }}>{to}</Text>
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
{
|
{
|
||||||
(error.endDate || error.startDate) && <Text style={[Styles.textInformation, { color: colors.error }, Styles.mt05]}>Tanggal tidak boleh kosong</Text>
|
(error.endDate || error.startDate) && <Text style={[Styles.textInformation, Styles.cError, Styles.mt05]}>Tanggal tidak boleh kosong</Text>
|
||||||
}
|
}
|
||||||
{/* <Pressable
|
<Pressable
|
||||||
style={[Styles.btnTab, Styles.btnLainnya, dsbButton && Styles.btnDisabled]}
|
style={[Styles.btnTab, Styles.btnLainnya, dsbButton && Styles.btnDisabled]}
|
||||||
disabled={dsbButton}
|
disabled={dsbButton}
|
||||||
onPress={() => { setModalDetail(true) }}
|
onPress={() => { setModalDetail(true) }}
|
||||||
>
|
>
|
||||||
<Text style={[dsbButton ? Styles.cGray : Styles.cWhite]}>Detail</Text>
|
<Text style={[dsbButton ? Styles.cGray : Styles.cWhite]}>Detail</Text>
|
||||||
</Pressable> */}
|
</Pressable>
|
||||||
<ButtonSelect value="Detail" onPress={() => { setModalDetail(true) }} />
|
|
||||||
</View>
|
</View>
|
||||||
<InputForm
|
<InputForm
|
||||||
label="Judul Tugas"
|
label="Judul Tugas"
|
||||||
type="default"
|
type="default"
|
||||||
placeholder="Judul Tugas"
|
placeholder="Judul Tugas"
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
bg="white"
|
||||||
value={title}
|
value={title}
|
||||||
error={error.title}
|
error={error.title}
|
||||||
errorText="Judul tidak boleh kosong"
|
errorText="Judul tidak boleh kosong"
|
||||||
|
|||||||
@@ -5,7 +5,6 @@ import Styles from "@/constants/Styles";
|
|||||||
import { apiCancelTask } from "@/lib/api";
|
import { apiCancelTask } from "@/lib/api";
|
||||||
import { setUpdateTask } from "@/lib/taskUpdate";
|
import { setUpdateTask } from "@/lib/taskUpdate";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { SafeAreaView, ScrollView, View } from "react-native";
|
import { SafeAreaView, ScrollView, View } from "react-native";
|
||||||
@@ -13,7 +12,6 @@ import Toast from "react-native-toast-message";
|
|||||||
import { useDispatch, useSelector } from "react-redux";
|
import { useDispatch, useSelector } from "react-redux";
|
||||||
|
|
||||||
export default function TaskDivisionCancel() {
|
export default function TaskDivisionCancel() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const { id, detail } = useLocalSearchParams<{ id: string; detail: string }>();
|
const { id, detail } = useLocalSearchParams<{ id: string; detail: string }>();
|
||||||
const { token, decryptToken } = useAuthSession();
|
const { token, decryptToken } = useAuthSession();
|
||||||
const dispatch = useDispatch();
|
const dispatch = useDispatch();
|
||||||
@@ -62,18 +60,16 @@ export default function TaskDivisionCancel() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error : any ) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Gagal membatalkan kegiatan"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={{ flex: 1, backgroundColor: colors.background }}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => (
|
// headerLeft: () => (
|
||||||
@@ -119,7 +115,7 @@ export default function TaskDivisionCancel() {
|
|||||||
type="default"
|
type="default"
|
||||||
placeholder="Alasan Pembatalan"
|
placeholder="Alasan Pembatalan"
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
bg="white"
|
||||||
error={error}
|
error={error}
|
||||||
errorText="Alasan pembatalan harus diisi"
|
errorText="Alasan pembatalan harus diisi"
|
||||||
onChange={(val) => onValidation(val)}
|
onChange={(val) => onValidation(val)}
|
||||||
|
|||||||
@@ -5,7 +5,6 @@ import Styles from "@/constants/Styles";
|
|||||||
import { apiEditTask, apiGetTaskOne } from "@/lib/api";
|
import { apiEditTask, apiGetTaskOne } from "@/lib/api";
|
||||||
import { setUpdateTask } from "@/lib/taskUpdate";
|
import { setUpdateTask } from "@/lib/taskUpdate";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { SafeAreaView, ScrollView, View } from "react-native";
|
import { SafeAreaView, ScrollView, View } from "react-native";
|
||||||
@@ -13,7 +12,6 @@ import Toast from "react-native-toast-message";
|
|||||||
import { useDispatch, useSelector } from "react-redux";
|
import { useDispatch, useSelector } from "react-redux";
|
||||||
|
|
||||||
export default function TaskDivisionEdit() {
|
export default function TaskDivisionEdit() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const { id, detail } = useLocalSearchParams<{ id: string; detail: string }>();
|
const { id, detail } = useLocalSearchParams<{ id: string; detail: string }>();
|
||||||
const { token, decryptToken } = useAuthSession();
|
const { token, decryptToken } = useAuthSession();
|
||||||
const [judul, setJudul] = useState("");
|
const [judul, setJudul] = useState("");
|
||||||
@@ -80,18 +78,16 @@ export default function TaskDivisionEdit() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error : any ) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Gagal mengubah data"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={{ flex: 1, backgroundColor: colors.background }}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => (
|
// headerLeft: () => (
|
||||||
@@ -132,7 +128,7 @@ export default function TaskDivisionEdit() {
|
|||||||
type="default"
|
type="default"
|
||||||
placeholder="Judul Kegiatan"
|
placeholder="Judul Kegiatan"
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
bg="white"
|
||||||
value={judul}
|
value={judul}
|
||||||
onChange={(val) => { onValidation(val) }}
|
onChange={(val) => { onValidation(val) }}
|
||||||
error={error}
|
error={error}
|
||||||
|
|||||||
@@ -10,7 +10,6 @@ import SectionTanggalTugasTask from "@/components/task/sectionTanggalTugasTask";
|
|||||||
import Styles from "@/constants/Styles";
|
import Styles from "@/constants/Styles";
|
||||||
import { apiGetDivisionOneFeature, apiGetTaskOne } from "@/lib/api";
|
import { apiGetDivisionOneFeature, apiGetTaskOne } from "@/lib/api";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { RefreshControl, SafeAreaView, ScrollView, View } from "react-native";
|
import { RefreshControl, SafeAreaView, ScrollView, View } from "react-native";
|
||||||
@@ -26,7 +25,6 @@ type Props = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export default function DetailTaskDivision() {
|
export default function DetailTaskDivision() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const { id, detail } = useLocalSearchParams<{ id: string, detail: string }>();
|
const { id, detail } = useLocalSearchParams<{ id: string, detail: string }>();
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
const [data, setData] = useState<Props>()
|
const [data, setData] = useState<Props>()
|
||||||
@@ -99,7 +97,7 @@ export default function DetailTaskDivision() {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={{ flex: 1, backgroundColor: colors.background }}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
||||||
@@ -127,7 +125,6 @@ export default function DetailTaskDivision() {
|
|||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={handleRefresh}
|
onRefresh={handleRefresh}
|
||||||
tintColor={colors.icon}
|
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
|
|||||||
@@ -5,7 +5,6 @@ import Styles from "@/constants/Styles";
|
|||||||
import { apiGetTaskOne, apiReportTask } from "@/lib/api";
|
import { apiGetTaskOne, apiReportTask } from "@/lib/api";
|
||||||
import { setUpdateTask } from "@/lib/taskUpdate";
|
import { setUpdateTask } from "@/lib/taskUpdate";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { SafeAreaView, ScrollView, View } from "react-native";
|
import { SafeAreaView, ScrollView, View } from "react-native";
|
||||||
@@ -13,7 +12,6 @@ import Toast from "react-native-toast-message";
|
|||||||
import { useDispatch, useSelector } from "react-redux";
|
import { useDispatch, useSelector } from "react-redux";
|
||||||
|
|
||||||
export default function TaskDivisionReport() {
|
export default function TaskDivisionReport() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const { id, detail } = useLocalSearchParams<{ id: string; detail: string }>();
|
const { id, detail } = useLocalSearchParams<{ id: string; detail: string }>();
|
||||||
const { token, decryptToken } = useAuthSession();
|
const { token, decryptToken } = useAuthSession();
|
||||||
const [laporan, setLaporan] = useState("");
|
const [laporan, setLaporan] = useState("");
|
||||||
@@ -80,18 +78,16 @@ export default function TaskDivisionReport() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error : any ) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Gagal mengubah data"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={[Styles.flex1, { backgroundColor: colors.background }]}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => (
|
// headerLeft: () => (
|
||||||
@@ -132,7 +128,7 @@ export default function TaskDivisionReport() {
|
|||||||
type="default"
|
type="default"
|
||||||
placeholder="Laporan Kegiatan"
|
placeholder="Laporan Kegiatan"
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
bg="white"
|
||||||
value={laporan}
|
value={laporan}
|
||||||
onChange={(val) => { onValidation(val) }}
|
onChange={(val) => { onValidation(val) }}
|
||||||
error={error}
|
error={error}
|
||||||
|
|||||||
@@ -16,7 +16,6 @@ import { setMemberChoose } from "@/lib/memberChoose";
|
|||||||
import { setTaskCreate } from "@/lib/taskCreate";
|
import { setTaskCreate } from "@/lib/taskCreate";
|
||||||
import { setUpdateTask } from "@/lib/taskUpdate";
|
import { setUpdateTask } from "@/lib/taskUpdate";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { Ionicons, MaterialCommunityIcons } from "@expo/vector-icons";
|
import { Ionicons, MaterialCommunityIcons } from "@expo/vector-icons";
|
||||||
import * as DocumentPicker from "expo-document-picker";
|
import * as DocumentPicker from "expo-document-picker";
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
@@ -27,7 +26,6 @@ import { useDispatch, useSelector } from "react-redux";
|
|||||||
|
|
||||||
|
|
||||||
export default function CreateTaskDivision() {
|
export default function CreateTaskDivision() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const { id } = useLocalSearchParams();
|
const { id } = useLocalSearchParams();
|
||||||
const { token, decryptToken } = useAuthSession();
|
const { token, decryptToken } = useAuthSession();
|
||||||
const dispatch = useDispatch();
|
const dispatch = useDispatch();
|
||||||
@@ -105,11 +103,9 @@ export default function CreateTaskDivision() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error : any ) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal menambahkan data"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
@@ -117,7 +113,7 @@ export default function CreateTaskDivision() {
|
|||||||
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={{ flex: 1, backgroundColor: colors.background }}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => (
|
// headerLeft: () => (
|
||||||
@@ -165,7 +161,6 @@ export default function CreateTaskDivision() {
|
|||||||
val == "" || val == "null" ? setError(true) : setError(false);
|
val == "" || val == "null" ? setError(true) : setError(false);
|
||||||
}}
|
}}
|
||||||
error={error}
|
error={error}
|
||||||
bg={colors.card}
|
|
||||||
errorText="Judul Tugas tidak boleh kosong"
|
errorText="Judul Tugas tidak boleh kosong"
|
||||||
/>
|
/>
|
||||||
<ButtonSelect value="Tambah Tanggal & Tugas" onPress={() => { router.push(`/division/${id}/task/create/task`); }} />
|
<ButtonSelect value="Tambah Tanggal & Tugas" onPress={() => { router.push(`/division/${id}/task/create/task`); }} />
|
||||||
@@ -176,13 +171,13 @@ export default function CreateTaskDivision() {
|
|||||||
fileForm.length > 0 && (
|
fileForm.length > 0 && (
|
||||||
<View style={[Styles.mb15]}>
|
<View style={[Styles.mb15]}>
|
||||||
<Text style={[Styles.textDefaultSemiBold, Styles.mv05]}>File</Text>
|
<Text style={[Styles.textDefaultSemiBold, Styles.mv05]}>File</Text>
|
||||||
<View style={[Styles.wrapPaper, { backgroundColor: colors.card, borderColor: colors.background }]}>
|
<View style={[Styles.wrapPaper]}>
|
||||||
{
|
{
|
||||||
fileForm.map((item, index) => (
|
fileForm.map((item, index) => (
|
||||||
<BorderBottomItem
|
<BorderBottomItem
|
||||||
key={index}
|
key={index}
|
||||||
borderType="all"
|
borderType="all"
|
||||||
icon={<MaterialCommunityIcons name="file-outline" size={25} color={colors.text} />}
|
icon={<MaterialCommunityIcons name="file-outline" size={25} color="black" />}
|
||||||
title={item.name}
|
title={item.name}
|
||||||
titleWeight="normal"
|
titleWeight="normal"
|
||||||
onPress={() => { setIndexDelFile(index); setModal(true) }}
|
onPress={() => { setIndexDelFile(index); setModal(true) }}
|
||||||
@@ -200,7 +195,7 @@ export default function CreateTaskDivision() {
|
|||||||
<Text>Total {entitiesMember.length} Anggota</Text>
|
<Text>Total {entitiesMember.length} Anggota</Text>
|
||||||
</View>
|
</View>
|
||||||
|
|
||||||
<View style={[Styles.borderAll, Styles.round05, Styles.p10, { backgroundColor: colors.card, borderColor: colors.background }]}>
|
<View style={[Styles.borderAll, Styles.round10, Styles.p10]}>
|
||||||
{entitiesMember.map(
|
{entitiesMember.map(
|
||||||
(item: { img: any; name: any }, index: any) => {
|
(item: { img: any; name: any }, index: any) => {
|
||||||
return (
|
return (
|
||||||
@@ -228,7 +223,7 @@ export default function CreateTaskDivision() {
|
|||||||
<DrawerBottom animation="slide" isVisible={isModal} setVisible={setModal} title="Menu">
|
<DrawerBottom animation="slide" isVisible={isModal} setVisible={setModal} title="Menu">
|
||||||
<View style={Styles.rowItemsCenter}>
|
<View style={Styles.rowItemsCenter}>
|
||||||
<MenuItemRow
|
<MenuItemRow
|
||||||
icon={<Ionicons name="trash-outline" color={colors.text} size={25} />}
|
icon={<Ionicons name="trash" color="black" size={25} />}
|
||||||
title="Hapus"
|
title="Hapus"
|
||||||
onPress={() => { deleteFile(indexDelFile) }}
|
onPress={() => { deleteFile(indexDelFile) }}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -9,7 +9,6 @@ import Styles from "@/constants/Styles";
|
|||||||
import { apiGetDivisionMember } from "@/lib/api";
|
import { apiGetDivisionMember } from "@/lib/api";
|
||||||
import { setMemberChoose } from "@/lib/memberChoose";
|
import { setMemberChoose } from "@/lib/memberChoose";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { AntDesign } from "@expo/vector-icons";
|
import { AntDesign } from "@expo/vector-icons";
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
@@ -24,7 +23,6 @@ type Props = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export default function AddMemberCreateTask() {
|
export default function AddMemberCreateTask() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const dispatch = useDispatch()
|
const dispatch = useDispatch()
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
const { id } = useLocalSearchParams<{ id: string, detail: string }>()
|
const { id } = useLocalSearchParams<{ id: string, detail: string }>()
|
||||||
@@ -99,7 +97,7 @@ export default function AddMemberCreateTask() {
|
|||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<View style={[Styles.p15, { flex: 1, backgroundColor: colors.background }]}>
|
<View style={[Styles.p15, { flex: 1 }]}>
|
||||||
<InputSearch onChange={(val) => setSearch(val)} value={search} />
|
<InputSearch onChange={(val) => setSearch(val)} value={search} />
|
||||||
|
|
||||||
{
|
{
|
||||||
@@ -121,7 +119,7 @@ export default function AddMemberCreateTask() {
|
|||||||
</View>
|
</View>
|
||||||
|
|
||||||
:
|
:
|
||||||
<Text style={[Styles.textDefault, Styles.pv05, { textAlign: 'center', color: colors.dimmed }]}>Tidak ada member yang dipilih</Text>
|
<Text style={[Styles.textDefault, Styles.cGray, Styles.pv05, { textAlign: 'center' }]}>Tidak ada member yang dipilih</Text>
|
||||||
}
|
}
|
||||||
<ScrollView
|
<ScrollView
|
||||||
showsVerticalScrollIndicator={false}
|
showsVerticalScrollIndicator={false}
|
||||||
@@ -133,7 +131,7 @@ export default function AddMemberCreateTask() {
|
|||||||
return (
|
return (
|
||||||
<Pressable
|
<Pressable
|
||||||
key={index}
|
key={index}
|
||||||
style={[Styles.itemSelectModal, { borderColor: colors.icon + '20' }]}
|
style={[Styles.itemSelectModal]}
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
onChoose(item.idUser, item.name, item.img)
|
onChoose(item.idUser, item.name, item.img)
|
||||||
}}
|
}}
|
||||||
@@ -145,7 +143,7 @@ export default function AddMemberCreateTask() {
|
|||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
{
|
{
|
||||||
selectMember.some((i: any) => i.idUser == item.idUser) && <AntDesign name="check" size={20} color={colors.text} />
|
selectMember.some((i: any) => i.idUser == item.idUser) && <AntDesign name="check" size={20} color={'black'} />
|
||||||
}
|
}
|
||||||
</Pressable>
|
</Pressable>
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
import AppHeader from "@/components/AppHeader";
|
import AppHeader from "@/components/AppHeader";
|
||||||
import ButtonSaveHeader from "@/components/buttonSaveHeader";
|
import ButtonSaveHeader from "@/components/buttonSaveHeader";
|
||||||
import ButtonSelect from "@/components/buttonSelect";
|
|
||||||
import { InputForm } from "@/components/inputForm";
|
import { InputForm } from "@/components/inputForm";
|
||||||
import ModalAddDetailTugasTask from "@/components/task/modalAddDetailTugasTask";
|
import ModalAddDetailTugasTask from "@/components/task/modalAddDetailTugasTask";
|
||||||
import Text from "@/components/Text";
|
import Text from "@/components/Text";
|
||||||
@@ -8,7 +7,6 @@ import Styles from "@/constants/Styles";
|
|||||||
import { formatDateOnly } from "@/lib/fun_formatDateOnly";
|
import { formatDateOnly } from "@/lib/fun_formatDateOnly";
|
||||||
import { getDatesInRange } from "@/lib/fun_getDatesInRange";
|
import { getDatesInRange } from "@/lib/fun_getDatesInRange";
|
||||||
import { setTaskCreate } from "@/lib/taskCreate";
|
import { setTaskCreate } from "@/lib/taskCreate";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { useHeaderHeight } from '@react-navigation/elements';
|
import { useHeaderHeight } from '@react-navigation/elements';
|
||||||
import { router, Stack } from "expo-router";
|
import { router, Stack } from "expo-router";
|
||||||
import 'intl';
|
import 'intl';
|
||||||
@@ -18,6 +16,7 @@ import { useEffect, useState } from "react";
|
|||||||
import {
|
import {
|
||||||
KeyboardAvoidingView,
|
KeyboardAvoidingView,
|
||||||
Platform,
|
Platform,
|
||||||
|
Pressable,
|
||||||
SafeAreaView,
|
SafeAreaView,
|
||||||
ScrollView,
|
ScrollView,
|
||||||
View
|
View
|
||||||
@@ -28,7 +27,6 @@ import DateTimePicker, {
|
|||||||
import { useDispatch, useSelector } from "react-redux";
|
import { useDispatch, useSelector } from "react-redux";
|
||||||
|
|
||||||
export default function CreateTaskAddTugas() {
|
export default function CreateTaskAddTugas() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const headerHeight = useHeaderHeight();
|
const headerHeight = useHeaderHeight();
|
||||||
const dispatch = useDispatch()
|
const dispatch = useDispatch()
|
||||||
const [disable, setDisable] = useState(true);
|
const [disable, setDisable] = useState(true);
|
||||||
@@ -120,7 +118,7 @@ export default function CreateTaskAddTugas() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={{ flex: 1, backgroundColor: colors.background }}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => (
|
// headerLeft: () => (
|
||||||
@@ -160,7 +158,7 @@ export default function CreateTaskAddTugas() {
|
|||||||
>
|
>
|
||||||
<ScrollView>
|
<ScrollView>
|
||||||
<View style={[Styles.p15, Styles.mb100]}>
|
<View style={[Styles.p15, Styles.mb100]}>
|
||||||
<View style={[Styles.wrapPaper, Styles.p10, { backgroundColor: colors.card, borderColor: colors.background }]}>
|
<View style={[Styles.wrapPaper, Styles.p10]}>
|
||||||
<DateTimePicker
|
<DateTimePicker
|
||||||
mode="range"
|
mode="range"
|
||||||
startDate={range.startDate}
|
startDate={range.startDate}
|
||||||
@@ -170,13 +168,13 @@ export default function CreateTaskAddTugas() {
|
|||||||
selected: Styles.selectedDate,
|
selected: Styles.selectedDate,
|
||||||
selected_label: Styles.cWhite,
|
selected_label: Styles.cWhite,
|
||||||
range_fill: Styles.selectRangeDate,
|
range_fill: Styles.selectRangeDate,
|
||||||
month_label: { color: colors.text },
|
month_label: Styles.cBlack,
|
||||||
month_selector_label: { color: colors.text },
|
month_selector_label: Styles.cBlack,
|
||||||
year_label: { color: colors.text },
|
year_label: Styles.cBlack,
|
||||||
year_selector_label: { color: colors.text },
|
year_selector_label: Styles.cBlack,
|
||||||
day_label: { color: colors.text },
|
day_label: Styles.cBlack,
|
||||||
time_label: { color: colors.text },
|
time_label: Styles.cBlack,
|
||||||
weekday_label: { color: colors.text },
|
weekday_label: Styles.cBlack,
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
</View>
|
</View>
|
||||||
@@ -184,39 +182,38 @@ export default function CreateTaskAddTugas() {
|
|||||||
<View style={[Styles.rowSpaceBetween]}>
|
<View style={[Styles.rowSpaceBetween]}>
|
||||||
<View style={[{ width: "48%" }]}>
|
<View style={[{ width: "48%" }]}>
|
||||||
<Text style={[Styles.mb05]}>
|
<Text style={[Styles.mb05]}>
|
||||||
Tanggal Mulai <Text style={{ color: colors.error }}>*</Text>
|
Tanggal Mulai <Text style={Styles.cError}>*</Text>
|
||||||
</Text>
|
</Text>
|
||||||
<View style={[Styles.wrapPaper, Styles.noShadow, Styles.borderAll, Styles.p10, { backgroundColor: colors.card, borderColor: colors.icon + '20' }]}>
|
<View style={[Styles.wrapPaper, Styles.p10]}>
|
||||||
<Text style={{ textAlign: "center" }}>{from}</Text>
|
<Text style={{ textAlign: "center" }}>{from}</Text>
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
<View style={[{ width: "48%" }]}>
|
<View style={[{ width: "48%" }]}>
|
||||||
<Text style={[Styles.mb05]}>
|
<Text style={[Styles.mb05]}>
|
||||||
Tanggal Berakhir <Text style={{ color: colors.error }}>*</Text>
|
Tanggal Berakhir <Text style={Styles.cError}>*</Text>
|
||||||
</Text>
|
</Text>
|
||||||
<View style={[Styles.wrapPaper, Styles.noShadow, Styles.borderAll, Styles.p10, { backgroundColor: colors.card, borderColor: colors.icon + '20' }]}>
|
<View style={[Styles.wrapPaper, Styles.p10]}>
|
||||||
<Text style={{ textAlign: "center" }}>{to}</Text>
|
<Text style={{ textAlign: "center" }}>{to}</Text>
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
{
|
{
|
||||||
(error.endDate || error.startDate) && <Text style={[Styles.textInformation, { color: colors.error }, Styles.mt05]}>Tanggal tidak boleh kosong</Text>
|
(error.endDate || error.startDate) && <Text style={[Styles.textInformation, Styles.cError, Styles.mt05]}>Tanggal tidak boleh kosong</Text>
|
||||||
}
|
}
|
||||||
{/* <Pressable
|
<Pressable
|
||||||
style={[Styles.btnTab, Styles.btnLainnya, dsbButton && Styles.btnDisabled]}
|
style={[Styles.btnTab, Styles.btnLainnya, dsbButton && Styles.btnDisabled]}
|
||||||
disabled={dsbButton}
|
disabled={dsbButton}
|
||||||
onPress={() => { setModalDetail(true) }}
|
onPress={() => { setModalDetail(true) }}
|
||||||
>
|
>
|
||||||
<Text style={[dsbButton ? Styles.cGray : Styles.cWhite]}>Detail</Text>
|
<Text style={[dsbButton ? Styles.cGray : Styles.cWhite]}>Detail</Text>
|
||||||
</Pressable> */}
|
</Pressable>
|
||||||
<ButtonSelect value="Detail" onPress={() => { setModalDetail(true) }} />
|
|
||||||
</View>
|
</View>
|
||||||
<InputForm
|
<InputForm
|
||||||
label="Judul Tugas"
|
label="Judul Tugas"
|
||||||
type="default"
|
type="default"
|
||||||
placeholder="Judul Tugas"
|
placeholder="Judul Tugas"
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
bg="white"
|
||||||
value={title}
|
value={title}
|
||||||
error={error.title}
|
error={error.title}
|
||||||
errorText="Judul tidak boleh kosong"
|
errorText="Judul tidak boleh kosong"
|
||||||
|
|||||||
@@ -11,7 +11,6 @@ import { ColorsStatus } from "@/constants/ColorsStatus";
|
|||||||
import Styles from "@/constants/Styles";
|
import Styles from "@/constants/Styles";
|
||||||
import { apiGetTask } from "@/lib/api";
|
import { apiGetTask } from "@/lib/api";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import {
|
import {
|
||||||
AntDesign,
|
AntDesign,
|
||||||
Ionicons,
|
Ionicons,
|
||||||
@@ -32,7 +31,6 @@ type Props = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
export default function ListTask() {
|
export default function ListTask() {
|
||||||
const { colors } = useTheme()
|
|
||||||
const { id, status, year } = useLocalSearchParams<{ id: string; status: string; year: string }>()
|
const { id, status, year } = useLocalSearchParams<{ id: string; status: string; year: string }>()
|
||||||
const [isList, setList] = useState(false)
|
const [isList, setList] = useState(false)
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
@@ -112,7 +110,7 @@ export default function ListTask() {
|
|||||||
})
|
})
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<View style={[Styles.p15, Styles.flex1, { backgroundColor: colors.background }]}>
|
<View style={[Styles.p15, { flex: 1 }]}>
|
||||||
<View>
|
<View>
|
||||||
<ScrollView horizontal style={[Styles.mb10]} showsHorizontalScrollIndicator={false}>
|
<ScrollView horizontal style={[Styles.mb10]} showsHorizontalScrollIndicator={false}>
|
||||||
<ButtonTab
|
<ButtonTab
|
||||||
@@ -123,7 +121,7 @@ export default function ListTask() {
|
|||||||
icon={
|
icon={
|
||||||
<MaterialCommunityIcons
|
<MaterialCommunityIcons
|
||||||
name="clock-alert-outline"
|
name="clock-alert-outline"
|
||||||
color={statusFix == "0" ? "white" : colors.dimmed}
|
color={statusFix == "0" ? "white" : "black"}
|
||||||
size={20}
|
size={20}
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
@@ -137,7 +135,7 @@ export default function ListTask() {
|
|||||||
icon={
|
icon={
|
||||||
<MaterialCommunityIcons
|
<MaterialCommunityIcons
|
||||||
name="progress-check"
|
name="progress-check"
|
||||||
color={statusFix == "1" ? "white" : colors.dimmed}
|
color={statusFix == "1" ? "white" : "black"}
|
||||||
size={20}
|
size={20}
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
@@ -151,7 +149,7 @@ export default function ListTask() {
|
|||||||
icon={
|
icon={
|
||||||
<Ionicons
|
<Ionicons
|
||||||
name="checkmark-done-circle-outline"
|
name="checkmark-done-circle-outline"
|
||||||
color={statusFix == "2" ? "white" : colors.dimmed}
|
color={statusFix == "2" ? "white" : "black"}
|
||||||
size={20}
|
size={20}
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
@@ -165,7 +163,7 @@ export default function ListTask() {
|
|||||||
icon={
|
icon={
|
||||||
<AntDesign
|
<AntDesign
|
||||||
name="closecircleo"
|
name="closecircleo"
|
||||||
color={statusFix == "3" ? "white" : colors.dimmed}
|
color={statusFix == "3" ? "white" : "black"}
|
||||||
size={20}
|
size={20}
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
@@ -181,7 +179,7 @@ export default function ListTask() {
|
|||||||
>
|
>
|
||||||
<MaterialCommunityIcons
|
<MaterialCommunityIcons
|
||||||
name={isList ? "format-list-bulleted" : "view-grid"}
|
name={isList ? "format-list-bulleted" : "view-grid"}
|
||||||
color={colors.text}
|
color={"black"}
|
||||||
size={30}
|
size={30}
|
||||||
/>
|
/>
|
||||||
</Pressable>
|
</Pressable>
|
||||||
@@ -190,7 +188,7 @@ export default function ListTask() {
|
|||||||
<View style={[Styles.mv05]}>
|
<View style={[Styles.mv05]}>
|
||||||
<View style={[Styles.rowOnly]}>
|
<View style={[Styles.rowOnly]}>
|
||||||
<Text style={[Styles.mr05]}>Filter :</Text>
|
<Text style={[Styles.mr05]}>Filter :</Text>
|
||||||
<LabelStatus size="small" category="secondary" text={isYear} style={[Styles.mr05]} />
|
<LabelStatus size="small" category="secondary" text={isYear} style={{ marginRight: 5 }} />
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
<View style={[{ flex: 2 }]}>
|
<View style={[{ flex: 2 }]}>
|
||||||
@@ -235,7 +233,6 @@ export default function ListTask() {
|
|||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={handleRefresh}
|
onRefresh={handleRefresh}
|
||||||
tintColor={colors.icon}
|
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
@@ -277,11 +274,11 @@ export default function ListTask() {
|
|||||||
<LabelStatus
|
<LabelStatus
|
||||||
size="default"
|
size="default"
|
||||||
category={
|
category={
|
||||||
item.status === 0 ? 'secondary' :
|
item.status === 0 ? 'primary' :
|
||||||
item.status === 1 ? 'warning' :
|
item.status === 1 ? 'warning' :
|
||||||
item.status === 2 ? 'success' :
|
item.status === 2 ? 'success' :
|
||||||
item.status === 3 ? 'error' :
|
item.status === 3 ? 'error' :
|
||||||
'secondary'
|
'primary'
|
||||||
}
|
}
|
||||||
text={
|
text={
|
||||||
item.status === 0 ? 'SEGERA' :
|
item.status === 0 ? 'SEGERA' :
|
||||||
@@ -302,7 +299,6 @@ export default function ListTask() {
|
|||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={handleRefresh}
|
onRefresh={handleRefresh}
|
||||||
tintColor={colors.icon}
|
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
@@ -342,13 +338,13 @@ export default function ListTask() {
|
|||||||
</View>
|
</View>
|
||||||
)
|
)
|
||||||
) : (
|
) : (
|
||||||
<Text style={[Styles.textDefault, Styles.textCenter]} >
|
<Text style={[Styles.textDefault, Styles.cGray, { textAlign: "center" },]} >
|
||||||
Tidak ada data
|
Tidak ada data
|
||||||
</Text>
|
</Text>
|
||||||
)
|
)
|
||||||
|
|
||||||
}
|
}
|
||||||
</View >
|
</View>
|
||||||
</View >
|
</View>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
import AppHeader from "@/components/AppHeader";
|
import AppHeader from "@/components/AppHeader";
|
||||||
import ButtonSaveHeader from "@/components/buttonSaveHeader";
|
import ButtonSaveHeader from "@/components/buttonSaveHeader";
|
||||||
import ButtonSelect from "@/components/buttonSelect";
|
|
||||||
import { InputForm } from "@/components/inputForm";
|
import { InputForm } from "@/components/inputForm";
|
||||||
import ModalAddDetailTugasTask from "@/components/task/modalAddDetailTugasTask";
|
import ModalAddDetailTugasTask from "@/components/task/modalAddDetailTugasTask";
|
||||||
import Text from "@/components/Text";
|
import Text from "@/components/Text";
|
||||||
@@ -10,7 +9,6 @@ import { formatDateOnly } from "@/lib/fun_formatDateOnly";
|
|||||||
import { getDatesInRange } from "@/lib/fun_getDatesInRange";
|
import { getDatesInRange } from "@/lib/fun_getDatesInRange";
|
||||||
import { setUpdateTask } from "@/lib/taskUpdate";
|
import { setUpdateTask } from "@/lib/taskUpdate";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { useHeaderHeight } from '@react-navigation/elements';
|
import { useHeaderHeight } from '@react-navigation/elements';
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
import 'intl';
|
import 'intl';
|
||||||
@@ -20,6 +18,7 @@ import { useEffect, useState } from "react";
|
|||||||
import {
|
import {
|
||||||
KeyboardAvoidingView,
|
KeyboardAvoidingView,
|
||||||
Platform,
|
Platform,
|
||||||
|
Pressable,
|
||||||
SafeAreaView,
|
SafeAreaView,
|
||||||
ScrollView,
|
ScrollView,
|
||||||
View
|
View
|
||||||
@@ -29,7 +28,6 @@ import DateTimePicker, { DateType } from "react-native-ui-datepicker";
|
|||||||
import { useDispatch, useSelector } from "react-redux";
|
import { useDispatch, useSelector } from "react-redux";
|
||||||
|
|
||||||
export default function UpdateProjectTaskDivision() {
|
export default function UpdateProjectTaskDivision() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const headerHeight = useHeaderHeight();
|
const headerHeight = useHeaderHeight();
|
||||||
const { detail } = useLocalSearchParams<{ detail: string }>();
|
const { detail } = useLocalSearchParams<{ detail: string }>();
|
||||||
const dispatch = useDispatch();
|
const dispatch = useDispatch();
|
||||||
@@ -125,11 +123,9 @@ export default function UpdateProjectTaskDivision() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error : any ) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Gagal mengubah data"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoadingSubmit(false)
|
setLoadingSubmit(false)
|
||||||
}
|
}
|
||||||
@@ -190,7 +186,7 @@ export default function UpdateProjectTaskDivision() {
|
|||||||
}, [range])
|
}, [range])
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={{ flex: 1, backgroundColor: colors.background }}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => (
|
// headerLeft: () => (
|
||||||
@@ -235,7 +231,7 @@ export default function UpdateProjectTaskDivision() {
|
|||||||
>
|
>
|
||||||
<ScrollView>
|
<ScrollView>
|
||||||
<View style={[Styles.p15, Styles.mb100]}>
|
<View style={[Styles.p15, Styles.mb100]}>
|
||||||
<View style={[Styles.wrapPaper, Styles.p10, { backgroundColor: colors.card, borderColor: colors.background }]}>
|
<View style={[Styles.wrapPaper, Styles.p10]}>
|
||||||
{!loading && (
|
{!loading && (
|
||||||
<DateTimePicker
|
<DateTimePicker
|
||||||
mode="range"
|
mode="range"
|
||||||
@@ -248,13 +244,13 @@ export default function UpdateProjectTaskDivision() {
|
|||||||
selected: Styles.selectedDate,
|
selected: Styles.selectedDate,
|
||||||
selected_label: Styles.cWhite,
|
selected_label: Styles.cWhite,
|
||||||
range_fill: Styles.selectRangeDate,
|
range_fill: Styles.selectRangeDate,
|
||||||
month_label: { color: colors.text },
|
month_label: Styles.cBlack,
|
||||||
month_selector_label: { color: colors.text },
|
month_selector_label: Styles.cBlack,
|
||||||
year_label: { color: colors.text },
|
year_label: Styles.cBlack,
|
||||||
year_selector_label: { color: colors.text },
|
year_selector_label: Styles.cBlack,
|
||||||
day_label: { color: colors.text },
|
day_label: Styles.cBlack,
|
||||||
time_label: { color: colors.text },
|
time_label: Styles.cBlack,
|
||||||
weekday_label: { color: colors.text },
|
weekday_label: Styles.cBlack,
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
@@ -263,41 +259,40 @@ export default function UpdateProjectTaskDivision() {
|
|||||||
<View style={[Styles.rowSpaceBetween]}>
|
<View style={[Styles.rowSpaceBetween]}>
|
||||||
<View style={[{ width: "48%" }]}>
|
<View style={[{ width: "48%" }]}>
|
||||||
<Text style={[Styles.mb05]}>
|
<Text style={[Styles.mb05]}>
|
||||||
Tanggal Mulai <Text style={{ color: colors.error }}>*</Text>
|
Tanggal Mulai <Text style={Styles.cError}>*</Text>
|
||||||
</Text>
|
</Text>
|
||||||
<View style={[Styles.wrapPaper, Styles.noShadow, Styles.borderAll, Styles.p10, { backgroundColor: colors.card, borderColor: colors.icon + '20' }]}>
|
<View style={[Styles.wrapPaper, Styles.p10]}>
|
||||||
<Text style={{ textAlign: "center" }}>{from}</Text>
|
<Text style={{ textAlign: "center" }}>{from}</Text>
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
<View style={[{ width: "48%" }]}>
|
<View style={[{ width: "48%" }]}>
|
||||||
<Text style={[Styles.mb05]}>
|
<Text style={[Styles.mb05]}>
|
||||||
Tanggal Berakhir <Text style={{ color: colors.error }}>*</Text>
|
Tanggal Berakhir <Text style={Styles.cError}>*</Text>
|
||||||
</Text>
|
</Text>
|
||||||
<View style={[Styles.wrapPaper, Styles.noShadow, Styles.borderAll, Styles.p10, { backgroundColor: colors.card, borderColor: colors.icon + '20' }]}>
|
<View style={[Styles.wrapPaper, Styles.p10]}>
|
||||||
<Text style={{ textAlign: "center" }}>{to}</Text>
|
<Text style={{ textAlign: "center" }}>{to}</Text>
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
{(error.endDate || error.startDate) && (
|
{(error.endDate || error.startDate) && (
|
||||||
<Text style={[Styles.textInformation, { color: colors.error }, Styles.mt05]} >
|
<Text style={[Styles.textInformation, Styles.cError, Styles.mt05]} >
|
||||||
Tanggal tidak boleh kosong
|
Tanggal tidak boleh kosong
|
||||||
</Text>
|
</Text>
|
||||||
)}
|
)}
|
||||||
{/* <Pressable
|
<Pressable
|
||||||
style={[Styles.btnTab, Styles.btnLainnya, dsbButton && Styles.btnDisabled]}
|
style={[Styles.btnTab, Styles.btnLainnya, dsbButton && Styles.btnDisabled]}
|
||||||
disabled={dsbButton}
|
disabled={dsbButton}
|
||||||
onPress={() => { setModalDetail(true) }}
|
onPress={() => { setModalDetail(true) }}
|
||||||
>
|
>
|
||||||
<Text style={[dsbButton ? Styles.cGray : Styles.cWhite]}>Detail</Text>
|
<Text style={[dsbButton ? Styles.cGray : Styles.cWhite]}>Detail</Text>
|
||||||
</Pressable> */}
|
</Pressable>
|
||||||
<ButtonSelect value="Detail" onPress={() => { setModalDetail(true) }} />
|
|
||||||
</View>
|
</View>
|
||||||
<InputForm
|
<InputForm
|
||||||
label="Judul Tugas"
|
label="Judul Tugas"
|
||||||
type="default"
|
type="default"
|
||||||
placeholder="Judul Tugas"
|
placeholder="Judul Tugas"
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
bg="white"
|
||||||
value={title}
|
value={title}
|
||||||
error={error.title}
|
error={error.title}
|
||||||
errorText="Judul tidak boleh kosong"
|
errorText="Judul tidak boleh kosong"
|
||||||
|
|||||||
@@ -8,7 +8,6 @@ import { ConstEnv } from "@/constants/ConstEnv";
|
|||||||
import Styles from "@/constants/Styles";
|
import Styles from "@/constants/Styles";
|
||||||
import { apiAddMemberDivision, apiGetDivisionOneDetail, apiGetUser } from "@/lib/api";
|
import { apiAddMemberDivision, apiGetDivisionOneDetail, apiGetUser } from "@/lib/api";
|
||||||
import { setUpdateDivision } from "@/lib/divisionUpdate";
|
import { setUpdateDivision } from "@/lib/divisionUpdate";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { AntDesign } from "@expo/vector-icons";
|
import { AntDesign } from "@expo/vector-icons";
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
@@ -24,7 +23,6 @@ type Props = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export default function AddMemberDivision() {
|
export default function AddMemberDivision() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
const { id } = useLocalSearchParams<{ id: string }>()
|
const { id } = useLocalSearchParams<{ id: string }>()
|
||||||
const [dataOld, setDataOld] = useState<Props[]>([])
|
const [dataOld, setDataOld] = useState<Props[]>([])
|
||||||
@@ -89,11 +87,9 @@ export default function AddMemberDivision() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal menambahkan anggota"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
@@ -134,7 +130,7 @@ export default function AddMemberDivision() {
|
|||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<View style={[Styles.p15, { flex: 1, backgroundColor: colors.background }]}>
|
<View style={[Styles.p15, { flex: 1 }]}>
|
||||||
<InputSearch onChange={(val) => handleSearch(val)} value={search} />
|
<InputSearch onChange={(val) => handleSearch(val)} value={search} />
|
||||||
|
|
||||||
{
|
{
|
||||||
@@ -156,7 +152,7 @@ export default function AddMemberDivision() {
|
|||||||
</View>
|
</View>
|
||||||
|
|
||||||
:
|
:
|
||||||
<Text style={[Styles.textDefault, Styles.pv05, { textAlign: 'center', color: colors.dimmed }]}>Tidak ada member yang dipilih</Text>
|
<Text style={[Styles.textDefault, Styles.cGray, Styles.pv05, { textAlign: 'center' }]}>Tidak ada member yang dipilih</Text>
|
||||||
}
|
}
|
||||||
<ScrollView
|
<ScrollView
|
||||||
showsVerticalScrollIndicator={false}
|
showsVerticalScrollIndicator={false}
|
||||||
@@ -179,12 +175,12 @@ export default function AddMemberDivision() {
|
|||||||
<View style={[Styles.ml10]}>
|
<View style={[Styles.ml10]}>
|
||||||
<Text style={[Styles.textDefault]} numberOfLines={1} ellipsizeMode="tail">{item.name}</Text>
|
<Text style={[Styles.textDefault]} numberOfLines={1} ellipsizeMode="tail">{item.name}</Text>
|
||||||
{
|
{
|
||||||
found && <Text style={[Styles.textInformation, { color: colors.dimmed }]}>sudah menjadi anggota</Text>
|
found && <Text style={[Styles.textInformation, Styles.cGray]}>sudah menjadi anggota</Text>
|
||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
{
|
{
|
||||||
selectMember.some((i: any) => i.idUser == item.id) && <AntDesign name="check" size={20} color={colors.text} />
|
selectMember.some((i: any) => i.idUser == item.id) && <AntDesign name="check" size={20} color={'black'} />
|
||||||
}
|
}
|
||||||
</Pressable>
|
</Pressable>
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -5,7 +5,6 @@ import Styles from "@/constants/Styles";
|
|||||||
import { apiEditDivision, apiGetDivisionOneDetail } from "@/lib/api";
|
import { apiEditDivision, apiGetDivisionOneDetail } from "@/lib/api";
|
||||||
import { setUpdateDivision } from "@/lib/divisionUpdate";
|
import { setUpdateDivision } from "@/lib/divisionUpdate";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { SafeAreaView, ScrollView, View } from "react-native";
|
import { SafeAreaView, ScrollView, View } from "react-native";
|
||||||
@@ -13,7 +12,6 @@ import Toast from "react-native-toast-message";
|
|||||||
import { useDispatch, useSelector } from "react-redux";
|
import { useDispatch, useSelector } from "react-redux";
|
||||||
|
|
||||||
export default function EditDivision() {
|
export default function EditDivision() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const dispatch = useDispatch()
|
const dispatch = useDispatch()
|
||||||
const update = useSelector((state: any) => state.divisionUpdate)
|
const update = useSelector((state: any) => state.divisionUpdate)
|
||||||
const { token, decryptToken } = useAuthSession();
|
const { token, decryptToken } = useAuthSession();
|
||||||
@@ -56,18 +54,16 @@ export default function EditDivision() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal mengubah data"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={{ flex: 1, backgroundColor: colors.background }}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => (
|
// headerLeft: () => (
|
||||||
@@ -102,7 +98,7 @@ export default function EditDivision() {
|
|||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<ScrollView style={{ backgroundColor: colors.background }}>
|
<ScrollView>
|
||||||
<View style={[Styles.p15, Styles.mb100]}>
|
<View style={[Styles.p15, Styles.mb100]}>
|
||||||
<InputForm
|
<InputForm
|
||||||
label="Nama Divisi"
|
label="Nama Divisi"
|
||||||
|
|||||||
@@ -8,7 +8,6 @@ import CaraouselHome from "@/components/home/carouselHome"
|
|||||||
import Styles from "@/constants/Styles"
|
import Styles from "@/constants/Styles"
|
||||||
import { apiGetDivisionOneDetail } from "@/lib/api"
|
import { apiGetDivisionOneDetail } from "@/lib/api"
|
||||||
import { useAuthSession } from "@/providers/AuthProvider"
|
import { useAuthSession } from "@/providers/AuthProvider"
|
||||||
import { useTheme } from "@/providers/ThemeProvider"
|
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router"
|
import { router, Stack, useLocalSearchParams } from "expo-router"
|
||||||
import { useEffect, useState } from "react"
|
import { useEffect, useState } from "react"
|
||||||
import { RefreshControl, SafeAreaView, ScrollView, View } from "react-native"
|
import { RefreshControl, SafeAreaView, ScrollView, View } from "react-native"
|
||||||
@@ -23,7 +22,6 @@ type Props = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export default function DetailDivisionFitur() {
|
export default function DetailDivisionFitur() {
|
||||||
const { colors } = useTheme()
|
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
const { id } = useLocalSearchParams<{ id: string }>()
|
const { id } = useLocalSearchParams<{ id: string }>()
|
||||||
const [data, setData] = useState<Props>()
|
const [data, setData] = useState<Props>()
|
||||||
@@ -56,7 +54,7 @@ export default function DetailDivisionFitur() {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={{ flex: 1, backgroundColor: colors.background }}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
||||||
@@ -78,7 +76,6 @@ export default function DetailDivisionFitur() {
|
|||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={handleRefresh}
|
onRefresh={handleRefresh}
|
||||||
tintColor={colors.icon}
|
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
showsVerticalScrollIndicator={false}
|
showsVerticalScrollIndicator={false}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import ModalConfirmation from "@/components/ModalConfirmation"
|
import AlertKonfirmasi from "@/components/alertKonfirmasi"
|
||||||
import AppHeader from "@/components/AppHeader"
|
import AppHeader from "@/components/AppHeader"
|
||||||
import BorderBottomItem from "@/components/borderBottomItem"
|
import BorderBottomItem from "@/components/borderBottomItem"
|
||||||
import HeaderRightDivisionInfo from "@/components/division/headerDivisionInfo"
|
import HeaderRightDivisionInfo from "@/components/division/headerDivisionInfo"
|
||||||
@@ -13,7 +13,6 @@ import { ConstEnv } from "@/constants/ConstEnv"
|
|||||||
import Styles from "@/constants/Styles"
|
import Styles from "@/constants/Styles"
|
||||||
import { apiDeleteMemberDivision, apiGetDivisionOneDetail, apiGetDivisionOneFeature, apiUpdateStatusAdminDivision } from "@/lib/api"
|
import { apiDeleteMemberDivision, apiGetDivisionOneDetail, apiGetDivisionOneFeature, apiUpdateStatusAdminDivision } from "@/lib/api"
|
||||||
import { useAuthSession } from "@/providers/AuthProvider"
|
import { useAuthSession } from "@/providers/AuthProvider"
|
||||||
import { useTheme } from "@/providers/ThemeProvider"
|
|
||||||
import { Feather, MaterialCommunityIcons, MaterialIcons } from "@expo/vector-icons"
|
import { Feather, MaterialCommunityIcons, MaterialIcons } from "@expo/vector-icons"
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router"
|
import { router, Stack, useLocalSearchParams } from "expo-router"
|
||||||
import { useEffect, useState } from "react"
|
import { useEffect, useState } from "react"
|
||||||
@@ -40,7 +39,6 @@ type PropsMember = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export default function InformationDivision() {
|
export default function InformationDivision() {
|
||||||
const { colors } = useTheme()
|
|
||||||
const [refreshing, setRefreshing] = useState(false)
|
const [refreshing, setRefreshing] = useState(false)
|
||||||
const entityUser = useSelector((state: any) => state.user)
|
const entityUser = useSelector((state: any) => state.user)
|
||||||
const { id } = useLocalSearchParams<{ id: string }>()
|
const { id } = useLocalSearchParams<{ id: string }>()
|
||||||
@@ -59,13 +57,14 @@ export default function InformationDivision() {
|
|||||||
name: '',
|
name: '',
|
||||||
isAdmin: false
|
isAdmin: false
|
||||||
})
|
})
|
||||||
const [showDeleteModal, setShowDeleteModal] = useState(false)
|
|
||||||
|
|
||||||
function handleMemberOut() {
|
function handleMemberOut() {
|
||||||
setModal(false)
|
setModal(false)
|
||||||
setTimeout(() => {
|
AlertKonfirmasi({
|
||||||
setShowDeleteModal(true)
|
title: 'Konfirmasi',
|
||||||
}, 600)
|
desc: 'Apakah anda yakin ingin mengeluarkan anggota?',
|
||||||
|
onPress: () => { memberOut() }
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
async function memberOut() {
|
async function memberOut() {
|
||||||
@@ -78,11 +77,9 @@ export default function InformationDivision() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal mengeluarkan anggota"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setModal(false)
|
setModal(false)
|
||||||
}
|
}
|
||||||
@@ -98,11 +95,9 @@ export default function InformationDivision() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal mengubah status admin"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setModal(false)
|
setModal(false)
|
||||||
}
|
}
|
||||||
@@ -166,7 +161,7 @@ export default function InformationDivision() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={{ flex: 1, backgroundColor: colors.background }}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
||||||
@@ -190,10 +185,9 @@ export default function InformationDivision() {
|
|||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={handleRefresh}
|
onRefresh={handleRefresh}
|
||||||
tintColor={colors.icon}
|
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
style={[Styles.h100, { backgroundColor: colors.background }]}
|
style={[Styles.h100]}
|
||||||
>
|
>
|
||||||
<View style={[Styles.p15]}>
|
<View style={[Styles.p15]}>
|
||||||
{
|
{
|
||||||
@@ -203,7 +197,7 @@ export default function InformationDivision() {
|
|||||||
}
|
}
|
||||||
<View style={[Styles.mb15]}>
|
<View style={[Styles.mb15]}>
|
||||||
<Text style={[Styles.textDefaultSemiBold, Styles.mb05]}>Deskripsi Divisi</Text>
|
<Text style={[Styles.textDefaultSemiBold, Styles.mb05]}>Deskripsi Divisi</Text>
|
||||||
<View style={[Styles.wrapPaper, Styles.noShadow, { backgroundColor: colors.card, borderWidth: 1, borderColor: colors.icon + '20' }]}>
|
<View style={[Styles.wrapPaper]}>
|
||||||
{loading ?
|
{loading ?
|
||||||
arrSkeleton.map((item, index) => {
|
arrSkeleton.map((item, index) => {
|
||||||
return (
|
return (
|
||||||
@@ -217,7 +211,7 @@ export default function InformationDivision() {
|
|||||||
</View>
|
</View>
|
||||||
<View style={[Styles.mb15]}>
|
<View style={[Styles.mb15]}>
|
||||||
<Text style={[Styles.textDefault, Styles.mv05]}>{dataMember.length} Anggota</Text>
|
<Text style={[Styles.textDefault, Styles.mv05]}>{dataMember.length} Anggota</Text>
|
||||||
<View style={[Styles.wrapPaper, Styles.noShadow, { backgroundColor: colors.card, borderWidth: 1, borderColor: colors.icon + '20' }]}>
|
<View style={[Styles.wrapPaper]}>
|
||||||
{
|
{
|
||||||
((entityUser.role != "user" && entityUser.role != "coadmin") || isAdminDivision) &&
|
((entityUser.role != "user" && entityUser.role != "coadmin") || isAdminDivision) &&
|
||||||
dataDetail?.isActive && (
|
dataDetail?.isActive && (
|
||||||
@@ -225,8 +219,8 @@ export default function InformationDivision() {
|
|||||||
onPress={() => { router.push(`/division/${id}/add-member`) }}
|
onPress={() => { router.push(`/division/${id}/add-member`) }}
|
||||||
borderType="none"
|
borderType="none"
|
||||||
icon={
|
icon={
|
||||||
<View style={[Styles.iconContent]}>
|
<View style={[Styles.iconContent, ColorsStatus.gray]}>
|
||||||
<Feather name="user-plus" size={25} color={'black'} />
|
<Feather name="user-plus" size={25} color={'#384288'} />
|
||||||
</View>
|
</View>
|
||||||
}
|
}
|
||||||
title="Tambah Anggota"
|
title="Tambah Anggota"
|
||||||
@@ -266,8 +260,8 @@ export default function InformationDivision() {
|
|||||||
<View>
|
<View>
|
||||||
<Pressable style={[Styles.wrapItemBorderBottom]} onPress={() => { handleMemberAdmin() }}>
|
<Pressable style={[Styles.wrapItemBorderBottom]} onPress={() => { handleMemberAdmin() }}>
|
||||||
<View style={[Styles.rowItemsCenter]}>
|
<View style={[Styles.rowItemsCenter]}>
|
||||||
<View style={[Styles.iconContent]}>
|
<View style={[Styles.iconContent, ColorsStatus.info]}>
|
||||||
<MaterialIcons name="verified-user" size={25} color={'black'} />
|
<MaterialIcons name="verified-user" size={25} color='#19345E' />
|
||||||
</View>
|
</View>
|
||||||
<View style={[Styles.rowSpaceBetween, { width: '88%' }]}>
|
<View style={[Styles.rowSpaceBetween, { width: '88%' }]}>
|
||||||
<View style={[Styles.ml10]}>
|
<View style={[Styles.ml10]}>
|
||||||
@@ -280,7 +274,7 @@ export default function InformationDivision() {
|
|||||||
<Pressable style={[Styles.wrapItemBorderBottom]} onPress={() => { handleMemberOut() }}>
|
<Pressable style={[Styles.wrapItemBorderBottom]} onPress={() => { handleMemberOut() }}>
|
||||||
<View style={[Styles.rowItemsCenter]}>
|
<View style={[Styles.rowItemsCenter]}>
|
||||||
<View style={[Styles.iconContent, ColorsStatus.info]}>
|
<View style={[Styles.iconContent, ColorsStatus.info]}>
|
||||||
<MaterialCommunityIcons name="close-circle" size={25} color={colors.primary} />
|
<MaterialCommunityIcons name="close-circle" size={25} color='#19345E' />
|
||||||
</View>
|
</View>
|
||||||
<View style={[Styles.rowSpaceBetween, { width: '88%' }]}>
|
<View style={[Styles.rowSpaceBetween, { width: '88%' }]}>
|
||||||
<View style={[Styles.ml10]}>
|
<View style={[Styles.ml10]}>
|
||||||
@@ -291,19 +285,6 @@ export default function InformationDivision() {
|
|||||||
</Pressable>
|
</Pressable>
|
||||||
</View>
|
</View>
|
||||||
</DrawerBottom>
|
</DrawerBottom>
|
||||||
|
|
||||||
<ModalConfirmation
|
|
||||||
visible={showDeleteModal}
|
|
||||||
title="Konfirmasi"
|
|
||||||
message="Apakah anda yakin ingin mengeluarkan anggota?"
|
|
||||||
onConfirm={() => {
|
|
||||||
setShowDeleteModal(false)
|
|
||||||
memberOut()
|
|
||||||
}}
|
|
||||||
onCancel={() => setShowDeleteModal(false)}
|
|
||||||
confirmText="Keluar"
|
|
||||||
cancelText="Batal"
|
|
||||||
/>
|
|
||||||
</SafeAreaView>
|
</SafeAreaView>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@@ -6,7 +6,6 @@ import { InputDate } from "@/components/inputDate"
|
|||||||
import Styles from "@/constants/Styles"
|
import Styles from "@/constants/Styles"
|
||||||
import { apiGetDivisionReport } from "@/lib/api"
|
import { apiGetDivisionReport } from "@/lib/api"
|
||||||
import { stringToDate } from "@/lib/fun_stringToDate"
|
import { stringToDate } from "@/lib/fun_stringToDate"
|
||||||
import { useTheme } from "@/providers/ThemeProvider"
|
|
||||||
import { useAuthSession } from "@/providers/AuthProvider"
|
import { useAuthSession } from "@/providers/AuthProvider"
|
||||||
import dayjs from "dayjs"
|
import dayjs from "dayjs"
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router"
|
import { router, Stack, useLocalSearchParams } from "expo-router"
|
||||||
@@ -15,7 +14,6 @@ import { SafeAreaView, ScrollView, View } from "react-native"
|
|||||||
import Toast from "react-native-toast-message"
|
import Toast from "react-native-toast-message"
|
||||||
|
|
||||||
export default function ReportDivision() {
|
export default function ReportDivision() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const { id } = useLocalSearchParams<{ id: string }>()
|
const { id } = useLocalSearchParams<{ id: string }>()
|
||||||
const { token, decryptToken } = useAuthSession();
|
const { token, decryptToken } = useAuthSession();
|
||||||
const [showReport, setShowReport] = useState(false);
|
const [showReport, setShowReport] = useState(false);
|
||||||
@@ -94,11 +92,8 @@ export default function ReportDivision() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, });
|
Toast.show({ type: 'small', text1: response.message, });
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Gagal mengambil data"
|
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -109,7 +104,7 @@ export default function ReportDivision() {
|
|||||||
}, [showReport]);
|
}, [showReport]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={{ flex: 1, backgroundColor: colors.background }}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
||||||
@@ -124,7 +119,7 @@ export default function ReportDivision() {
|
|||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<ScrollView style={{ backgroundColor: colors.background }}>
|
<ScrollView>
|
||||||
<View style={[Styles.p15, Styles.mb100]}>
|
<View style={[Styles.p15, Styles.mb100]}>
|
||||||
<InputDate
|
<InputDate
|
||||||
onChange={(val) => validationForm("date", val)}
|
onChange={(val) => validationForm("date", val)}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import ModalConfirmation from "@/components/ModalConfirmation";
|
import AlertKonfirmasi from "@/components/alertKonfirmasi";
|
||||||
import AppHeader from "@/components/AppHeader";
|
import AppHeader from "@/components/AppHeader";
|
||||||
import ButtonNextHeader from "@/components/buttonNextHeader";
|
import ButtonNextHeader from "@/components/buttonNextHeader";
|
||||||
import { InputForm } from "@/components/inputForm";
|
import { InputForm } from "@/components/inputForm";
|
||||||
@@ -8,7 +8,6 @@ import Styles from "@/constants/Styles";
|
|||||||
import { apiCheckDivisionName } from "@/lib/api";
|
import { apiCheckDivisionName } from "@/lib/api";
|
||||||
import { setFormCreateDivision } from "@/lib/divisionCreate";
|
import { setFormCreateDivision } from "@/lib/divisionCreate";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { router, Stack } from "expo-router";
|
import { router, Stack } from "expo-router";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { SafeAreaView, ScrollView, View } from "react-native";
|
import { SafeAreaView, ScrollView, View } from "react-native";
|
||||||
@@ -16,7 +15,6 @@ import Toast from "react-native-toast-message";
|
|||||||
import { useDispatch, useSelector } from "react-redux";
|
import { useDispatch, useSelector } from "react-redux";
|
||||||
|
|
||||||
export default function CreateDivision() {
|
export default function CreateDivision() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
const [isSelect, setSelect] = useState(false)
|
const [isSelect, setSelect] = useState(false)
|
||||||
const [chooseGroup, setChooseGroup] = useState({ val: "", label: "" })
|
const [chooseGroup, setChooseGroup] = useState({ val: "", label: "" })
|
||||||
@@ -25,7 +23,6 @@ export default function CreateDivision() {
|
|||||||
const entityUser = useSelector((state: any) => state.user)
|
const entityUser = useSelector((state: any) => state.user)
|
||||||
const userLogin = useSelector((state: any) => state.entities)
|
const userLogin = useSelector((state: any) => state.entities)
|
||||||
const [loadingBtn, setLoadingBtn] = useState(false)
|
const [loadingBtn, setLoadingBtn] = useState(false)
|
||||||
const [showWarningModal, setShowWarningModal] = useState(false)
|
|
||||||
const [error, setError] = useState({
|
const [error, setError] = useState({
|
||||||
idGroup: false,
|
idGroup: false,
|
||||||
name: false,
|
name: false,
|
||||||
@@ -70,18 +67,21 @@ export default function CreateDivision() {
|
|||||||
const response = await apiCheckDivisionName({ data: { ...dataForm }, user: hasil })
|
const response = await apiCheckDivisionName({ data: { ...dataForm }, user: hasil })
|
||||||
if (response.success) {
|
if (response.success) {
|
||||||
if (!response.available) {
|
if (!response.available) {
|
||||||
setShowWarningModal(true)
|
AlertKonfirmasi({
|
||||||
|
title: 'Peringatan',
|
||||||
|
category: 'warning',
|
||||||
|
desc: 'Nama divisi sudah ada. Tidak dapat membuat divisi dengan nama yang sama',
|
||||||
|
onPress: () => { }
|
||||||
|
})
|
||||||
} else {
|
} else {
|
||||||
handleSetData()
|
handleSetData()
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal menambahkan data"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoadingBtn(false)
|
setLoadingBtn(false)
|
||||||
}
|
}
|
||||||
@@ -99,7 +99,7 @@ export default function CreateDivision() {
|
|||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={{ flex: 1, backgroundColor: colors.background }}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => (
|
// headerLeft: () => (
|
||||||
@@ -131,7 +131,7 @@ export default function CreateDivision() {
|
|||||||
/>
|
/>
|
||||||
<ScrollView
|
<ScrollView
|
||||||
showsVerticalScrollIndicator={false}
|
showsVerticalScrollIndicator={false}
|
||||||
style={[Styles.h100, { backgroundColor: colors.background }]}
|
style={[Styles.h100]}
|
||||||
>
|
>
|
||||||
<View style={[Styles.p15]}>
|
<View style={[Styles.p15]}>
|
||||||
{
|
{
|
||||||
@@ -179,15 +179,6 @@ export default function CreateDivision() {
|
|||||||
open={isSelect}
|
open={isSelect}
|
||||||
valChoose={chooseGroup.val}
|
valChoose={chooseGroup.val}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<ModalConfirmation
|
|
||||||
visible={showWarningModal}
|
|
||||||
title="Peringatan"
|
|
||||||
message="Nama divisi sudah ada. Tidak dapat membuat divisi dengan nama yang sama"
|
|
||||||
onConfirm={() => setShowWarningModal(false)}
|
|
||||||
onCancel={() => setShowWarningModal(false)}
|
|
||||||
confirmText="Oke"
|
|
||||||
/>
|
|
||||||
</SafeAreaView>
|
</SafeAreaView>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,7 +8,6 @@ import { apiCreateDivision } from "@/lib/api";
|
|||||||
import { setFormCreateDivision } from "@/lib/divisionCreate";
|
import { setFormCreateDivision } from "@/lib/divisionCreate";
|
||||||
import { setUpdateDivision } from "@/lib/divisionUpdate";
|
import { setUpdateDivision } from "@/lib/divisionUpdate";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { AntDesign } from "@expo/vector-icons";
|
import { AntDesign } from "@expo/vector-icons";
|
||||||
import { StackActions, useNavigation } from "@react-navigation/native";
|
import { StackActions, useNavigation } from "@react-navigation/native";
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
@@ -24,7 +23,6 @@ type Props = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export default function CreateDivisionAddAdmin() {
|
export default function CreateDivisionAddAdmin() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
const navigation = useNavigation()
|
const navigation = useNavigation()
|
||||||
const { id } = useLocalSearchParams<{ id: string }>()
|
const { id } = useLocalSearchParams<{ id: string }>()
|
||||||
@@ -66,11 +64,9 @@ export default function CreateDivisionAddAdmin() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error : any ) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal menambahkan data"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
@@ -108,7 +104,7 @@ export default function CreateDivisionAddAdmin() {
|
|||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<View style={[Styles.p15, { flex: 1, backgroundColor: colors.background }]}>
|
<View style={[Styles.p15, { flex: 1 }]}>
|
||||||
<ScrollView>
|
<ScrollView>
|
||||||
{
|
{
|
||||||
data.length > 0 ?
|
data.length > 0 ?
|
||||||
@@ -127,12 +123,12 @@ export default function CreateDivisionAddAdmin() {
|
|||||||
<View style={[Styles.ml10]}>
|
<View style={[Styles.ml10]}>
|
||||||
<Text style={[Styles.textDefault]} numberOfLines={1} ellipsizeMode="tail">{item.name}</Text>
|
<Text style={[Styles.textDefault]} numberOfLines={1} ellipsizeMode="tail">{item.name}</Text>
|
||||||
{
|
{
|
||||||
found && <Text style={[Styles.textInformation, { color: colors.dimmed }]}>sudah menjadi anggota</Text>
|
found && <Text style={[Styles.textInformation, Styles.cGray]}>sudah menjadi anggota</Text>
|
||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
{
|
{
|
||||||
selectMember.some((i: any) => i == item.idUser) && <AntDesign name="check" size={20} color={colors.text} />
|
selectMember.some((i: any) => i == item.idUser) && <AntDesign name="check" size={20} color={'black'} />
|
||||||
}
|
}
|
||||||
</Pressable>
|
</Pressable>
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -10,7 +10,6 @@ import { apiGetUser } from "@/lib/api";
|
|||||||
import { setFormCreateDivision } from "@/lib/divisionCreate";
|
import { setFormCreateDivision } from "@/lib/divisionCreate";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { AntDesign } from "@expo/vector-icons";
|
import { AntDesign } from "@expo/vector-icons";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { Pressable, ScrollView, View } from "react-native";
|
import { Pressable, ScrollView, View } from "react-native";
|
||||||
@@ -23,7 +22,6 @@ type Props = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export default function CreateDivisionAddMember() {
|
export default function CreateDivisionAddMember() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
const { id } = useLocalSearchParams<{ id: string }>()
|
const { id } = useLocalSearchParams<{ id: string }>()
|
||||||
const [dataOld, setDataOld] = useState<Props[]>([])
|
const [dataOld, setDataOld] = useState<Props[]>([])
|
||||||
@@ -86,7 +84,7 @@ export default function CreateDivisionAddMember() {
|
|||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<View style={[Styles.p15, { flex: 1, backgroundColor: colors.background }]}>
|
<View style={[Styles.p15, { flex: 1 }]}>
|
||||||
<InputSearch onChange={(val) => setSearch(val)} value={search} />
|
<InputSearch onChange={(val) => setSearch(val)} value={search} />
|
||||||
|
|
||||||
{
|
{
|
||||||
@@ -108,7 +106,7 @@ export default function CreateDivisionAddMember() {
|
|||||||
</View>
|
</View>
|
||||||
|
|
||||||
:
|
:
|
||||||
<Text style={[Styles.textDefault, Styles.pv05, { textAlign: 'center', color: colors.dimmed }]}>Tidak ada member yang dipilih</Text>
|
<Text style={[Styles.textDefault, Styles.cGray, Styles.pv05, { textAlign: 'center' }]}>Tidak ada member yang dipilih</Text>
|
||||||
}
|
}
|
||||||
<ScrollView
|
<ScrollView
|
||||||
showsVerticalScrollIndicator={false}
|
showsVerticalScrollIndicator={false}
|
||||||
@@ -131,12 +129,12 @@ export default function CreateDivisionAddMember() {
|
|||||||
<View style={[Styles.ml10]}>
|
<View style={[Styles.ml10]}>
|
||||||
<Text style={[Styles.textDefault]} numberOfLines={1} ellipsizeMode="tail">{item.name}</Text>
|
<Text style={[Styles.textDefault]} numberOfLines={1} ellipsizeMode="tail">{item.name}</Text>
|
||||||
{
|
{
|
||||||
found && <Text style={[Styles.textInformation, { color: colors.dimmed }]}>sudah menjadi anggota</Text>
|
found && <Text style={[Styles.textInformation, Styles.cGray]}>sudah menjadi anggota</Text>
|
||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
{
|
{
|
||||||
selectMember.some((i: any) => i.idUser == item.id) && <AntDesign name="check" size={20} color={colors.text} />
|
selectMember.some((i: any) => i.idUser == item.id) && <AntDesign name="check" size={20} color={'black'} />
|
||||||
}
|
}
|
||||||
</Pressable>
|
</Pressable>
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -6,16 +6,16 @@ import PaperGridContent from "@/components/paperGridContent";
|
|||||||
import Skeleton from "@/components/skeleton";
|
import Skeleton from "@/components/skeleton";
|
||||||
import SkeletonTwoItem from "@/components/skeletonTwoItem";
|
import SkeletonTwoItem from "@/components/skeletonTwoItem";
|
||||||
import Text from "@/components/Text";
|
import Text from "@/components/Text";
|
||||||
import WrapTab from "@/components/wrapTab";
|
import { ColorsStatus } from "@/constants/ColorsStatus";
|
||||||
import Styles from "@/constants/Styles";
|
import Styles from "@/constants/Styles";
|
||||||
import { apiGetDivision } from "@/lib/api";
|
import { apiGetDivision } from "@/lib/api";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import {
|
import {
|
||||||
AntDesign,
|
AntDesign,
|
||||||
Feather,
|
Feather,
|
||||||
Ionicons,
|
Ionicons,
|
||||||
MaterialCommunityIcons
|
MaterialCommunityIcons,
|
||||||
|
MaterialIcons,
|
||||||
} from "@expo/vector-icons";
|
} from "@expo/vector-icons";
|
||||||
import { router, useLocalSearchParams } from "expo-router";
|
import { router, useLocalSearchParams } from "expo-router";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
@@ -38,11 +38,9 @@ export default function ListDivision() {
|
|||||||
const [isList, setList] = useState(false);
|
const [isList, setList] = useState(false);
|
||||||
const entityUser = useSelector((state: any) => state.user)
|
const entityUser = useSelector((state: any) => state.user)
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
const { colors } = useTheme();
|
|
||||||
const [search, setSearch] = useState("")
|
const [search, setSearch] = useState("")
|
||||||
const [nameGroup, setNameGroup] = useState("")
|
const [nameGroup, setNameGroup] = useState("")
|
||||||
const [data, setData] = useState<Props[]>([])
|
const [data, setData] = useState<Props[]>([])
|
||||||
// ... state same ...
|
|
||||||
const update = useSelector((state: any) => state.divisionUpdate)
|
const update = useSelector((state: any) => state.divisionUpdate)
|
||||||
const arrSkeleton = Array.from({ length: 3 }, (_, index) => index)
|
const arrSkeleton = Array.from({ length: 3 }, (_, index) => index)
|
||||||
const [loading, setLoading] = useState(false)
|
const [loading, setLoading] = useState(false)
|
||||||
@@ -116,11 +114,11 @@ export default function ListDivision() {
|
|||||||
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<View style={[Styles.p15, { flex: 1, backgroundColor: colors.background }]}>
|
<View style={[Styles.p15, { flex: 1 }]}>
|
||||||
<View>
|
<View>
|
||||||
{
|
{
|
||||||
entityUser.role != "user" && entityUser.role != "coadmin" ?
|
entityUser.role != "user" && entityUser.role != "coadmin" ?
|
||||||
<WrapTab>
|
<View style={[Styles.wrapBtnTab]}>
|
||||||
<ButtonTab
|
<ButtonTab
|
||||||
active={status == "false" ? "false" : "true"}
|
active={status == "false" ? "false" : "true"}
|
||||||
value="true"
|
value="true"
|
||||||
@@ -129,7 +127,7 @@ export default function ListDivision() {
|
|||||||
icon={
|
icon={
|
||||||
<Feather
|
<Feather
|
||||||
name="check-circle"
|
name="check-circle"
|
||||||
color={status == "false" ? colors.dimmed : "white"}
|
color={status == "false" ? "black" : "white"}
|
||||||
size={20}
|
size={20}
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
@@ -143,15 +141,15 @@ export default function ListDivision() {
|
|||||||
icon={
|
icon={
|
||||||
<AntDesign
|
<AntDesign
|
||||||
name="closecircleo"
|
name="closecircleo"
|
||||||
color={status == "true" ? colors.dimmed : "white"}
|
color={status == "true" ? "black" : "white"}
|
||||||
size={20}
|
size={20}
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
n={2}
|
n={2}
|
||||||
/>
|
/>
|
||||||
</WrapTab>
|
</View>
|
||||||
:
|
:
|
||||||
<WrapTab>
|
<View style={[Styles.wrapBtnTab]}>
|
||||||
<ButtonTab
|
<ButtonTab
|
||||||
active={category == "semua" ? "false" : "true"}
|
active={category == "semua" ? "false" : "true"}
|
||||||
value="true"
|
value="true"
|
||||||
@@ -160,7 +158,7 @@ export default function ListDivision() {
|
|||||||
icon={
|
icon={
|
||||||
<Ionicons
|
<Ionicons
|
||||||
name="file-tray-outline"
|
name="file-tray-outline"
|
||||||
color={category == "semua" ? colors.dimmed : "white"}
|
color={category == "semua" ? "black" : "white"}
|
||||||
size={20}
|
size={20}
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
@@ -174,13 +172,13 @@ export default function ListDivision() {
|
|||||||
icon={
|
icon={
|
||||||
<Ionicons
|
<Ionicons
|
||||||
name="file-tray-stacked-outline"
|
name="file-tray-stacked-outline"
|
||||||
color={category == "semua" ? "white" : colors.dimmed}
|
color={category == "semua" ? "white" : "black"}
|
||||||
size={20}
|
size={20}
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
n={2}
|
n={2}
|
||||||
/>
|
/>
|
||||||
</WrapTab>
|
</View>
|
||||||
}
|
}
|
||||||
|
|
||||||
<View style={[Styles.rowSpaceBetween, { alignItems: 'center' }]}>
|
<View style={[Styles.rowSpaceBetween, { alignItems: 'center' }]}>
|
||||||
@@ -192,7 +190,7 @@ export default function ListDivision() {
|
|||||||
>
|
>
|
||||||
<MaterialCommunityIcons
|
<MaterialCommunityIcons
|
||||||
name={isList ? "format-list-bulleted" : "view-grid"}
|
name={isList ? "format-list-bulleted" : "view-grid"}
|
||||||
color={colors.text}
|
color={"black"}
|
||||||
size={30}
|
size={30}
|
||||||
/>
|
/>
|
||||||
</Pressable>
|
</Pressable>
|
||||||
@@ -204,7 +202,7 @@ export default function ListDivision() {
|
|||||||
</View>
|
</View>
|
||||||
)}
|
)}
|
||||||
</View>
|
</View>
|
||||||
<View style={[{ flex: 2 }, Styles.mt10]}>
|
<View style={[{ flex: 2 }, Styles.mt05]}>
|
||||||
{
|
{
|
||||||
loading ?
|
loading ?
|
||||||
isList ?
|
isList ?
|
||||||
@@ -218,7 +216,7 @@ export default function ListDivision() {
|
|||||||
:
|
:
|
||||||
data.length == 0 ? (
|
data.length == 0 ? (
|
||||||
<View style={[Styles.mt15]}>
|
<View style={[Styles.mt15]}>
|
||||||
<Text style={[Styles.textDefault, { textAlign: 'center', color: colors.dimmed }]}>Tidak ada data</Text>
|
<Text style={[Styles.textDefault, Styles.cGray, { textAlign: 'center' }]}>Tidak ada data</Text>
|
||||||
</View>
|
</View>
|
||||||
) : (
|
) : (
|
||||||
isList ? (
|
isList ? (
|
||||||
@@ -234,13 +232,13 @@ export default function ListDivision() {
|
|||||||
key={index}
|
key={index}
|
||||||
onPress={() => { router.push(`/division/${item.id}`) }}
|
onPress={() => { router.push(`/division/${item.id}`) }}
|
||||||
borderType="bottom"
|
borderType="bottom"
|
||||||
bgColor="transparent"
|
|
||||||
icon={
|
icon={
|
||||||
<View style={[Styles.iconContent]}>
|
<View style={[Styles.iconContent, ColorsStatus.lightGreen]}>
|
||||||
<Feather name="users" size={25} color={'black'} />
|
<MaterialIcons name="group" size={25} color={"#384288"} />
|
||||||
</View>
|
</View>
|
||||||
}
|
}
|
||||||
title={item.name}
|
title={item.name}
|
||||||
|
titleWeight="normal"
|
||||||
/>
|
/>
|
||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
@@ -252,7 +250,6 @@ export default function ListDivision() {
|
|||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={handleRefresh}
|
onRefresh={handleRefresh}
|
||||||
tintColor={colors.icon}
|
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
@@ -288,7 +285,6 @@ export default function ListDivision() {
|
|||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={handleRefresh}
|
onRefresh={handleRefresh}
|
||||||
tintColor={colors.icon}
|
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -9,7 +9,6 @@ import Styles from "@/constants/Styles";
|
|||||||
import { apiGetDivisionReport } from "@/lib/api";
|
import { apiGetDivisionReport } from "@/lib/api";
|
||||||
import { stringToDate } from "@/lib/fun_stringToDate";
|
import { stringToDate } from "@/lib/fun_stringToDate";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import dayjs from "dayjs";
|
import dayjs from "dayjs";
|
||||||
import { router, Stack } from "expo-router";
|
import { router, Stack } from "expo-router";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
@@ -17,7 +16,6 @@ import { SafeAreaView, ScrollView, View } from "react-native";
|
|||||||
import Toast from "react-native-toast-message";
|
import Toast from "react-native-toast-message";
|
||||||
|
|
||||||
export default function Report() {
|
export default function Report() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const { token, decryptToken } = useAuthSession();
|
const { token, decryptToken } = useAuthSession();
|
||||||
const [chooseGroup, setChooseGroup] = useState({ val: "", label: "" });
|
const [chooseGroup, setChooseGroup] = useState({ val: "", label: "" });
|
||||||
const [showReport, setShowReport] = useState(false);
|
const [showReport, setShowReport] = useState(false);
|
||||||
@@ -112,11 +110,8 @@ export default function Report() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Gagal mengambil data"
|
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -127,7 +122,7 @@ export default function Report() {
|
|||||||
}, [showReport]);
|
}, [showReport]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={{ flex: 1, backgroundColor: colors.background }}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => (
|
// headerLeft: () => (
|
||||||
@@ -149,11 +144,11 @@ export default function Report() {
|
|||||||
/>
|
/>
|
||||||
<ScrollView
|
<ScrollView
|
||||||
showsVerticalScrollIndicator={false}
|
showsVerticalScrollIndicator={false}
|
||||||
style={[Styles.h100, { backgroundColor: colors.background }]}
|
style={[Styles.h100]}
|
||||||
>
|
>
|
||||||
<View style={[Styles.p15, Styles.mb50]}>
|
<View style={[Styles.p15, Styles.mb50]}>
|
||||||
<SelectForm
|
<SelectForm
|
||||||
bg={colors.card}
|
bg="white"
|
||||||
label="Lembaga Desa"
|
label="Lembaga Desa"
|
||||||
placeholder="Pilih Lembaga Desa"
|
placeholder="Pilih Lembaga Desa"
|
||||||
value={chooseGroup.label}
|
value={chooseGroup.label}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import AppHeader from "@/components/AppHeader";
|
import ButtonBackHeader from "@/components/buttonBackHeader";
|
||||||
import ButtonSaveHeader from "@/components/buttonSaveHeader";
|
import ButtonSaveHeader from "@/components/buttonSaveHeader";
|
||||||
import { InputForm } from "@/components/inputForm";
|
import { InputForm } from "@/components/inputForm";
|
||||||
import ModalSelect from "@/components/modalSelect";
|
import ModalSelect from "@/components/modalSelect";
|
||||||
@@ -10,7 +10,6 @@ import { apiEditProfile, apiGetProfile } from "@/lib/api";
|
|||||||
import { setEntities } from "@/lib/entitiesSlice";
|
import { setEntities } from "@/lib/entitiesSlice";
|
||||||
import { validateName } from "@/lib/fun_validateName";
|
import { validateName } from "@/lib/fun_validateName";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { MaterialCommunityIcons } from "@expo/vector-icons";
|
import { MaterialCommunityIcons } from "@expo/vector-icons";
|
||||||
import { useHeaderHeight } from "@react-navigation/elements";
|
import { useHeaderHeight } from "@react-navigation/elements";
|
||||||
import * as ImagePicker from "expo-image-picker";
|
import * as ImagePicker from "expo-image-picker";
|
||||||
@@ -44,11 +43,9 @@ type Props = {
|
|||||||
export default function EditProfile() {
|
export default function EditProfile() {
|
||||||
const headerHeight = useHeaderHeight()
|
const headerHeight = useHeaderHeight()
|
||||||
const dispatch = useDispatch()
|
const dispatch = useDispatch()
|
||||||
const { colors } = useTheme();
|
|
||||||
const entities = useSelector((state: any) => state.entities)
|
const entities = useSelector((state: any) => state.entities)
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
const [errorImg, setErrorImg] = useState(false)
|
const [errorImg, setErrorImg] = useState(false)
|
||||||
// ... keeping state same ...
|
|
||||||
const [selectedImage, setSelectedImage] = useState<string | undefined | { uri: string }>(undefined);
|
const [selectedImage, setSelectedImage] = useState<string | undefined | { uri: string }>(undefined);
|
||||||
const [choosePosition, setChoosePosition] = useState({ val: entities.idPosition, label: entities.position });
|
const [choosePosition, setChoosePosition] = useState({ val: entities.idPosition, label: entities.position });
|
||||||
const [chooseGender, setChooseGender] = useState({ val: entities.gender, label: entities.gender == "F" ? 'Perempuan' : 'Laki-laki' });
|
const [chooseGender, setChooseGender] = useState({ val: entities.gender, label: entities.gender == "F" ? 'Perempuan' : 'Laki-laki' });
|
||||||
@@ -188,14 +185,9 @@ export default function EditProfile() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal mengupdate data"
|
Toast.show({ type: 'small', text1: 'Gagal mengupdate data', })
|
||||||
|
|
||||||
Toast.show({
|
|
||||||
type: 'small',
|
|
||||||
text1: message
|
|
||||||
})
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
@@ -221,43 +213,27 @@ export default function EditProfile() {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={[Styles.flex1, { backgroundColor: colors.background }]}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => (
|
headerLeft: () => (
|
||||||
// <ButtonBackHeader
|
<ButtonBackHeader
|
||||||
// onPress={() => {
|
onPress={() => {
|
||||||
// router.back();
|
router.back();
|
||||||
// }}
|
}}
|
||||||
// />
|
/>
|
||||||
// ),
|
),
|
||||||
headerTitle: "Edit Profile",
|
headerTitle: "Edit Profile",
|
||||||
headerTitleAlign: "center",
|
headerTitleAlign: "center",
|
||||||
header: () => (
|
headerRight: () => (
|
||||||
<AppHeader
|
<ButtonSaveHeader
|
||||||
title="Edit Profile"
|
disable={disableBtn || loading ? true : false}
|
||||||
showBack={true}
|
category="update"
|
||||||
onPressLeft={() => router.back()}
|
onPress={() => {
|
||||||
right={
|
handleEdit()
|
||||||
<ButtonSaveHeader
|
}}
|
||||||
disable={disableBtn || loading ? true : false}
|
|
||||||
category="update"
|
|
||||||
onPress={() => {
|
|
||||||
handleEdit()
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
}
|
|
||||||
/>
|
/>
|
||||||
)
|
),
|
||||||
// headerRight: () => (
|
|
||||||
// <ButtonSaveHeader
|
|
||||||
// disable={disableBtn || loading ? true : false}
|
|
||||||
// category="update"
|
|
||||||
// onPress={() => {
|
|
||||||
// handleEdit()
|
|
||||||
// }}
|
|
||||||
// />
|
|
||||||
// ),
|
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<KeyboardAvoidingView
|
<KeyboardAvoidingView
|
||||||
@@ -267,7 +243,7 @@ export default function EditProfile() {
|
|||||||
>
|
>
|
||||||
<ScrollView showsVerticalScrollIndicator={false}>
|
<ScrollView showsVerticalScrollIndicator={false}>
|
||||||
<View style={[Styles.p15]}>
|
<View style={[Styles.p15]}>
|
||||||
<View style={[Styles.contentItemCenter]}>
|
<View style={{ justifyContent: "center", alignItems: "center" }}>
|
||||||
{
|
{
|
||||||
selectedImage != undefined ? (
|
selectedImage != undefined ? (
|
||||||
<Pressable onPress={pickImageAsync}>
|
<Pressable onPress={pickImageAsync}>
|
||||||
|
|||||||
@@ -1,18 +1,16 @@
|
|||||||
import AppHeader from "@/components/AppHeader";
|
import AppHeader from "@/components/AppHeader";
|
||||||
import { ButtonFiturMenu } from "@/components/buttonFiturMenu";
|
import { ButtonFiturMenu } from "@/components/buttonFiturMenu";
|
||||||
import Styles from "@/constants/Styles";
|
import Styles from "@/constants/Styles";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
import { AntDesign, Entypo, Ionicons, MaterialCommunityIcons, MaterialIcons } from "@expo/vector-icons";
|
||||||
import { AntDesign, Entypo, Feather, Ionicons, MaterialCommunityIcons, MaterialIcons } from "@expo/vector-icons";
|
|
||||||
import { router, Stack } from "expo-router";
|
import { router, Stack } from "expo-router";
|
||||||
import { SafeAreaView, View } from "react-native";
|
import { SafeAreaView, View } from "react-native";
|
||||||
import { useSelector } from "react-redux";
|
import { useSelector } from "react-redux";
|
||||||
|
|
||||||
export default function Feature() {
|
export default function Feature() {
|
||||||
const entityUser = useSelector((state: any) => state.user)
|
const entityUser = useSelector((state: any) => state.user)
|
||||||
const { colors } = useTheme();
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={{ flex: 1, backgroundColor: colors.background }}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
headerTitle: 'Fitur',
|
headerTitle: 'Fitur',
|
||||||
@@ -24,26 +22,32 @@ export default function Feature() {
|
|||||||
/>
|
/>
|
||||||
<View style={[Styles.p15]}>
|
<View style={[Styles.p15]}>
|
||||||
<View style={[Styles.rowSpaceBetween, Styles.mb15]}>
|
<View style={[Styles.rowSpaceBetween, Styles.mb15]}>
|
||||||
<ButtonFiturMenu icon={<Feather name="users" size={30} color={colors.icon} />} text="Divisi" onPress={() => { router.push('/division?active=true') }} />
|
<ButtonFiturMenu icon={<MaterialIcons name="group" size={35} color="black" />} text="Divisi" onPress={() => { router.push('/division?active=true') }} />
|
||||||
<ButtonFiturMenu icon={<Feather name="bar-chart" size={30} color={colors.icon} />} text="Kegiatan" onPress={() => { router.push('/project?status=0') }} />
|
<ButtonFiturMenu icon={<AntDesign name="areachart" size={35} color="black" />} text="Kegiatan" onPress={() => { router.push('/project?status=0') }} />
|
||||||
<ButtonFiturMenu icon={<Ionicons name="megaphone-outline" size={30} color={colors.icon} />} text="Pengumuman" onPress={() => { router.push('/announcement') }} />
|
<ButtonFiturMenu icon={<MaterialIcons name="campaign" size={35} color="black" />} text="Pengumuman" onPress={() => { router.push('/announcement') }} />
|
||||||
<ButtonFiturMenu icon={<Ionicons name="chatbubbles-outline" size={30} color={colors.icon} />} text="Diskusi" onPress={() => { router.push('/discussion?active=true') }} />
|
<ButtonFiturMenu icon={<Ionicons name="chatbubbles-sharp" size={35} color="black" />} text="Diskusi" onPress={() => { router.push('/discussion?active=true') }} />
|
||||||
</View>
|
</View>
|
||||||
<View style={[Styles.rowSpaceBetween, Styles.mb15, (entityUser.role == 'cosupadmin' ? Styles.w70 : entityUser.role == 'supadmin' || entityUser.role == 'developer' ? Styles.w100 : Styles.w40)]}>
|
<View style={[Styles.rowSpaceBetween, Styles.mb15, (entityUser.role == 'cosupadmin' ? Styles.w70 : entityUser.role == 'supadmin' || entityUser.role == 'developer' ? Styles.w100 : Styles.w40)]}>
|
||||||
<ButtonFiturMenu icon={<MaterialCommunityIcons name="account-group-outline" size={30} color={colors.icon} />} text="Anggota" onPress={() => { router.push('/member') }} />
|
<ButtonFiturMenu icon={<MaterialIcons name="groups" size={35} color="black" />} text="Anggota" onPress={() => { router.push('/member') }} />
|
||||||
<ButtonFiturMenu icon={<MaterialCommunityIcons name="account-tie-outline" size={30} color={colors.icon} />} text="Jabatan" onPress={() => { router.push('/position') }} />
|
<ButtonFiturMenu icon={<MaterialCommunityIcons name="account-tie" size={35} color="black" />} text="Jabatan" onPress={() => { router.push('/position') }} />
|
||||||
{
|
{
|
||||||
entityUser.role == "cosupadmin" && <ButtonFiturMenu icon={<Ionicons name="images-outline" size={30} color={colors.icon} />} text="Banner" onPress={() => { router.push('/banner') }} />
|
entityUser.role == "cosupadmin" && <ButtonFiturMenu icon={<Entypo name="image-inverted" size={35} color="black" />} text="Banner" onPress={() => { router.push('/banner') }} />
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
(entityUser.role == "supadmin" || entityUser.role == "developer") &&
|
(entityUser.role == "supadmin" || entityUser.role == "developer") &&
|
||||||
<>
|
<>
|
||||||
<ButtonFiturMenu icon={<Ionicons name="bookmarks-outline" size={30} color={colors.icon} />} text="Lembaga Desa" onPress={() => { router.push('/group') }} />
|
<ButtonFiturMenu icon={<AntDesign name="tags" size={35} color="black" />} text="Lembaga Desa" onPress={() => { router.push('/group') }} />
|
||||||
{/* <ButtonFiturMenu icon={<Ionicons name="color-palette-sharp" size={30} color={colors.icon} />} text="Tema" onPress={() => { }} /> */}
|
{/* <ButtonFiturMenu icon={<Ionicons name="color-palette-sharp" size={35} color="black" />} text="Tema" onPress={() => { }} /> */}
|
||||||
<ButtonFiturMenu icon={<Ionicons name="images-outline" size={30} color={colors.icon} />} text="Banner" onPress={() => { router.push('/banner') }} />
|
<ButtonFiturMenu icon={<Entypo name="image-inverted" size={35} color="black" />} text="Banner" onPress={() => { router.push('/banner') }} />
|
||||||
</>
|
</>
|
||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
|
{/* {
|
||||||
|
(entityUser.role == "supadmin" || entityUser.role == "developer") &&
|
||||||
|
<View style={[Styles.rowSpaceBetween, Styles.mb15]}>
|
||||||
|
<ButtonFiturMenu icon={<Entypo name="image-inverted" size={35} color="black" />} text="Banner" onPress={() => { router.push('/banner') }} />
|
||||||
|
</View>
|
||||||
|
} */}
|
||||||
</View>
|
</View>
|
||||||
</SafeAreaView>
|
</SafeAreaView>
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import ModalConfirmation from "@/components/ModalConfirmation";
|
import AlertKonfirmasi from "@/components/alertKonfirmasi";
|
||||||
import BorderBottomItem from "@/components/borderBottomItem";
|
import BorderBottomItem from "@/components/borderBottomItem";
|
||||||
import { ButtonForm } from "@/components/buttonForm";
|
import { ButtonForm } from "@/components/buttonForm";
|
||||||
import ButtonTab from "@/components/buttonTab";
|
import ButtonTab from "@/components/buttonTab";
|
||||||
@@ -8,13 +8,12 @@ import InputSearch from "@/components/inputSearch";
|
|||||||
import MenuItemRow from "@/components/menuItemRow";
|
import MenuItemRow from "@/components/menuItemRow";
|
||||||
import SkeletonTwoItem from "@/components/skeletonTwoItem";
|
import SkeletonTwoItem from "@/components/skeletonTwoItem";
|
||||||
import Text from "@/components/Text";
|
import Text from "@/components/Text";
|
||||||
import WrapTab from "@/components/wrapTab";
|
import { ColorsStatus } from "@/constants/ColorsStatus";
|
||||||
import Styles from "@/constants/Styles";
|
import Styles from "@/constants/Styles";
|
||||||
import { apiDeleteGroup, apiEditGroup, apiGetGroup } from "@/lib/api";
|
import { apiDeleteGroup, apiEditGroup, apiGetGroup } from "@/lib/api";
|
||||||
import { setUpdateGroup } from "@/lib/groupSlice";
|
import { setUpdateGroup } from "@/lib/groupSlice";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
import { AntDesign, Feather, MaterialCommunityIcons } from "@expo/vector-icons";
|
||||||
import { AntDesign, Feather, Ionicons, MaterialCommunityIcons } from "@expo/vector-icons";
|
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { RefreshControl, View, VirtualizedList } from "react-native";
|
import { RefreshControl, View, VirtualizedList } from "react-native";
|
||||||
import Toast from "react-native-toast-message";
|
import Toast from "react-native-toast-message";
|
||||||
@@ -28,14 +27,12 @@ type Props = {
|
|||||||
|
|
||||||
export default function Index() {
|
export default function Index() {
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
const { colors } = useTheme();
|
|
||||||
const [isModal, setModal] = useState(false)
|
const [isModal, setModal] = useState(false)
|
||||||
const [isVisibleEdit, setVisibleEdit] = useState(false)
|
const [isVisibleEdit, setVisibleEdit] = useState(false)
|
||||||
const [data, setData] = useState<Props[]>([])
|
const [data, setData] = useState<Props[]>([])
|
||||||
const [search, setSearch] = useState('')
|
const [search, setSearch] = useState('')
|
||||||
const arrSkeleton = Array.from({ length: 5 }, (_, index) => index)
|
const arrSkeleton = Array.from({ length: 5 }, (_, index) => index)
|
||||||
const [loading, setLoading] = useState(true)
|
const [loading, setLoading] = useState(true)
|
||||||
const [showDeleteModal, setShowDeleteModal] = useState(false)
|
|
||||||
const [status, setStatus] = useState<'true' | 'false'>('true')
|
const [status, setStatus] = useState<'true' | 'false'>('true')
|
||||||
const [loadingSubmit, setLoadingSubmit] = useState(false)
|
const [loadingSubmit, setLoadingSubmit] = useState(false)
|
||||||
const [idChoose, setIdChoose] = useState('')
|
const [idChoose, setIdChoose] = useState('')
|
||||||
@@ -130,27 +127,27 @@ export default function Index() {
|
|||||||
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<View style={[Styles.p15, { flex: 1, backgroundColor: colors.background }]}>
|
<View style={[Styles.p15, { flex: 1 }]}>
|
||||||
<View style={[Styles.mb10]}>
|
<View style={[Styles.mb10]}>
|
||||||
<WrapTab>
|
<View style={[Styles.wrapBtnTab]}>
|
||||||
<ButtonTab
|
<ButtonTab
|
||||||
active={status == "false" ? "false" : "true"}
|
active={status == "false" ? "false" : "true"}
|
||||||
value="true"
|
value="true"
|
||||||
onPress={() => { setStatus("true") }}
|
onPress={() => { setStatus("true") }}
|
||||||
label="Aktif"
|
label="Aktif"
|
||||||
icon={<Feather name="check-circle" color={status == "true" ? 'white' : colors.dimmed} size={20} />}
|
icon={<Feather name="check-circle" color={status == "true" ? 'white' : 'black'} size={20} />}
|
||||||
n={2} />
|
n={2} />
|
||||||
<ButtonTab
|
<ButtonTab
|
||||||
active={status == "false" ? "false" : "true"}
|
active={status == "false" ? "false" : "true"}
|
||||||
value="false"
|
value="false"
|
||||||
onPress={() => { setStatus("false") }}
|
onPress={() => { setStatus("false") }}
|
||||||
label="Tidak Aktif"
|
label="Tidak Aktif"
|
||||||
icon={<AntDesign name="closecircleo" color={status == "false" ? 'white' : colors.dimmed} size={20} />}
|
icon={<AntDesign name="closecircleo" color={status == "false" ? 'white' : 'black'} size={20} />}
|
||||||
n={2} />
|
n={2} />
|
||||||
</WrapTab>
|
</View>
|
||||||
<InputSearch onChange={setSearch} />
|
<InputSearch onChange={setSearch} />
|
||||||
</View>
|
</View>
|
||||||
<View style={[{ flex: 2 }, Styles.mt10]}>
|
<View style={[{ flex: 2 }, Styles.mt05]}>
|
||||||
{
|
{
|
||||||
loading ?
|
loading ?
|
||||||
arrSkeleton.map((item, index) => {
|
arrSkeleton.map((item, index) => {
|
||||||
@@ -176,8 +173,8 @@ export default function Index() {
|
|||||||
}}
|
}}
|
||||||
borderType="all"
|
borderType="all"
|
||||||
icon={
|
icon={
|
||||||
<View style={[Styles.iconContent]}>
|
<View style={[Styles.iconContent, ColorsStatus.lightGreen]}>
|
||||||
<Ionicons name="bookmark-outline" size={25} color={'black'} />
|
<MaterialCommunityIcons name="office-building-outline" size={25} color={'#384288'} />
|
||||||
</View>
|
</View>
|
||||||
}
|
}
|
||||||
title={item.name}
|
title={item.name}
|
||||||
@@ -190,29 +187,30 @@ export default function Index() {
|
|||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={handleRefresh}
|
onRefresh={handleRefresh}
|
||||||
tintColor={colors.icon}
|
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
:
|
:
|
||||||
<Text style={[Styles.textDefault, { textAlign: 'center', color: colors.dimmed }]}>Tidak ada data</Text>
|
<Text style={[Styles.textDefault, Styles.cGray, { textAlign: 'center' }]}>Tidak ada data</Text>
|
||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
|
|
||||||
<DrawerBottom animation="slide" isVisible={isModal} setVisible={() => setModal(false)} title={titleChoose}>
|
<DrawerBottom animation="slide" isVisible={isModal} setVisible={() => setModal(false)} title={titleChoose}>
|
||||||
<View style={Styles.rowItemsCenter}>
|
<View style={Styles.rowItemsCenter}>
|
||||||
<MenuItemRow
|
<MenuItemRow
|
||||||
icon={<MaterialCommunityIcons name="toggle-switch-off-outline" color={colors.text} size={25} />}
|
icon={<MaterialCommunityIcons name="toggle-switch-off-outline" color="black" size={25} />}
|
||||||
title={activeChoose ? "Non Aktifkan" : "Aktifkan"}
|
title={activeChoose ? "Non Aktifkan" : "Aktifkan"}
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
setModal(false)
|
setModal(false)
|
||||||
setTimeout(() => {
|
AlertKonfirmasi({
|
||||||
setShowDeleteModal(true)
|
title: 'Konfirmasi',
|
||||||
}, 600)
|
desc: activeChoose ? 'Apakah anda yakin ingin menonaktifkan data?' : 'Apakah anda yakin ingin mengaktifkan data?',
|
||||||
|
onPress: () => { handleDelete() }
|
||||||
|
})
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<MenuItemRow
|
<MenuItemRow
|
||||||
icon={<MaterialCommunityIcons name="pencil-outline" color={colors.text} size={25} />}
|
icon={<MaterialCommunityIcons name="pencil-outline" color="black" size={25} />}
|
||||||
title="Edit"
|
title="Edit"
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
setModal(false)
|
setModal(false)
|
||||||
@@ -234,7 +232,6 @@ export default function Index() {
|
|||||||
label="Lembaga Desa"
|
label="Lembaga Desa"
|
||||||
value={titleChoose}
|
value={titleChoose}
|
||||||
error={error.title}
|
error={error.title}
|
||||||
bg={"transparent"}
|
|
||||||
errorText="Lembaga Desa tidak boleh kosong & minimal 3 karakter"
|
errorText="Lembaga Desa tidak boleh kosong & minimal 3 karakter"
|
||||||
onChange={(val) => { validationForm(val, 'title') }} />
|
onChange={(val) => { validationForm(val, 'title') }} />
|
||||||
</View>
|
</View>
|
||||||
@@ -243,19 +240,6 @@ export default function Index() {
|
|||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
</DrawerBottom>
|
</DrawerBottom>
|
||||||
|
|
||||||
<ModalConfirmation
|
|
||||||
visible={showDeleteModal}
|
|
||||||
title="Konfirmasi"
|
|
||||||
message={activeChoose ? 'Apakah anda yakin ingin menonaktifkan data?' : 'Apakah anda yakin ingin mengaktifkan data?'}
|
|
||||||
onConfirm={() => {
|
|
||||||
setShowDeleteModal(false)
|
|
||||||
handleDelete()
|
|
||||||
}}
|
|
||||||
onCancel={() => setShowDeleteModal(false)}
|
|
||||||
confirmText={activeChoose ? "Nonaktifkan" : "Aktifkan"}
|
|
||||||
cancelText="Batal"
|
|
||||||
/>
|
|
||||||
</View >
|
</View >
|
||||||
|
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -1,9 +1,10 @@
|
|||||||
import CaraouselHome2 from "@/components/home/carouselHome2";
|
import CaraouselHome from "@/components/home/carouselHome";
|
||||||
import ChartDokumenHome from "@/components/home/chartDokumenHome";
|
import ChartDokumenHome from "@/components/home/chartDokumenHome";
|
||||||
import ChartProgresHome from "@/components/home/chartProgresHome";
|
import ChartProgresHome from "@/components/home/chartProgresHome";
|
||||||
import DisccussionHome from "@/components/home/discussionHome";
|
import DisccussionHome from "@/components/home/discussionHome";
|
||||||
import DivisionHome from "@/components/home/divisionHome";
|
import DivisionHome from "@/components/home/divisionHome";
|
||||||
import EventHome from "@/components/home/eventHome";
|
import EventHome from "@/components/home/eventHome";
|
||||||
|
import FiturHome from "@/components/home/fiturHome";
|
||||||
import { HeaderRightHome } from "@/components/home/headerRightHome";
|
import { HeaderRightHome } from "@/components/home/headerRightHome";
|
||||||
import ProjectHome from "@/components/home/projectHome";
|
import ProjectHome from "@/components/home/projectHome";
|
||||||
import Text from "@/components/Text";
|
import Text from "@/components/Text";
|
||||||
@@ -11,11 +12,9 @@ import Styles from "@/constants/Styles";
|
|||||||
import { apiGetProfile } from "@/lib/api";
|
import { apiGetProfile } from "@/lib/api";
|
||||||
import { setEntities } from "@/lib/entitiesSlice";
|
import { setEntities } from "@/lib/entitiesSlice";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { LinearGradient } from "expo-linear-gradient";
|
|
||||||
import { Stack } from "expo-router";
|
import { Stack } from "expo-router";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { Dimensions, Platform, RefreshControl, SafeAreaView, ScrollView, View } from "react-native";
|
import { Platform, RefreshControl, SafeAreaView, ScrollView, View } from "react-native";
|
||||||
import { useSafeAreaInsets } from "react-native-safe-area-context";
|
import { useSafeAreaInsets } from "react-native-safe-area-context";
|
||||||
import { useDispatch, useSelector } from "react-redux";
|
import { useDispatch, useSelector } from "react-redux";
|
||||||
|
|
||||||
@@ -24,7 +23,6 @@ export default function Home() {
|
|||||||
const entities = useSelector((state: any) => state.entities)
|
const entities = useSelector((state: any) => state.entities)
|
||||||
const dispatch = useDispatch()
|
const dispatch = useDispatch()
|
||||||
const { token, decryptToken, signOut } = useAuthSession()
|
const { token, decryptToken, signOut } = useAuthSession()
|
||||||
const { colors } = useTheme();
|
|
||||||
const insets = useSafeAreaInsets()
|
const insets = useSafeAreaInsets()
|
||||||
const [refreshing, setRefreshing] = useState(false)
|
const [refreshing, setRefreshing] = useState(false)
|
||||||
|
|
||||||
@@ -49,13 +47,13 @@ export default function Home() {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={[Styles.flex1, { backgroundColor: colors.background }]}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
title: 'Home',
|
title: 'Home',
|
||||||
headerTitle: entities.village,
|
headerTitle: entities.village,
|
||||||
header: () => (
|
header: () => (
|
||||||
<View style={[Styles.rowItemsCenter, Styles.ph20, Platform.OS === 'ios' ? Styles.pb07 : Styles.pb13, { backgroundColor: colors.header, paddingTop: Platform.OS === 'ios' ? insets.top : 10 }]}>
|
<View style={[Styles.rowItemsCenter, Styles.ph20, Platform.OS === 'ios' ? Styles.pb07 : Styles.pb13, { backgroundColor: '#19345E', paddingTop: Platform.OS === 'ios' ? insets.top : 10 }]}>
|
||||||
<Text style={Styles.textHeaderHome}>{entities.village}</Text>
|
<Text style={Styles.textHeaderHome}>{entities.village}</Text>
|
||||||
<HeaderRightHome />
|
<HeaderRightHome />
|
||||||
</View>
|
</View>
|
||||||
@@ -67,36 +65,19 @@ export default function Home() {
|
|||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={handleRefresh}
|
onRefresh={handleRefresh}
|
||||||
tintColor={colors.icon}
|
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
showsVerticalScrollIndicator={false}
|
showsVerticalScrollIndicator={false}
|
||||||
style={[Styles.h100, { backgroundColor: colors.background }]}
|
|
||||||
>
|
>
|
||||||
<LinearGradient
|
<CaraouselHome refreshing={refreshing}/>
|
||||||
colors={[colors.header, colors.header, colors.header, colors.header, colors.homeGradient]}
|
<View style={[Styles.ph15, Styles.mb100]}>
|
||||||
style={[
|
<FiturHome />
|
||||||
Styles.posAbsolute,
|
<ProjectHome refreshing={refreshing}/>
|
||||||
Styles.zIndexMinus1,
|
<DivisionHome refreshing={refreshing}/>
|
||||||
{
|
<ChartProgresHome refreshing={refreshing}/>
|
||||||
width: Dimensions.get('window').width * 1.5,
|
<ChartDokumenHome refreshing={refreshing}/>
|
||||||
height: Dimensions.get('window').width * 1.5,
|
<EventHome refreshing={refreshing}/>
|
||||||
borderRadius: Dimensions.get('window').width * 0.5,
|
<DisccussionHome refreshing={refreshing}/>
|
||||||
top: -Dimensions.get('window').width * 1, // Positioned to show the bottom part of the circle as an arc
|
|
||||||
left: -Dimensions.get('window').width * 0.25,
|
|
||||||
}
|
|
||||||
]}
|
|
||||||
/>
|
|
||||||
{/* <CaraouselHome refreshing={refreshing} /> */}
|
|
||||||
<View style={[Styles.ph15]}>
|
|
||||||
<CaraouselHome2 refreshing={refreshing} />
|
|
||||||
{/* <FiturHome /> */}
|
|
||||||
<ProjectHome refreshing={refreshing} />
|
|
||||||
<DivisionHome refreshing={refreshing} />
|
|
||||||
<ChartProgresHome refreshing={refreshing} />
|
|
||||||
<ChartDokumenHome refreshing={refreshing} />
|
|
||||||
<EventHome refreshing={refreshing} />
|
|
||||||
<DisccussionHome refreshing={refreshing} />
|
|
||||||
</View>
|
</View>
|
||||||
</ScrollView>
|
</ScrollView>
|
||||||
</SafeAreaView>
|
</SafeAreaView>
|
||||||
|
|||||||
@@ -10,8 +10,6 @@ import { ConstEnv } from "@/constants/ConstEnv";
|
|||||||
import { valueRoleUser } from "@/constants/RoleUser";
|
import { valueRoleUser } from "@/constants/RoleUser";
|
||||||
import Styles from "@/constants/Styles";
|
import Styles from "@/constants/Styles";
|
||||||
import { apiGetProfile } from "@/lib/api";
|
import { apiGetProfile } from "@/lib/api";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { LinearGradient } from "expo-linear-gradient";
|
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
import React, { useEffect, useState } from "react";
|
import React, { useEffect, useState } from "react";
|
||||||
import { Pressable, RefreshControl, SafeAreaView, ScrollView, View } from "react-native";
|
import { Pressable, RefreshControl, SafeAreaView, ScrollView, View } from "react-native";
|
||||||
@@ -35,7 +33,6 @@ type Props = {
|
|||||||
|
|
||||||
export default function MemberDetail() {
|
export default function MemberDetail() {
|
||||||
const { id } = useLocalSearchParams<{ id: string }>();
|
const { id } = useLocalSearchParams<{ id: string }>();
|
||||||
const { colors } = useTheme();
|
|
||||||
const [data, setData] = useState<Props>()
|
const [data, setData] = useState<Props>()
|
||||||
const [errorImg, setErrorImg] = useState(false)
|
const [errorImg, setErrorImg] = useState(false)
|
||||||
const entityUser = useSelector((state: any) => state.user)
|
const entityUser = useSelector((state: any) => state.user)
|
||||||
@@ -55,11 +52,9 @@ export default function MemberDetail() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message })
|
Toast.show({ type: 'small', text1: response.message })
|
||||||
}
|
}
|
||||||
} catch (error : any ) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal mengambil data"
|
Toast.show({ type: 'small', text1: 'Gagal mengambil data' })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
@@ -79,11 +74,13 @@ export default function MemberDetail() {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={[Styles.flex1, { backgroundColor: colors.background }]}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
|
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
||||||
headerTitle: 'Anggota',
|
headerTitle: 'Anggota',
|
||||||
headerTitleAlign: 'center',
|
headerTitleAlign: 'center',
|
||||||
|
// headerRight: () => (entityUser.role != "user") && isEdit ? <HeaderRightMemberDetail active={data?.isActive} id={id} /> : <></>,
|
||||||
header: () => (
|
header: () => (
|
||||||
<AppHeader title="Anggota"
|
<AppHeader title="Anggota"
|
||||||
showBack={true}
|
showBack={true}
|
||||||
@@ -96,19 +93,15 @@ export default function MemberDetail() {
|
|||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<ScrollView
|
<ScrollView
|
||||||
style={[Styles.h100, { backgroundColor: colors.background }]}
|
style={[Styles.h100]}
|
||||||
refreshControl={
|
refreshControl={
|
||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={handleRefresh}
|
onRefresh={handleRefresh}
|
||||||
tintColor={colors.icon}
|
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
<LinearGradient
|
<View style={[Styles.wrapHeadViewMember]}>
|
||||||
colors={[colors.header, colors.homeGradient]}
|
|
||||||
style={[Styles.wrapHeadViewMember]}
|
|
||||||
>
|
|
||||||
{
|
{
|
||||||
loading ?
|
loading ?
|
||||||
<>
|
<>
|
||||||
@@ -121,15 +114,15 @@ export default function MemberDetail() {
|
|||||||
<Pressable onPress={() => setPreview(true)}>
|
<Pressable onPress={() => setPreview(true)}>
|
||||||
<ImageUser src={`${ConstEnv.url_storage}/files/${data?.img}`} size="lg" onError={setErrorImg} />
|
<ImageUser src={`${ConstEnv.url_storage}/files/${data?.img}`} size="lg" onError={setErrorImg} />
|
||||||
</Pressable>
|
</Pressable>
|
||||||
<Text style={[Styles.textSubtitle, Styles.cWhite, Styles.mt10, Styles.textCenter]}>{data?.name}</Text>
|
<Text style={[Styles.textSubtitle, Styles.cWhite, Styles.mt10, { textAlign: 'center' }]}>{data?.name}</Text>
|
||||||
<Text style={[Styles.textMediumNormal, Styles.cWhite]}>{data?.role}</Text>
|
<Text style={[Styles.textMediumNormal, Styles.cWhite]}>{data?.role}</Text>
|
||||||
</>
|
</>
|
||||||
|
|
||||||
}
|
}
|
||||||
</LinearGradient>
|
</View>
|
||||||
<View style={[Styles.p15]}>
|
<View style={[Styles.p15]}>
|
||||||
<View style={[Styles.rowSpaceBetween]}>
|
<View style={[Styles.rowSpaceBetween]}>
|
||||||
<Text style={[Styles.textDefaultSemiBold, { color: colors.text }]}>Informasi</Text>
|
<Text style={[Styles.textDefaultSemiBold]}>Informasi</Text>
|
||||||
<LabelStatus
|
<LabelStatus
|
||||||
size="small"
|
size="small"
|
||||||
category={data?.isActive ? 'success' : 'error'}
|
category={data?.isActive ? 'success' : 'error'}
|
||||||
|
|||||||
@@ -1,7 +1,6 @@
|
|||||||
import AppHeader from "@/components/AppHeader";
|
import AppHeader from "@/components/AppHeader";
|
||||||
import ButtonSaveHeader from "@/components/buttonSaveHeader";
|
import ButtonSaveHeader from "@/components/buttonSaveHeader";
|
||||||
import { InputForm } from "@/components/inputForm";
|
import { InputForm } from "@/components/inputForm";
|
||||||
import LoadingCenter from "@/components/loadingCenter";
|
|
||||||
import ModalSelect from "@/components/modalSelect";
|
import ModalSelect from "@/components/modalSelect";
|
||||||
import SelectForm from "@/components/selectForm";
|
import SelectForm from "@/components/selectForm";
|
||||||
import Text from "@/components/Text";
|
import Text from "@/components/Text";
|
||||||
@@ -11,7 +10,6 @@ import { apiCreateUser } from "@/lib/api";
|
|||||||
import { validateName } from "@/lib/fun_validateName";
|
import { validateName } from "@/lib/fun_validateName";
|
||||||
import { setUpdateMember } from "@/lib/memberSlice";
|
import { setUpdateMember } from "@/lib/memberSlice";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { MaterialCommunityIcons } from "@expo/vector-icons";
|
import { MaterialCommunityIcons } from "@expo/vector-icons";
|
||||||
import { useHeaderHeight } from '@react-navigation/elements';
|
import { useHeaderHeight } from '@react-navigation/elements';
|
||||||
import * as ImagePicker from "expo-image-picker";
|
import * as ImagePicker from "expo-image-picker";
|
||||||
@@ -34,7 +32,6 @@ export default function CreateMember() {
|
|||||||
const dispatch = useDispatch()
|
const dispatch = useDispatch()
|
||||||
const update = useSelector((state: any) => state.memberUpdate)
|
const update = useSelector((state: any) => state.memberUpdate)
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
const { colors } = useTheme();
|
|
||||||
const [valSelect, setValSelect] = useState<"group" | "position" | "role" | "gender">("group");
|
const [valSelect, setValSelect] = useState<"group" | "position" | "role" | "gender">("group");
|
||||||
const [chooseGroup, setChooseGroup] = useState({ val: "", label: "" });
|
const [chooseGroup, setChooseGroup] = useState({ val: "", label: "" });
|
||||||
const [choosePosition, setChoosePosition] = useState({ val: "", label: "" });
|
const [choosePosition, setChoosePosition] = useState({ val: "", label: "" });
|
||||||
@@ -185,11 +182,9 @@ export default function CreateMember() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error : any ) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal menambahkan data"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
@@ -211,11 +206,25 @@ export default function CreateMember() {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={[Styles.flex1, { backgroundColor: colors.background }]}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
|
// headerLeft: () => (
|
||||||
|
// <ButtonBackHeader
|
||||||
|
// onPress={() => {
|
||||||
|
// router.back();
|
||||||
|
// }}
|
||||||
|
// />
|
||||||
|
// ),
|
||||||
headerTitle: "Tambah Anggota",
|
headerTitle: "Tambah Anggota",
|
||||||
headerTitleAlign: "center",
|
headerTitleAlign: "center",
|
||||||
|
// headerRight: () => (
|
||||||
|
// <ButtonSaveHeader
|
||||||
|
// disable={disableBtn || loading}
|
||||||
|
// category="create"
|
||||||
|
// onPress={() => { handleCreate() }}
|
||||||
|
// />
|
||||||
|
// ),
|
||||||
header: () => (
|
header: () => (
|
||||||
<AppHeader title="Anggota"
|
<AppHeader title="Anggota"
|
||||||
showBack={true}
|
showBack={true}
|
||||||
@@ -231,15 +240,14 @@ export default function CreateMember() {
|
|||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
{loading && <LoadingCenter />}
|
|
||||||
<KeyboardAvoidingView
|
<KeyboardAvoidingView
|
||||||
style={[Styles.h100, { backgroundColor: colors.background }]}
|
style={[Styles.h100]}
|
||||||
behavior={Platform.OS === 'ios' ? 'padding' : undefined}
|
behavior={Platform.OS === 'ios' ? 'padding' : undefined}
|
||||||
keyboardVerticalOffset={headerHeight}
|
keyboardVerticalOffset={headerHeight}
|
||||||
>
|
>
|
||||||
<ScrollView showsVerticalScrollIndicator={false}>
|
<ScrollView showsVerticalScrollIndicator={false}>
|
||||||
<View style={[Styles.p15]}>
|
<View style={[Styles.p15]}>
|
||||||
<View style={[Styles.contentItemCenter]}>
|
<View style={{ justifyContent: "center", alignItems: "center" }}>
|
||||||
{selectedImage != undefined ? (
|
{selectedImage != undefined ? (
|
||||||
<Pressable onPress={pickImageAsync}>
|
<Pressable onPress={pickImageAsync}>
|
||||||
<Image src={selectedImage} style={[Styles.userProfileBig]} />
|
<Image src={selectedImage} style={[Styles.userProfileBig]} />
|
||||||
@@ -263,7 +271,6 @@ export default function CreateMember() {
|
|||||||
placeholder="Pilih Lembaga Desa"
|
placeholder="Pilih Lembaga Desa"
|
||||||
value={chooseGroup.label}
|
value={chooseGroup.label}
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
setValChoose(chooseGroup.val);
|
setValChoose(chooseGroup.val);
|
||||||
setValSelect("group");
|
setValSelect("group");
|
||||||
@@ -278,7 +285,6 @@ export default function CreateMember() {
|
|||||||
placeholder="Pilih Jabatan"
|
placeholder="Pilih Jabatan"
|
||||||
value={choosePosition.label}
|
value={choosePosition.label}
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
setValChoose(choosePosition.val);
|
setValChoose(choosePosition.val);
|
||||||
setValSelect("position");
|
setValSelect("position");
|
||||||
@@ -292,7 +298,6 @@ export default function CreateMember() {
|
|||||||
placeholder="Pilih Role"
|
placeholder="Pilih Role"
|
||||||
value={chooseRole.label}
|
value={chooseRole.label}
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
setValChoose(chooseRole.val);
|
setValChoose(chooseRole.val);
|
||||||
setValSelect("role");
|
setValSelect("role");
|
||||||
@@ -306,7 +311,6 @@ export default function CreateMember() {
|
|||||||
type="numeric"
|
type="numeric"
|
||||||
placeholder="NIK"
|
placeholder="NIK"
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
|
||||||
error={error.nik}
|
error={error.nik}
|
||||||
errorText="NIK Harus 16 Karakter"
|
errorText="NIK Harus 16 Karakter"
|
||||||
onChange={val => {
|
onChange={val => {
|
||||||
@@ -318,7 +322,6 @@ export default function CreateMember() {
|
|||||||
type="default"
|
type="default"
|
||||||
placeholder="Nama"
|
placeholder="Nama"
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
|
||||||
error={error.name}
|
error={error.name}
|
||||||
errorText="Nama harus 3–50 karakter (huruf, angka, spasi, dan simbol ringan (. , ' _ -))"
|
errorText="Nama harus 3–50 karakter (huruf, angka, spasi, dan simbol ringan (. , ' _ -))"
|
||||||
onChange={val => {
|
onChange={val => {
|
||||||
@@ -330,7 +333,6 @@ export default function CreateMember() {
|
|||||||
type="default"
|
type="default"
|
||||||
placeholder="Email"
|
placeholder="Email"
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
|
||||||
error={error.email}
|
error={error.email}
|
||||||
errorText="Email tidak valid"
|
errorText="Email tidak valid"
|
||||||
onChange={val => {
|
onChange={val => {
|
||||||
@@ -342,8 +344,7 @@ export default function CreateMember() {
|
|||||||
type="numeric"
|
type="numeric"
|
||||||
placeholder="8XX-XXX-XXX"
|
placeholder="8XX-XXX-XXX"
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
itemLeft={<Text style={[Platform.OS === 'ios' && Styles.mt02]}>+62</Text>}
|
||||||
itemLeft={<Text style={[Platform.OS === 'ios' && Styles.mt02, { color: colors.text }]}>+62</Text>}
|
|
||||||
error={error.phone}
|
error={error.phone}
|
||||||
errorText="Nomor Telepon tidak valid"
|
errorText="Nomor Telepon tidak valid"
|
||||||
onChange={val => {
|
onChange={val => {
|
||||||
@@ -355,7 +356,6 @@ export default function CreateMember() {
|
|||||||
placeholder="Pilih Jenis Kelamin"
|
placeholder="Pilih Jenis Kelamin"
|
||||||
value={chooseGender.label}
|
value={chooseGender.label}
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
setValChoose(chooseGender.val);
|
setValChoose(chooseGender.val);
|
||||||
setValSelect("gender");
|
setValSelect("gender");
|
||||||
|
|||||||
@@ -1,7 +1,6 @@
|
|||||||
import AppHeader from "@/components/AppHeader";
|
import AppHeader from "@/components/AppHeader";
|
||||||
import ButtonSaveHeader from "@/components/buttonSaveHeader";
|
import ButtonSaveHeader from "@/components/buttonSaveHeader";
|
||||||
import { InputForm } from "@/components/inputForm";
|
import { InputForm } from "@/components/inputForm";
|
||||||
import LoadingCenter from "@/components/loadingCenter";
|
|
||||||
import ModalSelect from "@/components/modalSelect";
|
import ModalSelect from "@/components/modalSelect";
|
||||||
import SelectForm from "@/components/selectForm";
|
import SelectForm from "@/components/selectForm";
|
||||||
import Text from "@/components/Text";
|
import Text from "@/components/Text";
|
||||||
@@ -11,7 +10,6 @@ import { apiEditUser, apiGetProfile } from "@/lib/api";
|
|||||||
import { validateName } from "@/lib/fun_validateName";
|
import { validateName } from "@/lib/fun_validateName";
|
||||||
import { setUpdateMember } from "@/lib/memberSlice";
|
import { setUpdateMember } from "@/lib/memberSlice";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { MaterialCommunityIcons } from "@expo/vector-icons";
|
import { MaterialCommunityIcons } from "@expo/vector-icons";
|
||||||
import { useHeaderHeight } from '@react-navigation/elements';
|
import { useHeaderHeight } from '@react-navigation/elements';
|
||||||
import * as ImagePicker from "expo-image-picker";
|
import * as ImagePicker from "expo-image-picker";
|
||||||
@@ -49,7 +47,6 @@ export default function EditMember() {
|
|||||||
const update = useSelector((state: any) => state.memberUpdate)
|
const update = useSelector((state: any) => state.memberUpdate)
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
const { id } = useLocalSearchParams<{ id: string }>();
|
const { id } = useLocalSearchParams<{ id: string }>();
|
||||||
const { colors } = useTheme();
|
|
||||||
const [errorImg, setErrorImg] = useState(false)
|
const [errorImg, setErrorImg] = useState(false)
|
||||||
const [selectedImage, setSelectedImage] = useState<string | undefined | { uri: string }>(undefined);
|
const [selectedImage, setSelectedImage] = useState<string | undefined | { uri: string }>(undefined);
|
||||||
const [choosePosition, setChoosePosition] = useState({ val: "", label: "" });
|
const [choosePosition, setChoosePosition] = useState({ val: "", label: "" });
|
||||||
@@ -211,11 +208,9 @@ export default function EditMember() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error : any ) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal menambahkan data"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
@@ -241,11 +236,27 @@ export default function EditMember() {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={[Styles.flex1, { backgroundColor: colors.background }]}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
|
// headerLeft: () => (
|
||||||
|
// <ButtonBackHeader
|
||||||
|
// onPress={() => {
|
||||||
|
// router.back();
|
||||||
|
// }}
|
||||||
|
// />
|
||||||
|
// ),
|
||||||
headerTitle: "Edit Anggota",
|
headerTitle: "Edit Anggota",
|
||||||
headerTitleAlign: "center",
|
headerTitleAlign: "center",
|
||||||
|
// headerRight: () => (
|
||||||
|
// <ButtonSaveHeader
|
||||||
|
// disable={disableBtn || loading}
|
||||||
|
// category="update"
|
||||||
|
// onPress={() => {
|
||||||
|
// handleEdit()
|
||||||
|
// }}
|
||||||
|
// />
|
||||||
|
// ),
|
||||||
header: () => (
|
header: () => (
|
||||||
<AppHeader
|
<AppHeader
|
||||||
title="Edit Anggota"
|
title="Edit Anggota"
|
||||||
@@ -264,15 +275,15 @@ export default function EditMember() {
|
|||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
{loading && <LoadingCenter />}
|
|
||||||
<KeyboardAvoidingView
|
<KeyboardAvoidingView
|
||||||
style={[Styles.h100, { backgroundColor: colors.background }]}
|
style={[Styles.h100]}
|
||||||
behavior={Platform.OS === 'ios' ? 'padding' : undefined}
|
behavior={Platform.OS === 'ios' ? 'padding' : undefined}
|
||||||
keyboardVerticalOffset={headerHeight}
|
keyboardVerticalOffset={headerHeight}
|
||||||
>
|
>
|
||||||
<ScrollView showsVerticalScrollIndicator={false} style={[Styles.h100]}>
|
<ScrollView showsVerticalScrollIndicator={false} style={[Styles.h100]}>
|
||||||
<View style={[Styles.p15]}>
|
<View style={[Styles.p15]}>
|
||||||
<View style={[Styles.contentItemCenter]}>
|
<View style={{ justifyContent: "center", alignItems: "center" }}>
|
||||||
{
|
{
|
||||||
errorImg ?
|
errorImg ?
|
||||||
<Pressable onPress={pickImageAsync}>
|
<Pressable onPress={pickImageAsync}>
|
||||||
@@ -314,7 +325,6 @@ export default function EditMember() {
|
|||||||
placeholder="Pilih Jabatan"
|
placeholder="Pilih Jabatan"
|
||||||
value={choosePosition.label}
|
value={choosePosition.label}
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
setValChoose(choosePosition.val);
|
setValChoose(choosePosition.val);
|
||||||
setValSelect("position");
|
setValSelect("position");
|
||||||
@@ -328,7 +338,6 @@ export default function EditMember() {
|
|||||||
placeholder="Pilih Role"
|
placeholder="Pilih Role"
|
||||||
value={chooseRole.label}
|
value={chooseRole.label}
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
setValChoose(chooseRole.val);
|
setValChoose(chooseRole.val);
|
||||||
setValSelect("role");
|
setValSelect("role");
|
||||||
@@ -343,7 +352,6 @@ export default function EditMember() {
|
|||||||
placeholder="NIK"
|
placeholder="NIK"
|
||||||
required
|
required
|
||||||
value={data?.nik}
|
value={data?.nik}
|
||||||
bg={colors.card}
|
|
||||||
error={error.nik}
|
error={error.nik}
|
||||||
errorText="NIK Harus 16 Karakter"
|
errorText="NIK Harus 16 Karakter"
|
||||||
onChange={val => {
|
onChange={val => {
|
||||||
@@ -356,7 +364,6 @@ export default function EditMember() {
|
|||||||
placeholder="Nama"
|
placeholder="Nama"
|
||||||
required
|
required
|
||||||
value={data?.name}
|
value={data?.name}
|
||||||
bg={colors.card}
|
|
||||||
error={error.name}
|
error={error.name}
|
||||||
errorText="Nama harus 3–50 karakter (huruf, angka, spasi, dan simbol ringan (. , ' _ -))"
|
errorText="Nama harus 3–50 karakter (huruf, angka, spasi, dan simbol ringan (. , ' _ -))"
|
||||||
onChange={val => {
|
onChange={val => {
|
||||||
@@ -369,7 +376,6 @@ export default function EditMember() {
|
|||||||
placeholder="Email"
|
placeholder="Email"
|
||||||
required
|
required
|
||||||
value={data?.email}
|
value={data?.email}
|
||||||
bg={colors.card}
|
|
||||||
error={error.email}
|
error={error.email}
|
||||||
errorText="Email tidak valid"
|
errorText="Email tidak valid"
|
||||||
onChange={val => {
|
onChange={val => {
|
||||||
@@ -381,9 +387,8 @@ export default function EditMember() {
|
|||||||
type="numeric"
|
type="numeric"
|
||||||
placeholder="8XX-XXX-XXX"
|
placeholder="8XX-XXX-XXX"
|
||||||
required
|
required
|
||||||
itemLeft={<Text style={[Platform.OS === 'ios' && Styles.mt02, { color: colors.text }]}>+62</Text>}
|
itemLeft={<Text style={[Platform.OS === 'ios' && Styles.mt02]}>+62</Text>}
|
||||||
value={data?.phone}
|
value={data?.phone}
|
||||||
bg={colors.card}
|
|
||||||
error={error.phone}
|
error={error.phone}
|
||||||
errorText="Nomor Telepon tidak valid"
|
errorText="Nomor Telepon tidak valid"
|
||||||
onChange={val => {
|
onChange={val => {
|
||||||
@@ -395,7 +400,6 @@ export default function EditMember() {
|
|||||||
placeholder="Pilih Jenis Kelamin"
|
placeholder="Pilih Jenis Kelamin"
|
||||||
value={chooseGender.label}
|
value={chooseGender.label}
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
setValChoose(chooseGender.val);
|
setValChoose(chooseGender.val);
|
||||||
setValSelect("gender");
|
setValSelect("gender");
|
||||||
|
|||||||
@@ -5,12 +5,10 @@ import InputSearch from "@/components/inputSearch";
|
|||||||
import LabelStatus from "@/components/labelStatus";
|
import LabelStatus from "@/components/labelStatus";
|
||||||
import SkeletonTwoItem from "@/components/skeletonTwoItem";
|
import SkeletonTwoItem from "@/components/skeletonTwoItem";
|
||||||
import Text from "@/components/Text";
|
import Text from "@/components/Text";
|
||||||
import WrapTab from "@/components/wrapTab";
|
|
||||||
import { ConstEnv } from "@/constants/ConstEnv";
|
import { ConstEnv } from "@/constants/ConstEnv";
|
||||||
import Styles from "@/constants/Styles";
|
import Styles from "@/constants/Styles";
|
||||||
import { apiGetUser } from "@/lib/api";
|
import { apiGetUser } from "@/lib/api";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { AntDesign, Feather } from "@expo/vector-icons";
|
import { AntDesign, Feather } from "@expo/vector-icons";
|
||||||
import { router, useLocalSearchParams } from "expo-router";
|
import { router, useLocalSearchParams } from "expo-router";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
@@ -35,7 +33,6 @@ export default function Index() {
|
|||||||
const { active, group } = useLocalSearchParams<{ active?: string, group?: string }>()
|
const { active, group } = useLocalSearchParams<{ active?: string, group?: string }>()
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
const entityUser = useSelector((state: any) => state.user)
|
const entityUser = useSelector((state: any) => state.user)
|
||||||
const { colors } = useTheme();
|
|
||||||
const [search, setSearch] = useState('')
|
const [search, setSearch] = useState('')
|
||||||
const [nameGroup, setNameGroup] = useState('')
|
const [nameGroup, setNameGroup] = useState('')
|
||||||
const [data, setData] = useState<Props[]>([])
|
const [data, setData] = useState<Props[]>([])
|
||||||
@@ -107,24 +104,24 @@ export default function Index() {
|
|||||||
});
|
});
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<View style={[Styles.p15, { flex: 1, backgroundColor: colors.background }]}>
|
<View style={[Styles.p15, { flex: 1 }]}>
|
||||||
<View>
|
<View>
|
||||||
<WrapTab>
|
<View style={[Styles.wrapBtnTab]}>
|
||||||
<ButtonTab
|
<ButtonTab
|
||||||
active={status == "false" ? "false" : "true"}
|
active={status == "false" ? "false" : "true"}
|
||||||
value="true"
|
value="true"
|
||||||
onPress={() => { setStatus("true") }}
|
onPress={() => { setStatus("true") }}
|
||||||
label="Aktif"
|
label="Aktif"
|
||||||
icon={<Feather name="check-circle" color={status == "false" ? colors.dimmed : 'white'} size={20} />}
|
icon={<Feather name="check-circle" color={status == "false" ? 'black' : 'white'} size={20} />}
|
||||||
n={2} />
|
n={2} />
|
||||||
<ButtonTab
|
<ButtonTab
|
||||||
active={status == "false" ? "false" : "true"}
|
active={status == "false" ? "false" : "true"}
|
||||||
value="false"
|
value="false"
|
||||||
onPress={() => { setStatus("false") }}
|
onPress={() => { setStatus("false") }}
|
||||||
label="Tidak Aktif"
|
label="Tidak Aktif"
|
||||||
icon={<AntDesign name="closecircleo" color={status == "false" ? 'white' : colors.dimmed} size={20} />}
|
icon={<AntDesign name="closecircleo" color={status == "false" ? 'white' : 'black'} size={20} />}
|
||||||
n={2} />
|
n={2} />
|
||||||
</WrapTab>
|
</View>
|
||||||
<InputSearch onChange={setSearch} />
|
<InputSearch onChange={setSearch} />
|
||||||
{
|
{
|
||||||
(entityUser.role == "supadmin" || entityUser.role == "developer") &&
|
(entityUser.role == "supadmin" || entityUser.role == "developer") &&
|
||||||
@@ -134,7 +131,7 @@ export default function Index() {
|
|||||||
</View>
|
</View>
|
||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
<View style={[{ flex: 2 }, Styles.mt10]}>
|
<View style={[{ flex: 2 }, Styles.mt05]}>
|
||||||
{
|
{
|
||||||
loading ?
|
loading ?
|
||||||
arrSkeleton.map((item, index) => {
|
arrSkeleton.map((item, index) => {
|
||||||
@@ -171,12 +168,11 @@ export default function Index() {
|
|||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={handleRefresh}
|
onRefresh={handleRefresh}
|
||||||
tintColor={colors.icon}
|
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
:
|
:
|
||||||
<Text style={[Styles.textDefault, { textAlign: 'center', color: colors.dimmed }]}>Tidak ada data</Text>
|
<Text style={[Styles.textDefault, Styles.cGray, { textAlign: 'center' }]}>Tidak ada data</Text>
|
||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
import BorderBottomItem from "@/components/borderBottomItem";
|
import BorderBottomItem from "@/components/borderBottomItem";
|
||||||
import BorderBottomItemVertical from "@/components/borderBottomItemVertical";
|
|
||||||
import SkeletonTwoItem from "@/components/skeletonTwoItem";
|
import SkeletonTwoItem from "@/components/skeletonTwoItem";
|
||||||
import Text from "@/components/Text";
|
import Text from "@/components/Text";
|
||||||
import { ColorsStatus } from "@/constants/ColorsStatus";
|
import { ColorsStatus } from "@/constants/ColorsStatus";
|
||||||
@@ -8,7 +7,6 @@ import { apiGetNotification, apiReadOneNotification } from "@/lib/api";
|
|||||||
import { setUpdateNotification } from "@/lib/notificationSlice";
|
import { setUpdateNotification } from "@/lib/notificationSlice";
|
||||||
import { pushToPage } from "@/lib/pushToPage";
|
import { pushToPage } from "@/lib/pushToPage";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { Feather } from "@expo/vector-icons";
|
import { Feather } from "@expo/vector-icons";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { RefreshControl, SafeAreaView, View, VirtualizedList } from "react-native";
|
import { RefreshControl, SafeAreaView, View, VirtualizedList } from "react-native";
|
||||||
@@ -26,7 +24,6 @@ type Props = {
|
|||||||
|
|
||||||
export default function Notification() {
|
export default function Notification() {
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
const { colors } = useTheme();
|
|
||||||
const [loading, setLoading] = useState(false)
|
const [loading, setLoading] = useState(false)
|
||||||
const [data, setData] = useState<Props[]>([])
|
const [data, setData] = useState<Props[]>([])
|
||||||
const [page, setPage] = useState(1)
|
const [page, setPage] = useState(1)
|
||||||
@@ -100,7 +97,7 @@ export default function Notification() {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={[Styles.flex1, { backgroundColor: colors.background }]}>
|
<SafeAreaView>
|
||||||
<View style={[Styles.p15]}>
|
<View style={[Styles.p15]}>
|
||||||
{
|
{
|
||||||
loading ?
|
loading ?
|
||||||
@@ -117,21 +114,21 @@ export default function Notification() {
|
|||||||
getItem={getItem}
|
getItem={getItem}
|
||||||
renderItem={({ item, index }: { item: Props, index: number }) => {
|
renderItem={({ item, index }: { item: Props, index: number }) => {
|
||||||
return (
|
return (
|
||||||
<BorderBottomItemVertical
|
<BorderBottomItem
|
||||||
borderType="bottom"
|
borderType="bottom"
|
||||||
icon={
|
icon={
|
||||||
<View style={[Styles.iconContent, item.isRead && ColorsStatus.secondary]}>
|
<View style={[Styles.iconContent, item.isRead ? ColorsStatus.secondary : ColorsStatus.primary]}>
|
||||||
<Feather name="bell" size={25} color="black" />
|
<Feather name="bell" size={25} color="white" />
|
||||||
</View>
|
</View>
|
||||||
}
|
}
|
||||||
title={item.title}
|
title={item.title}
|
||||||
rightTopInfo={item.createdAt}
|
rightTopInfo={item.createdAt}
|
||||||
desc={item.desc}
|
desc={item.desc}
|
||||||
textColor={item.isRead ? 'gray' : colors.text}
|
textColor={item.isRead ? 'gray' : 'black'}
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
handleReadNotification(item.id, item.category, item.idContent)
|
handleReadNotification(item.id, item.category, item.idContent)
|
||||||
|
|
||||||
}}
|
}}
|
||||||
bgColor={'transparent'}
|
|
||||||
/>
|
/>
|
||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
@@ -143,12 +140,11 @@ export default function Notification() {
|
|||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={handleRefresh}
|
onRefresh={handleRefresh}
|
||||||
tintColor={colors.icon}
|
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
:
|
:
|
||||||
<Text style={[Styles.textDefault, Styles.textCenter, { color: colors.dimmed }]}>Tidak ada data</Text>
|
<Text style={[Styles.textDefault, Styles.cGray, { textAlign: 'center' }]}>Tidak ada data</Text>
|
||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
</SafeAreaView>
|
</SafeAreaView>
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
import AlertKonfirmasi from "@/components/alertKonfirmasi";
|
||||||
import BorderBottomItem from "@/components/borderBottomItem";
|
import BorderBottomItem from "@/components/borderBottomItem";
|
||||||
import { ButtonForm } from "@/components/buttonForm";
|
import { ButtonForm } from "@/components/buttonForm";
|
||||||
import ButtonTab from "@/components/buttonTab";
|
import ButtonTab from "@/components/buttonTab";
|
||||||
@@ -6,15 +7,13 @@ import { InputForm } from "@/components/inputForm";
|
|||||||
import InputSearch from "@/components/inputSearch";
|
import InputSearch from "@/components/inputSearch";
|
||||||
import LabelStatus from "@/components/labelStatus";
|
import LabelStatus from "@/components/labelStatus";
|
||||||
import MenuItemRow from "@/components/menuItemRow";
|
import MenuItemRow from "@/components/menuItemRow";
|
||||||
import ModalConfirmation from "@/components/ModalConfirmation";
|
|
||||||
import SkeletonTwoItem from "@/components/skeletonTwoItem";
|
import SkeletonTwoItem from "@/components/skeletonTwoItem";
|
||||||
import Text from "@/components/Text";
|
import Text from "@/components/Text";
|
||||||
import WrapTab from "@/components/wrapTab";
|
import { ColorsStatus } from "@/constants/ColorsStatus";
|
||||||
import Styles from "@/constants/Styles";
|
import Styles from "@/constants/Styles";
|
||||||
import { apiDeletePosition, apiEditPosition, apiGetPosition } from "@/lib/api";
|
import { apiDeletePosition, apiEditPosition, apiGetPosition } from "@/lib/api";
|
||||||
import { setUpdatePosition } from "@/lib/positionSlice";
|
import { setUpdatePosition } from "@/lib/positionSlice";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { AntDesign, Feather, MaterialCommunityIcons } from "@expo/vector-icons";
|
import { AntDesign, Feather, MaterialCommunityIcons } from "@expo/vector-icons";
|
||||||
import { useLocalSearchParams } from "expo-router";
|
import { useLocalSearchParams } from "expo-router";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
@@ -34,7 +33,6 @@ export default function Index() {
|
|||||||
const arrSkeleton = Array.from({ length: 5 }, (_, index) => index)
|
const arrSkeleton = Array.from({ length: 5 }, (_, index) => index)
|
||||||
const [loading, setLoading] = useState(true)
|
const [loading, setLoading] = useState(true)
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
const { colors } = useTheme()
|
|
||||||
const [status, setStatus] = useState<'true' | 'false'>('true')
|
const [status, setStatus] = useState<'true' | 'false'>('true')
|
||||||
const entityUser = useSelector((state: any) => state.user)
|
const entityUser = useSelector((state: any) => state.user)
|
||||||
const { active, group } = useLocalSearchParams<{ active?: string, group?: string }>()
|
const { active, group } = useLocalSearchParams<{ active?: string, group?: string }>()
|
||||||
@@ -49,7 +47,6 @@ export default function Index() {
|
|||||||
name: false,
|
name: false,
|
||||||
});
|
});
|
||||||
const [refreshing, setRefreshing] = useState(false)
|
const [refreshing, setRefreshing] = useState(false)
|
||||||
const [showDeleteModal, setShowDeleteModal] = useState(false)
|
|
||||||
|
|
||||||
const dispatch = useDispatch()
|
const dispatch = useDispatch()
|
||||||
const update = useSelector((state: any) => state.positionUpdate)
|
const update = useSelector((state: any) => state.positionUpdate)
|
||||||
@@ -88,11 +85,8 @@ export default function Index() {
|
|||||||
const hasil = await decryptToken(String(token?.current))
|
const hasil = await decryptToken(String(token?.current))
|
||||||
const response = await apiDeletePosition({ user: hasil, isActive: chooseData.active }, chooseData.id)
|
const response = await apiDeletePosition({ user: hasil, isActive: chooseData.active }, chooseData.id)
|
||||||
dispatch(setUpdatePosition(!update))
|
dispatch(setUpdatePosition(!update))
|
||||||
} catch (error : any ) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal menghapus data"
|
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setModal(false)
|
setModal(false)
|
||||||
Toast.show({ type: 'small', text1: 'Berhasil mengupdate data', })
|
Toast.show({ type: 'small', text1: 'Berhasil mengupdate data', })
|
||||||
@@ -110,11 +104,8 @@ export default function Index() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error : any ) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal mengubah data"
|
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoadingSubmit(false)
|
setLoadingSubmit(false)
|
||||||
setVisibleEdit(false)
|
setVisibleEdit(false)
|
||||||
@@ -155,24 +146,24 @@ export default function Index() {
|
|||||||
});
|
});
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<View style={[Styles.p15, Styles.flex1, { backgroundColor: colors.background }]}>
|
<View style={[Styles.p15, { flex: 1 }]}>
|
||||||
<View>
|
<View>
|
||||||
<WrapTab>
|
<View style={[Styles.wrapBtnTab]}>
|
||||||
<ButtonTab
|
<ButtonTab
|
||||||
active={status == "false" ? "false" : "true"}
|
active={status == "false" ? "false" : "true"}
|
||||||
value="true"
|
value="true"
|
||||||
onPress={() => { setStatus("true") }}
|
onPress={() => { setStatus("true") }}
|
||||||
label="Aktif"
|
label="Aktif"
|
||||||
icon={<Feather name="check-circle" color={status == "true" ? 'white' : colors.dimmed} size={20} />}
|
icon={<Feather name="check-circle" color={status == "true" ? 'white' : 'black'} size={20} />}
|
||||||
n={2} />
|
n={2} />
|
||||||
<ButtonTab
|
<ButtonTab
|
||||||
active={status == "false" ? "false" : "true"}
|
active={status == "false" ? "false" : "true"}
|
||||||
value="false"
|
value="false"
|
||||||
onPress={() => { setStatus("false") }}
|
onPress={() => { setStatus("false") }}
|
||||||
label="Tidak Aktif"
|
label="Tidak Aktif"
|
||||||
icon={<AntDesign name="closecircleo" color={status == "false" ? 'white' : colors.dimmed} size={20} />}
|
icon={<AntDesign name="closecircleo" color={status == "false" ? 'white' : 'black'} size={20} />}
|
||||||
n={2} />
|
n={2} />
|
||||||
</WrapTab>
|
</View>
|
||||||
<InputSearch onChange={setSearch} />
|
<InputSearch onChange={setSearch} />
|
||||||
{
|
{
|
||||||
(entityUser.role == "supadmin" || entityUser.role == "developer") &&
|
(entityUser.role == "supadmin" || entityUser.role == "developer") &&
|
||||||
@@ -182,7 +173,7 @@ export default function Index() {
|
|||||||
</View>
|
</View>
|
||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
<View style={[Styles.flex2, Styles.mt10]}>
|
<View style={[{ flex: 2 }, Styles.mt05]}>
|
||||||
{
|
{
|
||||||
loading ?
|
loading ?
|
||||||
arrSkeleton.map((item, index) => {
|
arrSkeleton.map((item, index) => {
|
||||||
@@ -206,8 +197,8 @@ export default function Index() {
|
|||||||
}}
|
}}
|
||||||
borderType="all"
|
borderType="all"
|
||||||
icon={
|
icon={
|
||||||
<View style={[Styles.iconContent]}>
|
<View style={[Styles.iconContent, ColorsStatus.lightGreen]}>
|
||||||
<MaterialCommunityIcons name="account-tie-outline" size={25} color={'black'} />
|
<MaterialCommunityIcons name="account-tie" size={25} color={'#384288'} />
|
||||||
</View>
|
</View>
|
||||||
}
|
}
|
||||||
title={item.name}
|
title={item.name}
|
||||||
@@ -221,28 +212,29 @@ export default function Index() {
|
|||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={handleRefresh}
|
onRefresh={handleRefresh}
|
||||||
tintColor={colors.icon}
|
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
:
|
:
|
||||||
<Text style={[Styles.textDefault, Styles.textCenter, { color: colors.dimmed }]}>Tidak ada data</Text>
|
<Text style={[Styles.textDefault, Styles.cGray, { textAlign: 'center' }]}>Tidak ada data</Text>
|
||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
<DrawerBottom animation="slide" isVisible={isModal} setVisible={() => setModal(false)} title={chooseData.name}>
|
<DrawerBottom animation="slide" isVisible={isModal} setVisible={() => setModal(false)} title={chooseData.name}>
|
||||||
<View style={Styles.rowItemsCenter}>
|
<View style={Styles.rowItemsCenter}>
|
||||||
<MenuItemRow
|
<MenuItemRow
|
||||||
icon={<MaterialCommunityIcons name="toggle-switch-off-outline" color={colors.text} size={25} />}
|
icon={<MaterialCommunityIcons name="toggle-switch-off-outline" color="black" size={25} />}
|
||||||
title={chooseData.active ? 'Non Aktifkan' : "Aktifkan"}
|
title={chooseData.active ? 'Non Aktifkan' : "Aktifkan"}
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
setModal(false)
|
setModal(false)
|
||||||
setTimeout(() => {
|
AlertKonfirmasi({
|
||||||
setShowDeleteModal(true)
|
title: 'Konfirmasi',
|
||||||
}, 600)
|
desc: chooseData.active ? 'Apakah anda yakin ingin menonaktifkan data?' : 'Apakah anda yakin ingin mengaktifkan data?',
|
||||||
|
onPress: () => { handleDelete() }
|
||||||
|
})
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<MenuItemRow
|
<MenuItemRow
|
||||||
icon={<MaterialCommunityIcons name="pencil-outline" color={colors.text} size={25} />}
|
icon={<MaterialCommunityIcons name="pencil-outline" color="black" size={25} />}
|
||||||
title="Edit"
|
title="Edit"
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
setModal(false)
|
setModal(false)
|
||||||
@@ -256,14 +248,13 @@ export default function Index() {
|
|||||||
|
|
||||||
|
|
||||||
<DrawerBottom animation="none" keyboard height={30} backdropPressable={false} isVisible={isVisibleEdit} setVisible={() => setVisibleEdit(false)} title="Edit Jabatan">
|
<DrawerBottom animation="none" keyboard height={30} backdropPressable={false} isVisible={isVisibleEdit} setVisible={() => setVisibleEdit(false)} title="Edit Jabatan">
|
||||||
<View style={[Styles.justifySpaceBetween, Styles.flex1]}>
|
<View style={{ justifyContent: 'space-between', flex: 1 }}>
|
||||||
<View>
|
<View>
|
||||||
<InputForm
|
<InputForm
|
||||||
type="default"
|
type="default"
|
||||||
placeholder="Nama Jabatan"
|
placeholder="Nama Jabatan"
|
||||||
required
|
required
|
||||||
label="Jabatan"
|
label="Jabatan"
|
||||||
bg={"transparent"}
|
|
||||||
value={chooseData.name}
|
value={chooseData.name}
|
||||||
onChange={(val) => { validationForm(val) }}
|
onChange={(val) => { validationForm(val) }}
|
||||||
error={error.name}
|
error={error.name}
|
||||||
@@ -275,19 +266,6 @@ export default function Index() {
|
|||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
</DrawerBottom>
|
</DrawerBottom>
|
||||||
|
|
||||||
<ModalConfirmation
|
|
||||||
visible={showDeleteModal}
|
|
||||||
title="Konfirmasi"
|
|
||||||
message={chooseData.active ? 'Apakah anda yakin ingin menonaktifkan data?' : 'Apakah anda yakin ingin mengaktifkan data?'}
|
|
||||||
onConfirm={() => {
|
|
||||||
setShowDeleteModal(false)
|
|
||||||
handleDelete()
|
|
||||||
}}
|
|
||||||
onCancel={() => setShowDeleteModal(false)}
|
|
||||||
confirmText={chooseData.active ? "Nonaktifkan" : "Aktifkan"}
|
|
||||||
cancelText="Batal"
|
|
||||||
/>
|
|
||||||
</View>
|
</View>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@@ -1,3 +1,4 @@
|
|||||||
|
import AlertKonfirmasi from "@/components/alertKonfirmasi";
|
||||||
import AppHeader from "@/components/AppHeader";
|
import AppHeader from "@/components/AppHeader";
|
||||||
import { ButtonHeader } from "@/components/buttonHeader";
|
import { ButtonHeader } from "@/components/buttonHeader";
|
||||||
import ItemDetailMember from "@/components/itemDetailMember";
|
import ItemDetailMember from "@/components/itemDetailMember";
|
||||||
@@ -5,45 +6,23 @@ import Text from "@/components/Text";
|
|||||||
import { assetUserImage } from "@/constants/AssetsError";
|
import { assetUserImage } from "@/constants/AssetsError";
|
||||||
import { ConstEnv } from "@/constants/ConstEnv";
|
import { ConstEnv } from "@/constants/ConstEnv";
|
||||||
import Styles from "@/constants/Styles";
|
import Styles from "@/constants/Styles";
|
||||||
import { apiGetProfile } from "@/lib/api";
|
|
||||||
import { setEntities } from "@/lib/entitiesSlice";
|
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
import { AntDesign } from "@expo/vector-icons";
|
||||||
import { Feather } from "@expo/vector-icons";
|
|
||||||
import { LinearGradient } from "expo-linear-gradient";
|
|
||||||
import { router, Stack } from "expo-router";
|
import { router, Stack } from "expo-router";
|
||||||
import { useState } from "react";
|
import { useState } from "react";
|
||||||
import { Image, Pressable, RefreshControl, SafeAreaView, ScrollView, View } from "react-native";
|
import { Image, Pressable, SafeAreaView, ScrollView, View } from "react-native";
|
||||||
import ImageViewing from 'react-native-image-viewing';
|
import ImageViewing from 'react-native-image-viewing';
|
||||||
import { useDispatch, useSelector } from 'react-redux';
|
import { useSelector } from 'react-redux';
|
||||||
|
|
||||||
export default function Profile() {
|
export default function Profile() {
|
||||||
const { colors } = useTheme();
|
const { signOut } = useAuthSession()
|
||||||
const entities = useSelector((state: any) => state.entities)
|
const entities = useSelector((state: any) => state.entities)
|
||||||
const [error, setError] = useState(false)
|
const [error, setError] = useState(false)
|
||||||
const [preview, setPreview] = useState(false)
|
const [preview, setPreview] = useState(false)
|
||||||
const [refreshing, setRefreshing] = useState(false)
|
|
||||||
const dispatch = useDispatch()
|
|
||||||
const { token, decryptToken } = useAuthSession()
|
|
||||||
|
|
||||||
async function handleUserLogin() {
|
|
||||||
const hasil = await decryptToken(String(token?.current))
|
|
||||||
apiGetProfile({ id: hasil })
|
|
||||||
.then((data) => dispatch(setEntities(data.data)))
|
|
||||||
.catch((error) => {
|
|
||||||
console.error(error)
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
const handleRefresh = async () => {
|
|
||||||
setRefreshing(true)
|
|
||||||
handleUserLogin()
|
|
||||||
await new Promise(resolve => setTimeout(resolve, 2000));
|
|
||||||
setRefreshing(false)
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={[Styles.flex1, { backgroundColor: colors.background }]}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
headerTitle: 'Profile',
|
headerTitle: 'Profile',
|
||||||
@@ -55,31 +34,33 @@ export default function Profile() {
|
|||||||
onPressLeft={() => router.back()}
|
onPressLeft={() => router.back()}
|
||||||
right={
|
right={
|
||||||
<ButtonHeader
|
<ButtonHeader
|
||||||
item={<Feather name="settings" size={20} color="white" />}
|
item={<AntDesign name="logout" size={20} color="white" />}
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
router.push('/setting')
|
AlertKonfirmasi({
|
||||||
|
title: 'Keluar',
|
||||||
|
desc: 'Apakah anda yakin ingin keluar?',
|
||||||
|
onPress: () => { signOut() }
|
||||||
|
})
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
)
|
)
|
||||||
|
// headerRight: () => <ButtonHeader
|
||||||
|
// item={<AntDesign name="logout" size={20} color="white" />}
|
||||||
|
// onPress={() => {
|
||||||
|
// AlertKonfirmasi({
|
||||||
|
// title: 'Keluar',
|
||||||
|
// desc: 'Apakah anda yakin ingin keluar?',
|
||||||
|
// onPress: () => { signOut() }
|
||||||
|
// })
|
||||||
|
// }}
|
||||||
|
// />
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<ScrollView
|
<ScrollView style={[Styles.h100]}>
|
||||||
refreshControl={
|
<View style={{ flexDirection: 'column' }}>
|
||||||
<RefreshControl
|
<View style={[Styles.wrapHeadViewMember]}>
|
||||||
refreshing={refreshing}
|
|
||||||
onRefresh={handleRefresh}
|
|
||||||
tintColor={colors.icon}
|
|
||||||
/>
|
|
||||||
}
|
|
||||||
style={[Styles.h100, { backgroundColor: colors.background }]}
|
|
||||||
>
|
|
||||||
<View style={[Styles.flexColumn]}>
|
|
||||||
<LinearGradient
|
|
||||||
colors={[colors.header, colors.homeGradient]}
|
|
||||||
style={[Styles.wrapHeadViewMember]}
|
|
||||||
>
|
|
||||||
<Pressable onPress={() => setPreview(true)}>
|
<Pressable onPress={() => setPreview(true)}>
|
||||||
<Image
|
<Image
|
||||||
source={error ? require("../../assets/images/user.jpg") : { uri: `${ConstEnv.url_storage}/files/${entities.img}` }}
|
source={error ? require("../../assets/images/user.jpg") : { uri: `${ConstEnv.url_storage}/files/${entities.img}` }}
|
||||||
@@ -89,12 +70,14 @@ export default function Profile() {
|
|||||||
</Pressable>
|
</Pressable>
|
||||||
<Text style={[Styles.textSubtitle, Styles.cWhite, Styles.mt10]}>{entities.name}</Text>
|
<Text style={[Styles.textSubtitle, Styles.cWhite, Styles.mt10]}>{entities.name}</Text>
|
||||||
<Text style={[Styles.textMediumNormal, Styles.cWhite]}>{entities.role}</Text>
|
<Text style={[Styles.textMediumNormal, Styles.cWhite]}>{entities.role}</Text>
|
||||||
</LinearGradient>
|
</View>
|
||||||
<View style={[Styles.p15]}>
|
<View style={[Styles.p15]}>
|
||||||
<View style={[Styles.rowSpaceBetween]}>
|
<View style={[Styles.rowSpaceBetween]}>
|
||||||
<Text style={[Styles.textDefaultSemiBold, { color: colors.text }]}>Informasi</Text>
|
<Text style={[Styles.textDefaultSemiBold]}>Informasi</Text>
|
||||||
|
{
|
||||||
|
entities.idUserRole != "developer" && <Text onPress={() => { router.push('/edit-profile') }} style={[Styles.textLink]}>Edit</Text>
|
||||||
|
}
|
||||||
</View>
|
</View>
|
||||||
{/* Note: ItemDetailMember might need updates to support dynamic colors if it uses default text colors */}
|
|
||||||
<ItemDetailMember category="nik" value={entities.nik} />
|
<ItemDetailMember category="nik" value={entities.nik} />
|
||||||
<ItemDetailMember category="group" value={entities.group} />
|
<ItemDetailMember category="group" value={entities.group} />
|
||||||
<ItemDetailMember category="position" value={entities.position} />
|
<ItemDetailMember category="position" value={entities.position} />
|
||||||
@@ -104,7 +87,6 @@ export default function Profile() {
|
|||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
</ScrollView>
|
</ScrollView>
|
||||||
|
|
||||||
<ImageViewing
|
<ImageViewing
|
||||||
images={[{ uri: error ? assetUserImage.uri : `${ConstEnv.url_storage}/files/${entities.img}` }]}
|
images={[{ uri: error ? assetUserImage.uri : `${ConstEnv.url_storage}/files/${entities.img}` }]}
|
||||||
imageIndex={0}
|
imageIndex={0}
|
||||||
|
|||||||
@@ -3,13 +3,11 @@ import BorderBottomItem from "@/components/borderBottomItem"
|
|||||||
import ButtonSaveHeader from "@/components/buttonSaveHeader"
|
import ButtonSaveHeader from "@/components/buttonSaveHeader"
|
||||||
import ButtonSelect from "@/components/buttonSelect"
|
import ButtonSelect from "@/components/buttonSelect"
|
||||||
import DrawerBottom from "@/components/drawerBottom"
|
import DrawerBottom from "@/components/drawerBottom"
|
||||||
import LoadingCenter from "@/components/loadingCenter"
|
|
||||||
import MenuItemRow from "@/components/menuItemRow"
|
import MenuItemRow from "@/components/menuItemRow"
|
||||||
import Styles from "@/constants/Styles"
|
import Styles from "@/constants/Styles"
|
||||||
import { apiAddFileProject, apiCheckFileProject } from "@/lib/api"
|
import { apiAddFileProject, apiCheckFileProject } from "@/lib/api"
|
||||||
import { setUpdateProject } from "@/lib/projectUpdate"
|
import { setUpdateProject } from "@/lib/projectUpdate"
|
||||||
import { useAuthSession } from "@/providers/AuthProvider"
|
import { useAuthSession } from "@/providers/AuthProvider"
|
||||||
import { useTheme } from "@/providers/ThemeProvider"
|
|
||||||
import { Ionicons, MaterialCommunityIcons } from "@expo/vector-icons"
|
import { Ionicons, MaterialCommunityIcons } from "@expo/vector-icons"
|
||||||
import * as DocumentPicker from "expo-document-picker"
|
import * as DocumentPicker from "expo-document-picker"
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router"
|
import { router, Stack, useLocalSearchParams } from "expo-router"
|
||||||
@@ -19,7 +17,6 @@ import Toast from "react-native-toast-message"
|
|||||||
import { useDispatch, useSelector } from "react-redux"
|
import { useDispatch, useSelector } from "react-redux"
|
||||||
|
|
||||||
export default function ProjectAddFile() {
|
export default function ProjectAddFile() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const { id } = useLocalSearchParams<{ id: string }>()
|
const { id } = useLocalSearchParams<{ id: string }>()
|
||||||
const [fileForm, setFileForm] = useState<any[]>([])
|
const [fileForm, setFileForm] = useState<any[]>([])
|
||||||
const [listFile, setListFile] = useState<any[]>([])
|
const [listFile, setListFile] = useState<any[]>([])
|
||||||
@@ -118,11 +115,9 @@ export default function ProjectAddFile() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error : any ) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Gagal menambahkan data"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
@@ -132,7 +127,7 @@ export default function ProjectAddFile() {
|
|||||||
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={[Styles.flex1, { backgroundColor: colors.background }]}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
||||||
@@ -158,23 +153,20 @@ export default function ProjectAddFile() {
|
|||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
{
|
<ScrollView>
|
||||||
loading && <LoadingCenter size="large" />
|
<View style={[Styles.p15, Styles.mb100]}>
|
||||||
}
|
|
||||||
<ScrollView style={[Styles.flex1, { backgroundColor: colors.background }]}>
|
|
||||||
<View style={[Styles.p15]}>
|
|
||||||
<ButtonSelect value="Upload File" onPress={pickDocumentAsync} />
|
<ButtonSelect value="Upload File" onPress={pickDocumentAsync} />
|
||||||
{
|
{
|
||||||
listFile.length > 0 && (
|
listFile.length > 0 && (
|
||||||
<View style={[Styles.mb15]}>
|
<View style={[Styles.mb15]}>
|
||||||
<Text style={[Styles.textDefaultSemiBold, Styles.mv05, { color: colors.text }]}>File</Text>
|
<Text style={[Styles.textDefaultSemiBold, Styles.mv05]}>File</Text>
|
||||||
<View style={[Styles.wrapPaper, { backgroundColor: colors.card, borderColor: colors.background }]}>
|
<View style={[Styles.wrapPaper]}>
|
||||||
{
|
{
|
||||||
listFile.map((item, index) => (
|
listFile.map((item, index) => (
|
||||||
<BorderBottomItem
|
<BorderBottomItem
|
||||||
key={index}
|
key={index}
|
||||||
borderType="all"
|
borderType="all"
|
||||||
icon={<MaterialCommunityIcons name="file-outline" size={25} color={colors.text} />}
|
icon={<MaterialCommunityIcons name="file-outline" size={25} color="black" />}
|
||||||
title={item}
|
title={item}
|
||||||
titleWeight="normal"
|
titleWeight="normal"
|
||||||
onPress={() => { setIndexDelFile(index); setModal(true) }}
|
onPress={() => { setIndexDelFile(index); setModal(true) }}
|
||||||
@@ -188,12 +180,15 @@ export default function ProjectAddFile() {
|
|||||||
{
|
{
|
||||||
loadingCheck && <ActivityIndicator size="small" />
|
loadingCheck && <ActivityIndicator size="small" />
|
||||||
}
|
}
|
||||||
|
{
|
||||||
|
loading && <ActivityIndicator size="large" />
|
||||||
|
}
|
||||||
</View>
|
</View>
|
||||||
</ScrollView>
|
</ScrollView>
|
||||||
<DrawerBottom animation="slide" isVisible={isModal} setVisible={setModal} title="Menu">
|
<DrawerBottom animation="slide" isVisible={isModal} setVisible={setModal} title="Menu">
|
||||||
<View style={Styles.rowItemsCenter}>
|
<View style={Styles.rowItemsCenter}>
|
||||||
<MenuItemRow
|
<MenuItemRow
|
||||||
icon={<Ionicons name="trash-outline" color={colors.text} size={25} />}
|
icon={<Ionicons name="trash" color="black" size={25} />}
|
||||||
title="Hapus"
|
title="Hapus"
|
||||||
onPress={() => { deleteFile(indexDelFile) }}
|
onPress={() => { deleteFile(indexDelFile) }}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -9,7 +9,6 @@ import Styles from "@/constants/Styles";
|
|||||||
import { apiAddMemberProject, apiGetProjectOne, apiGetUser } from "@/lib/api";
|
import { apiAddMemberProject, apiGetProjectOne, apiGetUser } from "@/lib/api";
|
||||||
import { setUpdateProject } from "@/lib/projectUpdate";
|
import { setUpdateProject } from "@/lib/projectUpdate";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { AntDesign } from "@expo/vector-icons";
|
import { AntDesign } from "@expo/vector-icons";
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
@@ -24,7 +23,6 @@ type Props = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export default function AddMemberProject() {
|
export default function AddMemberProject() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const dispatch = useDispatch()
|
const dispatch = useDispatch()
|
||||||
const update = useSelector((state: any) => state.projectUpdate)
|
const update = useSelector((state: any) => state.projectUpdate)
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
@@ -45,11 +43,9 @@ export default function AddMemberProject() {
|
|||||||
setIdGroup(responseGroup.data.idGroup)
|
setIdGroup(responseGroup.data.idGroup)
|
||||||
const responsemember = await apiGetUser({ user: hasil, active: "true", search: search, group: String(responseGroup.data.idGroup) })
|
const responsemember = await apiGetUser({ user: hasil, active: "true", search: search, group: String(responseGroup.data.idGroup) })
|
||||||
setData(responsemember.data.filter((i: any) => i.idUserRole != 'supadmin'))
|
setData(responsemember.data.filter((i: any) => i.idUserRole != 'supadmin'))
|
||||||
} catch (error : any ) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal mengambil data"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -88,11 +84,9 @@ export default function AddMemberProject() {
|
|||||||
dispatch(setUpdateProject({ ...update, member: !update.member }))
|
dispatch(setUpdateProject({ ...update, member: !update.member }))
|
||||||
router.back()
|
router.back()
|
||||||
}
|
}
|
||||||
} catch (error : any ) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal menambahkan anggota"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
@@ -104,7 +98,7 @@ export default function AddMemberProject() {
|
|||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
||||||
headerTitle: 'Tambah Anggota',
|
headerTitle: 'Tambah Anggota Kegiatan',
|
||||||
headerTitleAlign: 'center',
|
headerTitleAlign: 'center',
|
||||||
// headerRight: () => (
|
// headerRight: () => (
|
||||||
// <ButtonSaveHeader
|
// <ButtonSaveHeader
|
||||||
@@ -117,7 +111,7 @@ export default function AddMemberProject() {
|
|||||||
// )
|
// )
|
||||||
header: () => (
|
header: () => (
|
||||||
<AppHeader
|
<AppHeader
|
||||||
title="Tambah Anggota"
|
title="Tambah Anggota Kegiatan"
|
||||||
showBack={true}
|
showBack={true}
|
||||||
onPressLeft={() => router.back()}
|
onPressLeft={() => router.back()}
|
||||||
right={
|
right={
|
||||||
@@ -133,7 +127,7 @@ export default function AddMemberProject() {
|
|||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<View style={[Styles.p15, { flex: 1, backgroundColor: colors.background }]}>
|
<View style={[Styles.p15, { flex: 1 }]}>
|
||||||
<InputSearch onChange={(val) => handleSearch(val)} value={search} />
|
<InputSearch onChange={(val) => handleSearch(val)} value={search} />
|
||||||
{
|
{
|
||||||
selectMember.length > 0
|
selectMember.length > 0
|
||||||
@@ -154,11 +148,11 @@ export default function AddMemberProject() {
|
|||||||
</View>
|
</View>
|
||||||
|
|
||||||
:
|
:
|
||||||
<Text style={[Styles.textDefault, Styles.pv05, { textAlign: 'center', color: colors.dimmed }]}>Tidak ada member yang dipilih</Text>
|
<Text style={[Styles.textDefault, Styles.cGray, Styles.pv05, { textAlign: 'center' }]}>Tidak ada member yang dipilih</Text>
|
||||||
}
|
}
|
||||||
<ScrollView
|
<ScrollView
|
||||||
showsVerticalScrollIndicator={false}
|
showsVerticalScrollIndicator={false}
|
||||||
style={[Styles.h100, { backgroundColor: colors.background }]}
|
style={[Styles.h100]}
|
||||||
>
|
>
|
||||||
{
|
{
|
||||||
data.length > 0 ?
|
data.length > 0 ?
|
||||||
@@ -169,7 +163,7 @@ export default function AddMemberProject() {
|
|||||||
return (
|
return (
|
||||||
<Pressable
|
<Pressable
|
||||||
key={index}
|
key={index}
|
||||||
style={[Styles.itemSelectModal, { borderColor: colors.icon + '20' }]}
|
style={[Styles.itemSelectModal]}
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
!found && onChoose(item.id, item.name, item.img)
|
!found && onChoose(item.id, item.name, item.img)
|
||||||
}}
|
}}
|
||||||
@@ -179,12 +173,12 @@ export default function AddMemberProject() {
|
|||||||
<View style={[Styles.ml10]}>
|
<View style={[Styles.ml10]}>
|
||||||
<Text style={[Styles.textDefault]} numberOfLines={1}>{item.name}</Text>
|
<Text style={[Styles.textDefault]} numberOfLines={1}>{item.name}</Text>
|
||||||
{
|
{
|
||||||
found && <Text style={[Styles.textInformation, { color: colors.dimmed }]}>sudah menjadi anggota</Text>
|
found && <Text style={[Styles.textInformation, Styles.cGray]}>sudah menjadi anggota</Text>
|
||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
{
|
{
|
||||||
selectMember.some((i: any) => i.idUser == item.id) && <AntDesign name="check" size={20} color={colors.text} />
|
selectMember.some((i: any) => i.idUser == item.id) && <AntDesign name="check" size={20} color={'black'} />
|
||||||
}
|
}
|
||||||
</Pressable>
|
</Pressable>
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
import AppHeader from "@/components/AppHeader";
|
import AppHeader from "@/components/AppHeader";
|
||||||
import ButtonSaveHeader from "@/components/buttonSaveHeader";
|
import ButtonSaveHeader from "@/components/buttonSaveHeader";
|
||||||
import ButtonSelect from "@/components/buttonSelect";
|
|
||||||
import { InputForm } from "@/components/inputForm";
|
import { InputForm } from "@/components/inputForm";
|
||||||
import ModalAddDetailTugasProject from "@/components/project/modalAddDetailTugasProject";
|
import ModalAddDetailTugasProject from "@/components/project/modalAddDetailTugasProject";
|
||||||
import Text from "@/components/Text";
|
import Text from "@/components/Text";
|
||||||
@@ -10,7 +9,6 @@ import { formatDateOnly } from "@/lib/fun_formatDateOnly";
|
|||||||
import { getDatesInRange } from "@/lib/fun_getDatesInRange";
|
import { getDatesInRange } from "@/lib/fun_getDatesInRange";
|
||||||
import { setUpdateProject } from "@/lib/projectUpdate";
|
import { setUpdateProject } from "@/lib/projectUpdate";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { useHeaderHeight } from '@react-navigation/elements';
|
import { useHeaderHeight } from '@react-navigation/elements';
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
import 'intl';
|
import 'intl';
|
||||||
@@ -20,6 +18,7 @@ import { useEffect, useState } from "react";
|
|||||||
import {
|
import {
|
||||||
KeyboardAvoidingView,
|
KeyboardAvoidingView,
|
||||||
Platform,
|
Platform,
|
||||||
|
Pressable,
|
||||||
SafeAreaView,
|
SafeAreaView,
|
||||||
ScrollView,
|
ScrollView,
|
||||||
View
|
View
|
||||||
@@ -31,7 +30,6 @@ import DateTimePicker, {
|
|||||||
import { useDispatch, useSelector } from "react-redux";
|
import { useDispatch, useSelector } from "react-redux";
|
||||||
|
|
||||||
export default function ProjectAddTask() {
|
export default function ProjectAddTask() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const headerHeight = useHeaderHeight();
|
const headerHeight = useHeaderHeight();
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
const dispatch = useDispatch()
|
const dispatch = useDispatch()
|
||||||
@@ -126,18 +124,16 @@ export default function ProjectAddTask() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error : any ) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Gagal menambahkan data"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={[Styles.flex1, { backgroundColor: colors.background }]}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => (
|
// headerLeft: () => (
|
||||||
@@ -162,11 +158,11 @@ export default function ProjectAddTask() {
|
|||||||
showBack={true}
|
showBack={true}
|
||||||
onPressLeft={() => router.back()}
|
onPressLeft={() => router.back()}
|
||||||
right={
|
right={
|
||||||
<ButtonSaveHeader
|
<ButtonSaveHeader
|
||||||
disable={disable || loading}
|
disable={disable || loading}
|
||||||
category="create"
|
category="create"
|
||||||
onPress={() => { handleCreate() }}
|
onPress={() => { handleCreate() }}
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
)
|
)
|
||||||
@@ -176,9 +172,9 @@ export default function ProjectAddTask() {
|
|||||||
behavior={Platform.OS === 'ios' ? 'padding' : undefined}
|
behavior={Platform.OS === 'ios' ? 'padding' : undefined}
|
||||||
keyboardVerticalOffset={headerHeight}
|
keyboardVerticalOffset={headerHeight}
|
||||||
>
|
>
|
||||||
<ScrollView style={[Styles.h100, { backgroundColor: colors.background }]}>
|
<ScrollView>
|
||||||
<View style={[Styles.p15, Styles.mb100]}>
|
<View style={[Styles.p15, Styles.mb100]}>
|
||||||
<View style={[Styles.wrapPaper, Styles.p10, { backgroundColor: colors.card, borderColor: colors.background }]}>
|
<View style={[Styles.wrapPaper, Styles.p10]}>
|
||||||
<DateTimePicker
|
<DateTimePicker
|
||||||
mode="range"
|
mode="range"
|
||||||
startDate={range.startDate}
|
startDate={range.startDate}
|
||||||
@@ -188,53 +184,52 @@ export default function ProjectAddTask() {
|
|||||||
selected: Styles.selectedDate,
|
selected: Styles.selectedDate,
|
||||||
selected_label: Styles.cWhite,
|
selected_label: Styles.cWhite,
|
||||||
range_fill: Styles.selectRangeDate,
|
range_fill: Styles.selectRangeDate,
|
||||||
month_label: { color: colors.text },
|
month_label: Styles.cBlack,
|
||||||
month_selector_label: { color: colors.text },
|
month_selector_label: Styles.cBlack,
|
||||||
year_label: { color: colors.text },
|
year_label: Styles.cBlack,
|
||||||
year_selector_label: { color: colors.text },
|
year_selector_label: Styles.cBlack,
|
||||||
day_label: { color: colors.text },
|
day_label: Styles.cBlack,
|
||||||
time_label: { color: colors.text },
|
time_label: Styles.cBlack,
|
||||||
weekday_label: { color: colors.text },
|
weekday_label: Styles.cBlack,
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
</View>
|
</View>
|
||||||
<View style={[Styles.mv10]}>
|
<View style={[Styles.mv10]}>
|
||||||
<View style={[Styles.rowSpaceBetween, Styles.mb10]}>
|
<View style={[Styles.rowSpaceBetween]}>
|
||||||
<View style={[Styles.w48]}>
|
<View style={[{ width: "48%" }]}>
|
||||||
<Text style={[Styles.mb05]}>
|
<Text style={[Styles.mb05]}>
|
||||||
Tanggal Mulai <Text style={{ color: colors.error }}>*</Text>
|
Tanggal Mulai <Text style={Styles.cError}>*</Text>
|
||||||
</Text>
|
</Text>
|
||||||
<View style={[Styles.wrapPaper, Styles.noShadow, Styles.borderAll, Styles.p10, { backgroundColor: colors.card, borderColor: colors.icon + '20' }]}>
|
<View style={[Styles.wrapPaper, Styles.p10]}>
|
||||||
<Text style={Styles.textCenter}>{from}</Text>
|
<Text style={{ textAlign: "center" }}>{from}</Text>
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
<View style={[Styles.w48]}>
|
<View style={[{ width: "48%" }]}>
|
||||||
<Text style={[Styles.mb05]}>
|
<Text style={[Styles.mb05]}>
|
||||||
Tanggal Berakhir <Text style={{ color: colors.error }}>*</Text>
|
Tanggal Berakhir <Text style={Styles.cError}>*</Text>
|
||||||
</Text>
|
</Text>
|
||||||
<View style={[Styles.wrapPaper, Styles.noShadow, Styles.borderAll, Styles.p10, { backgroundColor: colors.card, borderColor: colors.icon + '20' }]}>
|
<View style={[Styles.wrapPaper, Styles.p10]}>
|
||||||
<Text style={Styles.textCenter}>{to}</Text>
|
<Text style={{ textAlign: "center" }}>{to}</Text>
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
{
|
{
|
||||||
(error.endDate || error.startDate) && <Text style={[Styles.textInformation, { color: colors.error }, Styles.mt05]}>Tanggal tidak boleh kosong</Text>
|
(error.endDate || error.startDate) && <Text style={[Styles.textInformation, Styles.cError, Styles.mt05]}>Tanggal tidak boleh kosong</Text>
|
||||||
}
|
}
|
||||||
{/* <Pressable
|
<Pressable
|
||||||
style={[Styles.btnTab, Styles.btnLainnya, dsbButton && Styles.btnDisabled]}
|
style={[Styles.btnTab, Styles.btnLainnya, dsbButton && Styles.btnDisabled]}
|
||||||
disabled={dsbButton}
|
disabled={dsbButton}
|
||||||
onPress={() => { setModalDetail(true) }}
|
onPress={() => { setModalDetail(true) }}
|
||||||
>
|
>
|
||||||
<Text style={[dsbButton ? Styles.cGray : Styles.cWhite]}>Detail</Text>
|
<Text style={[dsbButton ? Styles.cGray : Styles.cWhite]}>Detail</Text>
|
||||||
</Pressable> */}
|
</Pressable>
|
||||||
<ButtonSelect value="Detail" onPress={() => { setModalDetail(true) }} />
|
|
||||||
</View>
|
</View>
|
||||||
<InputForm
|
<InputForm
|
||||||
label="Judul Tugas"
|
label="Judul Tugas"
|
||||||
type="default"
|
type="default"
|
||||||
placeholder="Judul Tugas"
|
placeholder="Judul Tugas"
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
bg="white"
|
||||||
value={title}
|
value={title}
|
||||||
error={error.title}
|
error={error.title}
|
||||||
errorText="Judul tidak boleh kosong"
|
errorText="Judul tidak boleh kosong"
|
||||||
|
|||||||
@@ -5,7 +5,6 @@ import Styles from "@/constants/Styles";
|
|||||||
import { apiCancelProject } from "@/lib/api";
|
import { apiCancelProject } from "@/lib/api";
|
||||||
import { setUpdateProject } from "@/lib/projectUpdate";
|
import { setUpdateProject } from "@/lib/projectUpdate";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
import React, { useEffect, useState } from "react";
|
import React, { useEffect, useState } from "react";
|
||||||
import { SafeAreaView, ScrollView, View } from "react-native";
|
import { SafeAreaView, ScrollView, View } from "react-native";
|
||||||
@@ -13,7 +12,6 @@ import Toast from "react-native-toast-message";
|
|||||||
import { useDispatch, useSelector } from "react-redux";
|
import { useDispatch, useSelector } from "react-redux";
|
||||||
|
|
||||||
export default function ProjectCancel() {
|
export default function ProjectCancel() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const { token, decryptToken } = useAuthSession();
|
const { token, decryptToken } = useAuthSession();
|
||||||
const { id } = useLocalSearchParams<{ id: string }>();
|
const { id } = useLocalSearchParams<{ id: string }>();
|
||||||
const dispatch = useDispatch();
|
const dispatch = useDispatch();
|
||||||
@@ -58,18 +56,16 @@ export default function ProjectCancel() {
|
|||||||
Toast.show({ type: 'small', text1: 'Berhasil membatalkan kegiatan', })
|
Toast.show({ type: 'small', text1: 'Berhasil membatalkan kegiatan', })
|
||||||
router.back();
|
router.back();
|
||||||
}
|
}
|
||||||
} catch (error : any ) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Gagal membatalkan kegiatan"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={[Styles.flex1, { backgroundColor: colors.background }]}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => (
|
// headerLeft: () => (
|
||||||
@@ -110,7 +106,7 @@ export default function ProjectCancel() {
|
|||||||
/>
|
/>
|
||||||
<ScrollView
|
<ScrollView
|
||||||
showsVerticalScrollIndicator={false}
|
showsVerticalScrollIndicator={false}
|
||||||
style={[Styles.h100, { backgroundColor: colors.background }]}
|
style={[Styles.h100]}
|
||||||
>
|
>
|
||||||
<View style={[Styles.p15]}>
|
<View style={[Styles.p15]}>
|
||||||
<InputForm
|
<InputForm
|
||||||
@@ -118,7 +114,7 @@ export default function ProjectCancel() {
|
|||||||
type="default"
|
type="default"
|
||||||
placeholder="Alasan Pembatalan"
|
placeholder="Alasan Pembatalan"
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
bg="white"
|
||||||
error={error}
|
error={error}
|
||||||
errorText="Alasan pembatalan harus diisi"
|
errorText="Alasan pembatalan harus diisi"
|
||||||
onChange={(val) => onValidation(val)}
|
onChange={(val) => onValidation(val)}
|
||||||
|
|||||||
@@ -5,7 +5,6 @@ import Styles from "@/constants/Styles";
|
|||||||
import { apiEditProject, apiGetProjectOne } from "@/lib/api";
|
import { apiEditProject, apiGetProjectOne } from "@/lib/api";
|
||||||
import { setUpdateProject } from "@/lib/projectUpdate";
|
import { setUpdateProject } from "@/lib/projectUpdate";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { SafeAreaView, ScrollView, View } from "react-native";
|
import { SafeAreaView, ScrollView, View } from "react-native";
|
||||||
@@ -13,7 +12,6 @@ import Toast from "react-native-toast-message";
|
|||||||
import { useDispatch, useSelector } from "react-redux";
|
import { useDispatch, useSelector } from "react-redux";
|
||||||
|
|
||||||
export default function EditProject() {
|
export default function EditProject() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const { token, decryptToken } = useAuthSession();
|
const { token, decryptToken } = useAuthSession();
|
||||||
const { id } = useLocalSearchParams<{ id: string }>();
|
const { id } = useLocalSearchParams<{ id: string }>();
|
||||||
const dispatch = useDispatch()
|
const dispatch = useDispatch()
|
||||||
@@ -77,11 +75,9 @@ export default function EditProject() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error : any ) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Gagal mengubah data"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
@@ -90,7 +86,7 @@ export default function EditProject() {
|
|||||||
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={[Styles.flex1, { backgroundColor: colors.background }]}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => (
|
// headerLeft: () => (
|
||||||
@@ -125,14 +121,14 @@ export default function EditProject() {
|
|||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<ScrollView style={[Styles.h100, { backgroundColor: colors.background }]}>
|
<ScrollView>
|
||||||
<View style={[Styles.p15, Styles.mb100]}>
|
<View style={[Styles.p15, Styles.mb100]}>
|
||||||
<InputForm
|
<InputForm
|
||||||
label="Judul Kegiatan"
|
label="Judul Kegiatan"
|
||||||
type="default"
|
type="default"
|
||||||
placeholder="Judul Kegiatan"
|
placeholder="Judul Kegiatan"
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
bg="white"
|
||||||
value={judul}
|
value={judul}
|
||||||
onChange={(val) => { onValidation(val) }}
|
onChange={(val) => { onValidation(val) }}
|
||||||
error={error}
|
error={error}
|
||||||
|
|||||||
@@ -10,7 +10,6 @@ import SectionProgress from "@/components/sectionProgress";
|
|||||||
import Styles from "@/constants/Styles";
|
import Styles from "@/constants/Styles";
|
||||||
import { apiGetProjectOne } from "@/lib/api";
|
import { apiGetProjectOne } from "@/lib/api";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
import React, { useEffect, useState } from "react";
|
import React, { useEffect, useState } from "react";
|
||||||
import { RefreshControl, SafeAreaView, ScrollView, View } from "react-native";
|
import { RefreshControl, SafeAreaView, ScrollView, View } from "react-native";
|
||||||
@@ -33,7 +32,6 @@ type Props = {
|
|||||||
|
|
||||||
export default function DetailProject() {
|
export default function DetailProject() {
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
const { colors } = useTheme();
|
|
||||||
const { id } = useLocalSearchParams<{ id: string }>();
|
const { id } = useLocalSearchParams<{ id: string }>();
|
||||||
const [data, setData] = useState<Props>()
|
const [data, setData] = useState<Props>()
|
||||||
const [progress, setProgress] = useState(0)
|
const [progress, setProgress] = useState(0)
|
||||||
@@ -93,7 +91,7 @@ export default function DetailProject() {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={[Styles.flex1, { backgroundColor: colors.background }]}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
||||||
@@ -113,12 +111,10 @@ export default function DetailProject() {
|
|||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<ScrollView
|
<ScrollView
|
||||||
style={[Styles.h100, { backgroundColor: colors.background }]}
|
|
||||||
refreshControl={
|
refreshControl={
|
||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={handleRefresh}
|
onRefresh={handleRefresh}
|
||||||
tintColor={colors.icon}
|
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
|
|||||||
@@ -5,7 +5,6 @@ import Styles from "@/constants/Styles";
|
|||||||
import { apiGetProjectOne, apiReportProject } from "@/lib/api";
|
import { apiGetProjectOne, apiReportProject } from "@/lib/api";
|
||||||
import { setUpdateProject } from "@/lib/projectUpdate";
|
import { setUpdateProject } from "@/lib/projectUpdate";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { SafeAreaView, ScrollView, View } from "react-native";
|
import { SafeAreaView, ScrollView, View } from "react-native";
|
||||||
@@ -13,7 +12,6 @@ import Toast from "react-native-toast-message";
|
|||||||
import { useDispatch, useSelector } from "react-redux";
|
import { useDispatch, useSelector } from "react-redux";
|
||||||
|
|
||||||
export default function ReportProject() {
|
export default function ReportProject() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const { token, decryptToken } = useAuthSession();
|
const { token, decryptToken } = useAuthSession();
|
||||||
const { id } = useLocalSearchParams<{ id: string }>();
|
const { id } = useLocalSearchParams<{ id: string }>();
|
||||||
const dispatch = useDispatch()
|
const dispatch = useDispatch()
|
||||||
@@ -77,11 +75,9 @@ export default function ReportProject() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error : any ) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Gagal mengubah data"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
@@ -90,7 +86,7 @@ export default function ReportProject() {
|
|||||||
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={[Styles.flex1, { backgroundColor: colors.background }]}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => (
|
// headerLeft: () => (
|
||||||
@@ -127,7 +123,7 @@ export default function ReportProject() {
|
|||||||
/>
|
/>
|
||||||
<ScrollView
|
<ScrollView
|
||||||
showsVerticalScrollIndicator={false}
|
showsVerticalScrollIndicator={false}
|
||||||
style={[Styles.h100, { backgroundColor: colors.background }]}
|
style={[Styles.h100]}
|
||||||
>
|
>
|
||||||
<View style={[Styles.p15]}>
|
<View style={[Styles.p15]}>
|
||||||
<InputForm
|
<InputForm
|
||||||
@@ -135,7 +131,7 @@ export default function ReportProject() {
|
|||||||
type="default"
|
type="default"
|
||||||
placeholder="Laporan Kegiatan"
|
placeholder="Laporan Kegiatan"
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
bg="white"
|
||||||
value={laporan}
|
value={laporan}
|
||||||
onChange={(val) => { onValidation(val) }}
|
onChange={(val) => { onValidation(val) }}
|
||||||
error={error}
|
error={error}
|
||||||
|
|||||||
@@ -5,7 +5,6 @@ import ButtonSelect from "@/components/buttonSelect";
|
|||||||
import DrawerBottom from "@/components/drawerBottom";
|
import DrawerBottom from "@/components/drawerBottom";
|
||||||
import ImageUser from "@/components/imageNew";
|
import ImageUser from "@/components/imageNew";
|
||||||
import { InputForm } from "@/components/inputForm";
|
import { InputForm } from "@/components/inputForm";
|
||||||
import LoadingCenter from "@/components/loadingCenter";
|
|
||||||
import MenuItemRow from "@/components/menuItemRow";
|
import MenuItemRow from "@/components/menuItemRow";
|
||||||
import ModalSelect from "@/components/modalSelect";
|
import ModalSelect from "@/components/modalSelect";
|
||||||
import SectionListAddTask from "@/components/project/sectionListAddTask";
|
import SectionListAddTask from "@/components/project/sectionListAddTask";
|
||||||
@@ -19,7 +18,6 @@ import { setMemberChoose } from "@/lib/memberChoose";
|
|||||||
import { setUpdateProject } from "@/lib/projectUpdate";
|
import { setUpdateProject } from "@/lib/projectUpdate";
|
||||||
import { setTaskCreate } from "@/lib/taskCreate";
|
import { setTaskCreate } from "@/lib/taskCreate";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { Ionicons, MaterialCommunityIcons } from "@expo/vector-icons";
|
import { Ionicons, MaterialCommunityIcons } from "@expo/vector-icons";
|
||||||
import * as DocumentPicker from "expo-document-picker";
|
import * as DocumentPicker from "expo-document-picker";
|
||||||
import { router, Stack } from "expo-router";
|
import { router, Stack } from "expo-router";
|
||||||
@@ -33,7 +31,6 @@ import Toast from "react-native-toast-message";
|
|||||||
import { useDispatch, useSelector } from "react-redux";
|
import { useDispatch, useSelector } from "react-redux";
|
||||||
|
|
||||||
export default function CreateProject() {
|
export default function CreateProject() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const [loading, setLoading] = useState(false)
|
const [loading, setLoading] = useState(false)
|
||||||
const { token, decryptToken } = useAuthSession();
|
const { token, decryptToken } = useAuthSession();
|
||||||
const [chooseGroup, setChooseGroup] = useState({ val: "", label: "" });
|
const [chooseGroup, setChooseGroup] = useState({ val: "", label: "" });
|
||||||
@@ -151,11 +148,9 @@ export default function CreateProject() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error : any ) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal menambahkan data"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoading(false)
|
setLoading(false)
|
||||||
}
|
}
|
||||||
@@ -195,7 +190,7 @@ export default function CreateProject() {
|
|||||||
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={[Styles.flex1, { backgroundColor: colors.background }]}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => (
|
// headerLeft: () => (
|
||||||
@@ -233,12 +228,9 @@ export default function CreateProject() {
|
|||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
{
|
|
||||||
loading && <LoadingCenter />
|
|
||||||
}
|
|
||||||
<ScrollView
|
<ScrollView
|
||||||
showsVerticalScrollIndicator={false}
|
showsVerticalScrollIndicator={false}
|
||||||
style={[Styles.h100, { backgroundColor: colors.background }]}
|
style={[Styles.h100]}
|
||||||
>
|
>
|
||||||
<View style={[Styles.p15]}>
|
<View style={[Styles.p15]}>
|
||||||
{
|
{
|
||||||
@@ -250,7 +242,6 @@ export default function CreateProject() {
|
|||||||
placeholder="Pilih Lembaga Desa"
|
placeholder="Pilih Lembaga Desa"
|
||||||
value={chooseGroup.label}
|
value={chooseGroup.label}
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
setValChoose(chooseGroup.val);
|
setValChoose(chooseGroup.val);
|
||||||
setValSelect("group");
|
setValSelect("group");
|
||||||
@@ -266,7 +257,6 @@ export default function CreateProject() {
|
|||||||
type="default"
|
type="default"
|
||||||
placeholder="Nama Kegiatan"
|
placeholder="Nama Kegiatan"
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
|
||||||
value={dataForm.title}
|
value={dataForm.title}
|
||||||
error={error.title}
|
error={error.title}
|
||||||
errorText="Nama kegiatan tidak boleh kosong"
|
errorText="Nama kegiatan tidak boleh kosong"
|
||||||
@@ -304,13 +294,13 @@ export default function CreateProject() {
|
|||||||
fileForm.length > 0 && (
|
fileForm.length > 0 && (
|
||||||
<View style={[Styles.mb15]}>
|
<View style={[Styles.mb15]}>
|
||||||
<Text style={[Styles.textDefaultSemiBold, Styles.mv05]}>File</Text>
|
<Text style={[Styles.textDefaultSemiBold, Styles.mv05]}>File</Text>
|
||||||
<View style={[Styles.wrapPaper, { backgroundColor: colors.card, borderColor: colors.background }]}>
|
<View style={[Styles.wrapPaper]}>
|
||||||
{
|
{
|
||||||
fileForm.map((item, index) => (
|
fileForm.map((item, index) => (
|
||||||
<BorderBottomItem
|
<BorderBottomItem
|
||||||
key={index}
|
key={index}
|
||||||
borderType="all"
|
borderType="all"
|
||||||
icon={<MaterialCommunityIcons name="file-outline" size={25} color={colors.text} />}
|
icon={<MaterialCommunityIcons name="file-outline" size={25} color="black" />}
|
||||||
title={item.name}
|
title={item.name}
|
||||||
titleWeight="normal"
|
titleWeight="normal"
|
||||||
onPress={() => { setIndexDelFile(index); setModal(true) }}
|
onPress={() => { setIndexDelFile(index); setModal(true) }}
|
||||||
@@ -328,7 +318,7 @@ export default function CreateProject() {
|
|||||||
<Text>Total {entitiesMember.length} Anggota</Text>
|
<Text>Total {entitiesMember.length} Anggota</Text>
|
||||||
</View>
|
</View>
|
||||||
|
|
||||||
<View style={[Styles.borderAll, Styles.round05, Styles.p10, { borderColor: colors.icon + '20', backgroundColor: colors.card }]}>
|
<View style={[Styles.borderAll, Styles.round10, Styles.p10]}>
|
||||||
{entitiesMember.map(
|
{entitiesMember.map(
|
||||||
(item: { img: any; name: any }, index: any) => {
|
(item: { img: any; name: any }, index: any) => {
|
||||||
return (
|
return (
|
||||||
@@ -354,7 +344,7 @@ export default function CreateProject() {
|
|||||||
<DrawerBottom animation="slide" isVisible={isModal} setVisible={setModal} title="Menu">
|
<DrawerBottom animation="slide" isVisible={isModal} setVisible={setModal} title="Menu">
|
||||||
<View style={Styles.rowItemsCenter}>
|
<View style={Styles.rowItemsCenter}>
|
||||||
<MenuItemRow
|
<MenuItemRow
|
||||||
icon={<Ionicons name="trash-outline" color={colors.text} size={25} />}
|
icon={<Ionicons name="trash" color="black" size={25} />}
|
||||||
title="Hapus"
|
title="Hapus"
|
||||||
onPress={() => { deleteFile(indexDelFile) }}
|
onPress={() => { deleteFile(indexDelFile) }}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -9,7 +9,6 @@ import Styles from "@/constants/Styles";
|
|||||||
import { apiGetUser } from "@/lib/api";
|
import { apiGetUser } from "@/lib/api";
|
||||||
import { setMemberChoose } from "@/lib/memberChoose";
|
import { setMemberChoose } from "@/lib/memberChoose";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { AntDesign } from "@expo/vector-icons";
|
import { AntDesign } from "@expo/vector-icons";
|
||||||
import { router, Stack } from "expo-router";
|
import { router, Stack } from "expo-router";
|
||||||
import React, { useEffect, useState } from "react";
|
import React, { useEffect, useState } from "react";
|
||||||
@@ -24,7 +23,6 @@ type Props = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export default function AddMemberCreateProject() {
|
export default function AddMemberCreateProject() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const dispatch = useDispatch()
|
const dispatch = useDispatch()
|
||||||
const { token, decryptToken } = useAuthSession()
|
const { token, decryptToken } = useAuthSession()
|
||||||
const [data, setData] = useState<Props[]>([])
|
const [data, setData] = useState<Props[]>([])
|
||||||
@@ -105,7 +103,7 @@ export default function AddMemberCreateProject() {
|
|||||||
)
|
)
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<View style={[Styles.p15, Styles.flex1, { backgroundColor: colors.background }]}>
|
<View style={[Styles.p15, { flex: 1 }]}>
|
||||||
<InputSearch onChange={(val) => setSearch(val)} value={search} />
|
<InputSearch onChange={(val) => setSearch(val)} value={search} />
|
||||||
|
|
||||||
{
|
{
|
||||||
@@ -127,11 +125,10 @@ export default function AddMemberCreateProject() {
|
|||||||
</View>
|
</View>
|
||||||
|
|
||||||
:
|
:
|
||||||
<Text style={[Styles.textDefault, Styles.textCenter, { color: colors.dimmed }, Styles.pv05]}>Tidak ada member yang dipilih</Text>
|
<Text style={[Styles.textDefault, Styles.cGray, Styles.pv05, { textAlign: 'center' }]}>Tidak ada member yang dipilih</Text>
|
||||||
}
|
}
|
||||||
<ScrollView
|
<ScrollView
|
||||||
showsVerticalScrollIndicator={false}
|
showsVerticalScrollIndicator={false}
|
||||||
style={[Styles.h100, Styles.flex1, { backgroundColor: colors.background }]}
|
|
||||||
>
|
>
|
||||||
|
|
||||||
{
|
{
|
||||||
@@ -140,7 +137,7 @@ export default function AddMemberCreateProject() {
|
|||||||
return (
|
return (
|
||||||
<Pressable
|
<Pressable
|
||||||
key={index}
|
key={index}
|
||||||
style={[Styles.itemSelectModal, { borderColor: colors.icon + '20' }]}
|
style={[Styles.itemSelectModal]}
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
onChoose(item.id, item.name, item.img)
|
onChoose(item.id, item.name, item.img)
|
||||||
}}
|
}}
|
||||||
@@ -152,14 +149,14 @@ export default function AddMemberCreateProject() {
|
|||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
{
|
{
|
||||||
selectMember.some((i: any) => i.idUser == item.id) && <AntDesign name="check" size={20} color={colors.text} />
|
selectMember.some((i: any) => i.idUser == item.id) && <AntDesign name="check" size={20} color={'black'} />
|
||||||
}
|
}
|
||||||
</Pressable>
|
</Pressable>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
:
|
:
|
||||||
<Text style={[Styles.textDefault, Styles.textCenter]}>Tidak ada data</Text>
|
<Text style={[Styles.textDefault, { textAlign: 'center' }]}>Tidak ada data</Text>
|
||||||
}
|
}
|
||||||
</ScrollView>
|
</ScrollView>
|
||||||
</View>
|
</View>
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
import AppHeader from "@/components/AppHeader";
|
import AppHeader from "@/components/AppHeader";
|
||||||
import ButtonSaveHeader from "@/components/buttonSaveHeader";
|
import ButtonSaveHeader from "@/components/buttonSaveHeader";
|
||||||
import ButtonSelect from "@/components/buttonSelect";
|
|
||||||
import { InputForm } from "@/components/inputForm";
|
import { InputForm } from "@/components/inputForm";
|
||||||
import ModalAddDetailTugasProject from "@/components/project/modalAddDetailTugasProject";
|
import ModalAddDetailTugasProject from "@/components/project/modalAddDetailTugasProject";
|
||||||
import Text from "@/components/Text";
|
import Text from "@/components/Text";
|
||||||
@@ -8,7 +7,6 @@ import Styles from "@/constants/Styles";
|
|||||||
import { formatDateOnly } from "@/lib/fun_formatDateOnly";
|
import { formatDateOnly } from "@/lib/fun_formatDateOnly";
|
||||||
import { getDatesInRange } from "@/lib/fun_getDatesInRange";
|
import { getDatesInRange } from "@/lib/fun_getDatesInRange";
|
||||||
import { setTaskCreate } from "@/lib/taskCreate";
|
import { setTaskCreate } from "@/lib/taskCreate";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { useHeaderHeight } from '@react-navigation/elements';
|
import { useHeaderHeight } from '@react-navigation/elements';
|
||||||
import { router, Stack } from "expo-router";
|
import { router, Stack } from "expo-router";
|
||||||
import 'intl';
|
import 'intl';
|
||||||
@@ -18,6 +16,7 @@ import React, { useEffect, useState } from "react";
|
|||||||
import {
|
import {
|
||||||
KeyboardAvoidingView,
|
KeyboardAvoidingView,
|
||||||
Platform,
|
Platform,
|
||||||
|
Pressable,
|
||||||
SafeAreaView,
|
SafeAreaView,
|
||||||
ScrollView,
|
ScrollView,
|
||||||
View
|
View
|
||||||
@@ -28,7 +27,6 @@ import DateTimePicker, {
|
|||||||
import { useDispatch, useSelector } from "react-redux";
|
import { useDispatch, useSelector } from "react-redux";
|
||||||
|
|
||||||
export default function CreateProjectAddTask() {
|
export default function CreateProjectAddTask() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const headerHeight = useHeaderHeight();
|
const headerHeight = useHeaderHeight();
|
||||||
const dispatch = useDispatch()
|
const dispatch = useDispatch()
|
||||||
const [disable, setDisable] = useState(true);
|
const [disable, setDisable] = useState(true);
|
||||||
@@ -121,7 +119,7 @@ export default function CreateProjectAddTask() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={[Styles.flex1, { backgroundColor: colors.background }]}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => (
|
// headerLeft: () => (
|
||||||
@@ -160,9 +158,9 @@ export default function CreateProjectAddTask() {
|
|||||||
behavior={Platform.OS === 'ios' ? 'padding' : undefined}
|
behavior={Platform.OS === 'ios' ? 'padding' : undefined}
|
||||||
keyboardVerticalOffset={headerHeight}
|
keyboardVerticalOffset={headerHeight}
|
||||||
>
|
>
|
||||||
<ScrollView style={[Styles.h100, { backgroundColor: colors.background }]}>
|
<ScrollView>
|
||||||
<View style={[Styles.p15, Styles.mb100]}>
|
<View style={[Styles.p15, Styles.mb100]}>
|
||||||
<View style={[Styles.wrapPaper, Styles.p10, { backgroundColor: colors.card, borderColor: colors.background }]}>
|
<View style={[Styles.wrapPaper, Styles.p10]}>
|
||||||
<DateTimePicker
|
<DateTimePicker
|
||||||
mode="range"
|
mode="range"
|
||||||
startDate={range.startDate}
|
startDate={range.startDate}
|
||||||
@@ -172,53 +170,52 @@ export default function CreateProjectAddTask() {
|
|||||||
selected: Styles.selectedDate,
|
selected: Styles.selectedDate,
|
||||||
selected_label: Styles.cWhite,
|
selected_label: Styles.cWhite,
|
||||||
range_fill: Styles.selectRangeDate,
|
range_fill: Styles.selectRangeDate,
|
||||||
month_label: { color: colors.text },
|
month_label: Styles.cBlack,
|
||||||
month_selector_label: { color: colors.text },
|
month_selector_label: Styles.cBlack,
|
||||||
year_label: { color: colors.text },
|
year_label: Styles.cBlack,
|
||||||
year_selector_label: { color: colors.text },
|
year_selector_label: Styles.cBlack,
|
||||||
day_label: { color: colors.text },
|
day_label: Styles.cBlack,
|
||||||
time_label: { color: colors.text },
|
time_label: Styles.cBlack,
|
||||||
weekday_label: { color: colors.text },
|
weekday_label: Styles.cBlack,
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
</View>
|
</View>
|
||||||
<View style={[Styles.mv10]}>
|
<View style={[Styles.mv10]}>
|
||||||
<View style={[Styles.rowSpaceBetween, Styles.mb10]}>
|
<View style={[Styles.rowSpaceBetween]}>
|
||||||
<View style={[{ width: "48%" }]}>
|
<View style={[{ width: "48%" }]}>
|
||||||
<Text style={[Styles.mb05]}>
|
<Text style={[Styles.mb05]}>
|
||||||
Tanggal Mulai <Text style={{ color: colors.error }}>*</Text>
|
Tanggal Mulai <Text style={Styles.cError}>*</Text>
|
||||||
</Text>
|
</Text>
|
||||||
<View style={[Styles.wrapPaper, Styles.noShadow, Styles.borderAll, Styles.p10, { backgroundColor: colors.card, borderColor: colors.icon + '20' }]}>
|
<View style={[Styles.wrapPaper, Styles.p10]}>
|
||||||
<Text style={Styles.textCenter}>{from}</Text>
|
<Text style={{ textAlign: "center" }}>{from}</Text>
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
<View style={[{ width: "48%" }]}>
|
<View style={[{ width: "48%" }]}>
|
||||||
<Text style={[Styles.mb05]}>
|
<Text style={[Styles.mb05]}>
|
||||||
Tanggal Berakhir <Text style={{ color: colors.error }}>*</Text>
|
Tanggal Berakhir <Text style={Styles.cError}>*</Text>
|
||||||
</Text>
|
</Text>
|
||||||
<View style={[Styles.wrapPaper, Styles.noShadow, Styles.borderAll, Styles.p10, { backgroundColor: colors.card, borderColor: colors.icon + '20' }]}>
|
<View style={[Styles.wrapPaper, Styles.p10]}>
|
||||||
<Text style={Styles.textCenter}>{to}</Text>
|
<Text style={{ textAlign: "center" }}>{to}</Text>
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
{
|
{
|
||||||
(error.endDate || error.startDate) && <Text style={[Styles.textInformation, Styles.mt05, { color: colors.error }]}>Tanggal tidak boleh kosong</Text>
|
(error.endDate || error.startDate) && <Text style={[Styles.textInformation, Styles.cError, Styles.mt05]}>Tanggal tidak boleh kosong</Text>
|
||||||
}
|
}
|
||||||
{/* <Pressable
|
<Pressable
|
||||||
style={[Styles.btnTab, Styles.btnLainnya, dsbButton && Styles.btnDisabled]}
|
style={[Styles.btnTab, Styles.btnLainnya, dsbButton && Styles.btnDisabled]}
|
||||||
disabled={dsbButton}
|
disabled={dsbButton}
|
||||||
onPress={() => { setModalDetail(true) }}
|
onPress={() => { setModalDetail(true) }}
|
||||||
>
|
>
|
||||||
<Text style={[dsbButton ? Styles.cGray : Styles.cWhite]}>Detail</Text>
|
<Text style={[dsbButton ? Styles.cGray : Styles.cWhite]}>Detail</Text>
|
||||||
</Pressable> */}
|
</Pressable>
|
||||||
<ButtonSelect value="Detail" onPress={() => { setModalDetail(true) }} />
|
|
||||||
</View>
|
</View>
|
||||||
<InputForm
|
<InputForm
|
||||||
label="Judul Tugas"
|
label="Judul Tugas"
|
||||||
type="default"
|
type="default"
|
||||||
placeholder="Judul Tugas"
|
placeholder="Judul Tugas"
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
bg="white"
|
||||||
value={title}
|
value={title}
|
||||||
error={error.title}
|
error={error.title}
|
||||||
errorText="Judul tidak boleh kosong"
|
errorText="Judul tidak boleh kosong"
|
||||||
|
|||||||
@@ -7,12 +7,10 @@ import ProgressBar from "@/components/progressBar";
|
|||||||
import Skeleton from "@/components/skeleton";
|
import Skeleton from "@/components/skeleton";
|
||||||
import SkeletonTwoItem from "@/components/skeletonTwoItem";
|
import SkeletonTwoItem from "@/components/skeletonTwoItem";
|
||||||
import Text from "@/components/Text";
|
import Text from "@/components/Text";
|
||||||
import WrapTab from "@/components/wrapTab";
|
|
||||||
import { ColorsStatus } from "@/constants/ColorsStatus";
|
import { ColorsStatus } from "@/constants/ColorsStatus";
|
||||||
import Styles from "@/constants/Styles";
|
import Styles from "@/constants/Styles";
|
||||||
import { apiGetProject } from "@/lib/api";
|
import { apiGetProject } from "@/lib/api";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import {
|
import {
|
||||||
AntDesign,
|
AntDesign,
|
||||||
Ionicons,
|
Ionicons,
|
||||||
@@ -42,11 +40,9 @@ export default function ListProject() {
|
|||||||
}>();
|
}>();
|
||||||
const [statusFix, setStatusFix] = useState<'0' | '1' | '2' | '3'>('0')
|
const [statusFix, setStatusFix] = useState<'0' | '1' | '2' | '3'>('0')
|
||||||
const { token, decryptToken } = useAuthSession();
|
const { token, decryptToken } = useAuthSession();
|
||||||
const { colors } = useTheme();
|
|
||||||
const entityUser = useSelector((state: any) => state.user)
|
const entityUser = useSelector((state: any) => state.user)
|
||||||
const [search, setSearch] = useState("")
|
const [search, setSearch] = useState("")
|
||||||
const [nameGroup, setNameGroup] = useState("")
|
const [nameGroup, setNameGroup] = useState("")
|
||||||
// ... state same ...
|
|
||||||
const [isYear, setYear] = useState("")
|
const [isYear, setYear] = useState("")
|
||||||
const [data, setData] = useState<Props[]>([])
|
const [data, setData] = useState<Props[]>([])
|
||||||
const [isList, setList] = useState(false)
|
const [isList, setList] = useState(false)
|
||||||
@@ -126,70 +122,67 @@ export default function ListProject() {
|
|||||||
})
|
})
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<View style={[Styles.p15, Styles.flex1, { backgroundColor: colors.background }]}>
|
<View style={[Styles.p15, { flex: 1 }]}>
|
||||||
<View>
|
<View>
|
||||||
<WrapTab>
|
<ScrollView horizontal style={[Styles.mb10]} showsHorizontalScrollIndicator={false}>
|
||||||
<ScrollView horizontal showsHorizontalScrollIndicator={false} style={[Styles.round20]}>
|
<ButtonTab
|
||||||
<ButtonTab
|
active={statusFix}
|
||||||
active={statusFix}
|
value="0"
|
||||||
value="0"
|
onPress={() => { setStatusFix("0") }}
|
||||||
onPress={() => { setStatusFix("0") }}
|
label="Segera"
|
||||||
label="Segera"
|
icon={
|
||||||
icon={
|
<MaterialCommunityIcons
|
||||||
<MaterialCommunityIcons
|
name="clock-alert-outline"
|
||||||
name="clock-alert-outline"
|
color={statusFix == "0" ? "white" : "black"}
|
||||||
color={statusFix == "0" ? "white" : colors.dimmed}
|
size={20}
|
||||||
size={20}
|
/>
|
||||||
/>
|
}
|
||||||
}
|
n={4}
|
||||||
n={4}
|
/>
|
||||||
/>
|
<ButtonTab
|
||||||
<ButtonTab
|
active={statusFix}
|
||||||
active={statusFix}
|
value="1"
|
||||||
value="1"
|
onPress={() => { setStatusFix("1") }}
|
||||||
onPress={() => { setStatusFix("1") }}
|
label="Dikerjakan"
|
||||||
label="Dikerjakan"
|
icon={
|
||||||
icon={
|
<MaterialCommunityIcons
|
||||||
<MaterialCommunityIcons
|
name="progress-check"
|
||||||
name="progress-check"
|
color={statusFix == "1" ? "white" : "black"}
|
||||||
color={statusFix == "1" ? "white" : colors.dimmed}
|
size={20}
|
||||||
size={20}
|
/>
|
||||||
/>
|
}
|
||||||
}
|
n={4}
|
||||||
n={4}
|
/>
|
||||||
/>
|
<ButtonTab
|
||||||
<ButtonTab
|
active={statusFix}
|
||||||
active={statusFix}
|
value="2"
|
||||||
value="2"
|
onPress={() => { setStatusFix("2") }}
|
||||||
onPress={() => { setStatusFix("2") }}
|
label="Selesai"
|
||||||
label="Selesai"
|
icon={
|
||||||
icon={
|
<Ionicons
|
||||||
<Ionicons
|
name="checkmark-done-circle-outline"
|
||||||
name="checkmark-done-circle-outline"
|
color={statusFix == "2" ? "white" : "black"}
|
||||||
color={statusFix == "2" ? "white" : colors.dimmed}
|
size={20}
|
||||||
size={20}
|
/>
|
||||||
/>
|
}
|
||||||
}
|
n={4}
|
||||||
n={4}
|
/>
|
||||||
/>
|
<ButtonTab
|
||||||
<ButtonTab
|
active={statusFix}
|
||||||
active={statusFix}
|
value="3"
|
||||||
value="3"
|
onPress={() => { setStatusFix("3") }}
|
||||||
onPress={() => { setStatusFix("3") }}
|
label="Batal"
|
||||||
label="Batal"
|
icon={
|
||||||
icon={
|
<AntDesign
|
||||||
<AntDesign
|
name="closecircleo"
|
||||||
name="closecircleo"
|
color={statusFix == "3" ? "white" : "black"}
|
||||||
color={statusFix == "3" ? "white" : colors.dimmed}
|
size={20}
|
||||||
size={20}
|
/>
|
||||||
/>
|
}
|
||||||
}
|
n={4}
|
||||||
n={4}
|
/>
|
||||||
/>
|
</ScrollView>
|
||||||
</ScrollView>
|
<View style={[Styles.rowSpaceBetween, { alignItems: 'center' }]}>
|
||||||
</WrapTab>
|
|
||||||
|
|
||||||
<View style={[Styles.rowSpaceBetween, Styles.rowItemsCenter]}>
|
|
||||||
<InputSearch width={68} onChange={setSearch} />
|
<InputSearch width={68} onChange={setSearch} />
|
||||||
<Pressable
|
<Pressable
|
||||||
onPress={() => {
|
onPress={() => {
|
||||||
@@ -198,26 +191,26 @@ export default function ListProject() {
|
|||||||
>
|
>
|
||||||
<MaterialCommunityIcons
|
<MaterialCommunityIcons
|
||||||
name={isList ? "format-list-bulleted" : "view-grid"}
|
name={isList ? "format-list-bulleted" : "view-grid"}
|
||||||
color={colors.text}
|
color={"black"}
|
||||||
size={30}
|
size={30}
|
||||||
/>
|
/>
|
||||||
</Pressable>
|
</Pressable>
|
||||||
</View>
|
</View>
|
||||||
<View style={[Styles.mt10]}>
|
<View style={[Styles.mv05]}>
|
||||||
{
|
{
|
||||||
// entityUser.role != 'cosupadmin' && entityUser.role != 'admin' &&
|
// entityUser.role != 'cosupadmin' && entityUser.role != 'admin' &&
|
||||||
<View style={[Styles.rowOnly]}>
|
<View style={[Styles.rowOnly]}>
|
||||||
<Text style={[Styles.mr05]}>Filter :</Text>
|
<Text style={[Styles.mr05]}>Filter :</Text>
|
||||||
{
|
{
|
||||||
(entityUser.role == "supadmin" || entityUser.role == "developer") &&
|
(entityUser.role == "supadmin" || entityUser.role == "developer") &&
|
||||||
<LabelStatus size="small" category="secondary" text={nameGroup} style={[Styles.mr05]} />
|
<LabelStatus size="small" category="secondary" text={nameGroup} style={{ marginRight: 5 }} />
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
(entityUser.role == 'user' || entityUser.role == 'coadmin')
|
(entityUser.role == 'user' || entityUser.role == 'coadmin')
|
||||||
? (cat == 'null' || cat == 'undefined' || cat == undefined || cat == '' || cat == 'data-saya') ? <LabelStatus size="small" category="secondary" text="Kegiatan Saya" style={[Styles.mr05]} /> : <LabelStatus size="small" category="secondary" text="Semua Kegiatan" style={[Styles.mr05]} />
|
? (cat == 'null' || cat == 'undefined' || cat == undefined || cat == '' || cat == 'data-saya') ? <LabelStatus size="small" category="secondary" text="Kegiatan Saya" style={{ marginRight: 5 }} /> : <LabelStatus size="small" category="secondary" text="Semua Kegiatan" style={{ marginRight: 5 }} />
|
||||||
: ''
|
: ''
|
||||||
}
|
}
|
||||||
<LabelStatus size="small" category="secondary" text={isYear} style={[Styles.mr05]} />
|
<LabelStatus size="small" category="secondary" text={isYear} style={{ marginRight: 5 }} />
|
||||||
{/* {
|
{/* {
|
||||||
(entityUser.role == 'user' || entityUser.role == 'coadmin')
|
(entityUser.role == 'user' || entityUser.role == 'coadmin')
|
||||||
? (cat == 'null' || cat == 'undefined' || cat == undefined || cat == '' || cat == 'data-saya') ? <LabelStatus size="small" category="primary" text="Kegiatan Saya" /> : <LabelStatus size="small" category="primary" text="Semua Kegiatan" />
|
? (cat == 'null' || cat == 'undefined' || cat == undefined || cat == '' || cat == 'data-saya') ? <LabelStatus size="small" category="primary" text="Kegiatan Saya" /> : <LabelStatus size="small" category="primary" text="Semua Kegiatan" />
|
||||||
@@ -227,7 +220,7 @@ export default function ListProject() {
|
|||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
<View style={[Styles.flex2, Styles.mt10]}>
|
<View style={[{ flex: 2 }]}>
|
||||||
{
|
{
|
||||||
loading ?
|
loading ?
|
||||||
isList ?
|
isList ?
|
||||||
@@ -253,13 +246,12 @@ export default function ListProject() {
|
|||||||
key={index}
|
key={index}
|
||||||
onPress={() => { router.push(`/project/${item.id}`); }}
|
onPress={() => { router.push(`/project/${item.id}`); }}
|
||||||
borderType="bottom"
|
borderType="bottom"
|
||||||
bgColor="transparent"
|
|
||||||
icon={
|
icon={
|
||||||
<View style={[Styles.iconContent]} >
|
<View style={[Styles.iconContent, ColorsStatus.lightGreen]} >
|
||||||
<AntDesign
|
<AntDesign
|
||||||
name="areachart"
|
name="areachart"
|
||||||
size={25}
|
size={25}
|
||||||
color={"black"}
|
color={"#384288"}
|
||||||
/>
|
/>
|
||||||
</View>
|
</View>
|
||||||
}
|
}
|
||||||
@@ -275,7 +267,6 @@ export default function ListProject() {
|
|||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={handleRefresh}
|
onRefresh={handleRefresh}
|
||||||
tintColor={colors.icon}
|
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
@@ -319,21 +310,20 @@ export default function ListProject() {
|
|||||||
content="page"
|
content="page"
|
||||||
title={item.title}
|
title={item.title}
|
||||||
headerColor="primary"
|
headerColor="primary"
|
||||||
titleTail={2}
|
|
||||||
>
|
>
|
||||||
<ProgressBar value={item.progress} category="list" />
|
<ProgressBar value={item.progress} category="list" />
|
||||||
<View style={[Styles.rowSpaceBetween]}>
|
<View style={[Styles.rowSpaceBetween]}>
|
||||||
<Text style={[Styles.textDefault, { color: colors.dimmed }]}>
|
<Text style={[Styles.textDefault, Styles.cGray]}>
|
||||||
{item.createdAt}
|
{item.createdAt}
|
||||||
</Text>
|
</Text>
|
||||||
<LabelStatus
|
<LabelStatus
|
||||||
size="default"
|
size="default"
|
||||||
category={
|
category={
|
||||||
item.status === 0 ? 'secondary' :
|
item.status === 0 ? 'primary' :
|
||||||
item.status === 1 ? 'warning' :
|
item.status === 1 ? 'warning' :
|
||||||
item.status === 2 ? 'success' :
|
item.status === 2 ? 'success' :
|
||||||
item.status === 3 ? 'error' :
|
item.status === 3 ? 'error' :
|
||||||
'secondary'
|
'primary'
|
||||||
}
|
}
|
||||||
text={
|
text={
|
||||||
item.status === 0 ? 'SEGERA' :
|
item.status === 0 ? 'SEGERA' :
|
||||||
@@ -355,7 +345,6 @@ export default function ListProject() {
|
|||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={handleRefresh}
|
onRefresh={handleRefresh}
|
||||||
tintColor={colors.icon}
|
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
@@ -372,7 +361,7 @@ export default function ListProject() {
|
|||||||
>
|
>
|
||||||
<ProgressBar value={item.progress} category="list" />
|
<ProgressBar value={item.progress} category="list" />
|
||||||
<View style={[Styles.rowSpaceBetween]}>
|
<View style={[Styles.rowSpaceBetween]}>
|
||||||
<Text style={[Styles.textDefault, { color: colors.dimmed }]}>
|
<Text style={[Styles.textDefault, Styles.cGray]}>
|
||||||
{item.createdAt}
|
{item.createdAt}
|
||||||
</Text>
|
</Text>
|
||||||
<LabelStatus
|
<LabelStatus
|
||||||
@@ -400,7 +389,7 @@ export default function ListProject() {
|
|||||||
)
|
)
|
||||||
:
|
:
|
||||||
<View style={[Styles.mt15]}>
|
<View style={[Styles.mt15]}>
|
||||||
<Text style={[Styles.textDefault, Styles.textCenter, { color: colors.dimmed }]}>Tidak ada kegiatan</Text>
|
<Text style={[Styles.textDefault, Styles.cGray, { textAlign: 'center' }]}>Tidak ada kegiatan</Text>
|
||||||
</View>
|
</View>
|
||||||
}
|
}
|
||||||
</View>
|
</View>
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
import AppHeader from "@/components/AppHeader";
|
import AppHeader from "@/components/AppHeader";
|
||||||
import ButtonSaveHeader from "@/components/buttonSaveHeader";
|
import ButtonSaveHeader from "@/components/buttonSaveHeader";
|
||||||
import ButtonSelect from "@/components/buttonSelect";
|
|
||||||
import { InputForm } from "@/components/inputForm";
|
import { InputForm } from "@/components/inputForm";
|
||||||
import ModalAddDetailTugasProject from "@/components/project/modalAddDetailTugasProject";
|
import ModalAddDetailTugasProject from "@/components/project/modalAddDetailTugasProject";
|
||||||
import Text from "@/components/Text";
|
import Text from "@/components/Text";
|
||||||
@@ -10,20 +9,18 @@ import { formatDateOnly } from "@/lib/fun_formatDateOnly";
|
|||||||
import { getDatesInRange } from "@/lib/fun_getDatesInRange";
|
import { getDatesInRange } from "@/lib/fun_getDatesInRange";
|
||||||
import { setUpdateProject } from "@/lib/projectUpdate";
|
import { setUpdateProject } from "@/lib/projectUpdate";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { useHeaderHeight } from '@react-navigation/elements';
|
import { useHeaderHeight } from '@react-navigation/elements';
|
||||||
import { router, Stack, useLocalSearchParams } from "expo-router";
|
import { router, Stack, useLocalSearchParams } from "expo-router";
|
||||||
import 'intl';
|
import 'intl';
|
||||||
import 'intl/locale-data/jsonp/id';
|
import 'intl/locale-data/jsonp/id';
|
||||||
import moment from "moment";
|
import moment from "moment";
|
||||||
import React, { useEffect, useState } from "react";
|
import React, { useEffect, useState } from "react";
|
||||||
import { KeyboardAvoidingView, Platform, SafeAreaView, ScrollView, View } from "react-native";
|
import { KeyboardAvoidingView, Platform, Pressable, SafeAreaView, ScrollView, View } from "react-native";
|
||||||
import Toast from "react-native-toast-message";
|
import Toast from "react-native-toast-message";
|
||||||
import DateTimePicker, { DateType } from "react-native-ui-datepicker";
|
import DateTimePicker, { DateType } from "react-native-ui-datepicker";
|
||||||
import { useDispatch, useSelector } from "react-redux";
|
import { useDispatch, useSelector } from "react-redux";
|
||||||
|
|
||||||
export default function UpdateProjectTask() {
|
export default function UpdateProjectTask() {
|
||||||
const { colors } = useTheme();
|
|
||||||
const headerHeight = useHeaderHeight();
|
const headerHeight = useHeaderHeight();
|
||||||
const dispatch = useDispatch()
|
const dispatch = useDispatch()
|
||||||
const update = useSelector((state: any) => state.projectUpdate)
|
const update = useSelector((state: any) => state.projectUpdate)
|
||||||
@@ -118,11 +115,9 @@ export default function UpdateProjectTask() {
|
|||||||
} else {
|
} else {
|
||||||
Toast.show({ type: 'small', text1: response.message, })
|
Toast.show({ type: 'small', text1: response.message, })
|
||||||
}
|
}
|
||||||
} catch (error : any ) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
const message = error?.response?.data?.message || "Gagal mengubah data"
|
Toast.show({ type: 'small', text1: 'Terjadi kesalahan', })
|
||||||
|
|
||||||
Toast.show({ type: 'small', text1: message })
|
|
||||||
} finally {
|
} finally {
|
||||||
setLoadingSubmit(false)
|
setLoadingSubmit(false)
|
||||||
}
|
}
|
||||||
@@ -174,7 +169,7 @@ export default function UpdateProjectTask() {
|
|||||||
}, [range])
|
}, [range])
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SafeAreaView style={[Styles.flex1, { backgroundColor: colors.background }]}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
// headerLeft: () => <ButtonBackHeader onPress={() => { router.back() }} />,
|
||||||
@@ -205,9 +200,9 @@ export default function UpdateProjectTask() {
|
|||||||
behavior={Platform.OS === 'ios' ? 'padding' : undefined}
|
behavior={Platform.OS === 'ios' ? 'padding' : undefined}
|
||||||
keyboardVerticalOffset={headerHeight}
|
keyboardVerticalOffset={headerHeight}
|
||||||
>
|
>
|
||||||
<ScrollView style={[Styles.h100, { backgroundColor: colors.background }]}>
|
<ScrollView>
|
||||||
<View style={[Styles.p15, Styles.mb100]}>
|
<View style={[Styles.p15, Styles.mb100]}>
|
||||||
<View style={[Styles.wrapPaper, Styles.p10, { backgroundColor: colors.card, borderColor: colors.background }]}>
|
<View style={[Styles.wrapPaper, Styles.p10]}>
|
||||||
{
|
{
|
||||||
!loading
|
!loading
|
||||||
&&
|
&&
|
||||||
@@ -222,51 +217,50 @@ export default function UpdateProjectTask() {
|
|||||||
selected: Styles.selectedDate,
|
selected: Styles.selectedDate,
|
||||||
selected_label: Styles.cWhite,
|
selected_label: Styles.cWhite,
|
||||||
range_fill: Styles.selectRangeDate,
|
range_fill: Styles.selectRangeDate,
|
||||||
month_label: { color: colors.text },
|
month_label: Styles.cBlack,
|
||||||
month_selector_label: { color: colors.text },
|
month_selector_label: Styles.cBlack,
|
||||||
year_label: { color: colors.text },
|
year_label: Styles.cBlack,
|
||||||
year_selector_label: { color: colors.text },
|
year_selector_label: Styles.cBlack,
|
||||||
day_label: { color: colors.text },
|
day_label: Styles.cBlack,
|
||||||
time_label: { color: colors.text },
|
time_label: Styles.cBlack,
|
||||||
weekday_label: { color: colors.text },
|
weekday_label: Styles.cBlack,
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
|
|
||||||
</View>
|
</View>
|
||||||
<View style={[Styles.mv10]}>
|
<View style={[Styles.mv10]}>
|
||||||
<View style={[Styles.rowSpaceBetween, Styles.mb10]}>
|
<View style={[Styles.rowSpaceBetween]}>
|
||||||
<View style={[{ width: '48%' }]}>
|
<View style={[{ width: '48%' }]}>
|
||||||
<Text style={[Styles.mb05]}>Tanggal Mulai <Text style={{ color: colors.error }}>*</Text></Text>
|
<Text style={[Styles.mb05]}>Tanggal Mulai <Text style={Styles.cError}>*</Text></Text>
|
||||||
<View style={[Styles.wrapPaper, Styles.noShadow, Styles.borderAll, Styles.p10, { backgroundColor: colors.card, borderColor: colors.icon + '20' }]}>
|
<View style={[Styles.wrapPaper, Styles.p10]}>
|
||||||
<Text style={Styles.textCenter}>{from}</Text>
|
<Text style={{ textAlign: 'center' }}>{from}</Text>
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
<View style={[{ width: '48%' }]}>
|
<View style={[{ width: '48%' }]}>
|
||||||
<Text style={[Styles.mb05]}>Tanggal Berakhir <Text style={{ color: colors.error }}>*</Text></Text>
|
<Text style={[Styles.mb05]}>Tanggal Berakhir <Text style={Styles.cError}>*</Text></Text>
|
||||||
<View style={[Styles.wrapPaper, Styles.noShadow, Styles.borderAll, Styles.p10, { backgroundColor: colors.card, borderColor: colors.icon + '20' }]}>
|
<View style={[Styles.wrapPaper, Styles.p10]}>
|
||||||
<Text style={Styles.textCenter}>{to}</Text>
|
<Text style={{ textAlign: 'center' }}>{to}</Text>
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
{
|
{
|
||||||
(error.endDate || error.startDate) && <Text style={[Styles.textInformation, { color: colors.error }, Styles.mt05]}>Tanggal tidak boleh kosong</Text>
|
(error.endDate || error.startDate) && <Text style={[Styles.textInformation, Styles.cError, Styles.mt05]}>Tanggal tidak boleh kosong</Text>
|
||||||
}
|
}
|
||||||
{/* <Pressable
|
<Pressable
|
||||||
style={[Styles.btnTab, Styles.btnLainnya, dsbButton && Styles.btnDisabled]}
|
style={[Styles.btnTab, Styles.btnLainnya, dsbButton && Styles.btnDisabled]}
|
||||||
disabled={dsbButton}
|
disabled={dsbButton}
|
||||||
onPress={() => { setModalDetail(true) }}
|
onPress={() => { setModalDetail(true) }}
|
||||||
>
|
>
|
||||||
<Text style={[dsbButton ? Styles.cGray : Styles.cWhite]}>Detail</Text>
|
<Text style={[dsbButton ? Styles.cGray : Styles.cWhite]}>Detail</Text>
|
||||||
</Pressable> */}
|
</Pressable>
|
||||||
<ButtonSelect value="Detail" onPress={() => { setModalDetail(true) }} />
|
|
||||||
</View>
|
</View>
|
||||||
<InputForm
|
<InputForm
|
||||||
label="Judul Tugas"
|
label="Judul Tugas"
|
||||||
type="default"
|
type="default"
|
||||||
placeholder="Judul Tugas"
|
placeholder="Judul Tugas"
|
||||||
required
|
required
|
||||||
bg={colors.card}
|
bg="white"
|
||||||
value={title}
|
value={title}
|
||||||
error={error.title}
|
error={error.title}
|
||||||
errorText="Judul tidak boleh kosong"
|
errorText="Judul tidak boleh kosong"
|
||||||
|
|||||||
@@ -8,14 +8,12 @@ import { ConstEnv } from "@/constants/ConstEnv";
|
|||||||
import Styles from "@/constants/Styles";
|
import Styles from "@/constants/Styles";
|
||||||
import { apiGetSearch } from "@/lib/api";
|
import { apiGetSearch } from "@/lib/api";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { AntDesign, MaterialIcons } from "@expo/vector-icons";
|
import { AntDesign, MaterialIcons } from "@expo/vector-icons";
|
||||||
import { router, Stack } from "expo-router";
|
import { router, Stack } from "expo-router";
|
||||||
import React, { useState } from "react";
|
import React, { useState } from "react";
|
||||||
import { RefreshControl, SafeAreaView, ScrollView, View } from "react-native";
|
import { RefreshControl, SafeAreaView, ScrollView, View } from "react-native";
|
||||||
import Toast from "react-native-toast-message";
|
import Toast from "react-native-toast-message";
|
||||||
|
|
||||||
// ... types ...
|
|
||||||
type PropsUser = {
|
type PropsUser = {
|
||||||
id: string
|
id: string
|
||||||
name: string
|
name: string
|
||||||
@@ -45,7 +43,6 @@ export default function Search() {
|
|||||||
const [dataProject, setDataProject] = useState<PropProject[]>([])
|
const [dataProject, setDataProject] = useState<PropProject[]>([])
|
||||||
const [refreshing, setRefreshing] = useState(false)
|
const [refreshing, setRefreshing] = useState(false)
|
||||||
const [search, setSearch] = useState('')
|
const [search, setSearch] = useState('')
|
||||||
const { colors } = useTheme();
|
|
||||||
|
|
||||||
async function handleSearch(cari: string) {
|
async function handleSearch(cari: string) {
|
||||||
try {
|
try {
|
||||||
@@ -65,14 +62,9 @@ export default function Search() {
|
|||||||
setDataDivisi([])
|
setDataDivisi([])
|
||||||
setDataProject([])
|
setDataProject([])
|
||||||
}
|
}
|
||||||
} catch (error: any) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error)
|
||||||
const message = error?.response?.data?.message || "Gagal melakukan pencarian"
|
return Toast.show({ type: 'small', text1: 'Gagal melakukan pencarian', })
|
||||||
|
|
||||||
Toast.show({
|
|
||||||
type: 'small',
|
|
||||||
text1: message
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -87,7 +79,7 @@ export default function Search() {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<SafeAreaView style={[Styles.flex1, { backgroundColor: colors.background }]}>
|
<SafeAreaView>
|
||||||
<Stack.Screen
|
<Stack.Screen
|
||||||
options={{
|
options={{
|
||||||
headerTitle: 'Pencarian',
|
headerTitle: 'Pencarian',
|
||||||
@@ -108,7 +100,6 @@ export default function Search() {
|
|||||||
<RefreshControl
|
<RefreshControl
|
||||||
refreshing={refreshing}
|
refreshing={refreshing}
|
||||||
onRefresh={handleRefresh}
|
onRefresh={handleRefresh}
|
||||||
tintColor={colors.icon}
|
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
@@ -186,7 +177,7 @@ export default function Search() {
|
|||||||
</ScrollView>
|
</ScrollView>
|
||||||
:
|
:
|
||||||
<View style={[Styles.contentItemCenter, Styles.mt10]}>
|
<View style={[Styles.contentItemCenter, Styles.mt10]}>
|
||||||
<Text style={[Styles.textInformation, { color: colors.icon }]}>Tidak ada data</Text>
|
<Text style={[Styles.textInformation, Styles.cGray]}>Tidak ada data</Text>
|
||||||
</View>
|
</View>
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,201 +0,0 @@
|
|||||||
import ModalConfirmation from "@/components/ModalConfirmation";
|
|
||||||
import Text from "@/components/Text";
|
|
||||||
import ButtonSetting from "@/components/buttonSetting";
|
|
||||||
import DrawerBottom from "@/components/drawerBottom";
|
|
||||||
import Styles from "@/constants/Styles";
|
|
||||||
import { apiGetCheckToken, apiRegisteredToken, apiUnregisteredToken } from "@/lib/api";
|
|
||||||
import { checkPermission, getToken, openSettings } from "@/lib/useNotification";
|
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import { Feather, Ionicons } from "@expo/vector-icons";
|
|
||||||
import AsyncStorage from "@react-native-async-storage/async-storage";
|
|
||||||
import { router } from "expo-router";
|
|
||||||
import { useCallback, useEffect, useRef, useState } from "react";
|
|
||||||
import { AppState, AppStateStatus, Pressable, View } from "react-native";
|
|
||||||
import { useSelector } from "react-redux";
|
|
||||||
|
|
||||||
export default function ListSetting() {
|
|
||||||
const { theme, setTheme, colors } = useTheme()
|
|
||||||
const { signOut } = useAuthSession()
|
|
||||||
const [isNotificationEnabled, setIsNotificationEnabled] = useState<boolean | null>(null);
|
|
||||||
const entities = useSelector((state: any) => state.entities)
|
|
||||||
const [modalVisible, setModalVisible] = useState(false);
|
|
||||||
const [modalConfig, setModalConfig] = useState({
|
|
||||||
title: '',
|
|
||||||
message: '',
|
|
||||||
confirmText: 'Buka Pengaturan',
|
|
||||||
onConfirm: () => { }
|
|
||||||
});
|
|
||||||
|
|
||||||
const [showLogoutModal, setShowLogoutModal] = useState(false)
|
|
||||||
const [showThemeModal, setShowThemeModal] = useState(false)
|
|
||||||
const prevOsPermission = useRef<boolean | undefined>(undefined);
|
|
||||||
|
|
||||||
const registerToken = async () => {
|
|
||||||
try {
|
|
||||||
await AsyncStorage.setItem('@notification_permission', "true");
|
|
||||||
const token = await getToken();
|
|
||||||
if (token) {
|
|
||||||
await apiRegisteredToken({ user: entities.id, token, category: "register" });
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
console.warn('Error registering token:', error);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const unregisterToken = async () => {
|
|
||||||
try {
|
|
||||||
await AsyncStorage.setItem('@notification_permission', "false");
|
|
||||||
const token = await getToken();
|
|
||||||
if (token) {
|
|
||||||
await apiUnregisteredToken({ user: entities.id, token, category: "unregister" });
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
console.warn('Error unregistering token:', error);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const checkNotif = useCallback(async () => {
|
|
||||||
const osPermission = await checkPermission();
|
|
||||||
|
|
||||||
// Jika dari tidak diijinkan sistem kemudian diijinkan (setelah balik dari pengaturan device)
|
|
||||||
if (prevOsPermission.current === false && osPermission === true) {
|
|
||||||
await registerToken();
|
|
||||||
}
|
|
||||||
prevOsPermission.current = osPermission;
|
|
||||||
|
|
||||||
if (!osPermission) {
|
|
||||||
setIsNotificationEnabled(false);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
const token = await getToken();
|
|
||||||
if (token) {
|
|
||||||
const response = await apiGetCheckToken({ user: entities.id, token });
|
|
||||||
setIsNotificationEnabled(!!response.data);
|
|
||||||
} else {
|
|
||||||
setIsNotificationEnabled(false);
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
console.warn('Error checking token status:', error);
|
|
||||||
setIsNotificationEnabled(false);
|
|
||||||
}
|
|
||||||
}, [entities.id]);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
checkNotif();
|
|
||||||
|
|
||||||
const subscription = AppState.addEventListener('change', (nextAppState: AppStateStatus) => {
|
|
||||||
if (nextAppState === 'active') {
|
|
||||||
checkNotif();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
return () => {
|
|
||||||
subscription.remove();
|
|
||||||
};
|
|
||||||
}, [checkNotif]);
|
|
||||||
|
|
||||||
const handleToggleNotif = async () => {
|
|
||||||
const osPermission = await checkPermission();
|
|
||||||
|
|
||||||
if (!osPermission) {
|
|
||||||
setModalConfig({
|
|
||||||
title: "Aktifkan Notifikasi?",
|
|
||||||
message: "Izin notifikasi tidak diberikan. Buka pengaturan sistem untuk mengaktifkannya?",
|
|
||||||
confirmText: "Buka Pengaturan",
|
|
||||||
onConfirm: () => {
|
|
||||||
setModalVisible(false);
|
|
||||||
openSettings();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
setModalVisible(true);
|
|
||||||
} else {
|
|
||||||
// OS Permission is granted, perform in-app toggle
|
|
||||||
const targetState = !isNotificationEnabled;
|
|
||||||
if (targetState) {
|
|
||||||
await registerToken();
|
|
||||||
} else {
|
|
||||||
await unregisterToken();
|
|
||||||
}
|
|
||||||
// UI will be updated by checkNotif (triggered by state change or manually here)
|
|
||||||
setIsNotificationEnabled(targetState);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const ThemeOption = ({ label, value, icon }: { label: string, value: 'light' | 'dark' | 'system', icon: string }) => (
|
|
||||||
<Pressable
|
|
||||||
style={[Styles.itemSelectModal, { borderColor: colors.icon + '20' }]}
|
|
||||||
onPress={() => {
|
|
||||||
setTheme(value);
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<View style={Styles.rowItemsCenter}>
|
|
||||||
<Ionicons name={icon as any} size={20} color={colors.text} style={Styles.mr10} />
|
|
||||||
<Text style={{ color: colors.text }}>{label}</Text>
|
|
||||||
</View>
|
|
||||||
{theme === value && <Ionicons name="checkmark" size={20} color={colors.text} />}
|
|
||||||
</Pressable>
|
|
||||||
);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<View style={[Styles.p15, Styles.flex1, { backgroundColor: colors.background }]}>
|
|
||||||
<View style={[Styles.wrapPaper, { backgroundColor: colors.card, borderColor: colors.icon + '20' }, Styles.p0, Styles.round05]}>
|
|
||||||
{
|
|
||||||
entities.idUserRole != "developer" &&
|
|
||||||
<ButtonSetting
|
|
||||||
title="Edit Profile"
|
|
||||||
icon={<Feather name="user" size={20} color={colors.text} />}
|
|
||||||
onPress={() => { router.push('/edit-profile') }}
|
|
||||||
/>
|
|
||||||
}
|
|
||||||
<ButtonSetting
|
|
||||||
title="Tema Aplikasi"
|
|
||||||
icon={<Ionicons name="color-palette-outline" size={20} color={colors.text} />}
|
|
||||||
onPress={() => setShowThemeModal(true)}
|
|
||||||
value={theme === 'light' ? 'Terang' : theme === 'dark' ? 'Gelap' : 'Sistem'}
|
|
||||||
/>
|
|
||||||
<ButtonSetting
|
|
||||||
title="Notifikasi"
|
|
||||||
icon={<Feather name="bell" size={20} color={colors.text} />}
|
|
||||||
onPress={handleToggleNotif}
|
|
||||||
value={isNotificationEnabled === null ? 'Memuat...' : isNotificationEnabled ? 'Aktif' : 'Nonaktif'}
|
|
||||||
/>
|
|
||||||
<ButtonSetting
|
|
||||||
title="Keluar"
|
|
||||||
icon={<Feather name="log-out" size={20} color={colors.text} />}
|
|
||||||
onPress={() => setShowLogoutModal(true)}
|
|
||||||
borderBottom={false}
|
|
||||||
/>
|
|
||||||
</View>
|
|
||||||
|
|
||||||
<ModalConfirmation
|
|
||||||
visible={modalVisible}
|
|
||||||
title={modalConfig.title}
|
|
||||||
message={modalConfig.message}
|
|
||||||
confirmText={modalConfig.confirmText}
|
|
||||||
onConfirm={modalConfig.onConfirm}
|
|
||||||
onCancel={() => setModalVisible(false)}
|
|
||||||
/>
|
|
||||||
|
|
||||||
<ModalConfirmation
|
|
||||||
visible={showLogoutModal}
|
|
||||||
title="Keluar"
|
|
||||||
message="Apakah anda yakin ingin keluar?"
|
|
||||||
onConfirm={() => {
|
|
||||||
setShowLogoutModal(false)
|
|
||||||
signOut()
|
|
||||||
}}
|
|
||||||
onCancel={() => setShowLogoutModal(false)}
|
|
||||||
confirmText="Keluar"
|
|
||||||
cancelText="Batal"
|
|
||||||
/>
|
|
||||||
<DrawerBottom animation="slide" isVisible={showThemeModal} setVisible={setShowThemeModal} title="Tema Aplikasi">
|
|
||||||
<ThemeOption label="Terang" value="light" icon="sunny-outline" />
|
|
||||||
<ThemeOption label="Gelap" value="dark" icon="moon-outline" />
|
|
||||||
<ThemeOption label="Sistem" value="system" icon="phone-portrait-outline" />
|
|
||||||
</DrawerBottom>
|
|
||||||
</View>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
@@ -1,5 +1,4 @@
|
|||||||
import AuthProvider from '@/providers/AuthProvider';
|
import AuthProvider from '@/providers/AuthProvider';
|
||||||
import ThemeProvider, { useTheme } from '@/providers/ThemeProvider';
|
|
||||||
import { useFonts } from 'expo-font';
|
import { useFonts } from 'expo-font';
|
||||||
import { Stack } from 'expo-router';
|
import { Stack } from 'expo-router';
|
||||||
import * as SplashScreen from 'expo-splash-screen';
|
import * as SplashScreen from 'expo-splash-screen';
|
||||||
@@ -8,27 +7,10 @@ import { useEffect } from 'react';
|
|||||||
import { GestureHandlerRootView } from 'react-native-gesture-handler';
|
import { GestureHandlerRootView } from 'react-native-gesture-handler';
|
||||||
import { NotifierWrapper } from 'react-native-notifier';
|
import { NotifierWrapper } from 'react-native-notifier';
|
||||||
import 'react-native-reanimated';
|
import 'react-native-reanimated';
|
||||||
import Styles from '@/constants/Styles';
|
|
||||||
|
|
||||||
// Prevent the splash screen from auto-hiding before asset loading is complete.
|
// Prevent the splash screen from auto-hiding before asset loading is complete.
|
||||||
SplashScreen.preventAutoHideAsync();
|
SplashScreen.preventAutoHideAsync();
|
||||||
|
|
||||||
// Inner component - berada di dalam ThemeProvider, bisa pakai useTheme()
|
|
||||||
function AppStack() {
|
|
||||||
const { colors } = useTheme();
|
|
||||||
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
<Stack screenOptions={{ contentStyle: { backgroundColor: colors.header } }}>
|
|
||||||
<Stack.Screen name="index" options={{ headerShown: false }} />
|
|
||||||
<Stack.Screen name="verification" options={{ headerShown: false }} />
|
|
||||||
<Stack.Screen name="(application)" options={{ headerShown: false }} />
|
|
||||||
</Stack>
|
|
||||||
<StatusBar style="auto" />
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export default function RootLayout() {
|
export default function RootLayout() {
|
||||||
const [loaded] = useFonts({
|
const [loaded] = useFonts({
|
||||||
SpaceMono: require('../assets/fonts/SpaceMono-Regular.ttf'),
|
SpaceMono: require('../assets/fonts/SpaceMono-Regular.ttf'),
|
||||||
@@ -45,13 +27,16 @@ export default function RootLayout() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<GestureHandlerRootView style={Styles.flex1}>
|
<GestureHandlerRootView style={{ flex: 1 }}>
|
||||||
<NotifierWrapper>
|
<NotifierWrapper>
|
||||||
<ThemeProvider>
|
<AuthProvider>
|
||||||
<AuthProvider>
|
<Stack>
|
||||||
<AppStack />
|
<Stack.Screen name="index" options={{ headerShown: false }} />
|
||||||
</AuthProvider>
|
<Stack.Screen name="verification" options={{ headerShown: false }} />
|
||||||
</ThemeProvider>
|
<Stack.Screen name="(application)" options={{ headerShown: false }} />
|
||||||
|
</Stack>
|
||||||
|
<StatusBar style="auto" />
|
||||||
|
</AuthProvider>
|
||||||
</NotifierWrapper>
|
</NotifierWrapper>
|
||||||
</GestureHandlerRootView>
|
</GestureHandlerRootView>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -3,7 +3,6 @@ import Text from '@/components/Text';
|
|||||||
import { ConstEnv } from "@/constants/ConstEnv";
|
import { ConstEnv } from "@/constants/ConstEnv";
|
||||||
import Styles from "@/constants/Styles";
|
import Styles from "@/constants/Styles";
|
||||||
import { useAuthSession } from "@/providers/AuthProvider";
|
import { useAuthSession } from "@/providers/AuthProvider";
|
||||||
import { useTheme } from "@/providers/ThemeProvider";
|
|
||||||
import CryptoES from "crypto-es";
|
import CryptoES from "crypto-es";
|
||||||
import React, { useState } from "react";
|
import React, { useState } from "react";
|
||||||
import { Image, View } from "react-native";
|
import { Image, View } from "react-native";
|
||||||
@@ -16,24 +15,17 @@ export default function Index() {
|
|||||||
value,
|
value,
|
||||||
setValue,
|
setValue,
|
||||||
});
|
});
|
||||||
const { colors } = useTheme();
|
|
||||||
|
|
||||||
const { signIn } = useAuthSession();
|
const { signIn } = useAuthSession();
|
||||||
const login = (): void => {
|
const login = (): void => {
|
||||||
// WARNING: This is a hardcoded bypass for development purposes.
|
const random: string = 'contohLoginMobileDarmasaba';
|
||||||
// It should be removed or secured before production release.
|
var mytexttoEncryption = "contohLoginMobileDarmasaba"
|
||||||
if (__DEV__) {
|
const encrypted = CryptoES.AES.encrypt(mytexttoEncryption, ConstEnv.pass_encrypt).toString();
|
||||||
const random: string = 'contohLoginMobileDarmasaba';
|
signIn(encrypted);
|
||||||
var mytexttoEncryption = "contohLoginMobileDarmasaba"
|
|
||||||
const encrypted = CryptoES.AES.encrypt(mytexttoEncryption, ConstEnv.pass_encrypt).toString();
|
|
||||||
signIn(encrypted);
|
|
||||||
} else {
|
|
||||||
console.warn("Bypass login disabled in production.");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return (
|
return (
|
||||||
<View style={[Styles.wrapLogin, { backgroundColor: colors.background }]} >
|
<View style={Styles.wrapLogin} >
|
||||||
<View style={[Styles.rowItemsCenter, Styles.mv50]}>
|
<View style={{ alignItems: "center", marginVertical: 50 }}>
|
||||||
<Image
|
<Image
|
||||||
source={require("../assets/images/logo.png")}
|
source={require("../assets/images/logo.png")}
|
||||||
style={[{ width: 300, height: 150 }]}
|
style={[{ width: 300, height: 150 }]}
|
||||||
@@ -58,7 +50,7 @@ export default function Index() {
|
|||||||
renderCell={({ index, symbol, isFocused }) => (
|
renderCell={({ index, symbol, isFocused }) => (
|
||||||
<Text
|
<Text
|
||||||
key={index}
|
key={index}
|
||||||
style={[Styles.verificationCell, isFocused && Styles.verificationFocusCell, { borderColor: isFocused ? colors.tint : colors.icon, color: colors.text }]}
|
style={[Styles.verificationCell, isFocused && Styles.verificationFocusCell]}
|
||||||
onLayout={getCellOnLayoutHandler(index)}>
|
onLayout={getCellOnLayoutHandler(index)}>
|
||||||
{symbol || (isFocused ? <Cursor /> : null)}
|
{symbol || (isFocused ? <Cursor /> : null)}
|
||||||
</Text>
|
</Text>
|
||||||
@@ -69,7 +61,7 @@ export default function Index() {
|
|||||||
// onPress={() => { router.push("/home") }}
|
// onPress={() => { router.push("/home") }}
|
||||||
onPress={login}
|
onPress={login}
|
||||||
/>
|
/>
|
||||||
<Text style={[Styles.textInformation, Styles.mt05, Styles.cDefault, { textAlign: 'center', color: colors.tint }]}>
|
<Text style={[Styles.textInformation, Styles.mt05, Styles.cDefault, { textAlign: 'center' }]}>
|
||||||
Tidak Menerima kode verifikasi? Kirim Ulang
|
Tidak Menerima kode verifikasi? Kirim Ulang
|
||||||
</Text>
|
</Text>
|
||||||
</View>
|
</View>
|
||||||
|
|||||||