在无限滚动中延迟加载时锁定滚动位置(向上滚动)

时间:2016-06-23 16:04:25

标签: angularjs infinite-scroll

我在我的应用程序中有一个对话/ WhatsApp喜欢的功能。最新消息位于底部,最旧消息位于顶部。

当我在消息顶部附近滚动它会自动加载更多但是我想保持滚动位置。我创建了一个指令来做到这一点,但问题是它闪烁。似乎发生的是新消息被加载并且滚动位置被重置为顶部。然后我的指令开始并将滚动位置重置到原来的位置。所以它有效,但看起来并不好。

有没有人有任何线索如何解决这个问题?

HTML:

<div class="conversation-scroll-container" scroll-lock="{{ vm.isLoading }}">

VM(加载消息功能):

return this.messageService
    .getMessages(conversation, dates)
    .then(data => {
        this.selectedConversation.applyNewMessages(data);

        return data;
    })
    .finally(() => {
            this.isLoading = false;
    });

最后指令(在TS中):

export class ScrollLockDirective implements angular.IDirective {
    public restrict = "A";
    private scrollLock = false;
    private scrollLockOffset = 0;

    constructor(private $timeout: angular.ITimeoutService) { }

    static factory(): ng.IDirectiveFactory {
        const directive = ($timeout: angular.ITimeoutService) => new ScrollLockDirective($timeout);

        directive.$inject = ["$timeout"];

        return directive;
    }

    public link = (scope: angular.IScope, element: angular.IAugmentedJQuery, attributes: angular.IAttributes) => {

        attributes.$observe("scrollLock", () => {

            if (attributes["scrollLock"] === "true") {
                this.scrollLock = true;
            }
            else {
                this.doLockScroll(element);
                this.scrollLock = false;
            }
        });

        element.bind("scroll", () => {
            this.doLockScroll(element);
        });

        scope.$on("$destroy", () => {
            element.unbind("scroll");
        });
    }

    private doLockScroll(element: angular.IAugmentedJQuery) {

        if (this.scrollLock) {
            this.$timeout(() => {
                element.scrollTop(element[0].scrollHeight - this.scrollLockOffset);
            });
        } else {
            this.scrollLockOffset = element[0].scrollHeight - element.scrollTop();
        }
    }
}

1 个答案:

答案 0 :(得分:-1)

我遇到了和你一样的问题,我也尝试设置scrollTop位置等等,但解决方案非常简单:你只需要渲染新项目,而不是重新渲染所有项目。