Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Lists updates: curate lists and blocklists #1689

Merged
merged 67 commits into from
Nov 1, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
67 commits
Select commit Hold shift + click to select a range
833f525
Add lists screen
pfrazee Oct 9, 2023
5546ad8
Update Lists screen and List create/edit modal to support curate lists
pfrazee Oct 10, 2023
fa599a6
Rework the ProfileList screen and add curatelist support
pfrazee Oct 10, 2023
3f2b787
More ProfileList progress
pfrazee Oct 11, 2023
e41e65e
Update list modals
pfrazee Oct 11, 2023
207a861
Rename mutelists to modlists
pfrazee Oct 11, 2023
8e08f3d
Layout updates/fixes
pfrazee Oct 11, 2023
dc28bde
More layout fixes
pfrazee Oct 11, 2023
743ca9d
Modal fixes
pfrazee Oct 11, 2023
276aa7f
List list screen updates
pfrazee Oct 11, 2023
7c42ec9
Update feed page to give more info
pfrazee Oct 12, 2023
059d908
Layout fixes to ListAddUser modal
pfrazee Oct 12, 2023
744d0be
Layout fixes to FlatList and Feed on desktop
pfrazee Oct 12, 2023
fd7182b
Layout fix to LoadLatestBtn on Web
pfrazee Oct 12, 2023
6e2d37b
Handle did resolution before showing the ProfileList screen
pfrazee Oct 12, 2023
d7753e0
Rename the CustomFeed routes to ProfileFeed for consistency
pfrazee Oct 12, 2023
1c76aba
Fix layout issues with the pager and feeds
pfrazee Oct 12, 2023
922615c
Factor out some common code
pfrazee Oct 12, 2023
9bc1352
Fix UIs for mobile
pfrazee Oct 12, 2023
b843094
Fix user list rendering
pfrazee Oct 12, 2023
95340ca
Fix: dont bubble custom feed errors in the merge feed
pfrazee Oct 12, 2023
a30e846
Refactor feed models to reduce usage of the SavedFeeds model
pfrazee Oct 12, 2023
1baefab
Replace CustomFeedModel with FeedSourceModel which abstracts feed-gen…
pfrazee Oct 13, 2023
fc0932f
Add the ability to pin lists
pfrazee Oct 13, 2023
1ef1160
Add pinned lists to mobile
pfrazee Oct 13, 2023
f085a91
Remove dead code
pfrazee Oct 13, 2023
6d576c7
Rework the ProfileScreenHeader to create more real-estate for action …
pfrazee Oct 14, 2023
a823f94
Improve layout behavior on web mobile breakpoints
pfrazee Oct 14, 2023
cf57252
Refactor feed & list pages to use new Tabs layout component
pfrazee Oct 25, 2023
db6fbd6
Refactor to ProfileSubpageHeader
pfrazee Oct 26, 2023
7109d97
Implement modlist block and mute
pfrazee Oct 26, 2023
f11e9f3
Switch to new api and just modify state on modlist actions
pfrazee Oct 26, 2023
499eea2
Fix some UI overflows
pfrazee Oct 26, 2023
8204959
Fix: dont show edit buttons on lists you dont own
pfrazee Oct 26, 2023
bcf591b
Fix alignment issue on long titles
pfrazee Oct 26, 2023
e5b0658
Improve loading and error states for feeds & lists
pfrazee Oct 26, 2023
ff487f7
Update list dropdown icons for ios
pfrazee Oct 26, 2023
3edf5d6
Fetch feed display names in the mergefeed
pfrazee Oct 26, 2023
259e0b8
Improve rendering off offline feeds in the feed-listing page
pfrazee Oct 26, 2023
21d23fc
Update Feeds listing UI to react to changes in saved/pinned state
pfrazee Oct 26, 2023
8dd228d
Refresh list and feed on posts tab press
pfrazee Oct 26, 2023
17e308b
Fix pinned feed ordering UI
pfrazee Oct 26, 2023
0eb9b3a
Fixes to list pinning
pfrazee Oct 26, 2023
647d0a8
Remove view=simple qp
pfrazee Oct 26, 2023
9a28f2f
Add list to feed tuners
pfrazee Oct 26, 2023
9472ddc
Render richtext
pfrazee Oct 26, 2023
31fe65a
Add list href
pfrazee Oct 26, 2023
55990ea
Add 'view avatar'
pfrazee Oct 26, 2023
cb9954d
Remove unused import
pfrazee Oct 26, 2023
c104dba
Fix missing import
pfrazee Oct 26, 2023
a0a8f08
Correctly reflect block by list state
pfrazee Oct 31, 2023
dce0fe9
Replace the <Tabs> component with the more effective <PagerWithHeader…
pfrazee Nov 1, 2023
b9ed8d6
Improve the responsiveness of the PagerWithHeader
pfrazee Nov 1, 2023
05a4406
Fix visual jank in the feed loading state
pfrazee Nov 1, 2023
9ed23af
Improve performance of the PagerWithHeader
pfrazee Nov 1, 2023
ab89453
Fix a case that would cause the header to animate too aggressively
pfrazee Nov 1, 2023
b2bafc7
Add the ability to scroll to top by tapping the selected tab
pfrazee Nov 1, 2023
c5a265d
Fix unit test runner
pfrazee Nov 1, 2023
daf8f95
Update modlists test
pfrazee Nov 1, 2023
2b7b07c
Add curatelist tests
pfrazee Nov 1, 2023
db519a0
Fix: remove link behavior in ListAddUser modal
pfrazee Nov 1, 2023
4ce013b
Fix some layout jank in the PagerWithHeader on iOS
pfrazee Nov 1, 2023
0525ce0
Simplify ListItems header rendering
pfrazee Nov 1, 2023
5b73242
Wait for the appview to recognize the list before proceeding with lis…
pfrazee Nov 1, 2023
d43cefc
Fix glitch in the onPageSelecting index of the Pager
pfrazee Nov 1, 2023
f28aaac
Fix until()
pfrazee Nov 1, 2023
dd16aa9
Copy fix
pfrazee Nov 1, 2023
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
208 changes: 208 additions & 0 deletions __e2e__/tests/curate-lists.test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,208 @@
/* eslint-env detox/detox */

import {openApp, loginAsAlice, loginAsBob, createServer, sleep} from '../util'

describe('Curate lists', () => {
beforeAll(async () => {
await createServer('?users&follows&posts')
await openApp({
permissions: {notifications: 'YES', medialibrary: 'YES', photos: 'YES'},
})
})

it('Login and create a curatelists', async () => {
await expect(element(by.id('signInButton'))).toBeVisible()
await loginAsAlice()
await element(by.id('e2eGotoLists')).tap()
await element(by.id('newUserListBtn')).tap()
await expect(element(by.id('createOrEditListModal'))).toBeVisible()
await element(by.id('editNameInput')).typeText('Good Ppl')
await element(by.id('editDescriptionInput')).typeText('They good')
await element(by.id('saveBtn')).tap()
await expect(element(by.id('createOrEditListModal'))).not.toBeVisible()
await element(by.text('About')).tap()
await expect(element(by.id('headerTitle'))).toHaveText('Good Ppl')
await expect(element(by.id('listDescription'))).toHaveText('They good')
})

it('Edit display name and description via the edit curatelist modal', async () => {
await element(by.id('headerDropdownBtn')).tap()
await element(by.text('Edit List Details')).tap()
await expect(element(by.id('createOrEditListModal'))).toBeVisible()
await element(by.id('editNameInput')).clearText()
await element(by.id('editNameInput')).typeText('Bad Ppl')
await element(by.id('editDescriptionInput')).clearText()
await element(by.id('editDescriptionInput')).typeText('They bad')
await element(by.id('saveBtn')).tap()
await expect(element(by.id('createOrEditListModal'))).not.toBeVisible()
await expect(element(by.id('headerTitle'))).toHaveText('Bad Ppl')
await expect(element(by.id('listDescription'))).toHaveText('They bad')
// have to wait for the toast to clear
await waitFor(element(by.id('headerDropdownBtn')))
.toBeVisible()
.withTimeout(5000)
})

it('Remove description via the edit curatelist modal', async () => {
await element(by.id('headerDropdownBtn')).tap()
await element(by.text('Edit List Details')).tap()
await expect(element(by.id('createOrEditListModal'))).toBeVisible()
await element(by.id('editDescriptionInput')).clearText()
await element(by.id('saveBtn')).tap()
await expect(element(by.id('createOrEditListModal'))).not.toBeVisible()
await expect(element(by.id('listDescription'))).not.toBeVisible()
// have to wait for the toast to clear
await waitFor(element(by.id('headerDropdownBtn')))
.toBeVisible()
.withTimeout(5000)
})

it('Set avi via the edit curatelist modal', async () => {
await expect(element(by.id('userAvatarFallback'))).toExist()
await element(by.id('headerDropdownBtn')).tap()
await element(by.text('Edit List Details')).tap()
await expect(element(by.id('createOrEditListModal'))).toBeVisible()
await element(by.id('changeAvatarBtn')).tap()
await element(by.text('Library')).tap()
await sleep(3e3)
await element(by.id('saveBtn')).tap()
await expect(element(by.id('createOrEditListModal'))).not.toBeVisible()
await expect(element(by.id('userAvatarImage'))).toExist()
// have to wait for the toast to clear
await waitFor(element(by.id('headerDropdownBtn')))
.toBeVisible()
.withTimeout(5000)
})

it('Remove avi via the edit curatelist modal', async () => {
await expect(element(by.id('userAvatarImage'))).toExist()
await element(by.id('headerDropdownBtn')).tap()
await element(by.text('Edit List Details')).tap()
await expect(element(by.id('createOrEditListModal'))).toBeVisible()
await element(by.id('changeAvatarBtn')).tap()
await element(by.text('Remove')).tap()
await element(by.id('saveBtn')).tap()
await expect(element(by.id('createOrEditListModal'))).not.toBeVisible()
await expect(element(by.id('userAvatarFallback'))).toExist()
// have to wait for the toast to clear
await waitFor(element(by.id('headerDropdownBtn')))
.toBeVisible()
.withTimeout(5000)
})

it('Delete the curatelist', async () => {
await element(by.id('headerDropdownBtn')).tap()
await element(by.text('Delete List')).tap()
await element(by.id('confirmBtn')).tap()
await expect(element(by.id('listsEmpty'))).toBeVisible()
})

it('Create a new curatelist', async () => {
await element(by.id('newUserListBtn')).tap()
await expect(element(by.id('createOrEditListModal'))).toBeVisible()
await element(by.id('editNameInput')).typeText('Good Ppl')
await element(by.id('editDescriptionInput')).typeText('They good')
await element(by.id('saveBtn')).tap()
await expect(element(by.id('createOrEditListModal'))).not.toBeVisible()
await element(by.text('About')).tap()
await expect(element(by.id('headerTitle'))).toHaveText('Good Ppl')
await expect(element(by.id('listDescription'))).toHaveText('They good')
})

it('Adds users on curatelists from the list', async () => {
await element(by.text('About')).tap()
await element(by.id('addUserBtn')).tap()
await expect(element(by.id('listAddUserModal'))).toBeVisible()
await waitFor(element(by.id('user-bob.test-addBtn')))
.toBeVisible()
.withTimeout(5000)
await element(by.id('user-bob.test-addBtn')).tap()
await element(by.id('doneBtn')).tap()
await expect(element(by.id('listAddUserModal'))).not.toBeVisible()
await expect(element(by.id('user-bob.test'))).toBeVisible()
})

it('Shows posts by the users in the list', async () => {
await element(by.text('Posts')).tap()
await expect(element(by.id('feedItem-by-bob.test'))).toBeVisible()
})

it('Pins the list', async () => {
await element(by.id('pinBtn')).tap()
await element(by.id('e2eGotoHome')).tap()
await element(by.id('homeScreenFeedTabs-Good Ppl')).tap()
await expect(element(by.id('feedItem-by-bob.test'))).toBeVisible()

await element(by.id('bottomBarFeedsBtn')).tap()
await element(by.id('saved-feed-Good Ppl')).tap()
await expect(element(by.id('feedItem-by-bob.test'))).toBeVisible()

await element(by.id('unpinBtn')).tap()
await element(by.id('bottomBarHomeBtn')).tap()
await expect(
element(by.id('homeScreenFeedTabs-Good Ppl')),
).not.toBeVisible()

await element(by.id('e2eGotoLists')).tap()
await element(by.id('list-Good Ppl')).tap()
})

it('Removes users on curatelists from the list', async () => {
await element(by.text('About')).tap()
await expect(element(by.id('user-bob.test'))).toBeVisible()
await element(by.id('user-bob.test-editBtn')).tap()
await expect(element(by.id('userAddRemoveListsModal'))).toBeVisible()
await element(by.id('toggleBtn-Good Ppl')).tap()
await element(by.id('saveBtn')).tap()
await expect(element(by.id('userAddRemoveListsModal'))).not.toBeVisible()
})

it('Shows the curatelist on my profile', async () => {
await element(by.id('bottomBarProfileBtn')).tap()
await element(by.id('selector')).swipe('left')
await element(by.id('selector-4')).tap()
await element(by.id('list-Good Ppl')).tap()
})

it('Adds and removes users on curatelists from the profile', async () => {
await element(by.id('bottomBarSearchBtn')).tap()
await element(by.id('searchTextInput')).typeText('bob')
await element(by.id('searchAutoCompleteResult-bob.test')).tap()
await expect(element(by.id('profileView'))).toBeVisible()

await element(by.id('profileHeaderDropdownBtn')).tap()
await element(by.text('Add to Lists')).tap()
await expect(element(by.id('userAddRemoveListsModal'))).toBeVisible()
await element(by.id('toggleBtn-Good Ppl')).tap()
await element(by.id('saveBtn')).tap()
await expect(element(by.id('userAddRemoveListsModal'))).not.toBeVisible()

await element(by.id('profileHeaderDropdownBtn')).tap()
await element(by.text('Add to Lists')).tap()
await expect(element(by.id('userAddRemoveListsModal'))).toBeVisible()
await element(by.id('toggleBtn-Good Ppl')).tap()
await element(by.id('saveBtn')).tap()
await expect(element(by.id('userAddRemoveListsModal'))).not.toBeVisible()
})

it('Can report a user list', async () => {
await element(by.id('e2eGotoSettings')).tap()
await element(by.id('signOutBtn')).tap()
await loginAsBob()
await element(by.id('bottomBarSearchBtn')).tap()
await element(by.id('searchTextInput')).typeText('alice')
await element(by.id('searchAutoCompleteResult-alice.test')).tap()
await element(by.id('selector')).swipe('left')
await element(by.id('selector-3')).tap()
await element(by.id('list-Good Ppl')).tap()
await element(by.id('headerDropdownBtn')).tap()
await element(by.text('Report List')).tap()
await expect(element(by.id('reportModal'))).toBeVisible()
await expect(element(by.text('Report List'))).toBeVisible()
await element(
by.id('reportReasonRadios-com.atproto.moderation.defs#reasonRude'),
).tap()
await element(by.id('sendReportBtn')).tap()
await expect(element(by.id('reportModal'))).not.toBeVisible()
})
})
Loading