모든 댓글이 아닌 바로 추가된 댓글에 클래스를 추가하는 방법

안녕하세요, 제 댓글 섹션에 새로 추가된 댓글에 부트스트랩 효과 fadeInDown을 추가해야 합니다. 작동하지만 불행히도 페이지를 새로고침할 때 모든 댓글에 영향을 미치고 댓글을 편집할 때도 효과가 추가됩니다. 새로 생성된 댓글에만 이 특정 클래스를 추가할 수 있습니까? 나머지 댓글이나 다른 애플리케이션 기능에 영향을 미치지 않습니까? 나는 어떤 조언이라도 기뻐할 것입니다.
제가 작업하고 있는 구성 요소는 다음과 같습니다. 댓글 섹션 HTML

<h6>Comments: ({{commentsCount}})</h6>
<app-new-comment class="d-block mt-3" [requestId]="requestId" (update)="updateComments()"> 
</app-new-comment>
<app-comment *ngFor="let comment of comments; let i = first"
         [ngClass]="{'fadeInDown': i}"
         class="d-block mt-3 animated"
         [comment]="comment"
         (update)="updateComments()"></app-comment>


댓글 섹션 TS

export class CommentSectionComponent implements OnInit, OnDestroy {

@Input() requestId: string
comments: CommentDetailsDto[]
commentsCount = 0

private subscription: Subscription

constructor(
    private commentsService: CommentsService,
    private alert: AlertService
) {
}

ngOnInit(): void {
    this.updateComments()
}

ngOnDestroy(): void {
    this.unsubscribe()
}

updateComments(): void {
    this.unsubscribe()
    this.subscription = this.commentsService.getComments(this.requestId)
        .subscribe({
            next: (comments: CommentDetailsDto[]) => {
                this.comments = comments.reverse()
                this.commentsCount = this.getCommentsCount(comments)
            },
            error: (error: HttpErrorResponse) => {
                this.alert.handleHttpError(error)
            }
        })
}

private getCommentsCount(comments: CommentDetailsDto[]): number {
    return comments.reduce(
        (cnt, comment) => cnt + 1 + (comment.replies ? this.getCommentsCount(comment.replies) : 0),
        0
    )
}

private unsubscribe(): void {
    if (this.subscription instanceof Subscription) {
        this.subscription.unsubscribe()
    }
}

}


코멘트 섹션 SCSS

@keyframes fadeInDown {
from {
    opacity: 0;
    transform: translate3d(0, -100%, 0);
}
to {
    opacity: 1;
    transform: translate3d(0, 0, 0);
}
}

.fadeInDown {
    animation-name: fadeInDown;
}


새 댓글 구성 요소 HTML

<div class="d-flex">
<app-user-image class="mr-3"></app-user-image>
<textarea #textArea class="d-block" placeholder="Add comment" [(ngModel)]="newComment"> 
 </textarea>
 </div>
 <div class="text-right mt-3">
    <button class="btn btn-primary font-weight-bold px-5" type="button" 
 [disabled]="!newComment" (click)="submit()">
       Post
     </button>
 </div>


새로운 댓글 TS

export class NewCommentComponent implements OnInit {

@ViewChild('textArea') textArea: ElementRef<HTMLTextAreaElement>
@Input() requestId?: string
@Input() comment?: CommentDetailsDto
@Output() update = new EventEmitter()
newComment: string

private readonly commentResponseObserver = {
    error: (error: HttpErrorResponse) => {
        this.alert.handleHttpError(error)
    },
    complete: () => {
        delete this.newComment
        this.update.emit()
        this.alert.showSuccess('Comment submitted successfully')
    }
}

constructor(
    private commentsService: CommentsService,
    private alert: AlertService,
    private route: ActivatedRoute
) {
}

ngOnInit(): void {
    if (this.comment) {
        this.textArea.nativeElement.focus()
    }
}

submit(): void {
    if (this.requestId) {
        this.addComment()
    } else if (this.comment) {
        this.addReply()
    }
}

addComment(): void {
    if (this.newComment) {
        this.commentsService.addComment(this.requestId, this.newComment)
            .subscribe(this.commentResponseObserver)
    }
}

addReply(): void {
    if (this.newComment) {
        this.commentsService.addReply(this.route.snapshot.queryParamMap.get('requestId'), this.comment, this.newComment)
            .subscribe(this.commentResponseObserver)
    }
}
}


주석 구성 요소 HTML

<div class="comment">
<app-user-image></app-user-image>
<div class="position-relative d-inline-block flex-fill">
    <div class="d-flex justify-content-between align-items-center">
        <div><strong>{{comment.author.name}} / {{comment.author.id}}</strong><span
            class="date">{{comment.created | krakenDateTime}}</span></div>
        <div class="actions">
            <button *ngIf="this.hateoas.supports(comment, 'update') && !edit"
                    type="button" class="bg-transparent border-0" title="Edit"
                    (click)="toggleEdit()"><i class="icon-kraken icon-kraken-edit"></i></button>
            <button *ngIf="this.hateoas.supports(comment, 'delete')"
                    type="button" class="bg-transparent border-0" title="Delete"
                    (click)="displayDeletionConfirmation()"><i class="icon-kraken icon-kraken-trash"></i></button>
        </div>
    </div>
    <textarea *ngIf="edit; else readonlyComment"
              #textarea
              class="d-block w-100"
              style="min-height: 7rem;"
              [rows]="rows()"
              [(ngModel)]="commentContent"></textarea>
    <ng-template #readonlyComment>
        <div [innerHTML]="commentContentHtml()"></div>
    </ng-template>
    <strong *ngIf="showReplyButton"
            class="reply-button"
            (click)="toggleReplyComponent()"><a href="#temporaryLastRow">Reply</a></strong>
    </div>
 </div>
 <div *ngIf="edit" class="animated fadeIn text-right mt-3">
    <button class="btn btn-sm discard" (click)="cancelEdit()">Discard</button>
    <button class="btn btn-sm ml-2 update" (click)="updateComment()">Update</button>
 </div>
 <div class="replies">
     <app-new-comment *ngIf="showNewReplyWindow"
                 class="d-block mt-3"
                 [comment]="comment"
                 (update)="this.update.emit()"></app-new-comment>
    <app-comment *ngIf="firstReply"
             class="d-block my-3"
             [comment]="firstReply"
             (update)="update.emit()"></app-comment>
     <div *ngIf="moreReplies.length" class="replies-toggle" (click)="showMoreReplies = 
 !showMoreReplies">
    <div class="horizontal-bar"></div>
    <span class="mx-3">{{showMoreReplies ? 'Hide' : 'See'}} {{moreReplies.length}} more 
 comments</span>
    <div class="horizontal-bar"></div>
</div>
<div *ngIf="showMoreReplies">
    <app-comment *ngFor="let reply of moreReplies"
                 class="d-block my-3"
                 [comment]="reply"
                 (update)="update.emit()"></app-comment>
</div>
<span id="temporaryLastRow"></span>


코멘트 TS

export class CommentComponent implements OnInit {

@ViewChild('textarea', {static: true}) textarea: ElementRef<HTMLTextAreaElement>

@Input() comment: CommentDetailsDto
@Output() update = new EventEmitter()

edit = false
showReplyButton = false
showNewReplyWindow = false
showMoreReplies = false
commentContent: string
firstReply: CommentDetailsDto
moreReplies: CommentDetailsDto[]

constructor(
    private commentsService: CommentsService,
    private modalFactoryService: ModalFactoryService,
    public hateoas: HateoasService,
    private alert: AlertService
) {
}

ngOnInit(): void {
    this.commentContent = this.comment.content
    this.showReplyButton = this.hateoas.supports(this.comment, 'reply')
    this.moreReplies = this.comment.replies.reverse().slice(1)
    this.firstReply = this.comment.replies[0]
}

toggleEdit(): void {
    this.edit = !this.edit
}

updateComment(): void {
    this.commentsService.updateComment(this.comment, this.commentContent)
        .subscribe({
            error: (error: HttpErrorResponse) => {
                this.alert.handleHttpError(error)
            },
            complete: () => {
                this.alert.showSuccess('Comment updated successfully')
                this.update.emit()
            }
        })
}

cancelEdit(): void {
    this.edit = false
    this.commentContent = this.comment.content
}

displayDeletionConfirmation(): void {
    this.modalFactoryService.openConfirmationModal(
        'Delete Comment',
        {
            message: 'Are you sure you want to delete this comment?',
            yesOptionButton: 'Yes',
            noOptionButton: 'No'
        },
        () => {
            this.hateoas.execute(this.comment, 'delete')
                .subscribe({
                    error: (error: HttpErrorResponse) => {
                        this.alert.handleHttpError(error)
                    },
                    complete: () => {
                        this.alert.showSuccess('Comment deleted successfully')
                        this.update.emit()
                    }
                })
        }
    )
}

toggleReplyComponent(): void {
    this.showNewReplyWindow = !this.showNewReplyWindow
}

commentContentHtml(): string {
    return this.commentContent.replace(/\n/g, '<br>')
}

rows(): number {
    const newLines = this.commentContent.match(/\n/g) || []
    return newLines.length + 1
}
}

 export interface Author {
id: string
name: string
}

 export interface CommentDetailsDto extends Resource {
content: string
author: Author
replies: CommentDetailsDto[]
created: string
 }

좋은 웹페이지 즐겨찾기