Add WIP 'report post' modal

zio/stable
Paul Frazee 2022-12-18 17:28:28 -06:00
parent 36dc1c7525
commit 66a0f8e848
7 changed files with 211 additions and 2 deletions

View File

@ -51,6 +51,14 @@ export class ServerInputModal {
} }
} }
export class ReportPostModal {
name = 'report-post'
constructor(public postUrl: string) {
makeAutoObservable(this)
}
}
interface LightboxModel { interface LightboxModel {
canSwipeLeft: boolean canSwipeLeft: boolean
canSwipeRight: boolean canSwipeRight: boolean
@ -127,6 +135,7 @@ export class ShellUiModel {
| EditProfileModal | EditProfileModal
| CreateSceneModal | CreateSceneModal
| ServerInputModal | ServerInputModal
| ReportPostModal
| undefined | undefined
isLightboxActive = false isLightboxActive = false
activeLightbox: activeLightbox:
@ -154,7 +163,8 @@ export class ShellUiModel {
| ConfirmModal | ConfirmModal
| EditProfileModal | EditProfileModal
| CreateSceneModal | CreateSceneModal
| ServerInputModal, | ServerInputModal
| ReportPostModal,
) { ) {
this.isModalActive = true this.isModalActive = true
this.activeModal = modal this.activeModal = modal

View File

@ -12,6 +12,7 @@ import * as EditProfileModal from './EditProfile'
import * as CreateSceneModal from './CreateScene' import * as CreateSceneModal from './CreateScene'
import * as InviteToSceneModal from './InviteToScene' import * as InviteToSceneModal from './InviteToScene'
import * as ServerInputModal from './ServerInput' import * as ServerInputModal from './ServerInput'
import * as ReportPostModal from './ReportPost'
const CLOSED_SNAPPOINTS = ['10%'] const CLOSED_SNAPPOINTS = ['10%']
@ -70,6 +71,13 @@ export const Modal = observer(function Modal() {
{...(store.shell.activeModal as models.ServerInputModal)} {...(store.shell.activeModal as models.ServerInputModal)}
/> />
) )
} else if (store.shell.activeModal?.name === 'report-post') {
snapPoints = ReportPostModal.snapPoints
element = (
<ReportPostModal.Component
{...(store.shell.activeModal as models.ReportPostModal)}
/>
)
} else { } else {
element = <View /> element = <View />
} }

View File

@ -0,0 +1,94 @@
import React, {useState} from 'react'
import {
ActivityIndicator,
StyleSheet,
Text,
TouchableOpacity,
View,
} from 'react-native'
import LinearGradient from 'react-native-linear-gradient'
import {useStores} from '../../../state'
import {s, colors, gradients} from '../../lib/styles'
import {RadioGroup, RadioGroupItem} from '../util/forms/RadioGroup'
import {ErrorMessage} from '../util/ErrorMessage'
const ITEMS: RadioGroupItem[] = [
{key: 'spam', label: 'Spam or excessive repeat posts'},
{key: 'abuse', label: 'Abusive, rude, or hateful'},
{key: 'copyright', label: 'Contains copyrighted material'},
{key: 'illegal', label: 'Contains illegal content'},
]
export const snapPoints = ['50%']
export function Component({postUrl}: {postUrl: string}) {
const store = useStores()
const [isProcessing, setIsProcessing] = useState<boolean>(false)
const [error, setError] = useState<string>('')
const [issue, setIssue] = useState<string>('')
const onSelectIssue = (v: string) => setIssue(v)
const onPress = async () => {
setError('')
setIsProcessing(true)
try {
// TODO
store.shell.closeModal()
return
} catch (e: any) {
setError(e.toString())
setIsProcessing(false)
}
}
return (
<View style={[s.flex1, s.pl10, s.pr10]}>
<Text style={styles.title}>Report post</Text>
<Text style={styles.description}>What is the issue with this post?</Text>
<RadioGroup items={ITEMS} onSelect={onSelectIssue} />
{error ? (
<View style={s.mt10}>
<ErrorMessage message={error} />
</View>
) : undefined}
{isProcessing ? (
<View style={[styles.btn, s.mt10]}>
<ActivityIndicator />
</View>
) : issue ? (
<TouchableOpacity style={s.mt10} onPress={onPress}>
<LinearGradient
colors={[gradients.primary.start, gradients.primary.end]}
start={{x: 0, y: 0}}
end={{x: 1, y: 1}}
style={[styles.btn]}>
<Text style={[s.white, s.bold, s.f18]}>Send Report</Text>
</LinearGradient>
</TouchableOpacity>
) : undefined}
</View>
)
}
const styles = StyleSheet.create({
title: {
textAlign: 'center',
fontWeight: 'bold',
fontSize: 24,
marginBottom: 12,
},
description: {
textAlign: 'center',
fontSize: 17,
paddingHorizontal: 22,
color: colors.gray5,
marginBottom: 10,
},
btn: {
flexDirection: 'row',
alignItems: 'center',
justifyContent: 'center',
width: '100%',
borderRadius: 32,
padding: 14,
backgroundColor: colors.gray1,
},
})

View File

@ -15,7 +15,7 @@ import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome'
import {colors} from '../../lib/styles' import {colors} from '../../lib/styles'
import {toShareUrl} from '../../../lib/strings' import {toShareUrl} from '../../../lib/strings'
import {useStores} from '../../../state' import {useStores} from '../../../state'
import {ConfirmModal} from '../../../state/models/shell-ui' import {ReportPostModal, ConfirmModal} from '../../../state/models/shell-ui'
import {TABS_ENABLED} from '../../../build-flags' import {TABS_ENABLED} from '../../../build-flags'
const HITSLOP = {left: 10, top: 10, right: 10, bottom: 10} const HITSLOP = {left: 10, top: 10, right: 10, bottom: 10}
@ -116,6 +116,13 @@ export function PostDropdownBtn({
Share.share({url: toShareUrl(itemHref)}) Share.share({url: toShareUrl(itemHref)})
}, },
}, },
{
icon: 'circle-exclamation',
label: 'Report post',
onPress() {
store.shell.openModal(new ReportPostModal(itemHref))
},
},
isAuthor isAuthor
? { ? {
icon: ['far', 'trash-can'], icon: ['far', 'trash-can'],

View File

@ -0,0 +1,54 @@
import React from 'react'
import {StyleSheet, Text, TouchableOpacity, View} from 'react-native'
import {colors} from '../../../lib/styles'
export function RadioButton({
label,
isSelected,
onPress,
}: {
label: string
isSelected: boolean
onPress: () => void
}) {
return (
<TouchableOpacity style={styles.outer} onPress={onPress}>
<View style={styles.circle}>
{isSelected ? <View style={styles.circleFill} /> : undefined}
</View>
<Text style={styles.label}>{label}</Text>
</TouchableOpacity>
)
}
const styles = StyleSheet.create({
outer: {
flexDirection: 'row',
alignItems: 'center',
marginBottom: 5,
borderRadius: 8,
borderWidth: 1,
borderColor: colors.gray2,
paddingHorizontal: 10,
paddingVertical: 8,
},
circle: {
width: 30,
height: 30,
borderRadius: 15,
padding: 4,
borderWidth: 1,
borderColor: colors.gray3,
marginRight: 10,
},
circleFill: {
width: 20,
height: 20,
borderRadius: 10,
backgroundColor: colors.blue3,
},
label: {
flex: 1,
fontSize: 17,
},
})

View File

@ -0,0 +1,34 @@
import React, {useState} from 'react'
import {View} from 'react-native'
import {RadioButton} from './RadioButton'
export interface RadioGroupItem {
label: string
key: string
}
export function RadioGroup({
items,
onSelect,
}: {
items: RadioGroupItem[]
onSelect: (key: string) => void
}) {
const [selection, setSelection] = useState<string>('')
const onSelectInner = (key: string) => {
setSelection(key)
onSelect(key)
}
return (
<View>
{items.map(item => (
<RadioButton
key={item.key}
label={item.label}
isSelected={item.key === selection}
onPress={() => onSelectInner(item.key)}
/>
))}
</View>
)
}

View File

@ -21,6 +21,7 @@ import {faBookmark as farBookmark} from '@fortawesome/free-regular-svg-icons/faB
import {faCamera} from '@fortawesome/free-solid-svg-icons/faCamera' import {faCamera} from '@fortawesome/free-solid-svg-icons/faCamera'
import {faCheck} from '@fortawesome/free-solid-svg-icons/faCheck' import {faCheck} from '@fortawesome/free-solid-svg-icons/faCheck'
import {faCircleCheck} from '@fortawesome/free-regular-svg-icons/faCircleCheck' import {faCircleCheck} from '@fortawesome/free-regular-svg-icons/faCircleCheck'
import {faCircleExclamation} from '@fortawesome/free-solid-svg-icons/faCircleExclamation'
import {faCircleUser} from '@fortawesome/free-regular-svg-icons/faCircleUser' import {faCircleUser} from '@fortawesome/free-regular-svg-icons/faCircleUser'
import {faClone} from '@fortawesome/free-solid-svg-icons/faClone' import {faClone} from '@fortawesome/free-solid-svg-icons/faClone'
import {faClone as farClone} from '@fortawesome/free-regular-svg-icons/faClone' import {faClone as farClone} from '@fortawesome/free-regular-svg-icons/faClone'
@ -86,6 +87,7 @@ export function setup() {
faCamera, faCamera,
faCheck, faCheck,
faCircleCheck, faCircleCheck,
faCircleExclamation,
faCircleUser, faCircleUser,
faClone, faClone,
farClone, farClone,