frontend: Enhance Disconnected Component with user-friendly messages and animations

- Updated HTML structure for improved layout and branding.
- Added disconnect reason handling in TypeScript for better user feedback.
- Enhanced SCSS styles for a more visually appealing disconnected state.
- Introduced scaleIn animation for a smoother user experience.
This commit is contained in:
Carlos Santos 2025-07-02 17:05:46 +02:00
parent fbcb70dbc2
commit 4691888309
5 changed files with 245 additions and 27 deletions

View File

@ -1,10 +1,22 @@
<div class="disconnected-container">
<mat-card class="disconnected-card">
<mat-card-header>
<mat-card-title>You have left the room</mat-card-title>
</mat-card-header>
<mat-card-content>
<p>The meeting has ended.</p>
</mat-card-content>
</mat-card>
<div class="disconnect-content" role="main" aria-labelledby="disconnect-title">
<div class="disconnect-icon-container">
<img
src=""
alt="Brand Logo"
class="logo-image"
/>
</div>
<div class="disconnect-message">
<h1 id="disconnect-title" class="disconnect-title">Meeting Ended</h1>
<p class="disconnect-subtitle">
{{ disconnectReason || 'You have successfully left the video conference' }}
</p>
</div>
<div class="disconnect-footer">
<p class="footer-text">Thank you for using OpenVidu Meet</p>
</div>
</div>
</div>

View File

@ -1,33 +1,153 @@
@import '../../../../../../src/assets/styles/design-tokens';
.disconnected-container {
@include ov-theme-transition;
display: flex;
justify-content: center;
align-items: center;
height: 100vh;
background-color: var(--ov-meet-surface-background);
min-height: 100vh;
padding: var(--ov-meet-spacing-lg);
background: linear-gradient(135deg, var(--ov-meet-background-color) 0%, var(--ov-meet-background-secondary) 100%);
}
.disconnected-card {
width: 400px;
padding: 20px;
.disconnect-content {
@include ov-card;
max-width: 520px;
width: 100%;
text-align: center;
background-color: var(--ov-meet-surface-primary);
border-radius: var(--ov-meet-surface-radius);
padding: var(--ov-meet-spacing-xxl);
background: var(--ov-meet-surface-elevated);
box-shadow: var(--ov-meet-shadow-lg);
@include ov-mobile-down {
padding: var(--ov-meet-spacing-xl);
margin: var(--ov-meet-spacing-md);
max-width: calc(100vw - 2 * var(--ov-meet-spacing-md));
}
}
mat-card-header {
.disconnect-icon-container {
margin-bottom: var(--ov-meet-spacing-xl);
display: flex;
justify-content: center;
align-items: center;
.logo-image {
width: 50px;
height: auto;
object-fit: contain;
display: block;
transition: all var(--ov-meet-transition-normal);
@include ov-mobile-down {
width: 40px;
}
}
&::before {
content: '';
position: absolute;
width: 80px;
height: 80px;
background: rgba(25, 118, 210, 0.1);
border-radius: var(--ov-meet-radius-circle);
z-index: -1;
@include ov-mobile-down {
width: 64px;
height: 64px;
}
}
position: relative;
}
mat-card-title {
font-size: 1.5em;
.disconnect-message {
margin-bottom: var(--ov-meet-spacing-xl);
.disconnect-title {
font-size: var(--ov-meet-font-size-xxl);
font-weight: var(--ov-meet-font-weight-semibold);
color: var(--ov-meet-text-primary);
margin: 0 0 var(--ov-meet-spacing-sm) 0;
line-height: var(--ov-meet-line-height-tight);
@include ov-mobile-down {
font-size: var(--ov-meet-font-size-xl);
}
}
.disconnect-subtitle {
font-size: var(--ov-meet-font-size-md);
color: var(--ov-meet-text-secondary);
margin: 0;
line-height: var(--ov-meet-line-height-normal);
max-width: 400px;
margin-left: auto;
margin-right: auto;
@include ov-mobile-down {
font-size: var(--ov-meet-font-size-sm);
}
}
}
mat-card-content p {
font-size: 1em;
color: #555;
.disconnect-footer {
padding-top: var(--ov-meet-spacing-lg);
border-top: 1px solid var(--ov-meet-border-color-light);
.footer-text {
display: flex;
align-items: center;
justify-content: center;
gap: var(--ov-meet-spacing-xs);
font-size: var(--ov-meet-font-size-sm);
color: var(--ov-meet-text-hint);
margin: 0;
font-weight: var(--ov-meet-font-weight-medium);
}
@include ov-mobile-down {
.footer-text {
font-size: var(--ov-meet-font-size-xs);
flex-direction: column;
gap: var(--ov-meet-spacing-xs);
}
}
}
mat-card-actions {
margin-top: 20px;
.disconnect-content {
animation: fadeIn 0.6s ease-out;
}
.disconnect-icon-container {
animation: scaleIn 0.8s ease-out 0.2s both;
}
@media (prefers-contrast: high) {
.disconnect-footer {
border-top-width: 2px;
}
.disconnect-icon-container::before {
border: 2px solid var(--ov-meet-color-primary);
}
}
@media (prefers-reduced-motion: reduce) {
.disconnect-content,
.disconnect-icon-container {
animation: none;
}
.logo-image {
transition: none;
}
}
[data-theme='dark'] {
.disconnect-icon-container::before {
background: rgba(25, 118, 210, 0.2);
}
}

View File

@ -1,11 +1,53 @@
import { Component } from '@angular/core';
import { Component, OnInit } from '@angular/core';
import { CommonModule } from '@angular/common';
import { MatCardModule } from '@angular/material/card';
import { MatButtonModule } from '@angular/material/button';
import { MatIconModule } from '@angular/material/icon';
import { ActivatedRoute } from '@angular/router';
@Component({
selector: 'ov-disconnected',
standalone: true,
imports: [MatCardModule],
imports: [CommonModule, MatCardModule, MatButtonModule, MatIconModule],
templateUrl: './disconnected.component.html',
styleUrl: './disconnected.component.scss'
})
export class DisconnectedComponent {}
export class DisconnectedComponent implements OnInit {
disconnectReason?: string;
constructor(private route: ActivatedRoute) {}
ngOnInit(): void {
// Get disconnect reason from query parameters
this.getDisconnectReasonFromQueryParams();
}
/**
* Retrieves the disconnect reason from URL query parameters
*/
private getDisconnectReasonFromQueryParams(): void {
const reason = this.route.snapshot.queryParams['reason'];
if (reason) {
// Map technical reasons to user-friendly messages
this.disconnectReason = this.mapReasonToUserMessage(reason);
}
}
/**
* Maps technical disconnect reasons to user-friendly messages
*/
private mapReasonToUserMessage(reason: string): string {
const reasonMap: { [key: string]: string } = {
disconnect: 'You have successfully disconnected from the meeting',
forceDisconnectByUser: 'You were removed from the meeting by meeting host',
forceDisconnectByServer: 'Your connection was terminated by the server',
sessionClosedByServer: 'The meeting was ended by the host',
networkDisconnect: 'Connection lost due to network connectivity issues',
openviduDisconnect: 'The meeting ended due to technical difficulties',
roomDeleted: 'The meeting room has been deleted',
browserClosed: 'The meeting ended when your browser was closed'
};
return reasonMap[reason] || reasonMap['disconnect'];
}
}

View File

@ -267,7 +267,40 @@ export class VideoRoomComponent implements OnInit, OnDestroy {
this.sessionStorageService.removeModeratorSecret(event.roomName);
}
await this.navigationService.redirectTo(redirectURL, isExternalURL);
// Add disconnect reason as query parameter if redirecting to disconnect page
let finalRedirectURL = redirectURL;
if (!isExternalURL && (redirectURL === '/disconnected' || redirectURL.includes('/disconnected'))) {
const reasonParam = this.getReasonParamFromEvent(event.reason, isRoomDeleted);
const separator = redirectURL.includes('?') ? '&' : '?';
finalRedirectURL = `${redirectURL}${separator}reason=${encodeURIComponent(reasonParam)}`;
}
await this.navigationService.redirectTo(finalRedirectURL, isExternalURL);
}
/**
* Maps ParticipantLeftReason to a query parameter value
*/
private getReasonParamFromEvent(reason: ParticipantLeftReason, isRoomDeleted: boolean): string {
if (isRoomDeleted) {
return 'roomDeleted';
}
switch (reason) {
default:
case ParticipantLeftReason.LEAVE:
return 'disconnect';
case ParticipantLeftReason.PARTICIPANT_REMOVED:
return 'forceDisconnectByUser';
case ParticipantLeftReason.SERVER_SHUTDOWN:
return 'sessionClosedByServer';
case ParticipantLeftReason.NETWORK_DISCONNECT:
return 'networkDisconnect';
case ParticipantLeftReason.SIGNAL_CLOSE:
return 'openviduDisconnect';
case ParticipantLeftReason.BROWSER_UNLOAD:
return 'browserClosed';
}
}
async onRecordingStartRequested(event: RecordingStartRequestedEvent) {

View File

@ -91,6 +91,17 @@
}
}
@keyframes scaleIn {
from {
opacity: 0;
transform: scale(0.9);
}
to {
opacity: 1;
transform: scale(1);
}
}
// Transition classes
.fade-in {
animation: fadeIn 0.6s ease-out forwards;