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

Communication: Remember last scroll position when switching conversations #9614

Merged
merged 23 commits into from
Nov 9, 2024
Merged
Show file tree
Hide file tree
Changes from 9 commits
Commits
Show all changes
23 commits
Select commit Hold shift + click to select a range
0ad2e22
Add scroll position memorization
cremertim Oct 28, 2024
28a0dc6
Reduced timeout
cremertim Oct 28, 2024
a76e349
performance
cremertim Oct 28, 2024
9726517
Default scroll to bottom of the messages if you create a new message
cremertim Oct 29, 2024
9979342
Added test cases
cremertim Oct 29, 2024
d9b1024
First implementation via ID
cremertim Oct 29, 2024
21c0c1b
Merge branch 'develop' into bugfix/communication/adjust-channel-jumping
cremertim Oct 29, 2024
870d20f
optimization from coderabit
cremertim Oct 29, 2024
82aff7f
Merge remote-tracking branch 'origin/bugfix/communication/adjust-chan…
cremertim Oct 29, 2024
1b5de0b
Added integration Test for search
cremertim Oct 30, 2024
4ad6553
variable readonly
cremertim Oct 30, 2024
c98aef2
Added tests
cremertim Oct 30, 2024
7398ef2
Merge branch 'develop' into bugfix/communication/adjust-channel-jumping
cremertim Oct 30, 2024
674e3b5
Merge branch 'develop' into bugfix/communication/adjust-channel-jumping
cremertim Oct 30, 2024
3fb2349
Merge remote-tracking branch 'origin/develop' into bugfix/communicati…
Pablosanqt Oct 31, 2024
7d7ad2c
Small bugfix
Pablosanqt Oct 31, 2024
94e18ab
Fix for change between modules
Pablosanqt Oct 31, 2024
355a91c
Fix tests
Pablosanqt Oct 31, 2024
41d9509
Merge remote-tracking branch 'origin/develop' into bugfix/communicati…
PaRangger Nov 3, 2024
095f321
Fix for proper scroll position on load and on new message
PaRangger Nov 4, 2024
8fe911b
Merge remote-tracking branch 'origin/develop' into bugfix/communicati…
PaRangger Nov 8, 2024
b4da15b
Fix merge conflict issues
PaRangger Nov 8, 2024
0e18443
Merge branch 'develop' into bugfix/communication/adjust-channel-jumping
krusche Nov 9, 2024
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
Original file line number Diff line number Diff line change
Expand Up @@ -96,15 +96,10 @@
@if (getAsChannel(_activeConversation)?.isAnnouncementChannel) {
<div class="pt-2">
<button class="btn btn-md btn-primary" (click)="createEditModal.open()" jhiTranslate="artemisApp.metis.newAnnouncement"></button>
<jhi-post-create-edit-modal
#createEditModal
[posting]="newPost!"
[isCommunicationPage]="true"
(onCreate)="createEmptyPost(); scrollToBottomOfMessages()"
/>
<jhi-post-create-edit-modal #createEditModal [posting]="newPost!" [isCommunicationPage]="true" (onCreate)="handleNewMessageCreated()" />
</div>
} @else {
<jhi-message-inline-input class="message-input" [posting]="newPost!" (onCreate)="createEmptyPost(); scrollToBottomOfMessages()" />
<jhi-message-inline-input class="message-input" [posting]="newPost!" (onCreate)="handleNewMessageCreated()" />
}
</div>
}
Expand All @@ -114,15 +109,10 @@
@if (getAsChannel(_activeConversation)?.isAnnouncementChannel) {
<div class="pt-2">
<button class="btn btn-md btn-primary" (click)="createEditModal.open()" jhiTranslate="artemisApp.metis.newAnnouncement"></button>
<jhi-post-create-edit-modal
#createEditModal
[posting]="newPost!"
[isCommunicationPage]="true"
(onCreate)="createEmptyPost(); scrollToBottomOfMessages()"
/>
<jhi-post-create-edit-modal #createEditModal [posting]="newPost!" [isCommunicationPage]="true" (onCreate)="handleNewMessageCreated()" />
</div>
} @else {
<jhi-message-inline-input class="message-input" [posting]="newPost!" (onCreate)="createEmptyPost(); scrollToBottomOfMessages()" />
<jhi-message-inline-input class="message-input" [posting]="newPost!" (onCreate)="handleNewMessageCreated()" />
}
</div>
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ import {
OnInit,
Output,
QueryList,
Renderer2,
ViewChild,
ViewChildren,
ViewEncapsulation,
Expand All @@ -30,6 +31,7 @@ import { canCreateNewMessageInConversation } from 'app/shared/metis/conversation
import { debounceTime, distinctUntilChanged } from 'rxjs/operators';
import { LayoutService } from 'app/shared/breakpoints/layout.service';
import { CustomBreakpointNames } from 'app/shared/breakpoints/breakpoints.service';
import { PostingThreadComponent } from 'app/shared/metis/posting-thread/posting-thread.component';

@Component({
selector: 'jhi-conversation-messages',
Expand All @@ -39,16 +41,22 @@ import { CustomBreakpointNames } from 'app/shared/breakpoints/breakpoints.servic
})
export class ConversationMessagesComponent implements OnInit, AfterViewInit, OnDestroy {
private ngUnsubscribe = new Subject<void>();
sessionStorageKey = 'conversationId.scrollPosition.';
cremertim marked this conversation as resolved.
Show resolved Hide resolved

readonly PageType = PageType;
readonly ButtonType = ButtonType;

private scrollDebounceTime = 100; // ms
private scrollSubject = new Subject<number>();
private canStartSaving = false;

@Output() openThread = new EventEmitter<Post>();

@ViewChild('searchInput')
searchInput: ElementRef;

@ViewChildren('postingThread')
messages: QueryList<any>;
messages: QueryList<PostingThreadComponent>;
@ViewChild('container')
content: ElementRef;
@Input()
Expand All @@ -70,6 +78,7 @@ export class ConversationMessagesComponent implements OnInit, AfterViewInit, OnD
searchText = '';
_activeConversation?: ConversationDTO;

elementsAtScrollPosition: PostingThreadComponent[];
newPost?: Post;
posts: Post[] = [];
totalNumberOfPosts = 0;
Expand All @@ -83,6 +92,7 @@ export class ConversationMessagesComponent implements OnInit, AfterViewInit, OnD
isMobile = false;

private layoutService: LayoutService = inject(LayoutService);
private renderer = inject(Renderer2);

constructor(
public metisService: MetisService, // instance from course-conversations.component
Expand All @@ -94,6 +104,7 @@ export class ConversationMessagesComponent implements OnInit, AfterViewInit, OnD
this.subscribeToSearch();
this.subscribeToMetis();
this.subscribeToActiveConversation();
this.setupScrollDebounce();
this.isMobile = this.layoutService.isBreakpointActive(CustomBreakpointNames.extraSmall);

this.layoutService
Expand Down Expand Up @@ -137,11 +148,21 @@ export class ConversationMessagesComponent implements OnInit, AfterViewInit, OnD

ngAfterViewInit() {
this.messages.changes.pipe(takeUntil(this.ngUnsubscribe)).subscribe(this.handleScrollOnNewMessage);
this.messages.changes.pipe(takeUntil(this.ngUnsubscribe)).subscribe(() => {
if (this.posts.length > 0) {
const savedScrollId = sessionStorage.getItem(this.sessionStorageKey + this._activeConversation?.id) ?? '';
setTimeout(() => this.goToLastSelectedElement(parseInt(savedScrollId, 10)), 0);
}
});
cremertim marked this conversation as resolved.
Show resolved Hide resolved
this.content.nativeElement.addEventListener('scroll', () => {
this.findElementsAtScrollPosition();
});
cremertim marked this conversation as resolved.
Show resolved Hide resolved
}

ngOnDestroy(): void {
this.ngUnsubscribe.next();
this.ngUnsubscribe.complete();
this.content.nativeElement.removeEventListener('scroll', this.saveScrollPosition);
cremertim marked this conversation as resolved.
Show resolved Hide resolved
}

private onActiveConversationChange() {
Expand All @@ -150,6 +171,7 @@ export class ConversationMessagesComponent implements OnInit, AfterViewInit, OnD
this.searchInput.nativeElement.value = '';
this.searchText = '';
}
this.canStartSaving = false;
this.onSearch();
this.createEmptyPost();
}
Expand Down Expand Up @@ -261,4 +283,55 @@ export class ConversationMessagesComponent implements OnInit, AfterViewInit, OnD
this.searchInput.nativeElement.dispatchEvent(new Event('input'));
}
}

private setupScrollDebounce(): void {
this.scrollSubject.pipe(debounceTime(this.scrollDebounceTime), takeUntil(this.ngUnsubscribe)).subscribe((scrollTop) => {
if (this._activeConversation?.id) {
sessionStorage.setItem(this.sessionStorageKey + this._activeConversation.id, scrollTop.toString());
}
});
}
cremertim marked this conversation as resolved.
Show resolved Hide resolved

saveScrollPosition = (postId: number) => {
this.scrollSubject.next(postId);
};

handleNewMessageCreated() {
this.createEmptyPost();
this.scrollToBottomOfMessages();
}
PaRangger marked this conversation as resolved.
Show resolved Hide resolved
private async goToLastSelectedElement(lastScrollPosition: number) {
if (!lastScrollPosition) {
this.scrollToBottomOfMessages();
this.canStartSaving = true;
return;
}
const messageArray = this.messages.toArray();
const element = messageArray.find((message) => message.post.id === lastScrollPosition); // Suchen nach dem Post

if (!element) {
this.fetchNextPage();
} else {
await this.renderer.selectRootElement(element.elementRef.nativeElement, true).scrollIntoView({ behavior: 'instant', block: 'center' });
this.canStartSaving = true;
}
}
cremertim marked this conversation as resolved.
Show resolved Hide resolved

private findElementsAtScrollPosition() {
const messageArray = this.messages.toArray();
const containerRect = this.content.nativeElement.getBoundingClientRect();
const visibleMessages = [];
for (const message of messageArray) {
if (!message.elementRef?.nativeElement || !message.post?.id) continue;
const rect = message.elementRef.nativeElement.getBoundingClientRect();
if (rect.top >= containerRect.top && rect.bottom <= containerRect.bottom) {
visibleMessages.push(message);
break; // Only need the first visible message
}
}
this.elementsAtScrollPosition = visibleMessages;
if (this.elementsAtScrollPosition && this.canStartSaving) {
this.saveScrollPosition(this.elementsAtScrollPosition[0].post.id!);
}
}
}
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import { ChangeDetectionStrategy, Component, EventEmitter, Input, Output } from '@angular/core';
import { ChangeDetectionStrategy, Component, ElementRef, EventEmitter, Input, Output, inject } from '@angular/core';
import { Post } from 'app/entities/metis/post.model';
import dayjs from 'dayjs/esm';

Expand All @@ -17,4 +17,6 @@ export class PostingThreadComponent {
@Input() showChannelReference?: boolean;
@Input() hasChannelModerationRights = false;
@Output() openThread = new EventEmitter<Post>();

elementRef = inject(ElementRef);
}
Original file line number Diff line number Diff line change
Expand Up @@ -131,6 +131,25 @@ examples.forEach((activeConversation) => {
expect(getFilteredPostSpy).toHaveBeenCalledOnce();
}));

it('should save the scroll position in sessionStorage', fakeAsync(() => {
// Überwache sessionStorage.setItem
const setItemSpy = jest.spyOn(sessionStorage, 'setItem');
component.ngOnInit();
component.saveScrollPosition(15);
tick(100);
const expectedKey = `${component.sessionStorageKey}${component._activeConversation?.id}`;
const expectedValue = '15';
expect(setItemSpy).toHaveBeenCalledWith(expectedKey, expectedValue);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

could we be more specific here. This can be very helpful in debugging if there is a bug somewhere.

Suggested change
expect(setItemSpy).toHaveBeenCalledWith(expectedKey, expectedValue);
expect(setItemSpy).toHaveBeenCalledExactlyOnceWith(expectedKey, expectedValue);

}));

it('should scroll to the bottom when a new message is created', fakeAsync(() => {
component.content.nativeElement.scrollTop = 100;
fixture.detectChanges();
component.handleNewMessageCreated();
tick();
expect(component.content.nativeElement.scrollTop).toBe(component.content.nativeElement.scrollHeight);
}));

it('should create empty post with the correct conversation type', fakeAsync(() => {
const createEmptyPostForContextSpy = jest.spyOn(metisService, 'createEmptyPostForContext').mockReturnValue(new Post());
component.createEmptyPost();
Expand Down
Loading