Refactor feed manipulation and rendering to be more robust (#297)
This commit is contained in:
		
							parent
							
								
									93df983692
								
							
						
					
					
						commit
						c50a20d214
					
				
					 7 changed files with 360 additions and 260 deletions
				
			
		
							
								
								
									
										186
									
								
								src/lib/api/feed-manip.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										186
									
								
								src/lib/api/feed-manip.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,186 @@ | ||||||
|  | import {AppBskyFeedFeedViewPost} from '@atproto/api' | ||||||
|  | type FeedViewPost = AppBskyFeedFeedViewPost.Main | ||||||
|  | 
 | ||||||
|  | export type FeedTunerFn = ( | ||||||
|  |   tuner: FeedTuner, | ||||||
|  |   slices: FeedViewPostsSlice[], | ||||||
|  | ) => void | ||||||
|  | 
 | ||||||
|  | export class FeedViewPostsSlice { | ||||||
|  |   constructor(public items: FeedViewPost[] = []) {} | ||||||
|  | 
 | ||||||
|  |   get uri() { | ||||||
|  |     if (this.isReply) { | ||||||
|  |       return this.items[1].post.uri | ||||||
|  |     } | ||||||
|  |     return this.items[0].post.uri | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   get ts() { | ||||||
|  |     if (this.items[0].reason?.indexedAt) { | ||||||
|  |       return this.items[0].reason.indexedAt as string | ||||||
|  |     } | ||||||
|  |     return this.items[0].post.indexedAt | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   get isThread() { | ||||||
|  |     return ( | ||||||
|  |       this.items.length > 1 && | ||||||
|  |       this.items.every( | ||||||
|  |         item => item.post.author.did === this.items[0].post.author.did, | ||||||
|  |       ) | ||||||
|  |     ) | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   get isReply() { | ||||||
|  |     return this.items.length === 2 && !this.isThread | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   get rootItem() { | ||||||
|  |     if (this.isReply) { | ||||||
|  |       return this.items[1] | ||||||
|  |     } | ||||||
|  |     return this.items[0] | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   containsUri(uri: string) { | ||||||
|  |     return !!this.items.find(item => item.post.uri === uri) | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   insert(item: FeedViewPost) { | ||||||
|  |     const selfReplyUri = getSelfReplyUri(item) | ||||||
|  |     const i = this.items.findIndex(item2 => item2.post.uri === selfReplyUri) | ||||||
|  |     if (i !== -1) { | ||||||
|  |       this.items.splice(i + 1, 0, item) | ||||||
|  |     } else { | ||||||
|  |       this.items.push(item) | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   flattenReplyParent() { | ||||||
|  |     if (this.items[0].reply?.parent) { | ||||||
|  |       this.items.splice(0, 0, {post: this.items[0].reply?.parent}) | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   logSelf() { | ||||||
|  |     console.log( | ||||||
|  |       `- Slice ${this.items.length}${this.isThread ? ' (thread)' : ''} -`, | ||||||
|  |     ) | ||||||
|  |     for (const item of this.items) { | ||||||
|  |       console.log( | ||||||
|  |         `  ${item.reason ? `RP by ${item.reason.by.handle}: ` : ''}${ | ||||||
|  |           item.post.author.handle | ||||||
|  |         }: ${item.reply ? `(Reply ${item.reply.parent.author.handle}) ` : ''}${ | ||||||
|  |           item.post.record.text | ||||||
|  |         }`,
 | ||||||
|  |       ) | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export class FeedTuner { | ||||||
|  |   seenUris: Set<string> = new Set() | ||||||
|  | 
 | ||||||
|  |   constructor() {} | ||||||
|  | 
 | ||||||
|  |   reset() { | ||||||
|  |     this.seenUris.clear() | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   tune( | ||||||
|  |     feed: FeedViewPost[], | ||||||
|  |     tunerFns: FeedTunerFn[] = [], | ||||||
|  |   ): FeedViewPostsSlice[] { | ||||||
|  |     const slices: FeedViewPostsSlice[] = [] | ||||||
|  | 
 | ||||||
|  |     // arrange the posts into thread slices
 | ||||||
|  |     for (let i = feed.length - 1; i >= 0; i--) { | ||||||
|  |       const item = feed[i] | ||||||
|  | 
 | ||||||
|  |       const selfReplyUri = getSelfReplyUri(item) | ||||||
|  |       if (selfReplyUri) { | ||||||
|  |         const parent = slices.find(item2 => item2.containsUri(selfReplyUri)) | ||||||
|  |         if (parent) { | ||||||
|  |           parent.insert(item) | ||||||
|  |           continue | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |       slices.unshift(new FeedViewPostsSlice([item])) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // remove any items already "seen"
 | ||||||
|  |     for (let i = slices.length - 1; i >= 0; i--) { | ||||||
|  |       if (this.seenUris.has(slices[i].uri)) { | ||||||
|  |         slices.splice(i, 1) | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // turn non-threads with reply parents into threads
 | ||||||
|  |     for (const slice of slices) { | ||||||
|  |       if ( | ||||||
|  |         !slice.isThread && | ||||||
|  |         !slice.items[0].reason && | ||||||
|  |         slice.items[0].reply?.parent && | ||||||
|  |         !this.seenUris.has(slice.items[0].reply?.parent.uri) | ||||||
|  |       ) { | ||||||
|  |         slice.flattenReplyParent() | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // sort by slice roots' timestamps
 | ||||||
|  |     slices.sort((a, b) => b.ts.localeCompare(a.ts)) | ||||||
|  | 
 | ||||||
|  |     // run the custom tuners
 | ||||||
|  |     for (const tunerFn of tunerFns) { | ||||||
|  |       tunerFn(this, slices) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     for (const slice of slices) { | ||||||
|  |       for (const item of slice.items) { | ||||||
|  |         this.seenUris.add(item.post.uri) | ||||||
|  |       } | ||||||
|  |       slice.logSelf() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return slices | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   static dedupReposts(tuner: FeedTuner, slices: FeedViewPostsSlice[]) { | ||||||
|  |     // remove duplicates caused by reposts
 | ||||||
|  |     for (let i = 0; i < slices.length; i++) { | ||||||
|  |       const item1 = slices[i] | ||||||
|  |       for (let j = i + 1; j < slices.length; j++) { | ||||||
|  |         const item2 = slices[j] | ||||||
|  |         if (item2.isThread) { | ||||||
|  |           // dont dedup items that are rendering in a thread as this can cause rendering errors
 | ||||||
|  |           continue | ||||||
|  |         } | ||||||
|  |         if (item1.containsUri(item2.items[0].post.uri)) { | ||||||
|  |           slices.splice(j, 1) | ||||||
|  |           j-- | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   static likedRepliesOnly(tuner: FeedTuner, slices: FeedViewPostsSlice[]) { | ||||||
|  |     // remove any replies without any likes
 | ||||||
|  |     for (let i = slices.length - 1; i >= 0; i--) { | ||||||
|  |       if (slices[i].isThread) { | ||||||
|  |         continue | ||||||
|  |       } | ||||||
|  |       const item = slices[i].rootItem | ||||||
|  |       const isRepost = Boolean(item.reason) | ||||||
|  |       if (item.reply && !isRepost && item.post.upvoteCount === 0) { | ||||||
|  |         slices.splice(i, 1) | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function getSelfReplyUri(item: FeedViewPost): string | undefined { | ||||||
|  |   return item.reply?.parent.author.did === item.post.author.did | ||||||
|  |     ? item.reply?.parent.uri | ||||||
|  |     : undefined | ||||||
|  | } | ||||||
|  | @ -23,36 +23,27 @@ import { | ||||||
|   mergePosts, |   mergePosts, | ||||||
| } from 'lib/api/build-suggested-posts' | } from 'lib/api/build-suggested-posts' | ||||||
| 
 | 
 | ||||||
|  | import {FeedTuner, FeedViewPostsSlice} from 'lib/api/feed-manip' | ||||||
|  | 
 | ||||||
| const PAGE_SIZE = 30 | const PAGE_SIZE = 30 | ||||||
| 
 | 
 | ||||||
| let _idCounter = 0 | let _idCounter = 0 | ||||||
| 
 | 
 | ||||||
| type FeedViewPostWithThreadMeta = FeedViewPost & { |  | ||||||
|   _isThreadParent?: boolean |  | ||||||
|   _isThreadChildElided?: boolean |  | ||||||
|   _isThreadChild?: boolean |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| export class FeedItemModel { | export class FeedItemModel { | ||||||
|   // ui state
 |   // ui state
 | ||||||
|   _reactKey: string = '' |   _reactKey: string = '' | ||||||
|   _isThreadParent: boolean = false |  | ||||||
|   _isThreadChildElided: boolean = false |  | ||||||
|   _isThreadChild: boolean = false |  | ||||||
|   _hideParent: boolean = true // used to avoid dup post rendering while showing some parents
 |  | ||||||
| 
 | 
 | ||||||
|   // data
 |   // data
 | ||||||
|   post: PostView |   post: PostView | ||||||
|   postRecord?: AppBskyFeedPost.Record |   postRecord?: AppBskyFeedPost.Record | ||||||
|   reply?: FeedViewPost['reply'] |   reply?: FeedViewPost['reply'] | ||||||
|   replyParent?: FeedItemModel |  | ||||||
|   reason?: FeedViewPost['reason'] |   reason?: FeedViewPost['reason'] | ||||||
|   richText?: RichText |   richText?: RichText | ||||||
| 
 | 
 | ||||||
|   constructor( |   constructor( | ||||||
|     public rootStore: RootStoreModel, |     public rootStore: RootStoreModel, | ||||||
|     reactKey: string, |     reactKey: string, | ||||||
|     v: FeedViewPostWithThreadMeta, |     v: FeedViewPost, | ||||||
|   ) { |   ) { | ||||||
|     this._reactKey = reactKey |     this._reactKey = reactKey | ||||||
|     this.post = v.post |     this.post = v.post | ||||||
|  | @ -78,35 +69,21 @@ export class FeedItemModel { | ||||||
|       ) |       ) | ||||||
|     } |     } | ||||||
|     this.reply = v.reply |     this.reply = v.reply | ||||||
|     if (v.reply?.parent) { |  | ||||||
|       this.replyParent = new FeedItemModel(rootStore, '', { |  | ||||||
|         post: v.reply.parent, |  | ||||||
|       }) |  | ||||||
|     } |  | ||||||
|     this.reason = v.reason |     this.reason = v.reason | ||||||
|     this._isThreadParent = v._isThreadParent || false |  | ||||||
|     this._isThreadChild = v._isThreadChild || false |  | ||||||
|     this._isThreadChildElided = v._isThreadChildElided || false |  | ||||||
|     makeAutoObservable(this, {rootStore: false}) |     makeAutoObservable(this, {rootStore: false}) | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   copy(v: FeedViewPost) { |   copy(v: FeedViewPost) { | ||||||
|     this.post = v.post |     this.post = v.post | ||||||
|     this.reply = v.reply |     this.reply = v.reply | ||||||
|     if (v.reply?.parent) { |  | ||||||
|       this.replyParent = new FeedItemModel(this.rootStore, '', { |  | ||||||
|         post: v.reply.parent, |  | ||||||
|       }) |  | ||||||
|     } else { |  | ||||||
|       this.replyParent = undefined |  | ||||||
|     } |  | ||||||
|     this.reason = v.reason |     this.reason = v.reason | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   get _isRenderingAsThread() { |   copyMetrics(v: FeedViewPost) { | ||||||
|     return ( |     this.post.replyCount = v.post.replyCount | ||||||
|       this._isThreadParent || this._isThreadChild || this._isThreadChildElided |     this.post.repostCount = v.post.repostCount | ||||||
|     ) |     this.post.upvoteCount = v.post.upvoteCount | ||||||
|  |     this.post.viewer = v.post.viewer | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   get reasonRepost(): ReasonRepost | undefined { |   get reasonRepost(): ReasonRepost | undefined { | ||||||
|  | @ -192,6 +169,73 @@ export class FeedItemModel { | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | export class FeedSliceModel { | ||||||
|  |   // ui state
 | ||||||
|  |   _reactKey: string = '' | ||||||
|  | 
 | ||||||
|  |   // data
 | ||||||
|  |   items: FeedItemModel[] = [] | ||||||
|  | 
 | ||||||
|  |   constructor( | ||||||
|  |     public rootStore: RootStoreModel, | ||||||
|  |     reactKey: string, | ||||||
|  |     slice: FeedViewPostsSlice, | ||||||
|  |   ) { | ||||||
|  |     this._reactKey = reactKey | ||||||
|  |     for (const item of slice.items) { | ||||||
|  |       this.items.push( | ||||||
|  |         new FeedItemModel(rootStore, `item-${_idCounter++}`, item), | ||||||
|  |       ) | ||||||
|  |     } | ||||||
|  |     makeAutoObservable(this, {rootStore: false}) | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   get uri() { | ||||||
|  |     if (this.isReply) { | ||||||
|  |       return this.items[1].post.uri | ||||||
|  |     } | ||||||
|  |     return this.items[0].post.uri | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   get isThread() { | ||||||
|  |     return ( | ||||||
|  |       this.items.length > 1 && | ||||||
|  |       this.items.every( | ||||||
|  |         item => item.post.author.did === this.items[0].post.author.did, | ||||||
|  |       ) | ||||||
|  |     ) | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   get isReply() { | ||||||
|  |     return this.items.length === 2 && !this.isThread | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   get rootItem() { | ||||||
|  |     if (this.isReply) { | ||||||
|  |       return this.items[1] | ||||||
|  |     } | ||||||
|  |     return this.items[0] | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   containsUri(uri: string) { | ||||||
|  |     return !!this.items.find(item => item.post.uri === uri) | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   isThreadParentAt(i: number) { | ||||||
|  |     if (this.items.length === 1) { | ||||||
|  |       return false | ||||||
|  |     } | ||||||
|  |     return i < this.items.length - 1 | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   isThreadChildAt(i: number) { | ||||||
|  |     if (this.items.length === 1) { | ||||||
|  |       return false | ||||||
|  |     } | ||||||
|  |     return i > 0 | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
| export class FeedModel { | export class FeedModel { | ||||||
|   // state
 |   // state
 | ||||||
|   isLoading = false |   isLoading = false | ||||||
|  | @ -203,12 +247,13 @@ export class FeedModel { | ||||||
|   hasMore = true |   hasMore = true | ||||||
|   loadMoreCursor: string | undefined |   loadMoreCursor: string | undefined | ||||||
|   pollCursor: string | undefined |   pollCursor: string | undefined | ||||||
|  |   tuner = new FeedTuner() | ||||||
| 
 | 
 | ||||||
|   // used to linearize async modifications to state
 |   // used to linearize async modifications to state
 | ||||||
|   private lock = new AwaitLock() |   private lock = new AwaitLock() | ||||||
| 
 | 
 | ||||||
|   // data
 |   // data
 | ||||||
|   feed: FeedItemModel[] = [] |   slices: FeedSliceModel[] = [] | ||||||
| 
 | 
 | ||||||
|   constructor( |   constructor( | ||||||
|     public rootStore: RootStoreModel, |     public rootStore: RootStoreModel, | ||||||
|  | @ -228,7 +273,7 @@ export class FeedModel { | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   get hasContent() { |   get hasContent() { | ||||||
|     return this.feed.length !== 0 |     return this.slices.length !== 0 | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   get hasError() { |   get hasError() { | ||||||
|  | @ -241,34 +286,21 @@ export class FeedModel { | ||||||
| 
 | 
 | ||||||
|   get nonReplyFeed() { |   get nonReplyFeed() { | ||||||
|     if (this.feedType === 'author') { |     if (this.feedType === 'author') { | ||||||
|       return this.feed.filter(item => { |       return this.slices.filter(slice => { | ||||||
|         const params = this.params as GetAuthorFeed.QueryParams |         const params = this.params as GetAuthorFeed.QueryParams | ||||||
|  |         const item = slice.rootItem | ||||||
|         const isRepost = |         const isRepost = | ||||||
|           item.reply && |           item?.reasonRepost?.by?.handle === params.author || | ||||||
|           (item?.reasonRepost?.by?.handle === params.author || |           item?.reasonRepost?.by?.did === params.author | ||||||
|             item?.reasonRepost?.by?.did === params.author) |  | ||||||
| 
 |  | ||||||
|         return ( |         return ( | ||||||
|           !item.reply || // not a reply
 |           !item.reply || // not a reply
 | ||||||
|           isRepost || |           isRepost || // but allow if it's a repost
 | ||||||
|           ((item._isThreadParent || // but allow if it's a thread by the user
 |           (slice.isThread && // or a thread by the user
 | ||||||
|             item._isThreadChild) && |  | ||||||
|             item.reply?.root.author.did === item.post.author.did) |             item.reply?.root.author.did === item.post.author.did) | ||||||
|         ) |         ) | ||||||
|       }) |       }) | ||||||
|     } else if (this.feedType === 'home') { |  | ||||||
|       return this.feed.filter(item => { |  | ||||||
|         const isRepost = Boolean(item?.reasonRepost) |  | ||||||
|         return ( |  | ||||||
|           !item.reply || // not a reply
 |  | ||||||
|           isRepost || // but allow if it's a repost or thread
 |  | ||||||
|           item._isThreadParent || |  | ||||||
|           item._isThreadChild || |  | ||||||
|           item.post.upvoteCount >= 2 |  | ||||||
|         ) |  | ||||||
|       }) |  | ||||||
|     } else { |     } else { | ||||||
|       return this.feed |       return this.slices | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|  | @ -292,7 +324,8 @@ export class FeedModel { | ||||||
|     this.hasMore = true |     this.hasMore = true | ||||||
|     this.loadMoreCursor = undefined |     this.loadMoreCursor = undefined | ||||||
|     this.pollCursor = undefined |     this.pollCursor = undefined | ||||||
|     this.feed = [] |     this.slices = [] | ||||||
|  |     this.tuner.reset() | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   switchFeedType(feedType: 'home' | 'suggested') { |   switchFeedType(feedType: 'home' | 'suggested') { | ||||||
|  | @ -314,6 +347,7 @@ export class FeedModel { | ||||||
|     await this.lock.acquireAsync() |     await this.lock.acquireAsync() | ||||||
|     try { |     try { | ||||||
|       this.setHasNewLatest(false) |       this.setHasNewLatest(false) | ||||||
|  |       this.tuner.reset() | ||||||
|       this._xLoading(isRefreshing) |       this._xLoading(isRefreshing) | ||||||
|       try { |       try { | ||||||
|         const res = await this._getFeed({limit: PAGE_SIZE}) |         const res = await this._getFeed({limit: PAGE_SIZE}) | ||||||
|  | @ -401,11 +435,11 @@ export class FeedModel { | ||||||
|   update = bundleAsync(async () => { |   update = bundleAsync(async () => { | ||||||
|     await this.lock.acquireAsync() |     await this.lock.acquireAsync() | ||||||
|     try { |     try { | ||||||
|       if (!this.feed.length) { |       if (!this.slices.length) { | ||||||
|         return |         return | ||||||
|       } |       } | ||||||
|       this._xLoading() |       this._xLoading() | ||||||
|       let numToFetch = this.feed.length |       let numToFetch = this.slices.length | ||||||
|       let cursor |       let cursor | ||||||
|       try { |       try { | ||||||
|         do { |         do { | ||||||
|  | @ -464,9 +498,9 @@ export class FeedModel { | ||||||
|   onPostDeleted(uri: string) { |   onPostDeleted(uri: string) { | ||||||
|     let i |     let i | ||||||
|     do { |     do { | ||||||
|       i = this.feed.findIndex(item => item.post.uri === uri) |       i = this.slices.findIndex(slice => slice.containsUri(uri)) | ||||||
|       if (i !== -1) { |       if (i !== -1) { | ||||||
|         this.feed.splice(i, 1) |         this.slices.splice(i, 1) | ||||||
|       } |       } | ||||||
|     } while (i !== -1) |     } while (i !== -1) | ||||||
|   } |   } | ||||||
|  | @ -506,27 +540,29 @@ export class FeedModel { | ||||||
|   ) { |   ) { | ||||||
|     this.loadMoreCursor = res.data.cursor |     this.loadMoreCursor = res.data.cursor | ||||||
|     this.hasMore = !!this.loadMoreCursor |     this.hasMore = !!this.loadMoreCursor | ||||||
|     const orgLen = this.feed.length |  | ||||||
| 
 | 
 | ||||||
|     const reorgedFeed = preprocessFeed(res.data.feed) |     const slices = this.tuner.tune( | ||||||
|  |       res.data.feed, | ||||||
|  |       this.feedType === 'home' | ||||||
|  |         ? [FeedTuner.dedupReposts, FeedTuner.likedRepliesOnly] | ||||||
|  |         : [], | ||||||
|  |     ) | ||||||
| 
 | 
 | ||||||
|     const toAppend: FeedItemModel[] = [] |     const toAppend: FeedSliceModel[] = [] | ||||||
|     for (const item of reorgedFeed) { |     for (const slice of slices) { | ||||||
|       const itemModel = new FeedItemModel( |       const sliceModel = new FeedSliceModel( | ||||||
|         this.rootStore, |         this.rootStore, | ||||||
|         `item-${_idCounter++}`, |         `item-${_idCounter++}`, | ||||||
|         item, |         slice, | ||||||
|       ) |       ) | ||||||
|       toAppend.push(itemModel) |       toAppend.push(sliceModel) | ||||||
|     } |     } | ||||||
|     runInAction(() => { |     runInAction(() => { | ||||||
|       if (replace) { |       if (replace) { | ||||||
|         this.feed = toAppend |         this.slices = toAppend | ||||||
|       } else { |       } else { | ||||||
|         this.feed = this.feed.concat(toAppend) |         this.slices = this.slices.concat(toAppend) | ||||||
|       } |       } | ||||||
|       dedupReposts(this.feed) |  | ||||||
|       dedupParents(this.feed.slice(orgLen)) // we slice to avoid modifying rendering of already-shown posts
 |  | ||||||
|     }) |     }) | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|  | @ -535,35 +571,39 @@ export class FeedModel { | ||||||
|   ) { |   ) { | ||||||
|     this.pollCursor = res.data.feed[0]?.post.uri |     this.pollCursor = res.data.feed[0]?.post.uri | ||||||
| 
 | 
 | ||||||
|     const toPrepend: FeedItemModel[] = [] |     const slices = this.tuner.tune( | ||||||
|     for (const item of res.data.feed) { |       res.data.feed, | ||||||
|       if (this.feed.find(item2 => item2.post.uri === item.post.uri)) { |       this.feedType === 'home' | ||||||
|         break // stop here - we've hit a post we already have
 |         ? [FeedTuner.dedupReposts, FeedTuner.likedRepliesOnly] | ||||||
|       } |         : [], | ||||||
|  |     ) | ||||||
| 
 | 
 | ||||||
|       const itemModel = new FeedItemModel( |     const toPrepend: FeedSliceModel[] = [] | ||||||
|  |     for (const slice of slices) { | ||||||
|  |       const itemModel = new FeedSliceModel( | ||||||
|         this.rootStore, |         this.rootStore, | ||||||
|         `item-${_idCounter++}`, |         `item-${_idCounter++}`, | ||||||
|         item, |         slice, | ||||||
|       ) |       ) | ||||||
|       toPrepend.push(itemModel) |       toPrepend.push(itemModel) | ||||||
|     } |     } | ||||||
|     runInAction(() => { |     runInAction(() => { | ||||||
|       this.feed = toPrepend.concat(this.feed) |       this.slices = toPrepend.concat(this.slices) | ||||||
|     }) |     }) | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   private _updateAll(res: GetTimeline.Response | GetAuthorFeed.Response) { |   private _updateAll(res: GetTimeline.Response | GetAuthorFeed.Response) { | ||||||
|     for (const item of res.data.feed) { |     for (const item of res.data.feed) { | ||||||
|       const existingItem = this.feed.find( |       const existingSlice = this.slices.find(slice => | ||||||
|         // HACK: need to find the reposts' item, so we have to check for that -prf
 |         slice.containsUri(item.post.uri), | ||||||
|         item2 => |  | ||||||
|           item.post.uri === item2.post.uri && |  | ||||||
|           // @ts-ignore todo
 |  | ||||||
|           item.reason?.by?.did === item2.reason?.by?.did, |  | ||||||
|       ) |       ) | ||||||
|       if (existingItem) { |       if (existingSlice) { | ||||||
|         existingItem.copy(item) |         const existingItem = existingSlice.items.find( | ||||||
|  |           item2 => item2.post.uri === item.post.uri, | ||||||
|  |         ) | ||||||
|  |         if (existingItem) { | ||||||
|  |           existingItem.copyMetrics(item) | ||||||
|  |         } | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  | @ -601,147 +641,3 @@ export class FeedModel { | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 |  | ||||||
| interface Slice { |  | ||||||
|   index: number |  | ||||||
|   length: number |  | ||||||
| } |  | ||||||
| function preprocessFeed(feed: FeedViewPost[]): FeedViewPostWithThreadMeta[] { |  | ||||||
|   const reorg: FeedViewPostWithThreadMeta[] = [] |  | ||||||
| 
 |  | ||||||
|   // phase one: identify threads and reorganize them into the feed so
 |  | ||||||
|   // that they are in order and marked as part of a thread
 |  | ||||||
|   for (let i = feed.length - 1; i >= 0; i--) { |  | ||||||
|     const item = feed[i] as FeedViewPostWithThreadMeta |  | ||||||
| 
 |  | ||||||
|     const selfReplyUri = getSelfReplyUri(item) |  | ||||||
|     if (selfReplyUri) { |  | ||||||
|       const parentIndex = reorg.findIndex( |  | ||||||
|         item2 => item2.post.uri === selfReplyUri, |  | ||||||
|       ) |  | ||||||
|       if (parentIndex !== -1 && !reorg[parentIndex]._isThreadParent) { |  | ||||||
|         reorg[parentIndex]._isThreadParent = true |  | ||||||
|         item._isThreadChild = true |  | ||||||
|         reorg.splice(parentIndex + 1, 0, item) |  | ||||||
|         continue |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|     reorg.unshift(item) |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   // phase two: reorder the feed so that the timestamp of the
 |  | ||||||
|   // last post in a thread establishes its ordering
 |  | ||||||
|   let threadSlices: Slice[] = identifyThreadSlices(reorg) |  | ||||||
|   for (const slice of threadSlices) { |  | ||||||
|     const removed: FeedViewPostWithThreadMeta[] = reorg.splice( |  | ||||||
|       slice.index, |  | ||||||
|       slice.length, |  | ||||||
|     ) |  | ||||||
|     const targetDate = new Date(ts(removed[removed.length - 1])) |  | ||||||
|     let newIndex = reorg.findIndex(item => new Date(ts(item)) < targetDate) |  | ||||||
|     if (newIndex === -1) { |  | ||||||
|       newIndex = reorg.length |  | ||||||
|     } |  | ||||||
|     reorg.splice(newIndex, 0, ...removed) |  | ||||||
|     slice.index = newIndex |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   // phase three: compress any threads that are longer than 3 posts
 |  | ||||||
|   let removedCount = 0 |  | ||||||
|   // phase 2 moved posts around, so we need to re-identify the slice indices
 |  | ||||||
|   threadSlices = identifyThreadSlices(reorg) |  | ||||||
|   for (const slice of threadSlices) { |  | ||||||
|     if (slice.length > 3) { |  | ||||||
|       reorg.splice(slice.index - removedCount + 1, slice.length - 3) |  | ||||||
|       if (reorg[slice.index - removedCount]) { |  | ||||||
|         // ^ sanity check
 |  | ||||||
|         reorg[slice.index - removedCount]._isThreadChildElided = true |  | ||||||
|       } |  | ||||||
|       removedCount += slice.length - 3 |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   return reorg |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| function identifyThreadSlices(feed: FeedViewPost[]): Slice[] { |  | ||||||
|   let activeSlice = -1 |  | ||||||
|   let threadSlices: Slice[] = [] |  | ||||||
|   for (let i = 0; i < feed.length; i++) { |  | ||||||
|     const item = feed[i] as FeedViewPostWithThreadMeta |  | ||||||
|     if (activeSlice === -1) { |  | ||||||
|       if (item._isThreadParent) { |  | ||||||
|         activeSlice = i |  | ||||||
|       } |  | ||||||
|     } else { |  | ||||||
|       if (!item._isThreadChild) { |  | ||||||
|         threadSlices.push({index: activeSlice, length: i - activeSlice}) |  | ||||||
|         if (item._isThreadParent) { |  | ||||||
|           activeSlice = i |  | ||||||
|         } else { |  | ||||||
|           activeSlice = -1 |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|   if (activeSlice !== -1) { |  | ||||||
|     threadSlices.push({index: activeSlice, length: feed.length - activeSlice}) |  | ||||||
|   } |  | ||||||
|   return threadSlices |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // WARNING: mutates `feed`
 |  | ||||||
| function dedupReposts(feed: FeedItemModel[]) { |  | ||||||
|   // remove duplicates caused by reposts
 |  | ||||||
|   for (let i = 0; i < feed.length; i++) { |  | ||||||
|     const item1 = feed[i] |  | ||||||
|     for (let j = i + 1; j < feed.length; j++) { |  | ||||||
|       const item2 = feed[j] |  | ||||||
|       if (item2._isRenderingAsThread) { |  | ||||||
|         // dont dedup items that are rendering in a thread as this can cause rendering errors
 |  | ||||||
|         continue |  | ||||||
|       } |  | ||||||
|       if (item1.post.uri === item2.post.uri) { |  | ||||||
|         feed.splice(j, 1) |  | ||||||
|         j-- |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // WARNING: mutates `feed`
 |  | ||||||
| function dedupParents(feed: FeedItemModel[]) { |  | ||||||
|   // only show parents that aren't already in the feed
 |  | ||||||
|   for (let i = 0; i < feed.length; i++) { |  | ||||||
|     const item1 = feed[i] |  | ||||||
|     if (!item1.replyParent || item1._isThreadChild) { |  | ||||||
|       continue |  | ||||||
|     } |  | ||||||
|     let hideParent = false |  | ||||||
|     for (let j = 0; j < feed.length; j++) { |  | ||||||
|       const item2 = feed[j] |  | ||||||
|       if ( |  | ||||||
|         item1.replyParent.post.uri === item2.post.uri || // the post itself is there
 |  | ||||||
|         (j < i && item1.replyParent.post.uri === item2.replyParent?.post.uri) // another reply already showed it
 |  | ||||||
|       ) { |  | ||||||
|         hideParent = true |  | ||||||
|         break |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|     item1._hideParent = hideParent |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| function getSelfReplyUri(item: FeedViewPost): string | undefined { |  | ||||||
|   return item.reply?.parent.author.did === item.post.author.did |  | ||||||
|     ? item.reply?.parent.uri |  | ||||||
|     : undefined |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| function ts(item: FeedViewPost | FeedItemModel): string { |  | ||||||
|   if (item.reason?.indexedAt) { |  | ||||||
|     // @ts-ignore need better type checks
 |  | ||||||
|     return item.reason.indexedAt |  | ||||||
|   } |  | ||||||
|   return item.post.indexedAt |  | ||||||
| } |  | ||||||
|  |  | ||||||
|  | @ -100,7 +100,7 @@ export class ProfileUiModel { | ||||||
|           if (this.selectedView === Sections.Posts) { |           if (this.selectedView === Sections.Posts) { | ||||||
|             arr = this.feed.nonReplyFeed |             arr = this.feed.nonReplyFeed | ||||||
|           } else { |           } else { | ||||||
|             arr = this.feed.feed.slice() |             arr = this.feed.slices.slice() | ||||||
|           } |           } | ||||||
|           if (!this.feed.hasMore) { |           if (!this.feed.hasMore) { | ||||||
|             arr = arr.concat([ProfileUiModel.END_ITEM]) |             arr = arr.concat([ProfileUiModel.END_ITEM]) | ||||||
|  |  | ||||||
|  | @ -16,7 +16,7 @@ import {Text} from '../util/text/Text' | ||||||
| import {ErrorMessage} from '../util/error/ErrorMessage' | import {ErrorMessage} from '../util/error/ErrorMessage' | ||||||
| import {Button} from '../util/forms/Button' | import {Button} from '../util/forms/Button' | ||||||
| import {FeedModel} from 'state/models/feed-view' | import {FeedModel} from 'state/models/feed-view' | ||||||
| import {FeedItem} from './FeedItem' | import {FeedSlice} from './FeedSlice' | ||||||
| import {OnScrollCb} from 'lib/hooks/useOnMainScroll' | import {OnScrollCb} from 'lib/hooks/useOnMainScroll' | ||||||
| import {s} from 'lib/styles' | import {s} from 'lib/styles' | ||||||
| import {useAnalytics} from 'lib/analytics' | import {useAnalytics} from 'lib/analytics' | ||||||
|  | @ -61,11 +61,11 @@ export const Feed = observer(function Feed({ | ||||||
|       if (feed.isEmpty) { |       if (feed.isEmpty) { | ||||||
|         feedItems = feedItems.concat([EMPTY_FEED_ITEM]) |         feedItems = feedItems.concat([EMPTY_FEED_ITEM]) | ||||||
|       } else { |       } else { | ||||||
|         feedItems = feedItems.concat(feed.nonReplyFeed) |         feedItems = feedItems.concat(feed.slices) | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     return feedItems |     return feedItems | ||||||
|   }, [feed.hasError, feed.hasLoaded, feed.isEmpty, feed.nonReplyFeed]) |   }, [feed.hasError, feed.hasLoaded, feed.isEmpty, feed.slices]) | ||||||
| 
 | 
 | ||||||
|   // events
 |   // events
 | ||||||
|   // =
 |   // =
 | ||||||
|  | @ -92,10 +92,6 @@ export const Feed = observer(function Feed({ | ||||||
|   // rendering
 |   // rendering
 | ||||||
|   // =
 |   // =
 | ||||||
| 
 | 
 | ||||||
|   // TODO optimize renderItem or FeedItem, we're getting this notice from RN: -prf
 |  | ||||||
|   //   VirtualizedList: You have a large list that is slow to update - make sure your
 |  | ||||||
|   //   renderItem function renders components that follow React performance best practices
 |  | ||||||
|   //   like PureComponent, shouldComponentUpdate, etc
 |  | ||||||
|   const renderItem = React.useCallback( |   const renderItem = React.useCallback( | ||||||
|     ({item}: {item: any}) => { |     ({item}: {item: any}) => { | ||||||
|       if (item === EMPTY_FEED_ITEM) { |       if (item === EMPTY_FEED_ITEM) { | ||||||
|  | @ -138,7 +134,7 @@ export const Feed = observer(function Feed({ | ||||||
|           /> |           /> | ||||||
|         ) |         ) | ||||||
|       } |       } | ||||||
|       return <FeedItem item={item} showFollowBtn={showPostFollowBtn} /> |       return <FeedSlice slice={item} showFollowBtn={showPostFollowBtn} /> | ||||||
|     }, |     }, | ||||||
|     [feed, onPressTryAgain, showPostFollowBtn, pal, palInverted, navigation], |     [feed, onPressTryAgain, showPostFollowBtn, pal, palInverted, navigation], | ||||||
|   ) |   ) | ||||||
|  |  | ||||||
|  | @ -26,11 +26,14 @@ import {useAnalytics} from 'lib/analytics' | ||||||
| 
 | 
 | ||||||
| export const FeedItem = observer(function ({ | export const FeedItem = observer(function ({ | ||||||
|   item, |   item, | ||||||
|   showReplyLine, |   isThreadChild, | ||||||
|  |   isThreadParent, | ||||||
|   showFollowBtn, |   showFollowBtn, | ||||||
|   ignoreMuteFor, |   ignoreMuteFor, | ||||||
| }: { | }: { | ||||||
|   item: FeedItemModel |   item: FeedItemModel | ||||||
|  |   isThreadChild?: boolean | ||||||
|  |   isThreadParent?: boolean | ||||||
|   showReplyLine?: boolean |   showReplyLine?: boolean | ||||||
|   showFollowBtn?: boolean |   showFollowBtn?: boolean | ||||||
|   ignoreMuteFor?: string |   ignoreMuteFor?: string | ||||||
|  | @ -110,10 +113,8 @@ export const FeedItem = observer(function ({ | ||||||
|     return <View /> |     return <View /> | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   const isChild = |   const isSmallTop = isThreadChild | ||||||
|     item._isThreadChild || (!item.reason && !item._hideParent && item.reply) |   const isNoTop = false //isChild && !item._isThreadChild
 | ||||||
|   const isSmallTop = isChild && item._isThreadChild |  | ||||||
|   const isNoTop = isChild && !item._isThreadChild |  | ||||||
|   const isMuted = |   const isMuted = | ||||||
|     item.post.author.viewer?.muted && ignoreMuteFor !== item.post.author.did |     item.post.author.viewer?.muted && ignoreMuteFor !== item.post.author.did | ||||||
|   const outerStyles = [ |   const outerStyles = [ | ||||||
|  | @ -122,25 +123,18 @@ export const FeedItem = observer(function ({ | ||||||
|     {borderColor: pal.colors.border}, |     {borderColor: pal.colors.border}, | ||||||
|     isSmallTop ? styles.outerSmallTop : undefined, |     isSmallTop ? styles.outerSmallTop : undefined, | ||||||
|     isNoTop ? styles.outerNoTop : undefined, |     isNoTop ? styles.outerNoTop : undefined, | ||||||
|     item._isThreadParent ? styles.outerNoBottom : undefined, |     isThreadParent ? styles.outerNoBottom : undefined, | ||||||
|   ] |   ] | ||||||
| 
 | 
 | ||||||
|   return ( |   return ( | ||||||
|     <PostMutedWrapper isMuted={isMuted}> |     <PostMutedWrapper isMuted={isMuted}> | ||||||
|       {isChild && !item._isThreadChild && item.replyParent ? ( |  | ||||||
|         <FeedItem |  | ||||||
|           item={item.replyParent} |  | ||||||
|           showReplyLine |  | ||||||
|           ignoreMuteFor={ignoreMuteFor} |  | ||||||
|         /> |  | ||||||
|       ) : undefined} |  | ||||||
|       <Link style={outerStyles} href={itemHref} title={itemTitle} noFeedback> |       <Link style={outerStyles} href={itemHref} title={itemTitle} noFeedback> | ||||||
|         {item._isThreadChild && ( |         {isThreadChild && ( | ||||||
|           <View |           <View | ||||||
|             style={[styles.topReplyLine, {borderColor: pal.colors.replyLine}]} |             style={[styles.topReplyLine, {borderColor: pal.colors.replyLine}]} | ||||||
|           /> |           /> | ||||||
|         )} |         )} | ||||||
|         {(showReplyLine || item._isThreadParent) && ( |         {isThreadParent && ( | ||||||
|           <View |           <View | ||||||
|             style={[ |             style={[ | ||||||
|               styles.bottomReplyLine, |               styles.bottomReplyLine, | ||||||
|  | @ -199,7 +193,7 @@ export const FeedItem = observer(function ({ | ||||||
|               declarationCid={item.post.author.declaration.cid} |               declarationCid={item.post.author.declaration.cid} | ||||||
|               showFollowBtn={showFollowBtn} |               showFollowBtn={showFollowBtn} | ||||||
|             /> |             /> | ||||||
|             {!isChild && replyAuthorDid !== '' && ( |             {!isThreadChild && replyAuthorDid !== '' && ( | ||||||
|               <View style={[s.flexRow, s.mb2, s.alignCenter]}> |               <View style={[s.flexRow, s.mb2, s.alignCenter]}> | ||||||
|                 <FontAwesomeIcon |                 <FontAwesomeIcon | ||||||
|                   icon="reply" |                   icon="reply" | ||||||
|  | @ -259,7 +253,7 @@ export const FeedItem = observer(function ({ | ||||||
|           </View> |           </View> | ||||||
|         </View> |         </View> | ||||||
|       </Link> |       </Link> | ||||||
|       {item._isThreadChildElided ? ( |       {false /*isThreadChildElided*/ ? ( | ||||||
|         <Link |         <Link | ||||||
|           style={[pal.view, styles.viewFullThread]} |           style={[pal.view, styles.viewFullThread]} | ||||||
|           href={itemHref} |           href={itemHref} | ||||||
|  |  | ||||||
							
								
								
									
										28
									
								
								src/view/com/posts/FeedSlice.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								src/view/com/posts/FeedSlice.tsx
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,28 @@ | ||||||
|  | import React from 'react' | ||||||
|  | import {FeedSliceModel} from 'state/models/feed-view' | ||||||
|  | import {FeedItem} from './FeedItem' | ||||||
|  | 
 | ||||||
|  | export function FeedSlice({ | ||||||
|  |   slice, | ||||||
|  |   showFollowBtn, | ||||||
|  |   ignoreMuteFor, | ||||||
|  | }: { | ||||||
|  |   slice: FeedSliceModel | ||||||
|  |   showFollowBtn?: boolean | ||||||
|  |   ignoreMuteFor?: string | ||||||
|  | }) { | ||||||
|  |   return ( | ||||||
|  |     <> | ||||||
|  |       {slice.items.map((item, i) => ( | ||||||
|  |         <FeedItem | ||||||
|  |           key={item._reactKey} | ||||||
|  |           item={item} | ||||||
|  |           isThreadParent={slice.isThreadParentAt(i)} | ||||||
|  |           isThreadChild={slice.isThreadChildAt(i)} | ||||||
|  |           showFollowBtn={showFollowBtn} | ||||||
|  |           ignoreMuteFor={ignoreMuteFor} | ||||||
|  |         /> | ||||||
|  |       ))} | ||||||
|  |     </> | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  | @ -6,11 +6,11 @@ import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types' | ||||||
| import {withAuthRequired} from 'view/com/auth/withAuthRequired' | import {withAuthRequired} from 'view/com/auth/withAuthRequired' | ||||||
| import {ViewSelector} from '../com/util/ViewSelector' | import {ViewSelector} from '../com/util/ViewSelector' | ||||||
| import {CenteredView} from '../com/util/Views' | import {CenteredView} from '../com/util/Views' | ||||||
| import {ProfileUiModel, Sections} from 'state/models/ui/profile' | import {ProfileUiModel} from 'state/models/ui/profile' | ||||||
| import {useStores} from 'state/index' | import {useStores} from 'state/index' | ||||||
| import {FeedItemModel} from 'state/models/feed-view' | import {FeedSliceModel} from 'state/models/feed-view' | ||||||
| import {ProfileHeader} from '../com/profile/ProfileHeader' | import {ProfileHeader} from '../com/profile/ProfileHeader' | ||||||
| import {FeedItem} from '../com/posts/FeedItem' | import {FeedSlice} from '../com/posts/FeedSlice' | ||||||
| import {PostFeedLoadingPlaceholder} from '../com/util/LoadingPlaceholder' | import {PostFeedLoadingPlaceholder} from '../com/util/LoadingPlaceholder' | ||||||
| import {ErrorScreen} from '../com/util/error/ErrorScreen' | import {ErrorScreen} from '../com/util/error/ErrorScreen' | ||||||
| import {ErrorMessage} from '../com/util/error/ErrorMessage' | import {ErrorMessage} from '../com/util/error/ErrorMessage' | ||||||
|  | @ -123,8 +123,8 @@ export const ProfileScreen = withAuthRequired( | ||||||
|               style={styles.emptyState} |               style={styles.emptyState} | ||||||
|             /> |             /> | ||||||
|           ) |           ) | ||||||
|         } else if (item instanceof FeedItemModel) { |         } else if (item instanceof FeedSliceModel) { | ||||||
|           return <FeedItem item={item} ignoreMuteFor={uiState.profile.did} /> |           return <FeedSlice slice={item} ignoreMuteFor={uiState.profile.did} /> | ||||||
|         } |         } | ||||||
|         return <View /> |         return <View /> | ||||||
|       }, |       }, | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue