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: Add emoji support to messages #9595

Merged
merged 10 commits into from
Oct 27, 2024
Original file line number Diff line number Diff line change
Expand Up @@ -229,7 +229,7 @@
</div>

<ng-template #simpleActionButton let-action="action">
<button type="button" class="btn btn-sm py-0" [ngbTooltip]="action.icon ? (action.translationKey | artemisTranslate) : undefined" (click)="action.executeInCurrentEditor()">
<button type="button" class="btn btn-sm py-0" [ngbTooltip]="action.icon ? (action.translationKey | artemisTranslate) : undefined" (click)="handleActionClick($event, action)">
@if (action.icon) {
<fa-icon [icon]="action.icon" />
} @else {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,7 @@ import { SafeHtml } from '@angular/platform-browser';
import { ArtemisMarkdownService } from 'app/shared/markdown.service';
import { parseMarkdownForDomainActions } from 'app/shared/markdown-editor/monaco/markdown-editor-parsing.helper';
import { COMMUNICATION_MARKDOWN_EDITOR_OPTIONS, DEFAULT_MARKDOWN_EDITOR_OPTIONS } from 'app/shared/monaco-editor/monaco-editor-option.helper';
import { EmojiAction } from 'app/shared/monaco-editor/model/actions/emoji.action';

export enum MarkdownEditorHeight {
INLINE = 100,
Expand Down Expand Up @@ -279,6 +280,16 @@ export class MarkdownEditorMonacoComponent implements AfterContentInit, AfterVie
return action?.hideInEditor ? undefined : action;
}

handleActionClick(event: MouseEvent, action: TextEditorAction): void {
const x = event.clientX;
const y = event.clientY;
if (action instanceof EmojiAction) {
action.setPoint({ x, y });
}

action.executeInCurrentEditor();
}
krusche marked this conversation as resolved.
Show resolved Hide resolved

ngAfterViewInit(): void {
this.adjustEditorDimensions();
this.monacoEditor.setWordWrap(true);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ import {
ViewEncapsulation,
forwardRef,
} from '@angular/core';
import { ViewContainerRef } from '@angular/core';
import { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';
import { MetisService } from 'app/shared/metis/metis.service';
import { LectureService } from 'app/lecture/lecture.service';
Expand All @@ -30,6 +31,8 @@ import { ChannelReferenceAction } from 'app/shared/monaco-editor/model/actions/c
import { UserMentionAction } from 'app/shared/monaco-editor/model/actions/communication/user-mention.action';
import { ExerciseReferenceAction } from 'app/shared/monaco-editor/model/actions/communication/exercise-reference.action';
import { LectureAttachmentReferenceAction } from 'app/shared/monaco-editor/model/actions/communication/lecture-attachment-reference.action';
import { EmojiAction } from 'app/shared/monaco-editor/model/actions/emoji.action';
import { Overlay, OverlayPositionBuilder } from '@angular/cdk/overlay';

@Component({
selector: 'jhi-posting-markdown-editor',
Expand Down Expand Up @@ -65,6 +68,9 @@ export class PostingMarkdownEditorComponent implements OnInit, ControlValueAcces
private courseManagementService: CourseManagementService,
private lectureService: LectureService,
private channelService: ChannelService,
public viewContainerRef: ViewContainerRef,
private overlay: Overlay,
private positionBuilder: OverlayPositionBuilder,
asliayk marked this conversation as resolved.
Show resolved Hide resolved
) {}

/**
Expand All @@ -79,6 +85,7 @@ export class PostingMarkdownEditorComponent implements OnInit, ControlValueAcces
new BoldAction(),
new ItalicAction(),
new UnderlineAction(),
new EmojiAction(this.viewContainerRef, this.overlay, this.positionBuilder),
new QuoteAction(),
new CodeAction(),
new CodeBlockAction(),
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,119 @@
import { TextEditorAction } from 'app/shared/monaco-editor/model/actions/text-editor-action.model';
import { faSmile } from '@fortawesome/free-solid-svg-icons';
import { TextEditor } from 'app/shared/monaco-editor/model/actions/adapter/text-editor.interface';
import { ViewContainerRef } from '@angular/core';
import { EmojiPickerComponent } from 'app/shared/metis/emoji/emoji-picker.component';
import { Overlay, OverlayPositionBuilder, OverlayRef } from '@angular/cdk/overlay';
import { ComponentPortal } from '@angular/cdk/portal';
import { TextEditorPosition } from 'app/shared/monaco-editor/model/actions/adapter/text-editor-position.model';
import { TextEditorRange } from 'app/shared/monaco-editor/model/actions/adapter/text-editor-range.model';

/**
* Action to open the emoji picker and insert the selected emoji into the editor.
*/
export class EmojiAction extends TextEditorAction {
static readonly ID = 'emoji.action';
private overlayRef: OverlayRef | null = null;
private position?: { x: number; y: number };

constructor(
private viewContainerRef: ViewContainerRef,
private overlay: Overlay,
private positionBuilder: OverlayPositionBuilder,
) {
super(EmojiAction.ID, 'artemisApp.multipleChoiceQuestion.editor.emoji', faSmile, undefined);
krusche marked this conversation as resolved.
Show resolved Hide resolved
asliayk marked this conversation as resolved.
Show resolved Hide resolved
}

/**
* Sets the position where the emoji picker should appear.
* @param param The {x, y} coordinates.
*/
setPoint(param: { x: number; y: number }): void {
this.position = { x: param.x, y: param.y };
}
krusche marked this conversation as resolved.
Show resolved Hide resolved

/**
* Triggers the opening of the emoji picker and attaches it to the view container.
* @param editor The editor in which to insert the emoji.
*/
run(editor: TextEditor): void {
if (this.overlayRef) {
this.destroyEmojiPicker();
return;
}

if (this.position) {
this.createEmojiPicker(editor, this.position);
}
}
krusche marked this conversation as resolved.
Show resolved Hide resolved

/**
* Creates and attaches the emoji picker component dynamically and handles emoji selection.
* @param editor The editor instance where the emoji will be inserted.
* @param position The {x, y} coordinates where the picker should appear.
*/
private createEmojiPicker(editor: TextEditor, position: { x: number; y: number }): void {
const positionStrategy = this.positionBuilder
.global()
.left(`${position.x - 15}px`)
.top(`${position.y - 15}px`);

this.overlayRef = this.overlay.create({
positionStrategy,
hasBackdrop: true,
backdropClass: 'cdk-overlay-transparent-backdrop',
scrollStrategy: this.overlay.scrollStrategies.reposition(),
width: '0',
});

const emojiPickerPortal = new ComponentPortal(EmojiPickerComponent, this.viewContainerRef);
const componentRef = this.overlayRef.attach(emojiPickerPortal);
const pickerElement = componentRef.location.nativeElement;
pickerElement.style.transform = 'translate(-100%, -100%)';

componentRef.instance.emojiSelect.subscribe((selection: { emoji: any; event: PointerEvent }) => {
this.insertEmojiAtCursor(editor, selection.emoji.native);
this.destroyEmojiPicker();
});
krusche marked this conversation as resolved.
Show resolved Hide resolved

this.overlayRef.backdropClick().subscribe(() => {
this.destroyEmojiPicker();
});
}

/**
* Inserts the selected emoji into the editor at the current cursor position.
* @param editor The editor instance.
* @param emoji The emoji to insert.
*/
insertEmojiAtCursor(editor: TextEditor, emoji: string): void {
const position = editor.getPosition();
if (!position) return;

this.insertTextAtPosition(editor, position, emoji);
krusche marked this conversation as resolved.
Show resolved Hide resolved

const newPosition = new TextEditorPosition(position.getLineNumber(), position.getColumn() + 2);
krusche marked this conversation as resolved.
Show resolved Hide resolved
editor.setPosition(newPosition);
editor.focus();
}

/**
* Inserts the given emoji text at the current cursor position.
* @param editor The editor instance.
* @param position The current cursor position.
* @param emoji The emoji text to insert.
*/
insertTextAtPosition(editor: TextEditor, position: TextEditorPosition, emoji: string): void {
this.replaceTextAtRange(editor, new TextEditorRange(position, position), emoji);
}

/**
* Destroys the emoji picker component after an emoji is selected or toggled.
*/
private destroyEmojiPicker(): void {
if (this.overlayRef) {
this.overlayRef.dispose();
this.overlayRef = null;
}
}
}
3 changes: 2 additions & 1 deletion src/main/webapp/i18n/de/multipleChoiceQuestion.json
Original file line number Diff line number Diff line change
Expand Up @@ -45,7 +45,8 @@
"style": "Überschriften",
"codeBlock": "Code Block",
"code": "Code",
"color": "Farbe"
"color": "Farbe",
"emoji": "Emoji"
},
"visualEditor": {
"hintTooltip": "Füge hier einen Hinweis hinzu (sichtbar während des Quiz über den ?-Button)",
Expand Down
3 changes: 2 additions & 1 deletion src/main/webapp/i18n/en/multipleChoiceQuestion.json
Original file line number Diff line number Diff line change
Expand Up @@ -45,7 +45,8 @@
"codeBlock": "Code Block",
"code": "Code",
"color": "Color",
"style": "Style"
"style": "Style",
"emoji": "Emoji"
krusche marked this conversation as resolved.
Show resolved Hide resolved
},
"visualEditor": {
"hintTooltip": "Add a hint here (visible during the quiz via ?-Button)",
Expand Down
Loading
Loading