This commit is contained in:
khk 2020-01-23 12:26:45 +09:00
commit ddb52c3fb4
16 changed files with 534 additions and 335 deletions

View File

@ -28,7 +28,8 @@ import {
NotificationChannel,
ChatChannel,
MessengerChannel,
MessageChannel
MessageChannel,
AppChannel
} from '@ucap-webmessenger/native-electron';
import { ElectronNotificationService } from '@ucap-webmessenger/electron-notification';
import { ElectronUpdateWindowService } from '@ucap-webmessenger/electron-update-window';
@ -231,8 +232,7 @@ function createTray() {
// selector: 'terminate:',
click: () => {
// 메신저에 로그아웃 후 종료
appWindow = null;
app.exit();
appExit();
}
}
]);
@ -715,6 +715,10 @@ ipcMain.on(
}
);
ipcMain.on(AppChannel.Exit, (event: IpcMainEvent, ...args: any[]) => {
appExit();
});
autoUpdater.on('checking-for-update', () => {
log.info('Checking for update...');
});
@ -748,3 +752,8 @@ autoUpdater.on('update-downloaded', info => {
updateWindowService.setDownloadComplete();
autoUpdater.quitAndInstall(true, true);
});
function appExit() {
appWindow = null;
app.exit();
}

View File

@ -563,7 +563,8 @@ export class GroupComponent implements OnInit, OnDestroy {
>(SelectGroupDialogComponent, {
width: '600px',
data: {
title: this.translateService.instant('group.selectTargetGroup')
title: this.translateService.instant('group.selectTargetGroup'),
ignoreGroup: [group]
}
});

View File

@ -88,10 +88,10 @@
</mat-tab-group>
</div>
<div [style.display]="isSearch ? 'block' : 'none'">
<div class="message-section" [style.display]="isSearch ? 'block' : 'none'">
<div class="search-sub">
<form [formGroup]="fgSearchType" class="w-100-p">
<mat-form-field>
<mat-form-field style="width: 100px;">
<mat-select
formControlName="searchMessageType"
(selectionChange)="onChangeSelection($event)"
@ -127,13 +127,15 @@
</mat-radio-group>
</form>
</div>
<div>
<ucap-message-list-item
*ngFor="let message of messageSearchList$ | async"
[message]="message"
(click)="onClickDetail(message)"
class="message-item"
></ucap-message-list-item>
<div style="height: calc(100% - 65.5px)">
<perfect-scrollbar fxFlex="1 1 auto">
<ucap-message-list-item
*ngFor="let message of messageSearchList$ | async"
[message]="message"
(click)="onClickDetail(message)"
class="message-item"
></ucap-message-list-item>
</perfect-scrollbar>
</div>
</div>
</div>

View File

@ -81,6 +81,7 @@
(click)="onToggleUser(userInfo)"
(contextmenu)="onContextMenuOrgUser($event, userInfo)"
[matTooltip]="getTooltip(userInfo)"
matTooltipShowDelay="500"
matTooltipPosition="after"
>
</ucap-profile-user-list-item>
@ -105,6 +106,7 @@
(click)="onToggleUser(userInfo)"
(contextmenu)="onContextMenuOrgUser($event, userInfo)"
[matTooltip]="getTooltip(userInfo)"
matTooltipShowDelay="500"
matTooltipPosition="after"
>
</ucap-profile-user-list-item>

View File

@ -638,170 +638,190 @@ export class MessagesComponent implements OnInit, OnDestroy, AfterViewInit {
if (!!this.isShowTranslation && this.destLocale.trim().length > 0) {
/** CASE : Translation */
const destLocale = this.destLocale;
const original = message;
const roomSeq = this.roomInfoSubject.value.roomSeq;
if (!!this.isTranslationProcess) {
return;
// 번역할 대화 없이 스티커만 전송할 경우.
if (!message || message.trim().length === 0) {
this.sendMessageOfSticker(message);
} else {
this.sendMessageOfTranslate(message);
}
this.isTranslationProcess = true;
this.commonApiService
.translationSave({
userSeq: this.loginResSubject.value.userSeq,
deviceType: this.environmentsInfo.deviceType,
token: this.loginResSubject.value.tokenString,
roomSeq,
original,
srcLocale: '',
destLocale
} as TranslationSaveRequest)
.pipe(
take(1),
map(res => {
if (res.statusCode === StatusCode.Success) {
let sentMessage = '';
let eventType = EventType.Translation;
let previewObject:
| TranslationEventJson
| MassTranslationEventJson;
if (res.translationSeq > 0) {
// Mass Text Translation
previewObject = res;
sentMessage = res.returnJson;
eventType = EventType.MassTranslation;
} else {
// Normal Text Translation
previewObject = {
locale: destLocale,
original,
translation: res.translation,
stickername: '',
stickerfile: !!this.selectedSticker
? this.selectedSticker.index
: ''
};
sentMessage = JSON.stringify(previewObject);
eventType = EventType.Translation;
}
if (!!this.translationPreview) {
// preview
this.translationPreviewInfo = {
previewInfo: res,
translationType: eventType
};
} else {
// direct send
this.store.dispatch(
EventStore.send({
senderSeq: this.loginResSubject.value.userSeq,
req: {
roomSeq,
eventType,
sentMessage
}
})
);
if (!!this.translationPreviewInfo) {
this.translationPreviewInfo = null;
}
}
if (!!this.selectedSticker) {
this.isShowStickerSelector = false;
this.setStickerHistory(this.selectedSticker);
this.selectedSticker = null;
}
} else {
this.logger.error(res);
}
}),
catchError(error => of(this.logger.error(error)))
)
.subscribe(() => {
this.isTranslationProcess = false;
});
} else if (!!this.selectedSticker) {
/** CASE : Sticker */
if (
!!message &&
message.trim().length >
environment.productConfig.CommonSetting.masstextLength
) {
const result = await this.dialogService.open<
AlertDialogComponent,
AlertDialogData,
AlertDialogResult
>(AlertDialogComponent, {
width: '360px',
data: {
title: this.translateService.instant('chat.errors.label'),
message: this.translateService.instant(
'chat.errors.maxLengthOfMassText',
{
maxLength:
environment.productConfig.CommonSetting.masstextLength
}
)
}
});
return;
}
const stickerJson: StickerEventJson = {
name: '스티커',
file: this.selectedSticker.index,
chat: !!message ? message.trim() : ''
};
this.store.dispatch(
EventStore.send({
senderSeq: this.loginResSubject.value.userSeq,
req: {
roomSeq: this.roomInfoSubject.value.roomSeq,
eventType: EventType.Sticker,
sentMessage: JSON.stringify(stickerJson)
}
})
);
this.isShowStickerSelector = false;
this.setStickerHistory(this.selectedSticker);
this.selectedSticker = null;
this.sendMessageOfSticker(message);
} else if (
message.trim().length >
environment.productConfig.CommonSetting.masstextLength
) {
/** CASE : MASS TEXT */
this.store.dispatch(
EventStore.sendMass({
senderSeq: this.loginResSubject.value.userSeq,
req: {
roomSeq: this.roomInfoSubject.value.roomSeq,
eventType: EventType.MassText,
// sentMessage: message.replace(/\n/gi, '\r\n')
sentMessage: message
}
})
);
this.sendMessageOfMassText(message);
} else {
/** CASE : Normal Text */
this.store.dispatch(
EventStore.send({
senderSeq: this.loginResSubject.value.userSeq,
req: {
roomSeq: this.roomInfoSubject.value.roomSeq,
eventType: EventType.Character,
sentMessage: message
}
})
);
this.sendMessageOfNormal(message);
}
}
/** Send Normal message */
sendMessageOfNormal(message: string) {
this.store.dispatch(
EventStore.send({
senderSeq: this.loginResSubject.value.userSeq,
req: {
roomSeq: this.roomInfoSubject.value.roomSeq,
eventType: EventType.Character,
sentMessage: message
}
})
);
}
/** Send Sticker message */
async sendMessageOfSticker(message: string, isCheck: boolean = true) {
if (
!!isCheck &&
!!message &&
message.trim().length >
environment.productConfig.CommonSetting.masstextLength
) {
const result = await this.dialogService.open<
AlertDialogComponent,
AlertDialogData,
AlertDialogResult
>(AlertDialogComponent, {
width: '360px',
data: {
title: this.translateService.instant('chat.errors.label'),
message: this.translateService.instant(
'chat.errors.maxLengthOfMassText',
{
maxLength: environment.productConfig.CommonSetting.masstextLength
}
)
}
});
return;
}
const stickerJson: StickerEventJson = {
name: '스티커',
file: this.selectedSticker.index,
chat: !!message ? message.trim() : ''
};
this.store.dispatch(
EventStore.send({
senderSeq: this.loginResSubject.value.userSeq,
req: {
roomSeq: this.roomInfoSubject.value.roomSeq,
eventType: EventType.Sticker,
sentMessage: JSON.stringify(stickerJson)
}
})
);
this.isShowStickerSelector = false;
this.setStickerHistory(this.selectedSticker);
this.selectedSticker = null;
}
/** Send Masstext message */
sendMessageOfMassText(message: string) {
this.store.dispatch(
EventStore.sendMass({
senderSeq: this.loginResSubject.value.userSeq,
req: {
roomSeq: this.roomInfoSubject.value.roomSeq,
eventType: EventType.MassText,
// sentMessage: message.replace(/\n/gi, '\r\n')
sentMessage: message
}
})
);
}
/** Send Translation message */
sendMessageOfTranslate(message: string) {
const destLocale = this.destLocale;
const original = message;
const roomSeq = this.roomInfoSubject.value.roomSeq;
if (!!this.isTranslationProcess) {
return;
}
this.isTranslationProcess = true;
this.commonApiService
.translationSave({
userSeq: this.loginResSubject.value.userSeq,
deviceType: this.environmentsInfo.deviceType,
token: this.loginResSubject.value.tokenString,
roomSeq,
original,
srcLocale: '',
destLocale
} as TranslationSaveRequest)
.pipe(
take(1),
map(res => {
if (res.statusCode === StatusCode.Success) {
let sentMessage = '';
let eventType = EventType.Translation;
let previewObject: TranslationEventJson | MassTranslationEventJson;
if (res.translationSeq > 0) {
// Mass Text Translation
previewObject = res;
sentMessage = res.returnJson;
eventType = EventType.MassTranslation;
} else {
// Normal Text Translation
previewObject = {
locale: destLocale,
original,
translation: res.translation,
stickername: '',
stickerfile: !!this.selectedSticker
? this.selectedSticker.index
: ''
};
sentMessage = JSON.stringify(previewObject);
eventType = EventType.Translation;
}
if (!!this.translationPreview) {
// preview
this.translationPreviewInfo = {
previewInfo: res,
translationType: eventType
};
} else {
// direct send
this.store.dispatch(
EventStore.send({
senderSeq: this.loginResSubject.value.userSeq,
req: {
roomSeq,
eventType,
sentMessage
}
})
);
if (!!this.translationPreviewInfo) {
this.translationPreviewInfo = null;
}
}
if (!!this.selectedSticker) {
this.isShowStickerSelector = false;
this.setStickerHistory(this.selectedSticker);
this.selectedSticker = null;
}
} else {
this.logger.error(res);
}
}),
catchError(error => of(this.logger.error(error)))
)
.subscribe(() => {
this.isTranslationProcess = false;
});
}
onClickReceiveAlarm() {
this.store.dispatch(
RoomStore.updateOnlyAlarm({ roomInfo: this.roomInfoSubject.value })
@ -1592,6 +1612,7 @@ export class MessagesComponent implements OnInit, OnDestroy, AfterViewInit {
const trgtUserSeq: number[] = [];
result.group.userSeqs.map(seq => trgtUserSeq.push(seq));
this.userInfoListSubject.value
.filter(v => v.isJoinRoom)
.filter(v => result.group.userSeqs.indexOf(v.seq) < 0)
.forEach(user => {
trgtUserSeq.push(user.seq);

View File

@ -63,6 +63,7 @@
<mat-list-option
*ngFor="let groupBuddy of groupBuddyList$ | async"
[value]="groupBuddy.group"
[disabled]="getDisabled(groupBuddy)"
class="group-name"
>
<span class="title-name ellipsis"> {{ groupBuddy.group.name }} </span>

View File

@ -24,6 +24,7 @@ import { TranslateService } from '@ngx-translate/core';
export interface SelectGroupDialogData {
title: string;
ignoreGroup?: GroupDetailData[];
}
export interface SelectGroupDialogResult {
@ -71,6 +72,44 @@ export class SelectGroupDialogComponent implements OnInit {
)
]).pipe(
map(([buddyList, groupList]) => {
// sort..
if (!!groupList && groupList.length > 0) {
const tempOrderArr: GroupDetailData[] = [];
let myDeptGroup: GroupDetailData;
let defaultGroup: GroupDetailData;
const normalGroup: GroupDetailData[] = [];
groupList.forEach(group => {
if (
!!environment.productConfig.CommonSetting.useMyDeptGroup &&
environment.productConfig.CommonSetting.myDeptGroupSeq ===
group.seq
) {
myDeptGroup = group;
} else if (0 === group.seq) {
defaultGroup = group;
} else {
normalGroup.push(group);
}
});
if (!!myDeptGroup) {
tempOrderArr.push(myDeptGroup);
}
tempOrderArr.push(
...normalGroup.sort((a, b) =>
a.name < b.name ? -1 : a.name > b.name ? 1 : 0
)
);
if (!!defaultGroup) {
tempOrderArr.push(defaultGroup);
}
groupList = tempOrderArr;
}
/** 수정불가 그룹 */
let fixedGroupSeqs: number[];
if (!!environment.productConfig.CommonSetting.fixedGroupSeqs) {
@ -90,6 +129,17 @@ export class SelectGroupDialogComponent implements OnInit {
}
}
/** 선택 무시 그룹 필터링. */
if (!!this.data.ignoreGroup && this.data.ignoreGroup.length > 0) {
if (
this.data.ignoreGroup.filter(
groupDetail => groupDetail.seq === group.seq
).length > 0
) {
continue;
}
}
groupBuddyList.push({
group,
buddyList: buddyList.filter(buddy => {
@ -140,4 +190,18 @@ export class SelectGroupDialogComponent implements OnInit {
group: this.selectedGroup
});
}
getDisabled(groupBuddyList: {
group: GroupDetailData;
buddyList: UserInfo[];
}): boolean {
if (!!this.data.ignoreGroup && this.data.ignoreGroup.length > 0) {
return (
this.data.ignoreGroup.filter(
groupDetail => groupDetail.seq === groupBuddyList.group.seq
).length > 0
);
}
return false;
}
}

View File

@ -257,27 +257,28 @@
yPosition="below"
(closed)="onClosedProfileMenu($event)"
>
<div class="setting">
<button mat-menu-item [matMenuTriggerFor]="presenseMenu">
<ng-container [ngSwitch]="myStatus.pcStatus">
<ng-container *ngSwitchCase="StatusCode.OnLine">
{{ 'presence.online' | translate }}
<ng-template matMenuContent>
<div class="setting" style="width: 200px;">
<button mat-menu-item [matMenuTriggerFor]="presenseMenu">
<ng-container [ngSwitch]="myStatus.pcStatus">
<ng-container *ngSwitchCase="StatusCode.OnLine">
{{ 'presence.online' | translate }}
</ng-container>
<ng-container *ngSwitchCase="StatusCode.Away">
{{ 'presence.away' | translate }}
</ng-container>
<ng-container *ngSwitchCase="StatusCode.Busy">
{{ myStatus.statusMessage }}
</ng-container>
</ng-container>
<ng-container *ngSwitchCase="StatusCode.Away">
{{ 'presence.away' | translate }}
</ng-container>
<ng-container *ngSwitchCase="StatusCode.Busy">
{{ myStatus.statusMessage }}
</ng-container>
</ng-container>
</button>
</div>
<div class="setting">
<button mat-menu-item (click)="onClickOpenProfile($event)">
{{ 'profile.open' | translate }}
</button>
</div>
<!-- <div class="setting">
</button>
</div>
<div class="setting">
<button mat-menu-item (click)="onClickOpenProfile($event)">
{{ 'profile.open' | translate }}
</button>
</div>
<!-- <div class="setting">
<button
mat-menu-item
class="zoom minus-square"
@ -293,148 +294,163 @@
확대
</button>
</div> -->
<div class="setting">
<button mat-menu-item (click)="onClickNotice()">
{{ 'notice.label' | translate }}
</button>
</div>
<div class="setting">
<button mat-menu-item (click)="onClickRemoteSupport($event)">
{{ 'profile.remoteSupport' | translate }}
</button>
</div>
<div class="setting">
<button mat-menu-item (click)="onClickSettings()">
{{ 'settings.label' | translate }}
</button>
</div>
<mat-divider></mat-divider>
<div class="setting">
<button
mat-menu-item
[matMenuTriggerFor]="informationMenu"
(menuOpened)="onMenuOpenedinformationMenu()"
>
{{ 'information.checkForUpdates' | translate }}
</button>
</div>
<mat-divider></mat-divider>
<div class="setting">
<button mat-menu-item (click)="onClickLogout()">
{{ 'accounts.logout' | translate }}
</button>
</div>
<div class="setting">
<button mat-menu-item (click)="onClickNotice()">
{{ 'notice.label' | translate }}
</button>
</div>
<div class="setting">
<button mat-menu-item (click)="onClickRemoteSupport($event)">
{{ 'profile.remoteSupport' | translate }}
</button>
</div>
<div class="setting">
<button mat-menu-item (click)="onClickSettings()">
{{ 'settings.label' | translate }}
</button>
</div>
<mat-divider></mat-divider>
<div class="setting">
<button
mat-menu-item
[matMenuTriggerFor]="informationMenu"
(menuOpened)="onMenuOpenedinformationMenu()"
>
{{ 'information.checkForUpdates' | translate }}
</button>
</div>
<mat-divider></mat-divider>
<div class="setting">
<button mat-menu-item (click)="onClickLogout()">
{{ 'accounts.logout' | translate }}
</button>
</div>
<div class="setting">
<button mat-menu-item (click)="onClickQuit()">
{{ 'common.messages.quit' | translate }}
</button>
</div>
</ng-template>
</mat-menu>
<mat-menu #presenseMenu="matMenu" class="status-pc-set">
<div class="setting">
<button mat-menu-item (click)="onClickStatusOnline($event)">
<span class="presence pcOn"> </span>
{{ 'presence.online' | translate }}
</button>
</div>
<div class="setting">
<button mat-menu-item (click)="onClickStatusAway($event)">
<span class="presence pcOut"> </span>
{{ 'presence.away' | translate }}
</button>
<button
mat-menu-item
class="clock"
[matMenuTriggerFor]="awayTimeMenu"
></button>
</div>
<div class="setting">
<button mat-menu-item (click)="onClickStatusBusy($event, 1)">
<span class="presence pcOther"> </span>
<ucap-inline-edit-input
(apply)="
$event.stopPropagation();
onApplyStatusMessage(1, statusMessage1.value)
"
(edit)="$event.stopPropagation()"
(cancel)="$event.stopPropagation()"
class="form-eidt"
>
<span ucapInlineEditInput="view">{{ loginRes?.statusMessage1 }}</span>
<span ucapInlineEditInput="edit"
><input
matInput
#statusMessage1
type="text"
maxlength="5"
[value]="loginRes?.statusMessage1"
(click)="$event.stopPropagation()"
/></span>
</ucap-inline-edit-input>
</button>
</div>
<div class="setting">
<button mat-menu-item (click)="onClickStatusBusy($event, 2)">
<span class="presence pcOther"> </span>
<ucap-inline-edit-input
(apply)="
$event.stopPropagation();
onApplyStatusMessage(2, statusMessage2.value)
"
(edit)="$event.stopPropagation()"
(cancel)="$event.stopPropagation()"
class="form-eidt"
>
<span ucapInlineEditInput="view">{{ loginRes?.statusMessage2 }}</span>
<span ucapInlineEditInput="edit"
><input
matInput
#statusMessage2
type="text"
maxlength="5"
[value]="loginRes?.statusMessage2"
(click)="$event.stopPropagation()"
/></span>
</ucap-inline-edit-input>
</button>
</div>
<div class="setting">
<button mat-menu-item (click)="onClickStatusBusy($event, 3)">
<span class="presence pcOther"> </span>
<ucap-inline-edit-input
(apply)="
$event.stopPropagation();
onApplyStatusMessage(3, statusMessage3.value)
"
(edit)="$event.stopPropagation()"
(cancel)="$event.stopPropagation()"
class="form-eidt"
>
<span ucapInlineEditInput="view">{{ loginRes?.statusMessage3 }}</span>
<span ucapInlineEditInput="edit"
><input
matInput
#statusMessage3
type="text"
maxlength="5"
[value]="loginRes?.statusMessage3"
(click)="$event.stopPropagation()"
/></span>
</ucap-inline-edit-input>
</button>
</div>
<ng-template matMenuContent>
<div class="setting">
<button mat-menu-item (click)="onClickStatusOnline($event)">
<span class="presence pcOn"> </span>
{{ 'presence.online' | translate }}
</button>
</div>
<div class="setting">
<button mat-menu-item (click)="onClickStatusAway($event)">
<span class="presence pcOut"> </span>
{{ 'presence.away' | translate }}
</button>
<button
mat-menu-item
class="clock"
[matMenuTriggerFor]="awayTimeMenu"
></button>
</div>
<div class="setting">
<button mat-menu-item (click)="onClickStatusBusy($event, 1)">
<span class="presence pcOther"> </span>
<ucap-inline-edit-input
(apply)="
$event.stopPropagation();
onApplyStatusMessage(1, statusMessage1.value)
"
(edit)="$event.stopPropagation()"
(cancel)="$event.stopPropagation()"
class="form-eidt"
>
<span ucapInlineEditInput="view">{{ loginRes?.statusMessage1 }}</span>
<span ucapInlineEditInput="edit"
><input
matInput
#statusMessage1
type="text"
maxlength="5"
[value]="loginRes?.statusMessage1"
(click)="$event.stopPropagation()"
/></span>
</ucap-inline-edit-input>
</button>
</div>
<div class="setting">
<button mat-menu-item (click)="onClickStatusBusy($event, 2)">
<span class="presence pcOther"> </span>
<ucap-inline-edit-input
(apply)="
$event.stopPropagation();
onApplyStatusMessage(2, statusMessage2.value)
"
(edit)="$event.stopPropagation()"
(cancel)="$event.stopPropagation()"
class="form-eidt"
>
<span ucapInlineEditInput="view">{{ loginRes?.statusMessage2 }}</span>
<span ucapInlineEditInput="edit"
><input
matInput
#statusMessage2
type="text"
maxlength="5"
[value]="loginRes?.statusMessage2"
(click)="$event.stopPropagation()"
/></span>
</ucap-inline-edit-input>
</button>
</div>
<div class="setting">
<button mat-menu-item (click)="onClickStatusBusy($event, 3)">
<span class="presence pcOther"> </span>
<ucap-inline-edit-input
(apply)="
$event.stopPropagation();
onApplyStatusMessage(3, statusMessage3.value)
"
(edit)="$event.stopPropagation()"
(cancel)="$event.stopPropagation()"
class="form-eidt"
>
<span ucapInlineEditInput="view">{{ loginRes?.statusMessage3 }}</span>
<span ucapInlineEditInput="edit"
><input
matInput
#statusMessage3
type="text"
maxlength="5"
[value]="loginRes?.statusMessage3"
(click)="$event.stopPropagation()"
/></span>
</ucap-inline-edit-input>
</button>
</div>
</ng-template>
</mat-menu>
<mat-menu #awayTimeMenu="matMenu">
<div mat-menu-item (click)="$event.stopPropagation()">
{{ 'presence.settingOfAwayTime' | translate }}
</div>
<mat-radio-group aria-label="Select an option" [value]="myIdleCheckTime">
<div mat-menu-item *ngFor="let awayTime of awayTimeList">
<mat-radio-button [value]="awayTime" (change)="onChangeAwayTime($event)">
{{ awayTime }}{{ 'common.units.minute' | translate }}</mat-radio-button
>
<ng-template matMenuContent>
<div mat-menu-item (click)="$event.stopPropagation()">
{{ 'presence.settingOfAwayTime' | translate }}
</div>
</mat-radio-group>
<mat-radio-group aria-label="Select an option" [value]="myIdleCheckTime">
<div mat-menu-item *ngFor="let awayTime of awayTimeList">
<mat-radio-button
[value]="awayTime"
(change)="onChangeAwayTime($event)"
>
{{ awayTime
}}{{ 'common.units.minute' | translate }}</mat-radio-button
>
</div>
</mat-radio-group>
</ng-template>
</mat-menu>
<mat-menu #informationMenu="matMenu">
<<<<<<< HEAD
<div
class="version-info-container menu-item"
(click)="$event.stopPropagation()"
@ -461,8 +477,45 @@
<span class="version-info-button">
<button *ngIf="checkingUpdateIsExist" mat-flat-button
class="mat-primary" (click)="onClickApplyUpdate($event)">{{ 'information.applyUpdates' | translate }}</button>
=======
<ng-template matMenuContent>
<div
class="version-info-container menu-item"
(click)="$event.stopPropagation()"
>
<div>
<span>
{{ 'information.installedVersion' | translate }}:{{ appVersion }}
>>>>>>> 86c2a7cb2a2195023425b1653f1cc8472b09a8d9
</span>
</div>
<div *ngIf="checkingUpdate">
<div *ngIf="checkingUpdateIsProcessing" style="display: flex;">
<mat-spinner
*ngIf="checkingUpdate && checkingUpdateIsProcessing"
diameter="20"
strokeWidth="1"
></mat-spinner>
{{ 'information.checkForUpdatesInProgress' | translate }}
</div>
<div *ngIf="!checkingUpdateIsProcessing" style="display: flex;">
<span>
{{ 'information.latestVersion' | translate }}:{{
checkingUpdateAppVersion
}}
</span>
<span>
<button
*ngIf="checkingUpdateIsExist"
mat-button
(click)="onClickApplyUpdate($event)"
>
{{ 'information.applyUpdates' | translate }}
</button>
</span>
</div>
</div>
</div>
</div>
</ng-template>
</mat-menu>

View File

@ -309,6 +309,10 @@ export class TopBarComponent implements OnInit, OnDestroy {
this.store.dispatch(AuthenticationStore.logoutConfirmation());
}
onClickQuit(): void {
this.nativeService.appExit();
}
getMyProfileImageWidget(): string {
if (!!this.loginRes) {
return this.loginRes.userInfo.profileImageFile;

View File

@ -406,6 +406,7 @@
}
},
"messages": {
"quit": "Quit",
"yes": "Yes",
"no": "No",
"confirm": "Confirm",

View File

@ -406,6 +406,7 @@
}
},
"messages": {
"quit": "종료",
"yes": "예",
"no": "아니오",
"confirm": "확인",

View File

@ -221,6 +221,8 @@ export class BrowserNativeService extends NativeService {
windowMaximize(): void {}
appExit(): void {}
zoomTo(factor: number): Promise<number> {
return new Promise<number>((resolve, reject) => {
resolve(-1);

View File

@ -22,7 +22,8 @@ import {
ChatChannel,
MessengerChannel,
MessageChannel,
ProcessChannel
ProcessChannel,
AppChannel
} from '../types/channel.type';
import { Injectable } from '@angular/core';
import { TranslateLoaderService } from '../translate/electron-loader';
@ -388,6 +389,10 @@ export class ElectronNativeService implements NativeService {
}
}
appExit(): void {
this.ipcRenderer.send(AppChannel.Exit);
}
zoomTo(factor: number): Promise<number> {
return new Promise<number>((resolve, reject) => {
try {

View File

@ -53,3 +53,7 @@ export enum IdleStateChannel {
StartCheck = 'UCAP::idleState::startCheck',
ChangeLimitTime = 'UCAP::idleState::changeLimitTime'
}
export enum AppChannel {
Exit = 'UCAP::app::exit'
}

View File

@ -68,6 +68,7 @@ export abstract class NativeService {
abstract windowMinimize(): void;
abstract windowMaximize(): void;
abstract zoomTo(factor: number): Promise<number>;
abstract appExit(): void;
abstract idleStateChanged(): Observable<WindowIdle>;
abstract changeLimitOfIdleState(limitTime: number): void;

View File

@ -142,6 +142,7 @@ export class MessagesComponent implements OnInit, OnDestroy {
readToHereEvent: Info<EventJson>;
swapped = false;
hidden = false;
constructor(
private logger: NGXLogger,
@ -384,9 +385,10 @@ export class MessagesComponent implements OnInit, OnDestroy {
to: Info<EventJson>,
preCallback: () => void,
postCallback: () => void,
useSwap: boolean = true
useHide: boolean,
useSwap: boolean
) {
this.preSwapScroll(useSwap);
this.preSwapScroll(useHide, useSwap);
if (!!preCallback) {
preCallback();
}
@ -395,12 +397,12 @@ export class MessagesComponent implements OnInit, OnDestroy {
if (!!postCallback) {
postCallback();
}
this.postSwapScroll(useSwap);
this.postSwapScroll(useHide, useSwap);
}, 100);
});
}
preSwapScroll(useSwap: boolean = true) {
preSwapScroll(useHide: boolean, useSwap: boolean) {
if (useSwap && !this.swapped) {
this.chatMessagesBuffer.nativeElement.innerHTML = this.chatMessagesContainer.nativeElement.innerHTML;
this.chatMessagesBuffer.nativeElement.scrollTop = this.chatMessagesContainer.nativeElement.scrollTop;
@ -408,10 +410,13 @@ export class MessagesComponent implements OnInit, OnDestroy {
this.swapped = true;
}
this.chatMessagesContainer.nativeElement.classList.add('hide');
if (useHide && !this.hidden) {
this.chatMessagesContainer.nativeElement.classList.add('hide');
this.hidden = true;
}
}
postSwapScroll(useSwap: boolean = true) {
postSwapScroll(useHide: boolean, useSwap: boolean) {
if (useSwap && this.swapped) {
this.chatMessagesBuffer.nativeElement.innerHTML = '';
this.chatMessagesBuffer.nativeElement.scrollTop = 0;
@ -419,7 +424,10 @@ export class MessagesComponent implements OnInit, OnDestroy {
this.swapped = false;
}
this.chatMessagesContainer.nativeElement.classList.remove('hide');
if (useHide && this.hidden) {
this.chatMessagesContainer.nativeElement.classList.remove('hide');
this.hidden = false;
}
}
ready(): void {
@ -434,7 +442,9 @@ export class MessagesComponent implements OnInit, OnDestroy {
() => {},
() => {
this.firstCheckReadHere = false;
}
},
true,
true
);
} else {
this.swapScroll(
@ -442,7 +452,9 @@ export class MessagesComponent implements OnInit, OnDestroy {
() => {},
() => {
this.storedScrollItem = undefined;
}
},
true,
true
);
}
} else if (this.scrollUpInitalized) {
@ -460,15 +472,31 @@ export class MessagesComponent implements OnInit, OnDestroy {
() => {},
() => {
this.firstCheckReadHere = false;
}
},
true,
true
);
} else {
this.swapScroll(
this.eventList[this.eventList.length - 1],
() => {},
() => {},
false
);
if (
this.virtualScroller.viewPortInfo.endIndex ===
this.eventList.length - 2
) {
this.swapScroll(
this.eventList[this.eventList.length - 1],
() => {},
() => {},
false,
false
);
} else {
this.swapScroll(
this.eventList[this.eventList.length - 1],
() => {},
() => {},
true,
false
);
}
}
}
}
@ -499,7 +527,7 @@ export class MessagesComponent implements OnInit, OnDestroy {
if (this.scrollUpInitalized && this.eventRemained) {
this.storeScrollPosition();
this.preSwapScroll();
this.preSwapScroll(true, true);
this.moreEvent.emit(this.eventList[0].seq);