FIX: Store scroll position when using Back button (#16658)
For some pages, when navigating to a topic and then pressing the Back
browser button to go back to the topic list, the scroll position was
not preserved and the user was taken to the beginning of the list.
This happened because the application failed to detect when the user
used the Back button and whether the topic list should be fetch from
the cache or not. The scroll position is preserved only for cached
topic lists.
Other improvements:
* Improve isPoppedState
* Reset position for topic-lists from user-activity page
* Remove usage of jQuery
* Make sure the scrollTo function has effect
Follow up to 618a1ba571
.
This commit is contained in:
parent
6f00feaea0
commit
1d76c5ef5d
|
@ -1,6 +1,6 @@
|
||||||
import Component from "@ember/component";
|
import Component from "@ember/component";
|
||||||
import { action } from "@ember/object";
|
import { action } from "@ember/object";
|
||||||
import { schedule } from "@ember/runloop";
|
import { next, schedule } from "@ember/runloop";
|
||||||
import bootbox from "bootbox";
|
import bootbox from "bootbox";
|
||||||
import { openBookmarkModal } from "discourse/controllers/bookmark";
|
import { openBookmarkModal } from "discourse/controllers/bookmark";
|
||||||
import { ajax } from "discourse/lib/ajax";
|
import { ajax } from "discourse/lib/ajax";
|
||||||
|
@ -28,10 +28,10 @@ export default Component.extend(Scrolling, {
|
||||||
|
|
||||||
scrollToLastPosition() {
|
scrollToLastPosition() {
|
||||||
const scrollTo = this.session.bookmarkListScrollPosition;
|
const scrollTo = this.session.bookmarkListScrollPosition;
|
||||||
if (scrollTo > 0) {
|
if (scrollTo >= 0) {
|
||||||
schedule("afterRender", () => {
|
schedule("afterRender", () => {
|
||||||
if (this.element && !this.isDestroying && !this.isDestroyed) {
|
if (this.element && !this.isDestroying && !this.isDestroyed) {
|
||||||
window.scrollTo(0, scrollTo + 1);
|
next(() => window.scrollTo(0, scrollTo + 1));
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
import { observes, on } from "discourse-common/utils/decorators";
|
import { observes, on } from "discourse-common/utils/decorators";
|
||||||
import { schedule, scheduleOnce } from "@ember/runloop";
|
import { next, schedule, scheduleOnce } from "@ember/runloop";
|
||||||
import Component from "@ember/component";
|
import Component from "@ember/component";
|
||||||
import LoadMore from "discourse/mixins/load-more";
|
import LoadMore from "discourse/mixins/load-more";
|
||||||
import UrlRefresh from "discourse/mixins/url-refresh";
|
import UrlRefresh from "discourse/mixins/url-refresh";
|
||||||
|
@ -14,8 +14,12 @@ export default Component.extend(UrlRefresh, LoadMore, {
|
||||||
@observes("model")
|
@observes("model")
|
||||||
_readjustScrollPosition() {
|
_readjustScrollPosition() {
|
||||||
const scrollTo = this.session.topicListScrollPosition;
|
const scrollTo = this.session.topicListScrollPosition;
|
||||||
if (scrollTo > 0) {
|
if (scrollTo >= 0) {
|
||||||
schedule("afterRender", () => $(window).scrollTop(scrollTo + 1));
|
schedule("afterRender", () => {
|
||||||
|
if (this.element && !this.isDestroying && !this.isDestroyed) {
|
||||||
|
next(() => window.scrollTo(0, scrollTo + 1));
|
||||||
|
}
|
||||||
|
});
|
||||||
} else {
|
} else {
|
||||||
scheduleOnce("afterRender", this, this.loadMoreUnlessFull);
|
scheduleOnce("afterRender", this, this.loadMoreUnlessFull);
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,7 +3,7 @@ import discourseComputed, { observes } from "discourse-common/utils/decorators";
|
||||||
import Component from "@ember/component";
|
import Component from "@ember/component";
|
||||||
import LoadMore from "discourse/mixins/load-more";
|
import LoadMore from "discourse/mixins/load-more";
|
||||||
import { on } from "@ember/object/evented";
|
import { on } from "@ember/object/evented";
|
||||||
import { schedule } from "@ember/runloop";
|
import { next, schedule } from "@ember/runloop";
|
||||||
import showModal from "discourse/lib/show-modal";
|
import showModal from "discourse/lib/show-modal";
|
||||||
|
|
||||||
export default Component.extend(LoadMore, {
|
export default Component.extend(LoadMore, {
|
||||||
|
@ -75,10 +75,10 @@ export default Component.extend(LoadMore, {
|
||||||
}
|
}
|
||||||
|
|
||||||
const scrollTo = this.session.topicListScrollPosition;
|
const scrollTo = this.session.topicListScrollPosition;
|
||||||
if (scrollTo > 0) {
|
if (scrollTo >= 0) {
|
||||||
schedule("afterRender", () => {
|
schedule("afterRender", () => {
|
||||||
if (this.element && !this.isDestroying && !this.isDestroyed) {
|
if (this.element && !this.isDestroying && !this.isDestroyed) {
|
||||||
$(window).scrollTop(scrollTo + 1);
|
next(() => window.scrollTo(0, scrollTo + 1));
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
|
@ -11,7 +11,7 @@ import { getOwner } from "discourse-common/lib/get-owner";
|
||||||
import { observes } from "discourse-common/utils/decorators";
|
import { observes } from "discourse-common/utils/decorators";
|
||||||
import { on } from "@ember/object/evented";
|
import { on } from "@ember/object/evented";
|
||||||
import { popupAjaxError } from "discourse/lib/ajax-error";
|
import { popupAjaxError } from "discourse/lib/ajax-error";
|
||||||
import { schedule } from "@ember/runloop";
|
import { next, schedule } from "@ember/runloop";
|
||||||
|
|
||||||
export default Component.extend(LoadMore, {
|
export default Component.extend(LoadMore, {
|
||||||
tagName: "ul",
|
tagName: "ul",
|
||||||
|
@ -74,10 +74,10 @@ export default Component.extend(LoadMore, {
|
||||||
|
|
||||||
_scrollToLastPosition() {
|
_scrollToLastPosition() {
|
||||||
const scrollTo = this.session.userStreamScrollPosition;
|
const scrollTo = this.session.userStreamScrollPosition;
|
||||||
if (scrollTo > 0) {
|
if (scrollTo >= 0) {
|
||||||
schedule("afterRender", () => {
|
schedule("afterRender", () => {
|
||||||
if (this.element && !this.isDestroying && !this.isDestroyed) {
|
if (this.element && !this.isDestroying && !this.isDestroyed) {
|
||||||
window.scrollTo(0, scrollTo + 1);
|
next(() => window.scrollTo(0, scrollTo + 1));
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
|
@ -67,7 +67,7 @@ export default Controller.extend({
|
||||||
this.set("loadingMore", true);
|
this.set("loadingMore", true);
|
||||||
|
|
||||||
return this._loadMoreBookmarks(this.q)
|
return this._loadMoreBookmarks(this.q)
|
||||||
.then((response) => this._processLoadResponse(response))
|
.then((response) => this._processLoadResponse(this.q, response))
|
||||||
.catch(() => this._bookmarksListDenied())
|
.catch(() => this._bookmarksListDenied())
|
||||||
.finally(() => this.set("loadingMore", false));
|
.finally(() => this.set("loadingMore", false));
|
||||||
},
|
},
|
||||||
|
@ -91,12 +91,13 @@ export default Controller.extend({
|
||||||
this.set("permissionDenied", true);
|
this.set("permissionDenied", true);
|
||||||
},
|
},
|
||||||
|
|
||||||
_processLoadResponse(response) {
|
_processLoadResponse(searchTerm, response) {
|
||||||
if (!response || !response.user_bookmark_list) {
|
if (!response || !response.user_bookmark_list) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
response = response.user_bookmark_list;
|
response = response.user_bookmark_list;
|
||||||
|
this.model.searchTerm = searchTerm;
|
||||||
this.model.loadMoreUrl = response.more_bookmarks_url;
|
this.model.loadMoreUrl = response.more_bookmarks_url;
|
||||||
|
|
||||||
if (response.bookmarks) {
|
if (response.bookmarks) {
|
||||||
|
|
|
@ -92,7 +92,10 @@ const DiscourseRoute = Route.extend({
|
||||||
},
|
},
|
||||||
|
|
||||||
isPoppedState(transition) {
|
isPoppedState(transition) {
|
||||||
return !transition._discourse_intercepted && !!transition.intent.url;
|
return (
|
||||||
|
!transition._discourse_intercepted &&
|
||||||
|
(!!transition.intent.url || !!transition.queryParamsOnly)
|
||||||
|
);
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -12,7 +12,11 @@ export default DiscourseRoute.extend({
|
||||||
model(params, transition) {
|
model(params, transition) {
|
||||||
const controller = this.controllerFor("user-activity-bookmarks");
|
const controller = this.controllerFor("user-activity-bookmarks");
|
||||||
|
|
||||||
if (this.isPoppedState(transition) && this.session.bookmarksModel) {
|
if (
|
||||||
|
this.isPoppedState(transition) &&
|
||||||
|
this.session.bookmarksModel &&
|
||||||
|
this.session.bookmarksModel.searchTerm === params.q
|
||||||
|
) {
|
||||||
return Promise.resolve(this.session.bookmarksModel);
|
return Promise.resolve(this.session.bookmarksModel);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -23,6 +23,12 @@ export default UserTopicListRoute.extend({
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
|
afterModel(model, transition) {
|
||||||
|
if (!this.isPoppedState(transition)) {
|
||||||
|
this.session.set("topicListScrollPosition", null);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
emptyState() {
|
emptyState() {
|
||||||
const title = I18n.t("user_activity.no_read_topics_title");
|
const title = I18n.t("user_activity.no_read_topics_title");
|
||||||
const body = I18n.t("user_activity.no_read_topics_body", {
|
const body = I18n.t("user_activity.no_read_topics_body", {
|
||||||
|
|
|
@ -22,6 +22,12 @@ export default UserTopicListRoute.extend({
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
|
afterModel(model, transition) {
|
||||||
|
if (!this.isPoppedState(transition)) {
|
||||||
|
this.session.set("topicListScrollPosition", null);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
emptyState() {
|
emptyState() {
|
||||||
const user = this.modelFor("user");
|
const user = this.modelFor("user");
|
||||||
const title = this.isCurrentUser(user)
|
const title = this.isCurrentUser(user)
|
||||||
|
|
Loading…
Reference in New Issue