[APP-657] Add share list functionality (#863)

* replace delete list button text with icon

* fix mute list styling on desktop

* add share button to nav bar on a list

* fix styling when on profile

* bug: add key to ImageHorzList

* clean up code & refactor

* fix styling for ListItems

* create a reusable ListActions component for actions on a list

* remove dead styles

* add keys to ListActions

* add helpers to set list embed

* render list embeds

* fix list sharing on web

* make style prop optional in ListCard

* update `@atproto/api` to `0.3.13`
zio/stable
Ansh 2023-06-26 10:15:39 -07:00 committed by GitHub
parent 1666a747eb
commit b9abd444e5
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
18 changed files with 320 additions and 145 deletions

View File

@ -23,7 +23,7 @@
"e2e:run": "detox test --configuration ios.sim.debug --take-screenshots all" "e2e:run": "detox test --configuration ios.sim.debug --take-screenshots all"
}, },
"dependencies": { "dependencies": {
"@atproto/api": "0.3.8", "@atproto/api": "^0.3.13",
"@bam.tech/react-native-image-resizer": "^3.0.4", "@bam.tech/react-native-image-resizer": "^3.0.4",
"@braintree/sanitize-url": "^6.0.2", "@braintree/sanitize-url": "^6.0.2",
"@expo/html-elements": "^0.4.2", "@expo/html-elements": "^0.4.2",

View File

@ -155,3 +155,29 @@ export async function getFeedAsEmbed(
}, },
} }
} }
export async function getListAsEmbed(
store: RootStoreModel,
url: string,
): Promise<apilib.ExternalEmbedDraft> {
url = convertBskyAppUrlIfNeeded(url)
const [_0, user, _1, rkey] = url.split('/').filter(Boolean)
const list = makeRecordUri(user, 'app.bsky.graph.list', rkey)
const res = await store.agent.app.bsky.graph.getList({list})
return {
isLoading: false,
uri: list,
meta: {
url: list,
likelyType: LikelyType.AtpData,
title: res.data.list.name,
},
embed: {
$type: 'app.bsky.embed.record',
record: {
uri: res.data.list.uri,
cid: res.data.list.cid,
},
},
}
}

View File

@ -94,6 +94,20 @@ export function isBskyCustomFeedUrl(url: string): boolean {
return false return false
} }
export function isBskyListUrl(url: string): boolean {
if (isBskyAppUrl(url)) {
try {
const urlp = new URL(url)
return /profile\/(?<name>[^/]+)\/lists\/(?<rkey>[^/]+)/i.test(
urlp.pathname,
)
} catch {
console.error('Unexpected error in isBskyListUrl()', url)
}
}
return false
}
export function convertBskyAppUrlIfNeeded(url: string): string { export function convertBskyAppUrlIfNeeded(url: string): string {
if (isBskyAppUrl(url)) { if (isBskyAppUrl(url)) {
try { try {

View File

@ -97,6 +97,10 @@ export class ListModel {
return this.list?.creator.did === this.rootStore.me.did return this.list?.creator.did === this.rootStore.me.did
} }
get isSubscribed() {
return this.list?.viewer?.muted
}
// public api // public api
// = // =

View File

@ -3,9 +3,17 @@ import {useStores} from 'state/index'
import {ImageModel} from 'state/models/media/image' import {ImageModel} from 'state/models/media/image'
import * as apilib from 'lib/api/index' import * as apilib from 'lib/api/index'
import {getLinkMeta} from 'lib/link-meta/link-meta' import {getLinkMeta} from 'lib/link-meta/link-meta'
import {getPostAsQuote, getFeedAsEmbed} from 'lib/link-meta/bsky' import {
getPostAsQuote,
getFeedAsEmbed,
getListAsEmbed,
} from 'lib/link-meta/bsky'
import {downloadAndResize} from 'lib/media/manip' import {downloadAndResize} from 'lib/media/manip'
import {isBskyPostUrl, isBskyCustomFeedUrl} from 'lib/strings/url-helpers' import {
isBskyPostUrl,
isBskyCustomFeedUrl,
isBskyListUrl,
} from 'lib/strings/url-helpers'
import {ComposerOpts} from 'state/models/ui/shell' import {ComposerOpts} from 'state/models/ui/shell'
import {POST_IMG_MAX} from 'lib/constants' import {POST_IMG_MAX} from 'lib/constants'
@ -60,6 +68,24 @@ export function useExternalLinkFetch({
setExtLink(undefined) setExtLink(undefined)
}, },
) )
} else if (isBskyListUrl(extLink.uri)) {
getListAsEmbed(store, extLink.uri).then(
({embed, meta}) => {
if (aborted) {
return
}
setExtLink({
uri: extLink.uri,
isLoading: false,
meta,
embed,
})
},
err => {
store.log.error('Failed to fetch list for embedding', {err})
setExtLink(undefined)
},
)
} else { } else {
getLinkMeta(store, extLink.uri).then(meta => { getLinkMeta(store, extLink.uri).then(meta => {
if (aborted) { if (aborted) {

View File

@ -0,0 +1,83 @@
import React from 'react'
import {StyleSheet, View} from 'react-native'
import {Button} from '../util/forms/Button'
import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome'
import {usePalette} from 'lib/hooks/usePalette'
export const ListActions = ({
muted,
onToggleSubscribed,
onPressEditList,
isOwner,
onPressDeleteList,
onPressShareList,
reversed = false, // Default value of reversed is false
}: {
isOwner: boolean
muted?: boolean
onToggleSubscribed?: () => void
onPressEditList?: () => void
onPressDeleteList?: () => void
onPressShareList?: () => void
reversed?: boolean // New optional prop
}) => {
const pal = usePalette('default')
let buttons = [
<Button
key="subscribeButton"
type={muted ? 'inverted' : 'primary'}
label={muted ? 'Unsubscribe' : 'Subscribe & Mute'}
accessibilityLabel={muted ? 'Unsubscribe' : 'Subscribe and mute'}
accessibilityHint=""
onPress={onToggleSubscribed}
/>,
isOwner && (
<Button
key="editListButton"
type="default"
label="Edit List"
accessibilityLabel="Edit list"
accessibilityHint=""
onPress={onPressEditList}
/>
),
isOwner && (
<Button
key="deleteListButton"
type="default"
testID="deleteListBtn"
accessibilityLabel="Delete list"
accessibilityHint=""
onPress={onPressDeleteList}>
<FontAwesomeIcon icon={['far', 'trash-can']} style={[pal.text]} />
</Button>
),
<Button
key="shareListButton"
type="default"
testID="shareListBtn"
accessibilityLabel="Share list"
accessibilityHint=""
onPress={onPressShareList}>
<FontAwesomeIcon icon={'share'} style={[pal.text]} />
</Button>,
]
// If reversed is true, reverse the array to reverse the order of the buttons
if (reversed) {
buttons = buttons.filter(Boolean).reverse() // filterting out any falsey values and reversing the array
} else {
buttons = buttons.filter(Boolean) // filterting out any falsey values
}
return <View style={styles.headerBtns}>{buttons}</View>
}
const styles = StyleSheet.create({
headerBtns: {
flexDirection: 'row',
gap: 8,
marginTop: 12,
},
})

View File

@ -1,5 +1,5 @@
import React from 'react' import React from 'react'
import {StyleSheet, View} from 'react-native' import {StyleProp, StyleSheet, View, ViewStyle} from 'react-native'
import {AtUri, AppBskyGraphDefs, RichText} from '@atproto/api' import {AtUri, AppBskyGraphDefs, RichText} from '@atproto/api'
import {Link} from '../util/Link' import {Link} from '../util/Link'
import {Text} from '../util/text/Text' import {Text} from '../util/text/Text'
@ -16,12 +16,14 @@ export const ListCard = ({
noBg, noBg,
noBorder, noBorder,
renderButton, renderButton,
style,
}: { }: {
testID?: string testID?: string
list: AppBskyGraphDefs.ListView list: AppBskyGraphDefs.ListView
noBg?: boolean noBg?: boolean
noBorder?: boolean noBorder?: boolean
renderButton?: () => JSX.Element renderButton?: () => JSX.Element
style?: StyleProp<ViewStyle>
}) => { }) => {
const pal = usePalette('default') const pal = usePalette('default')
const store = useStores() const store = useStores()
@ -53,6 +55,7 @@ export const ListCard = ({
pal.border, pal.border,
noBorder && styles.outerNoBorder, noBorder && styles.outerNoBorder,
!noBg && pal.view, !noBg && pal.view,
style,
]} ]}
href={`/profile/${list.creator.did}/lists/${rkey}`} href={`/profile/${list.creator.did}/lists/${rkey}`}
title={list.name} title={list.name}

View File

@ -6,10 +6,10 @@ import {
StyleSheet, StyleSheet,
View, View,
ViewStyle, ViewStyle,
FlatList,
} from 'react-native' } from 'react-native'
import {AppBskyActorDefs, AppBskyGraphDefs, RichText} from '@atproto/api' import {AppBskyActorDefs, AppBskyGraphDefs, RichText} from '@atproto/api'
import {observer} from 'mobx-react-lite' import {observer} from 'mobx-react-lite'
import {FlatList} from '../util/Views'
import {ProfileCardFeedLoadingPlaceholder} from '../util/LoadingPlaceholder' import {ProfileCardFeedLoadingPlaceholder} from '../util/LoadingPlaceholder'
import {ErrorMessage} from '../util/error/ErrorMessage' import {ErrorMessage} from '../util/error/ErrorMessage'
import {LoadMoreRetryBtn} from '../util/LoadMoreRetryBtn' import {LoadMoreRetryBtn} from '../util/LoadMoreRetryBtn'
@ -25,6 +25,7 @@ import {usePalette} from 'lib/hooks/usePalette'
import {useStores} from 'state/index' import {useStores} from 'state/index'
import {s} from 'lib/styles' import {s} from 'lib/styles'
import {isDesktopWeb} from 'platform/detection' import {isDesktopWeb} from 'platform/detection'
import {ListActions} from './ListActions'
const LOADING_ITEM = {_reactKey: '__loading__'} const LOADING_ITEM = {_reactKey: '__loading__'}
const HEADER_ITEM = {_reactKey: '__header__'} const HEADER_ITEM = {_reactKey: '__header__'}
@ -41,6 +42,7 @@ export const ListItems = observer(
onToggleSubscribed, onToggleSubscribed,
onPressEditList, onPressEditList,
onPressDeleteList, onPressDeleteList,
onPressShareList,
renderEmptyState, renderEmptyState,
testID, testID,
headerOffset = 0, headerOffset = 0,
@ -49,9 +51,10 @@ export const ListItems = observer(
style?: StyleProp<ViewStyle> style?: StyleProp<ViewStyle>
scrollElRef?: MutableRefObject<FlatList<any> | null> scrollElRef?: MutableRefObject<FlatList<any> | null>
onPressTryAgain?: () => void onPressTryAgain?: () => void
onToggleSubscribed?: () => void onToggleSubscribed: () => void
onPressEditList?: () => void onPressEditList: () => void
onPressDeleteList?: () => void onPressDeleteList: () => void
onPressShareList: () => void
renderEmptyState?: () => JSX.Element renderEmptyState?: () => JSX.Element
testID?: string testID?: string
headerOffset?: number headerOffset?: number
@ -163,6 +166,7 @@ export const ListItems = observer(
onToggleSubscribed={onToggleSubscribed} onToggleSubscribed={onToggleSubscribed}
onPressEditList={onPressEditList} onPressEditList={onPressEditList}
onPressDeleteList={onPressDeleteList} onPressDeleteList={onPressDeleteList}
onPressShareList={onPressShareList}
/> />
) : null ) : null
} else if (item === ERROR_ITEM) { } else if (item === ERROR_ITEM) {
@ -193,14 +197,17 @@ export const ListItems = observer(
) )
}, },
[ [
list,
onPressTryAgain,
onPressRetryLoadMore,
renderMemberButton, renderMemberButton,
renderEmptyState,
list.list,
list.isOwner,
list.error,
onToggleSubscribed,
onPressEditList, onPressEditList,
onPressDeleteList, onPressDeleteList,
onToggleSubscribed, onPressShareList,
renderEmptyState, onPressTryAgain,
onPressRetryLoadMore,
], ],
) )
@ -257,12 +264,14 @@ const ListHeader = observer(
onToggleSubscribed, onToggleSubscribed,
onPressEditList, onPressEditList,
onPressDeleteList, onPressDeleteList,
onPressShareList,
}: { }: {
list: AppBskyGraphDefs.ListView list: AppBskyGraphDefs.ListView
isOwner: boolean isOwner: boolean
onToggleSubscribed?: () => void onToggleSubscribed: () => void
onPressEditList?: () => void onPressEditList: () => void
onPressDeleteList?: () => void onPressDeleteList: () => void
onPressShareList: () => void
}) => { }) => {
const pal = usePalette('default') const pal = usePalette('default')
const store = useStores() const store = useStores()
@ -301,43 +310,14 @@ const ListHeader = observer(
/> />
)} )}
{isDesktopWeb && ( {isDesktopWeb && (
<View style={styles.headerBtns}> <ListActions
{list.viewer?.muted ? ( isOwner={isOwner}
<Button muted={list.viewer?.muted}
type="inverted" onPressDeleteList={onPressDeleteList}
label="Unsubscribe" onPressEditList={onPressEditList}
accessibilityLabel="Unsubscribe" onToggleSubscribed={onToggleSubscribed}
accessibilityHint="" onPressShareList={onPressShareList}
onPress={onToggleSubscribed} />
/>
) : (
<Button
type="primary"
label="Subscribe & Mute"
accessibilityLabel="Subscribe and mute"
accessibilityHint=""
onPress={onToggleSubscribed}
/>
)}
{isOwner && (
<Button
type="default"
label="Edit List"
accessibilityLabel="Edit list"
accessibilityHint=""
onPress={onPressEditList}
/>
)}
{isOwner && (
<Button
type="default"
label="Delete List"
accessibilityLabel="Delete list"
accessibilityHint=""
onPress={onPressDeleteList}
/>
)}
</View>
)} )}
</View> </View>
<View> <View>

View File

@ -6,6 +6,7 @@ import {
StyleSheet, StyleSheet,
View, View,
ViewStyle, ViewStyle,
FlatList,
} from 'react-native' } from 'react-native'
import {observer} from 'mobx-react-lite' import {observer} from 'mobx-react-lite'
import { import {
@ -13,7 +14,6 @@ import {
FontAwesomeIconStyle, FontAwesomeIconStyle,
} from '@fortawesome/react-native-fontawesome' } from '@fortawesome/react-native-fontawesome'
import {AppBskyGraphDefs as GraphDefs} from '@atproto/api' import {AppBskyGraphDefs as GraphDefs} from '@atproto/api'
import {FlatList} from '../util/Views'
import {ListCard} from './ListCard' import {ListCard} from './ListCard'
import {ProfileCardFeedLoadingPlaceholder} from '../util/LoadingPlaceholder' import {ProfileCardFeedLoadingPlaceholder} from '../util/LoadingPlaceholder'
import {ErrorMessage} from '../util/error/ErrorMessage' import {ErrorMessage} from '../util/error/ErrorMessage'
@ -149,7 +149,11 @@ export const ListsList = observer(
return renderItem ? ( return renderItem ? (
renderItem(item) renderItem(item)
) : ( ) : (
<ListCard list={item} testID={`list-${item.name}`} /> <ListCard
list={item}
testID={`list-${item.name}`}
style={styles.item}
/>
) )
}, },
[ [
@ -193,7 +197,7 @@ export const ListsList = observer(
progressViewOffset={headerOffset} progressViewOffset={headerOffset}
/> />
} }
contentContainerStyle={s.contentContainer} contentContainerStyle={[s.contentContainer]}
style={{paddingTop: headerOffset}} style={{paddingTop: headerOffset}}
onEndReached={onEndReached} onEndReached={onEndReached}
onEndReachedThreshold={0.6} onEndReachedThreshold={0.6}
@ -237,4 +241,7 @@ const styles = StyleSheet.create({
gap: 8, gap: 8,
}, },
feedFooter: {paddingTop: 20}, feedFooter: {paddingTop: 20},
item: {
paddingHorizontal: 18,
},
}) })

View File

@ -13,6 +13,7 @@ export function ImageHorzList({images, style}: Props) {
<View style={[styles.flexRow, style]}> <View style={[styles.flexRow, style]}>
{images.map(({thumb, alt}) => ( {images.map(({thumb, alt}) => (
<Image <Image
key={thumb}
source={{uri: thumb}} source={{uri: thumb}}
style={styles.image} style={styles.image}
accessible={true} accessible={true}

View File

@ -0,0 +1,37 @@
import React, {useMemo} from 'react'
import {AppBskyFeedDefs} from '@atproto/api'
import {usePalette} from 'lib/hooks/usePalette'
import {StyleSheet} from 'react-native'
import {useStores} from 'state/index'
import {CustomFeedModel} from 'state/models/feeds/custom-feed'
import {CustomFeed} from 'view/com/feeds/CustomFeed'
export function CustomFeedEmbed({
record,
}: {
record: AppBskyFeedDefs.GeneratorView
}) {
const pal = usePalette('default')
const store = useStores()
const item = useMemo(
() => new CustomFeedModel(store, record),
[store, record],
)
return (
<CustomFeed
item={item}
style={[pal.view, pal.border, styles.customFeedOuter]}
showLikes
/>
)
}
const styles = StyleSheet.create({
customFeedOuter: {
borderWidth: 1,
borderRadius: 8,
marginTop: 4,
paddingHorizontal: 12,
paddingVertical: 12,
},
})

View File

@ -0,0 +1,35 @@
import React from 'react'
import {StyleProp, StyleSheet, View, ViewStyle} from 'react-native'
import {usePalette} from 'lib/hooks/usePalette'
import {observer} from 'mobx-react-lite'
import {ListCard} from 'view/com/lists/ListCard'
import {AppBskyGraphDefs} from '@atproto/api'
import {s} from 'lib/styles'
export const ListEmbed = observer(
({
item,
style,
}: {
item: AppBskyGraphDefs.ListView
style?: StyleProp<ViewStyle>
}) => {
const pal = usePalette('default')
return (
<View style={[pal.view, pal.border, s.border1, styles.container]}>
<ListCard list={item} style={[style, styles.card]} />
</View>
)
},
)
const styles = StyleSheet.create({
container: {
borderRadius: 8,
},
card: {
borderTopWidth: 0,
borderRadius: 8,
},
})

View File

@ -14,6 +14,7 @@ import {
AppBskyEmbedRecordWithMedia, AppBskyEmbedRecordWithMedia,
AppBskyFeedPost, AppBskyFeedPost,
AppBskyFeedDefs, AppBskyFeedDefs,
AppBskyGraphDefs,
} from '@atproto/api' } from '@atproto/api'
import {Link} from '../Link' import {Link} from '../Link'
import {ImageLayoutGrid} from '../images/ImageLayoutGrid' import {ImageLayoutGrid} from '../images/ImageLayoutGrid'
@ -25,8 +26,8 @@ import {ExternalLinkEmbed} from './ExternalLinkEmbed'
import {getYoutubeVideoId} from 'lib/strings/url-helpers' import {getYoutubeVideoId} from 'lib/strings/url-helpers'
import QuoteEmbed from './QuoteEmbed' import QuoteEmbed from './QuoteEmbed'
import {AutoSizedImage} from '../images/AutoSizedImage' import {AutoSizedImage} from '../images/AutoSizedImage'
import {CustomFeed} from 'view/com/feeds/CustomFeed' import {CustomFeedEmbed} from './CustomFeedEmbed'
import {CustomFeedModel} from 'state/models/feeds/custom-feed' import {ListEmbed} from './ListEmbed'
type Embed = type Embed =
| AppBskyEmbedRecord.View | AppBskyEmbedRecord.View
@ -144,6 +145,23 @@ export function PostEmbeds({
} }
} }
// custom feed embed (i.e. generator view)
// =
if (
AppBskyEmbedRecord.isView(embed) &&
AppBskyFeedDefs.isGeneratorView(embed.record)
) {
return <CustomFeedEmbed record={embed.record} />
}
// list embed (e.g. mute lists; i.e. ListView)
if (
AppBskyEmbedRecord.isView(embed) &&
AppBskyGraphDefs.isListView(embed.record)
) {
return <ListEmbed item={embed.record} />
}
// external link embed // external link embed
// = // =
if (AppBskyEmbedExternal.isView(embed)) { if (AppBskyEmbedExternal.isView(embed)) {
@ -164,34 +182,9 @@ export function PostEmbeds({
) )
} }
// custom feed embed (i.e. generator view)
// =
if (
AppBskyEmbedRecord.isView(embed) &&
AppBskyFeedDefs.isGeneratorView(embed.record)
) {
return <CustomFeedEmbed record={embed.record} />
}
return <View /> return <View />
} }
function CustomFeedEmbed({record}: {record: AppBskyFeedDefs.GeneratorView}) {
const pal = usePalette('default')
const store = useStores()
const item = React.useMemo(
() => new CustomFeedModel(store, record),
[store, record],
)
return (
<CustomFeed
item={item}
style={[pal.view, pal.border, styles.customFeedOuter]}
showLikes
/>
)
}
const styles = StyleSheet.create({ const styles = StyleSheet.create({
stackContainer: { stackContainer: {
gap: 6, gap: 6,
@ -208,13 +201,6 @@ const styles = StyleSheet.create({
borderRadius: 8, borderRadius: 8,
marginTop: 4, marginTop: 4,
}, },
customFeedOuter: {
borderWidth: 1,
borderRadius: 8,
marginTop: 4,
paddingHorizontal: 12,
paddingVertical: 12,
},
alt: { alt: {
backgroundColor: 'rgba(0, 0, 0, 0.75)', backgroundColor: 'rgba(0, 0, 0, 0.75)',
borderRadius: 6, borderRadius: 6,

View File

@ -87,9 +87,9 @@ export const ModerationMuteListsScreen = withAuthRequired(({}: Props) => {
<CenteredView <CenteredView
style={[ style={[
styles.container, styles.container,
isDesktopWeb && styles.containerDesktop,
pal.view, pal.view,
pal.border, pal.border,
isDesktopWeb && styles.containerDesktop,
]} ]}
testID="moderationMutelistsScreen"> testID="moderationMutelistsScreen">
<ViewHeader <ViewHeader

View File

@ -1,5 +1,5 @@
import React from 'react' import React from 'react'
import {StyleSheet, View} from 'react-native' import {StyleSheet} from 'react-native'
import {useFocusEffect} from '@react-navigation/native' import {useFocusEffect} from '@react-navigation/native'
import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types' import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types'
import {useNavigation} from '@react-navigation/native' import {useNavigation} from '@react-navigation/native'
@ -9,7 +9,6 @@ import {ViewHeader} from 'view/com/util/ViewHeader'
import {CenteredView} from 'view/com/util/Views' import {CenteredView} from 'view/com/util/Views'
import {ListItems} from 'view/com/lists/ListItems' import {ListItems} from 'view/com/lists/ListItems'
import {EmptyState} from 'view/com/util/EmptyState' import {EmptyState} from 'view/com/util/EmptyState'
import {Button} from 'view/com/util/forms/Button'
import * as Toast from 'view/com/util/Toast' import * as Toast from 'view/com/util/Toast'
import {ListModel} from 'state/models/content/list' import {ListModel} from 'state/models/content/list'
import {useStores} from 'state/index' import {useStores} from 'state/index'
@ -17,6 +16,9 @@ import {usePalette} from 'lib/hooks/usePalette'
import {useSetTitle} from 'lib/hooks/useSetTitle' import {useSetTitle} from 'lib/hooks/useSetTitle'
import {NavigationProp} from 'lib/routes/types' import {NavigationProp} from 'lib/routes/types'
import {isDesktopWeb} from 'platform/detection' import {isDesktopWeb} from 'platform/detection'
import {toShareUrl} from 'lib/strings/url-helpers'
import {shareUrl} from 'lib/sharing'
import {ListActions} from 'view/com/lists/ListActions'
type Props = NativeStackScreenProps<CommonNavigatorParams, 'ProfileList'> type Props = NativeStackScreenProps<CommonNavigatorParams, 'ProfileList'>
export const ProfileListScreen = withAuthRequired( export const ProfileListScreen = withAuthRequired(
@ -71,7 +73,7 @@ export const ProfileListScreen = withAuthRequired(
store.shell.openModal({ store.shell.openModal({
name: 'confirm', name: 'confirm',
title: 'Delete List', title: 'Delete List',
message: 'Are you sure?', message: 'Are you sure',
async onPressConfirm() { async onPressConfirm() {
await list.delete() await list.delete()
if (navigation.canGoBack()) { if (navigation.canGoBack()) {
@ -83,59 +85,33 @@ export const ProfileListScreen = withAuthRequired(
}) })
}, [store, list, navigation]) }, [store, list, navigation])
const onPressShareList = React.useCallback(() => {
const url = toShareUrl(`/profile/${name}/lists/${rkey}`)
shareUrl(url)
}, [name, rkey])
const renderEmptyState = React.useCallback(() => { const renderEmptyState = React.useCallback(() => {
return <EmptyState icon="users-slash" message="This list is empty!" /> return <EmptyState icon="users-slash" message="This list is empty!" />
}, []) }, [])
const renderHeaderBtns = React.useCallback(() => { const renderHeaderBtns = React.useCallback(() => {
return ( return (
<View style={styles.headerBtns}> <ListActions
{list?.isOwner && ( muted={list.list?.viewer?.muted}
<Button isOwner={list.isOwner}
type="default" onPressDeleteList={onPressDeleteList}
label="Delete List" onPressEditList={onPressEditList}
testID="deleteListBtn" onToggleSubscribed={onToggleSubscribed}
accessibilityLabel="Delete list" onPressShareList={onPressShareList}
accessibilityHint="" reversed={true}
onPress={onPressDeleteList} />
/>
)}
{list?.isOwner && (
<Button
type="default"
label="Edit List"
testID="editListBtn"
accessibilityLabel="Edit list"
accessibilityHint=""
onPress={onPressEditList}
/>
)}
{list.list?.viewer?.muted ? (
<Button
type="inverted"
label="Unsubscribe"
testID="unsubscribeListBtn"
accessibilityLabel="Unsubscribe from list"
accessibilityHint=""
onPress={onToggleSubscribed}
/>
) : (
<Button
type="primary"
label="Subscribe & Mute"
testID="subscribeListBtn"
accessibilityLabel="Subscribe to this list"
accessibilityHint="Mutes the users included in this list"
onPress={onToggleSubscribed}
/>
)}
</View>
) )
}, [ }, [
list?.isOwner, list.isOwner,
list.list?.viewer?.muted, list.list?.viewer?.muted,
onPressDeleteList, onPressDeleteList,
onPressEditList, onPressEditList,
onPressShareList,
onToggleSubscribed, onToggleSubscribed,
]) ])
@ -155,6 +131,7 @@ export const ProfileListScreen = withAuthRequired(
onToggleSubscribed={onToggleSubscribed} onToggleSubscribed={onToggleSubscribed}
onPressEditList={onPressEditList} onPressEditList={onPressEditList}
onPressDeleteList={onPressDeleteList} onPressDeleteList={onPressDeleteList}
onPressShareList={onPressShareList}
/> />
</CenteredView> </CenteredView>
) )
@ -162,10 +139,6 @@ export const ProfileListScreen = withAuthRequired(
) )
const styles = StyleSheet.create({ const styles = StyleSheet.create({
headerBtns: {
flexDirection: 'row',
gap: 8,
},
container: { container: {
flex: 1, flex: 1,
paddingBottom: isDesktopWeb ? 0 : 100, paddingBottom: isDesktopWeb ? 0 : 100,

View File

@ -189,6 +189,7 @@ export const BottomBar = observer(({navigation}: BottomTabBarProps) => {
pal.text, pal.text,
styles.profileIcon, styles.profileIcon,
styles.onProfile, styles.onProfile,
{borderColor: pal.text.color},
]}> ]}>
<UserAvatar avatar={store.me.avatar} size={27} /> <UserAvatar avatar={store.me.avatar} size={27} />
</View> </View>

View File

@ -59,7 +59,6 @@ export const styles = StyleSheet.create({
top: -4, top: -4,
}, },
onProfile: { onProfile: {
borderColor: colors.black,
borderWidth: 1, borderWidth: 1,
borderRadius: 100, borderRadius: 100,
}, },

View File

@ -40,10 +40,10 @@
tlds "^1.234.0" tlds "^1.234.0"
typed-emitter "^2.1.0" typed-emitter "^2.1.0"
"@atproto/api@0.3.8": "@atproto/api@^0.3.13":
version "0.3.8" version "0.3.13"
resolved "https://registry.yarnpkg.com/@atproto/api/-/api-0.3.8.tgz#3fc0ebd092cc212c2d0b31a600fe1945a02f9cf7" resolved "https://registry.yarnpkg.com/@atproto/api/-/api-0.3.13.tgz#e5ccaa83bb909e662286cdf74a77a76de6562a47"
integrity sha512-7qaIZGEP5J9FW4z8bXezzAmLRzHSXXHo6bWP9Jyu5MLp8tYt9vG6yR2N0QA7GvO0xSYqP87Q5vblPjYXGqtDKg== integrity sha512-smDlomgipca16G+jKXAZSMfsAmA5wG8WR3Z1dj29ZShVJlhs6+HHdxX7dWVDYEdSeb2rp/wyHN/tQhxGDAkz/g==
dependencies: dependencies:
"@atproto/common-web" "*" "@atproto/common-web" "*"
"@atproto/uri" "*" "@atproto/uri" "*"