Forum Screens (User Phase 5 - 17 files):
- Migrate all forum list, detail, create, and report screens to OS_Wrapper.
- ViewBeranda, ViewBeranda2, ViewBeranda3: List screens with pull-to-refresh.
- DetailForum, DetailForum2: Comment sections with headers (apply disableFlexGrow fix).
- create, edit, report-*, other-report-*, preview-report-*: Forms with keyboard handling.
Admin Phase 9 (User Access - 2 files):
- index.tsx: List with search and pagination.
- [id]/index.tsx: Detail with status toggle footer.
Scroll Fixes (Critical Bugs):
- Fix "Ghost Scroll" in Android FlatList: Removed TouchableWithoutFeedback and KeyboardAvoidingView wrappers in List Mode.
- Fix Large Header Cut-off: Added optional disableFlexGrow={true} to OS_Wrapper for screens with complex ListHeaderComponents (e.g., Forum Detail).
- Fix Keyboard Dismiss: Changed keyboardShouldPersistTaps to "handled" so taps on empty areas dismiss the keyboard while allowing scroll.
Documentation:
- Update TASK-005 with complete Phase 5 details and new progress totals.
Co-authored-by: Qwen-Coder <qwen-coder@alibabacloud.com>
Co-authored-by: Qwen-Coder <qwen-coder@alibabacloud.com>
255 lines
7.6 KiB
TypeScript
255 lines
7.6 KiB
TypeScript
// @/components/AndroidWrapper.tsx
|
|
// Android Wrapper - Based on NewWrapper_V2 (with keyboard handling for Android)
|
|
import { MainColor } from "@/constants/color-palet";
|
|
import { OS_HEIGHT } from "@/constants/constans-value";
|
|
import { GStyles } from "@/styles/global-styles";
|
|
import {
|
|
ImageBackground,
|
|
Keyboard,
|
|
KeyboardAvoidingView,
|
|
ScrollView,
|
|
FlatList,
|
|
TouchableWithoutFeedback,
|
|
View,
|
|
StyleProp,
|
|
ViewStyle,
|
|
} from "react-native";
|
|
import {
|
|
NativeSafeAreaViewProps,
|
|
SafeAreaView,
|
|
} from "react-native-safe-area-context";
|
|
import type { ScrollViewProps, FlatListProps } from "react-native";
|
|
import { useKeyboardForm, cloneChildrenWithFocusHandler } from "@/hooks/useKeyboardForm";
|
|
|
|
// --- Base Props ---
|
|
interface BaseProps {
|
|
withBackground?: boolean;
|
|
headerComponent?: React.ReactNode;
|
|
footerComponent?: React.ReactNode;
|
|
floatingButton?: React.ReactNode;
|
|
hideFooter?: boolean;
|
|
edgesFooter?: NativeSafeAreaViewProps["edges"];
|
|
style?: StyleProp<ViewStyle>;
|
|
refreshControl?: ScrollViewProps["refreshControl"];
|
|
/**
|
|
* Enable keyboard handling with auto-scroll (Android only)
|
|
* @default false
|
|
*/
|
|
enableKeyboardHandling?: boolean;
|
|
/**
|
|
* Scroll offset when keyboard appears (Android only)
|
|
* @default 100
|
|
*/
|
|
keyboardScrollOffset?: number;
|
|
/**
|
|
* Extra padding bottom for content to avoid navigation bar (Android only)
|
|
* @default 80
|
|
*/
|
|
contentPaddingBottom?: number;
|
|
/**
|
|
* Padding untuk content container (Android only)
|
|
* Set to 0 untuk tidak ada padding, atau custom value sesuai kebutuhan
|
|
* @default 16
|
|
*/
|
|
contentPadding?: number;
|
|
/**
|
|
* Disable flexGrow: 1 in contentContainerStyle
|
|
* Use this for screens with very large headers to fix scroll issues
|
|
* @default false
|
|
*/
|
|
disableFlexGrow?: boolean;
|
|
}
|
|
|
|
interface StaticModeProps extends BaseProps {
|
|
children: React.ReactNode;
|
|
listData?: never;
|
|
renderItem?: never;
|
|
}
|
|
|
|
interface ListModeProps extends BaseProps {
|
|
children?: never;
|
|
listData?: any[];
|
|
renderItem?: FlatListProps<any>["renderItem"];
|
|
onEndReached?: () => void;
|
|
ListHeaderComponent?: React.ReactElement | null;
|
|
ListFooterComponent?: React.ReactElement | null;
|
|
ListEmptyComponent?: React.ReactElement | null;
|
|
keyExtractor?: FlatListProps<any>["keyExtractor"];
|
|
}
|
|
|
|
type AndroidWrapperProps = StaticModeProps | ListModeProps;
|
|
|
|
export function AndroidWrapper(props: AndroidWrapperProps) {
|
|
const {
|
|
withBackground = false,
|
|
headerComponent,
|
|
footerComponent,
|
|
floatingButton,
|
|
hideFooter = false,
|
|
edgesFooter = [],
|
|
style,
|
|
refreshControl,
|
|
enableKeyboardHandling = false,
|
|
keyboardScrollOffset,
|
|
contentPaddingBottom,
|
|
contentPadding,
|
|
disableFlexGrow = false,
|
|
} = props;
|
|
|
|
// Default values (should be set by OS_Wrapper, but fallback for direct usage)
|
|
const finalKeyboardScrollOffset = keyboardScrollOffset ?? 100;
|
|
const finalContentPaddingBottom = contentPaddingBottom ?? 250;
|
|
const finalContentPadding = contentPadding ?? 0;
|
|
|
|
const assetBackground = require("../../assets/images/main-background.png");
|
|
|
|
// Use keyboard hook if enabled
|
|
const keyboardForm = enableKeyboardHandling
|
|
? useKeyboardForm(finalKeyboardScrollOffset)
|
|
: null;
|
|
|
|
const renderContainer = (content: React.ReactNode) => {
|
|
if (withBackground) {
|
|
return (
|
|
<ImageBackground
|
|
source={assetBackground}
|
|
resizeMode="cover"
|
|
style={GStyles.imageBackground}
|
|
>
|
|
<View style={[GStyles.containerWithBackground, style]}>
|
|
{content}
|
|
</View>
|
|
</ImageBackground>
|
|
);
|
|
}
|
|
return <View style={[GStyles.container, style]}>{content}</View>;
|
|
};
|
|
|
|
// 🔹 Mode Dinamis (FlatList)
|
|
if ("listData" in props) {
|
|
const listProps = props as ListModeProps;
|
|
|
|
return (
|
|
<View style={{ flex: 1, backgroundColor: MainColor.darkblue }}>
|
|
{headerComponent && (
|
|
<View style={GStyles.stickyHeader}>{headerComponent}</View>
|
|
)}
|
|
<FlatList
|
|
style={{ flex: 1 }}
|
|
data={listProps.listData}
|
|
renderItem={listProps.renderItem}
|
|
keyExtractor={
|
|
listProps.keyExtractor ||
|
|
((item, index) => `${String(item.id)}-${index}`)
|
|
}
|
|
refreshControl={refreshControl}
|
|
onEndReached={listProps.onEndReached}
|
|
onEndReachedThreshold={0.5}
|
|
ListHeaderComponent={listProps.ListHeaderComponent}
|
|
ListFooterComponent={listProps.ListFooterComponent}
|
|
ListEmptyComponent={listProps.ListEmptyComponent}
|
|
contentContainerStyle={{
|
|
flexGrow: disableFlexGrow ? 0 : 1,
|
|
paddingBottom:
|
|
(footerComponent && !hideFooter ? OS_HEIGHT : 0) +
|
|
finalContentPaddingBottom,
|
|
padding: finalContentPadding,
|
|
}}
|
|
keyboardShouldPersistTaps="handled"
|
|
removeClippedSubviews={false}
|
|
stickyHeaderIndices={[]}
|
|
nestedScrollEnabled={true}
|
|
/>
|
|
|
|
{/* Footer - Fixed di bawah dengan width 100% */}
|
|
{footerComponent && !hideFooter && (
|
|
<SafeAreaView
|
|
edges={["bottom"]}
|
|
style={{ backgroundColor: MainColor.darkblue, width: "100%" }}
|
|
>
|
|
<View style={{ width: "100%" }}>{footerComponent}</View>
|
|
</SafeAreaView>
|
|
)}
|
|
|
|
{!footerComponent && !hideFooter && (
|
|
<SafeAreaView
|
|
edges={["bottom"]}
|
|
style={{ backgroundColor: MainColor.darkblue }}
|
|
/>
|
|
)}
|
|
|
|
{floatingButton && (
|
|
<View style={GStyles.floatingContainer}>{floatingButton}</View>
|
|
)}
|
|
</View>
|
|
);
|
|
}
|
|
|
|
// 🔹 Mode Statis (ScrollView)
|
|
const staticProps = props as StaticModeProps;
|
|
|
|
// Inject focus handler jika keyboard handling enabled
|
|
const childrenWithFocus = enableKeyboardHandling && keyboardForm
|
|
? cloneChildrenWithFocusHandler(staticProps.children, keyboardForm.handleInputFocus)
|
|
: staticProps.children;
|
|
|
|
return (
|
|
<View style={{ flex: 1, backgroundColor: MainColor.darkblue }}>
|
|
{headerComponent && (
|
|
<View style={GStyles.stickyHeader}>{headerComponent}</View>
|
|
)}
|
|
|
|
<ScrollView
|
|
ref={keyboardForm?.scrollViewRef}
|
|
onScroll={keyboardForm?.handleScroll}
|
|
scrollEventThrottle={16}
|
|
refreshControl={refreshControl}
|
|
style={{ flex: 1 }}
|
|
contentContainerStyle={{
|
|
flexGrow: disableFlexGrow ? 0 : 1,
|
|
paddingBottom:
|
|
(footerComponent && !hideFooter ? OS_HEIGHT : 0) +
|
|
finalContentPaddingBottom,
|
|
padding: finalContentPadding,
|
|
}}
|
|
keyboardShouldPersistTaps="handled"
|
|
showsVerticalScrollIndicator={false}
|
|
>
|
|
<TouchableWithoutFeedback onPress={Keyboard.dismiss}>
|
|
{renderContainer(childrenWithFocus)}
|
|
</TouchableWithoutFeedback>
|
|
</ScrollView>
|
|
|
|
{/* Footer - Fixed di bawah dengan width 100% */}
|
|
{footerComponent && !hideFooter && (
|
|
<SafeAreaView
|
|
edges={["bottom"]}
|
|
style={{
|
|
backgroundColor: MainColor.darkblue,
|
|
width: "100%",
|
|
position: "absolute",
|
|
bottom: 0,
|
|
left: 0,
|
|
right: 0,
|
|
}}
|
|
>
|
|
<View style={{ width: "100%" }}>{footerComponent}</View>
|
|
</SafeAreaView>
|
|
)}
|
|
|
|
{!footerComponent && !hideFooter && (
|
|
<SafeAreaView
|
|
edges={["bottom"]}
|
|
style={{ backgroundColor: MainColor.darkblue }}
|
|
/>
|
|
)}
|
|
|
|
{floatingButton && (
|
|
<View style={GStyles.floatingContainer}>{floatingButton}</View>
|
|
)}
|
|
</View>
|
|
);
|
|
}
|
|
|
|
export default AndroidWrapper;
|