FIX: Decorate posts that are loaded after the initial render in post stream (#14600)
To clarify, this problem is not about the topic posts stream, it's about posts streams like the user Activity one in the profile page (or in technical terms anything using the `{{user-stream}}` component). Post decorations are currently applied inside a `didInsertElement` hook of the `{{user-stream}}` component. However, when the user scrolls the component will load more posts but these will be missing decorations because the `didInsertElement` is only fired once at the beginning of the component lifecycle. This PR makes the component keep track of the last decorated post/DOM node, and when new posts are loaded the component fire an event for each new post and pass the post's DOM node with the event. Our plugin API (I noticed this problem when I was working on https://github.com/discourse/discourse-follow/pull/37) Co-authored-by: Robin Ward <robin.ward@gmail.com>
This commit is contained in:
parent
82945a3d21
commit
7f3468e7d5
|
@ -14,6 +14,7 @@ import { schedule } from "@ember/runloop";
|
||||||
|
|
||||||
export default Component.extend(LoadMore, {
|
export default Component.extend(LoadMore, {
|
||||||
tagName: "ul",
|
tagName: "ul",
|
||||||
|
_lastDecoratedElement: null,
|
||||||
|
|
||||||
_initialize: on("init", function () {
|
_initialize: on("init", function () {
|
||||||
const filter = this.get("stream.filter");
|
const filter = this.get("stream.filter");
|
||||||
|
@ -47,6 +48,7 @@ export default Component.extend(LoadMore, {
|
||||||
$(this.element).on("click.discourse-redirect", ".excerpt a", (e) => {
|
$(this.element).on("click.discourse-redirect", ".excerpt a", (e) => {
|
||||||
return ClickTrack.trackClick(e, this.siteSettings);
|
return ClickTrack.trackClick(e, this.siteSettings);
|
||||||
});
|
});
|
||||||
|
this._updateLastDecoratedElement();
|
||||||
}),
|
}),
|
||||||
|
|
||||||
// This view is being removed. Shut down operations
|
// This view is being removed. Shut down operations
|
||||||
|
@ -59,6 +61,18 @@ export default Component.extend(LoadMore, {
|
||||||
$(this.element).off("click.discourse-redirect", ".excerpt a");
|
$(this.element).off("click.discourse-redirect", ".excerpt a");
|
||||||
}),
|
}),
|
||||||
|
|
||||||
|
_updateLastDecoratedElement() {
|
||||||
|
const nodes = this.element.querySelectorAll(".user-stream-item");
|
||||||
|
if (nodes.length === 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const lastElement = nodes[nodes.length - 1];
|
||||||
|
if (lastElement === this._lastDecoratedElement) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this._lastDecoratedElement = lastElement;
|
||||||
|
},
|
||||||
|
|
||||||
actions: {
|
actions: {
|
||||||
removeBookmark(userAction) {
|
removeBookmark(userAction) {
|
||||||
const stream = this.stream;
|
const stream = this.stream;
|
||||||
|
@ -123,7 +137,15 @@ export default Component.extend(LoadMore, {
|
||||||
|
|
||||||
this.set("loading", true);
|
this.set("loading", true);
|
||||||
const stream = this.stream;
|
const stream = this.stream;
|
||||||
stream.findItems().then(() => this.set("loading", false));
|
stream.findItems().then(() => {
|
||||||
|
this.set("loading", false);
|
||||||
|
let element = this._lastDecoratedElement?.nextElementSibling;
|
||||||
|
while (element) {
|
||||||
|
this.trigger("user-stream:new-item-inserted", element);
|
||||||
|
element = element.nextElementSibling;
|
||||||
|
}
|
||||||
|
this._updateLastDecoratedElement();
|
||||||
|
});
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
|
@ -311,12 +311,10 @@ class PluginApi {
|
||||||
if (!opts.onlyStream) {
|
if (!opts.onlyStream) {
|
||||||
decorate(ComposerEditor, "previewRefreshed", callback, opts.id);
|
decorate(ComposerEditor, "previewRefreshed", callback, opts.id);
|
||||||
decorate(DiscourseBanner, "didInsertElement", callback, opts.id);
|
decorate(DiscourseBanner, "didInsertElement", callback, opts.id);
|
||||||
decorate(
|
["didInsertElement", "user-stream:new-item-inserted"].forEach((event) => {
|
||||||
this.container.factoryFor("component:user-stream").class,
|
const klass = this.container.factoryFor("component:user-stream").class;
|
||||||
"didInsertElement",
|
decorate(klass, event, callback, opts.id);
|
||||||
callback,
|
});
|
||||||
opts.id
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue