stream-chat-angular 6.2.0 → 6.3.0-beta.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/assets/i18n/en.d.ts +42 -0
- package/assets/version.d.ts +1 -1
- package/esm2020/assets/i18n/en.mjs +43 -1
- package/esm2020/assets/version.mjs +2 -2
- package/esm2020/lib/channel-list/channel-list.component.mjs +3 -3
- package/esm2020/lib/channel-preview/channel-preview.component.mjs +4 -1
- package/esm2020/lib/channel.service.mjs +6 -3
- package/esm2020/lib/custom-templates.service.mjs +9 -1
- package/esm2020/lib/icon/icon.component.mjs +1 -1
- package/esm2020/lib/message/message.component.mjs +21 -6
- package/esm2020/lib/message-actions.service.mjs +5 -1
- package/esm2020/lib/message-input/message-input.component.mjs +30 -5
- package/esm2020/lib/message-preview.mjs +3 -2
- package/esm2020/lib/modal/modal.component.mjs +3 -3
- package/esm2020/lib/modal/stream-modal.module.mjs +19 -0
- package/esm2020/lib/notification-list/stream-notification.module.mjs +20 -0
- package/esm2020/lib/paginated-list/stream-paginated-list.module.mjs +20 -0
- package/esm2020/lib/polls/base-poll.component.mjs +87 -0
- package/esm2020/lib/polls/poll/poll.component.mjs +34 -0
- package/esm2020/lib/polls/poll-actions/add-option/add-option.component.mjs +67 -0
- package/esm2020/lib/polls/poll-actions/poll-actions.component.mjs +137 -0
- package/esm2020/lib/polls/poll-actions/poll-answers-list/poll-answers-list.component.mjs +82 -0
- package/esm2020/lib/polls/poll-actions/poll-results/poll-results-list/poll-results-list.component.mjs +63 -0
- package/esm2020/lib/polls/poll-actions/poll-results/poll-vote/poll-vote.component.mjs +33 -0
- package/esm2020/lib/polls/poll-actions/poll-results/poll-vote-results-list/poll-vote-results-list.component.mjs +82 -0
- package/esm2020/lib/polls/poll-actions/upsert-answer/upsert-answer.component.mjs +60 -0
- package/esm2020/lib/polls/poll-composer/poll-composer.component.mjs +134 -0
- package/esm2020/lib/polls/poll-composer/validators.mjs +18 -0
- package/esm2020/lib/polls/poll-header/poll-header.component.mjs +80 -0
- package/esm2020/lib/polls/poll-option-selector/poll-option-selector.component.mjs +137 -0
- package/esm2020/lib/polls/poll-options-list/poll-options-list.component.mjs +39 -0
- package/esm2020/lib/polls/poll-preview/poll-preview.component.mjs +31 -0
- package/esm2020/lib/polls/stream-polls.module.mjs +108 -0
- package/esm2020/lib/polls/unique.validator.mjs +13 -0
- package/esm2020/lib/stream-chat.module.mjs +26 -19
- package/esm2020/lib/types.mjs +1 -1
- package/esm2020/public-api.mjs +18 -1
- package/fesm2015/stream-chat-angular.mjs +1333 -61
- package/fesm2015/stream-chat-angular.mjs.map +1 -1
- package/fesm2020/stream-chat-angular.mjs +1291 -59
- package/fesm2020/stream-chat-angular.mjs.map +1 -1
- package/lib/channel.service.d.ts +2 -1
- package/lib/custom-templates.service.d.ts +14 -0
- package/lib/icon/icon.component.d.ts +1 -1
- package/lib/message/message.component.d.ts +1 -1
- package/lib/message-actions.service.d.ts +4 -0
- package/lib/message-input/message-input.component.d.ts +11 -1
- package/lib/message-preview.d.ts +1 -1
- package/lib/modal/stream-modal.module.d.ts +9 -0
- package/lib/notification-list/stream-notification.module.d.ts +10 -0
- package/lib/paginated-list/stream-paginated-list.module.d.ts +10 -0
- package/lib/polls/base-poll.component.d.ts +43 -0
- package/lib/polls/poll/poll.component.d.ts +12 -0
- package/lib/polls/poll-actions/add-option/add-option.component.d.ts +22 -0
- package/lib/polls/poll-actions/poll-actions.component.d.ts +50 -0
- package/lib/polls/poll-actions/poll-answers-list/poll-answers-list.component.d.ts +25 -0
- package/lib/polls/poll-actions/poll-results/poll-results-list/poll-results-list.component.d.ts +22 -0
- package/lib/polls/poll-actions/poll-results/poll-vote/poll-vote.component.d.ts +20 -0
- package/lib/polls/poll-actions/poll-results/poll-vote-results-list/poll-vote-results-list.component.d.ts +24 -0
- package/lib/polls/poll-actions/upsert-answer/upsert-answer.component.d.ts +27 -0
- package/lib/polls/poll-composer/poll-composer.component.d.ts +43 -0
- package/lib/polls/poll-composer/validators.d.ts +3 -0
- package/lib/polls/poll-header/poll-header.component.d.ts +19 -0
- package/lib/polls/poll-option-selector/poll-option-selector.component.d.ts +27 -0
- package/lib/polls/poll-options-list/poll-options-list.component.d.ts +17 -0
- package/lib/polls/poll-preview/poll-preview.component.d.ts +13 -0
- package/lib/polls/stream-polls.module.d.ts +26 -0
- package/lib/polls/unique.validator.d.ts +2 -0
- package/lib/stream-chat.module.d.ts +20 -19
- package/lib/types.d.ts +2 -1
- package/package.json +6 -2
- package/public-api.d.ts +17 -0
- package/src/assets/i18n/en.ts +46 -0
- package/src/assets/styles/css/index.css +1 -1
- package/src/assets/styles/css/index.layout.css +1 -1
- package/src/assets/styles/scss/AttachmentList/AttachmentList-layout.scss +50 -0
- package/src/assets/styles/scss/AttachmentList/AttachmentList-theme.scss +56 -0
- package/src/assets/styles/scss/AttachmentPreviewList/AttachmentPreviewList-layout.scss +4 -1
- package/src/assets/styles/scss/AttachmentPreviewList/AttachmentPreviewList-theme.scss +11 -0
- package/src/assets/styles/scss/Autocomplete/Autocomplete-layout.scss +14 -0
- package/src/assets/styles/scss/Autocomplete/Autocomplete-theme.scss +11 -0
- package/src/assets/styles/scss/Dialog/Dialog-layout.scss +13 -5
- package/src/assets/styles/scss/DropzoneContainer/DropzoneContainer-layout.scss +14 -0
- package/src/assets/styles/scss/DropzoneContainer/DropzoneContainer-theme.scss +17 -0
- package/src/assets/styles/scss/Form/Form-layout.scss +40 -0
- package/src/assets/styles/scss/Form/Form-theme.scss +75 -10
- package/src/assets/styles/scss/Icon/Icon-layout.scss +6 -0
- package/src/assets/styles/scss/Icon/Icon-theme.scss +4 -0
- package/src/assets/styles/scss/LinkPreview/LinkPreview-layout.scss +18 -0
- package/src/assets/styles/scss/LinkPreview/LinkPreview-theme.scss +15 -0
- package/src/assets/styles/scss/Location/Location-layout.scss +52 -0
- package/src/assets/styles/scss/Location/Location-theme.scss +32 -0
- package/src/assets/styles/scss/Message/Message-layout.scss +37 -2
- package/src/assets/styles/scss/Message/Message-theme.scss +40 -1
- package/src/assets/styles/scss/MessageActionsBox/MessageActionsBox-theme.scss +8 -0
- package/src/assets/styles/scss/MessageInput/MessageInput-layout.scss +32 -13
- package/src/assets/styles/scss/MessageInput/MessageInput-theme.scss +28 -20
- package/src/assets/styles/scss/Modal/Modal-layout.scss +2 -0
- package/src/assets/styles/scss/Modal/Modal-theme.scss +21 -6
- package/src/assets/styles/scss/Poll/Poll-layout.scss +45 -44
- package/src/assets/styles/scss/Poll/Poll-theme.scss +8 -36
- package/src/assets/styles/scss/_icons.scss +1 -0
- package/src/assets/styles/scss/index.layout.scss +3 -1
- package/src/assets/styles/scss/index.scss +2 -0
- package/src/assets/version.ts +1 -1
- /package/src/assets/styles/scss/DragAndDropContainer/{DragAmdDropContainer-layout.scss → DragAndDropContainer-layout.scss} +0 -0
|
@@ -0,0 +1,134 @@
|
|
|
1
|
+
import { Component, EventEmitter, Output, ViewChild, } from '@angular/core';
|
|
2
|
+
import { FormArray, FormControl, FormGroup, Validators } from '@angular/forms';
|
|
3
|
+
import { VotingVisibility } from 'stream-chat';
|
|
4
|
+
import { atLeastOneOption, maximumNumberOfVotes } from './validators';
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
import * as i1 from "../../custom-templates.service";
|
|
7
|
+
import * as i2 from "../../chat-client.service";
|
|
8
|
+
import * as i3 from "../../notification.service";
|
|
9
|
+
import * as i4 from "@angular/common";
|
|
10
|
+
import * as i5 from "../../modal/modal.component";
|
|
11
|
+
import * as i6 from "../../notification-list/notification-list.component";
|
|
12
|
+
import * as i7 from "@angular/forms";
|
|
13
|
+
import * as i8 from "@ngx-translate/core";
|
|
14
|
+
/**
|
|
15
|
+
*
|
|
16
|
+
*/
|
|
17
|
+
export class PollComposerComponent {
|
|
18
|
+
constructor(customTemplatesService, chatService, notificationService) {
|
|
19
|
+
this.customTemplatesService = customTemplatesService;
|
|
20
|
+
this.chatService = chatService;
|
|
21
|
+
this.notificationService = notificationService;
|
|
22
|
+
/**
|
|
23
|
+
* Emitted when a poll is created, the poll id is emitted
|
|
24
|
+
*/
|
|
25
|
+
this.pollCompose = new EventEmitter();
|
|
26
|
+
/**
|
|
27
|
+
* Emitted when the poll composing is cancelled
|
|
28
|
+
*/
|
|
29
|
+
this.cancel = new EventEmitter();
|
|
30
|
+
this.formGroup = new FormGroup({
|
|
31
|
+
name: new FormControl('', [Validators.required]),
|
|
32
|
+
options: new FormArray([new FormControl('')], [atLeastOneOption()]),
|
|
33
|
+
multiple_answers: new FormControl(false),
|
|
34
|
+
maximum_number_of_votes: new FormControl(null, [
|
|
35
|
+
Validators.min(2),
|
|
36
|
+
Validators.max(10),
|
|
37
|
+
]),
|
|
38
|
+
is_anonymous: new FormControl(false),
|
|
39
|
+
allow_user_suggested_options: new FormControl(false),
|
|
40
|
+
allow_answers: new FormControl(false),
|
|
41
|
+
});
|
|
42
|
+
this.isModalOpen = true;
|
|
43
|
+
this.formGroup
|
|
44
|
+
.get('maximum_number_of_votes')
|
|
45
|
+
?.addValidators([
|
|
46
|
+
maximumNumberOfVotes(this.formGroup.get('multiple_answers')),
|
|
47
|
+
]);
|
|
48
|
+
this.formGroup.get('maximum_number_of_votes')?.disable();
|
|
49
|
+
this.formGroup.valueChanges.subscribe((value) => {
|
|
50
|
+
if (value.multiple_answers &&
|
|
51
|
+
this.formGroup.get('maximum_number_of_votes')?.disabled) {
|
|
52
|
+
this.formGroup.get('maximum_number_of_votes')?.enable();
|
|
53
|
+
}
|
|
54
|
+
else if (!value.multiple_answers &&
|
|
55
|
+
this.formGroup.get('maximum_number_of_votes')?.enabled) {
|
|
56
|
+
this.formGroup.get('maximum_number_of_votes')?.disable();
|
|
57
|
+
}
|
|
58
|
+
});
|
|
59
|
+
}
|
|
60
|
+
optionChanged(index) {
|
|
61
|
+
const control = this.options.at(index);
|
|
62
|
+
const penultimateIndex = this.options.length - 2;
|
|
63
|
+
if (index === this.options.length - 1 && control.value?.length === 1) {
|
|
64
|
+
this.addOption();
|
|
65
|
+
}
|
|
66
|
+
else if (index === penultimateIndex &&
|
|
67
|
+
control.value?.length === 0 &&
|
|
68
|
+
this.options.at(this.options.length - 1).value?.length === 0) {
|
|
69
|
+
this.removeLastOption();
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
get options() {
|
|
73
|
+
return this.formGroup.get('options');
|
|
74
|
+
}
|
|
75
|
+
addOption() {
|
|
76
|
+
const control = new FormControl('', []);
|
|
77
|
+
this.options.push(control);
|
|
78
|
+
}
|
|
79
|
+
removeLastOption() {
|
|
80
|
+
this.options.removeAt(this.options.length - 1);
|
|
81
|
+
}
|
|
82
|
+
getModalContext() {
|
|
83
|
+
return {
|
|
84
|
+
isOpen: this.isModalOpen,
|
|
85
|
+
isOpenChangeHandler: (isOpen) => {
|
|
86
|
+
if (!isOpen) {
|
|
87
|
+
this.cancel.emit();
|
|
88
|
+
}
|
|
89
|
+
this.isModalOpen = isOpen;
|
|
90
|
+
},
|
|
91
|
+
content: this.formContent,
|
|
92
|
+
};
|
|
93
|
+
}
|
|
94
|
+
async createPoll() {
|
|
95
|
+
try {
|
|
96
|
+
const maxVotesControl = this.formGroup.get('maximum_number_of_votes');
|
|
97
|
+
const response = await this.chatService.chatClient.polls.createPoll({
|
|
98
|
+
name: this.formGroup.get('name').value,
|
|
99
|
+
options: this.formGroup
|
|
100
|
+
.get('options')
|
|
101
|
+
.value.filter((v) => !!v)
|
|
102
|
+
.map((v) => ({ text: v })),
|
|
103
|
+
enforce_unique_vote: !this.formGroup.get('multiple_answers')?.value,
|
|
104
|
+
max_votes_allowed: maxVotesControl?.value
|
|
105
|
+
? +maxVotesControl.value
|
|
106
|
+
: undefined,
|
|
107
|
+
voting_visibility: this.formGroup.get('is_anonymous')?.value
|
|
108
|
+
? VotingVisibility.anonymous
|
|
109
|
+
: VotingVisibility.public,
|
|
110
|
+
allow_user_suggested_options: !!this.formGroup.get('allow_user_suggested_options')?.value,
|
|
111
|
+
allow_answers: !!this.formGroup.get('allow_answers')?.value,
|
|
112
|
+
});
|
|
113
|
+
this.pollCompose.emit(response?.id);
|
|
114
|
+
}
|
|
115
|
+
catch (error) {
|
|
116
|
+
this.notificationService.addTemporaryNotification('streamChat.Failed to create poll');
|
|
117
|
+
throw error;
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
PollComposerComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: PollComposerComponent, deps: [{ token: i1.CustomTemplatesService }, { token: i2.ChatClientService }, { token: i3.NotificationService }], target: i0.ɵɵFactoryTarget.Component });
|
|
122
|
+
PollComposerComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.0.4", type: PollComposerComponent, selector: "stream-poll-composer", outputs: { pollCompose: "pollCompose", cancel: "cancel" }, viewQueries: [{ propertyName: "formContent", first: true, predicate: ["formContent"], descendants: true }], ngImport: i0, template: "<ng-container>\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.modalTemplate$ | async) || defaultModal;\n context: getModalContext()\n \"\n ></ng-container>\n</ng-container>\n\n<ng-template\n #defaultModal\n let-isOpen=\"isOpen\"\n let-isOpenChangeHandler=\"isOpenChangeHandler\"\n let-content=\"content\"\n>\n <stream-modal\n class=\"str-chat-angular__create-poll-modal str-chat__create-poll-modal\"\n [isOpen]=\"isOpen\"\n [content]=\"content\"\n (isOpenChange)=\"isOpenChangeHandler($event)\"\n >\n </stream-modal>\n</ng-template>\n\n<ng-template #formContent>\n <div\n class=\"str-chat__dialog str-chat__poll-creation-dialog\"\n data-testid=\"poll-creation-dialog\"\n >\n <div class=\"str-chat__modal-header\">\n <div class=\"str-chat__modal-header__title\">\n {{ \"streamChat.Create poll\" | translate }}\n </div>\n </div>\n <div class=\"str-chat__dialog__body\">\n <form [formGroup]=\"formGroup\" autocomplete=\"off\">\n <div\n class=\"str-chat__form__field str-chat__form__input-field str-chat__form__input-field--with-label\"\n [class.str-chat__form__input-field--has-error]=\"\n formGroup.get('name')?.errors && formGroup.get('name')?.touched\n \"\n >\n <label class=\"str-chat__form__field-label\" htmlFor=\"name\">\n {{ \"streamChat.Question\" | translate }}\n </label>\n <div class=\"str-chat__form__input-field__value\">\n <div\n class=\"str-chat__form-field-error str-chat__form__input-field__error\"\n >\n {{\n (formGroup.get(\"name\")?.errors?.required &&\n formGroup.get(\"name\")?.touched\n ? \"streamChat.Question is required\"\n : \"\"\n ) | translate\n }}\n </div>\n <input\n id=\"name\"\n type=\"text\"\n formControlName=\"name\"\n placeholder=\"{{ 'streamChat.Ask a question' | translate }}\"\n />\n </div>\n </div>\n <fieldset class=\"str-chat__form__field str-chat__form__input-fieldset\">\n <legend class=\"str-chat__form__field-label\">\n {{ \"streamChat.Options\" | translate }}\n </legend>\n <div\n class=\"str-chat__form-field-error str-chat__form__input-field__error\"\n >\n {{\n (formGroup.get(\"options\")?.errors?.atLeastOne &&\n formGroup.get(\"options\")?.touched\n ? \"streamChat.Provide at least one option\"\n : \"\"\n ) | translate\n }}\n </div>\n <ng-container formArrayName=\"options\">\n <div class=\"str-chat__form__input-fieldset__values\">\n <!-- eslint-disable @angular-eslint/template/use-track-by-function -->\n <ng-container\n *ngFor=\"let option of options.controls; let i = index\"\n >\n <!-- eslint-enable @angular-eslint/template/use-track-by-function -->\n <div class=\"str-chat__drag-and-drop-container__item\">\n <div class=\"str-chat__form__input-field\">\n <input\n id=\"option{{ i }}\"\n name=\"option{{ i }}\"\n type=\"text\"\n [formControl]=\"option\"\n (input)=\"optionChanged(i)\"\n placeholder=\"{{ 'streamChat.Add an option' | translate }}\"\n />\n </div>\n </div>\n </ng-container>\n </div>\n </ng-container>\n </fieldset>\n <ng-container\n *ngTemplateOutlet=\"\n switch;\n context: {\n name: 'multiple_answers',\n control: formGroup.get('multiple_answers'),\n label: 'streamChat.Multiple answers' | translate\n }\n \"\n ></ng-container>\n <div\n *ngIf=\"formGroup.get('multiple_answers')?.value\"\n class=\"str-chat__form__field str-chat__form__input-field str-chat__form__input-field--with-label\"\n [class.str-chat__form__input-field--has-error]=\"\n formGroup.get('maximum_number_of_votes')?.errors &&\n formGroup.get('maximum_number_of_votes')?.touched\n \"\n >\n <label class=\"str-chat__form__field-label\" htmlFor=\"name\">\n {{ \"streamChat.Maximum number of votes\" | translate }}\n </label>\n <div class=\"str-chat__form__input-field__value\">\n <div\n class=\"str-chat__form-field-error str-chat__form__input-field__error\"\n >\n {{\n (formGroup.get(\"maximum_number_of_votes\")?.errors &&\n formGroup.get(\"maximum_number_of_votes\")?.touched\n ? \"streamChat.Provide a value between {{ min }}\n and {{ max }}\" : \"\" ) | translate: { min: 2, max: 10 } }}\n </div>\n <input\n id=\"maximum_number_of_votes\"\n type=\"text\"\n formControlName=\"maximum_number_of_votes\"\n placeholder=\"{{ 'streamChat.Provide a value between ' \n + '{{ min }} and {{ max }}' | translate: { min: 2, max: 10 } }}\"\n />\n </div>\n </div>\n <ng-container\n *ngTemplateOutlet=\"\n switch;\n context: {\n name: 'is_anonymous',\n control: formGroup.get('is_anonymous'),\n label: 'streamChat.Anonymous poll' | translate\n }\n \"\n ></ng-container>\n <ng-container\n *ngTemplateOutlet=\"\n switch;\n context: {\n name: 'allow_user_suggested_options',\n control: formGroup.get('allow_user_suggested_options'),\n label: 'streamChat.Allow option suggestions' | translate\n }\n \"\n ></ng-container>\n <ng-container\n *ngTemplateOutlet=\"\n switch;\n context: {\n name: 'allow_answers',\n control: formGroup.get('allow_answers'),\n label: 'streamChat.Allow comments' | translate\n }\n \"\n ></ng-container>\n </form>\n </div>\n <stream-notification-list></stream-notification-list>\n <div class=\"str-chat__dialog__controls\">\n <button\n class=\"str-chat__dialog__controls-button str-chat__dialog__controls-button--cancel\"\n (click)=\"cancel.emit()\"\n type=\"button\"\n >\n {{ \"streamChat.Cancel\" | translate }}\n </button>\n <button\n class=\"str-chat__dialog__controls-button str-chat__dialog__controls-button--submit\"\n (click)=\"createPoll()\"\n [disabled]=\"formGroup.invalid\"\n type=\"submit\"\n >\n {{ \"streamChat.Create\" | translate }}\n </button>\n </div>\n </div>\n</ng-template>\n\n<ng-template #switch let-control=\"control\" let-label=\"label\" let-name=\"name\">\n <div class=\"str-chat__form__field str-chat__form__switch-field\">\n <label>\n <div class=\"str-chat__form__field str-chat__form__switch-field-content\">\n <div class=\"str-chat__form__field str-chat__form__switch-field__text\">\n {{ label | translate }}\n </div>\n </div>\n <input\n type=\"checkbox\"\n [checked]=\"control.value\"\n id=\"{{ name }}\"\n name=\"{{ name }}\"\n />\n <div\n class=\"str-chat__form__switch-field__switch\"\n [class.str-chat__form__switch-field__switch--on]=\"control.value\"\n (click)=\"control.setValue(!control.value, { emitEvent: true })\"\n (keyup.enter)=\"control.setValue(!control.value, { emitEvent: true })\"\n >\n <div class=\"str-chat__form__switch-field__switch-handle\"></div>\n </div>\n </label>\n </div>\n</ng-template>\n", dependencies: [{ kind: "directive", type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i4.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i5.ModalComponent, selector: "stream-modal", inputs: ["isOpen", "content"], outputs: ["isOpenChange"] }, { kind: "component", type: i6.NotificationListComponent, selector: "stream-notification-list" }, { kind: "directive", type: i7.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i7.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i7.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i7.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i7.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i7.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i7.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "directive", type: i7.FormArrayName, selector: "[formArrayName]", inputs: ["formArrayName"] }, { kind: "pipe", type: i4.AsyncPipe, name: "async" }, { kind: "pipe", type: i8.TranslatePipe, name: "translate" }] });
|
|
123
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: PollComposerComponent, decorators: [{
|
|
124
|
+
type: Component,
|
|
125
|
+
args: [{ selector: 'stream-poll-composer', template: "<ng-container>\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.modalTemplate$ | async) || defaultModal;\n context: getModalContext()\n \"\n ></ng-container>\n</ng-container>\n\n<ng-template\n #defaultModal\n let-isOpen=\"isOpen\"\n let-isOpenChangeHandler=\"isOpenChangeHandler\"\n let-content=\"content\"\n>\n <stream-modal\n class=\"str-chat-angular__create-poll-modal str-chat__create-poll-modal\"\n [isOpen]=\"isOpen\"\n [content]=\"content\"\n (isOpenChange)=\"isOpenChangeHandler($event)\"\n >\n </stream-modal>\n</ng-template>\n\n<ng-template #formContent>\n <div\n class=\"str-chat__dialog str-chat__poll-creation-dialog\"\n data-testid=\"poll-creation-dialog\"\n >\n <div class=\"str-chat__modal-header\">\n <div class=\"str-chat__modal-header__title\">\n {{ \"streamChat.Create poll\" | translate }}\n </div>\n </div>\n <div class=\"str-chat__dialog__body\">\n <form [formGroup]=\"formGroup\" autocomplete=\"off\">\n <div\n class=\"str-chat__form__field str-chat__form__input-field str-chat__form__input-field--with-label\"\n [class.str-chat__form__input-field--has-error]=\"\n formGroup.get('name')?.errors && formGroup.get('name')?.touched\n \"\n >\n <label class=\"str-chat__form__field-label\" htmlFor=\"name\">\n {{ \"streamChat.Question\" | translate }}\n </label>\n <div class=\"str-chat__form__input-field__value\">\n <div\n class=\"str-chat__form-field-error str-chat__form__input-field__error\"\n >\n {{\n (formGroup.get(\"name\")?.errors?.required &&\n formGroup.get(\"name\")?.touched\n ? \"streamChat.Question is required\"\n : \"\"\n ) | translate\n }}\n </div>\n <input\n id=\"name\"\n type=\"text\"\n formControlName=\"name\"\n placeholder=\"{{ 'streamChat.Ask a question' | translate }}\"\n />\n </div>\n </div>\n <fieldset class=\"str-chat__form__field str-chat__form__input-fieldset\">\n <legend class=\"str-chat__form__field-label\">\n {{ \"streamChat.Options\" | translate }}\n </legend>\n <div\n class=\"str-chat__form-field-error str-chat__form__input-field__error\"\n >\n {{\n (formGroup.get(\"options\")?.errors?.atLeastOne &&\n formGroup.get(\"options\")?.touched\n ? \"streamChat.Provide at least one option\"\n : \"\"\n ) | translate\n }}\n </div>\n <ng-container formArrayName=\"options\">\n <div class=\"str-chat__form__input-fieldset__values\">\n <!-- eslint-disable @angular-eslint/template/use-track-by-function -->\n <ng-container\n *ngFor=\"let option of options.controls; let i = index\"\n >\n <!-- eslint-enable @angular-eslint/template/use-track-by-function -->\n <div class=\"str-chat__drag-and-drop-container__item\">\n <div class=\"str-chat__form__input-field\">\n <input\n id=\"option{{ i }}\"\n name=\"option{{ i }}\"\n type=\"text\"\n [formControl]=\"option\"\n (input)=\"optionChanged(i)\"\n placeholder=\"{{ 'streamChat.Add an option' | translate }}\"\n />\n </div>\n </div>\n </ng-container>\n </div>\n </ng-container>\n </fieldset>\n <ng-container\n *ngTemplateOutlet=\"\n switch;\n context: {\n name: 'multiple_answers',\n control: formGroup.get('multiple_answers'),\n label: 'streamChat.Multiple answers' | translate\n }\n \"\n ></ng-container>\n <div\n *ngIf=\"formGroup.get('multiple_answers')?.value\"\n class=\"str-chat__form__field str-chat__form__input-field str-chat__form__input-field--with-label\"\n [class.str-chat__form__input-field--has-error]=\"\n formGroup.get('maximum_number_of_votes')?.errors &&\n formGroup.get('maximum_number_of_votes')?.touched\n \"\n >\n <label class=\"str-chat__form__field-label\" htmlFor=\"name\">\n {{ \"streamChat.Maximum number of votes\" | translate }}\n </label>\n <div class=\"str-chat__form__input-field__value\">\n <div\n class=\"str-chat__form-field-error str-chat__form__input-field__error\"\n >\n {{\n (formGroup.get(\"maximum_number_of_votes\")?.errors &&\n formGroup.get(\"maximum_number_of_votes\")?.touched\n ? \"streamChat.Provide a value between {{ min }}\n and {{ max }}\" : \"\" ) | translate: { min: 2, max: 10 } }}\n </div>\n <input\n id=\"maximum_number_of_votes\"\n type=\"text\"\n formControlName=\"maximum_number_of_votes\"\n placeholder=\"{{ 'streamChat.Provide a value between ' \n + '{{ min }} and {{ max }}' | translate: { min: 2, max: 10 } }}\"\n />\n </div>\n </div>\n <ng-container\n *ngTemplateOutlet=\"\n switch;\n context: {\n name: 'is_anonymous',\n control: formGroup.get('is_anonymous'),\n label: 'streamChat.Anonymous poll' | translate\n }\n \"\n ></ng-container>\n <ng-container\n *ngTemplateOutlet=\"\n switch;\n context: {\n name: 'allow_user_suggested_options',\n control: formGroup.get('allow_user_suggested_options'),\n label: 'streamChat.Allow option suggestions' | translate\n }\n \"\n ></ng-container>\n <ng-container\n *ngTemplateOutlet=\"\n switch;\n context: {\n name: 'allow_answers',\n control: formGroup.get('allow_answers'),\n label: 'streamChat.Allow comments' | translate\n }\n \"\n ></ng-container>\n </form>\n </div>\n <stream-notification-list></stream-notification-list>\n <div class=\"str-chat__dialog__controls\">\n <button\n class=\"str-chat__dialog__controls-button str-chat__dialog__controls-button--cancel\"\n (click)=\"cancel.emit()\"\n type=\"button\"\n >\n {{ \"streamChat.Cancel\" | translate }}\n </button>\n <button\n class=\"str-chat__dialog__controls-button str-chat__dialog__controls-button--submit\"\n (click)=\"createPoll()\"\n [disabled]=\"formGroup.invalid\"\n type=\"submit\"\n >\n {{ \"streamChat.Create\" | translate }}\n </button>\n </div>\n </div>\n</ng-template>\n\n<ng-template #switch let-control=\"control\" let-label=\"label\" let-name=\"name\">\n <div class=\"str-chat__form__field str-chat__form__switch-field\">\n <label>\n <div class=\"str-chat__form__field str-chat__form__switch-field-content\">\n <div class=\"str-chat__form__field str-chat__form__switch-field__text\">\n {{ label | translate }}\n </div>\n </div>\n <input\n type=\"checkbox\"\n [checked]=\"control.value\"\n id=\"{{ name }}\"\n name=\"{{ name }}\"\n />\n <div\n class=\"str-chat__form__switch-field__switch\"\n [class.str-chat__form__switch-field__switch--on]=\"control.value\"\n (click)=\"control.setValue(!control.value, { emitEvent: true })\"\n (keyup.enter)=\"control.setValue(!control.value, { emitEvent: true })\"\n >\n <div class=\"str-chat__form__switch-field__switch-handle\"></div>\n </div>\n </label>\n </div>\n</ng-template>\n" }]
|
|
126
|
+
}], ctorParameters: function () { return [{ type: i1.CustomTemplatesService }, { type: i2.ChatClientService }, { type: i3.NotificationService }]; }, propDecorators: { pollCompose: [{
|
|
127
|
+
type: Output
|
|
128
|
+
}], cancel: [{
|
|
129
|
+
type: Output
|
|
130
|
+
}], formContent: [{
|
|
131
|
+
type: ViewChild,
|
|
132
|
+
args: ['formContent']
|
|
133
|
+
}] } });
|
|
134
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"poll-composer.component.js","sourceRoot":"","sources":["../../../../../../projects/stream-chat-angular/src/lib/polls/poll-composer/poll-composer.component.ts","../../../../../../projects/stream-chat-angular/src/lib/polls/poll-composer/poll-composer.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  EventEmitter,\n  Output,\n  TemplateRef,\n  ViewChild,\n} from '@angular/core';\nimport { FormArray, FormControl, FormGroup, Validators } from '@angular/forms';\nimport { VotingVisibility } from 'stream-chat';\nimport { CustomTemplatesService } from '../../custom-templates.service';\nimport { ModalContext } from '../../types';\nimport { ChatClientService } from '../../chat-client.service';\nimport { NotificationService } from '../../notification.service';\nimport { atLeastOneOption, maximumNumberOfVotes } from './validators';\n\n/**\n *\n */\n@Component({\n  selector: 'stream-poll-composer',\n  templateUrl: './poll-composer.component.html',\n  styles: [],\n})\nexport class PollComposerComponent {\n  /**\n   * Emitted when a poll is created, the poll id is emitted\n   */\n  @Output() pollCompose = new EventEmitter<string>();\n  /**\n   * Emitted when the poll composing is cancelled\n   */\n  @Output() cancel = new EventEmitter<void>();\n  formGroup = new FormGroup({\n    name: new FormControl('', [Validators.required]),\n    options: new FormArray<FormControl<string | null>>(\n      [new FormControl<string | null>('')],\n      [atLeastOneOption()]\n    ),\n    multiple_answers: new FormControl(false),\n    maximum_number_of_votes: new FormControl(null, [\n      Validators.min(2),\n      Validators.max(10),\n    ]),\n    is_anonymous: new FormControl(false),\n    allow_user_suggested_options: new FormControl(false),\n    allow_answers: new FormControl(false),\n  });\n  isModalOpen = true;\n  @ViewChild('formContent') private formContent!: TemplateRef<void>;\n\n  constructor(\n    readonly customTemplatesService: CustomTemplatesService,\n    private chatService: ChatClientService,\n    private notificationService: NotificationService\n  ) {\n    this.formGroup\n      .get('maximum_number_of_votes')\n      ?.addValidators([\n        maximumNumberOfVotes(\n          this.formGroup.get('multiple_answers') as FormControl<boolean>\n        ),\n      ]);\n    this.formGroup.get('maximum_number_of_votes')?.disable();\n    this.formGroup.valueChanges.subscribe((value) => {\n      if (\n        value.multiple_answers &&\n        this.formGroup.get('maximum_number_of_votes')?.disabled\n      ) {\n        this.formGroup.get('maximum_number_of_votes')?.enable();\n      } else if (\n        !value.multiple_answers &&\n        this.formGroup.get('maximum_number_of_votes')?.enabled\n      ) {\n        this.formGroup.get('maximum_number_of_votes')?.disable();\n      }\n    });\n  }\n\n  optionChanged(index: number) {\n    const control = this.options.at(index);\n    const penultimateIndex = this.options.length - 2;\n    if (index === this.options.length - 1 && control.value?.length === 1) {\n      this.addOption();\n    } else if (\n      index === penultimateIndex &&\n      control.value?.length === 0 &&\n      this.options.at(this.options.length - 1).value?.length === 0\n    ) {\n      this.removeLastOption();\n    }\n  }\n\n  get options() {\n    return this.formGroup.get('options') as FormArray<\n      FormControl<string | null>\n    >;\n  }\n\n  addOption() {\n    const control = new FormControl<string | null>('', []);\n    this.options.push(control);\n  }\n\n  removeLastOption() {\n    this.options.removeAt(this.options.length - 1);\n  }\n\n  getModalContext(): ModalContext {\n    return {\n      isOpen: this.isModalOpen,\n      isOpenChangeHandler: (isOpen: boolean) => {\n        if (!isOpen) {\n          this.cancel.emit();\n        }\n        this.isModalOpen = isOpen;\n      },\n      content: this.formContent,\n    };\n  }\n\n  async createPoll() {\n    try {\n      const maxVotesControl = this.formGroup.get('maximum_number_of_votes');\n      const response = await this.chatService.chatClient.polls.createPoll({\n        name: this.formGroup.get('name')!.value!,\n        options: this.formGroup\n          .get('options')!\n          .value.filter((v) => !!v)\n          .map((v) => ({ text: v! })),\n        enforce_unique_vote: !this.formGroup.get('multiple_answers')?.value,\n        max_votes_allowed: maxVotesControl?.value\n          ? +maxVotesControl.value\n          : undefined,\n        voting_visibility: this.formGroup.get('is_anonymous')?.value\n          ? VotingVisibility.anonymous\n          : VotingVisibility.public,\n        allow_user_suggested_options: !!this.formGroup.get(\n          'allow_user_suggested_options'\n        )?.value,\n        allow_answers: !!this.formGroup.get('allow_answers')?.value,\n      });\n      this.pollCompose.emit(response?.id);\n    } catch (error) {\n      this.notificationService.addTemporaryNotification(\n        'streamChat.Failed to create poll'\n      );\n      throw error;\n    }\n  }\n}\n","<ng-container>\n  <ng-container\n    *ngTemplateOutlet=\"\n      (customTemplatesService.modalTemplate$ | async) || defaultModal;\n      context: getModalContext()\n    \"\n  ></ng-container>\n</ng-container>\n\n<ng-template\n  #defaultModal\n  let-isOpen=\"isOpen\"\n  let-isOpenChangeHandler=\"isOpenChangeHandler\"\n  let-content=\"content\"\n>\n  <stream-modal\n    class=\"str-chat-angular__create-poll-modal str-chat__create-poll-modal\"\n    [isOpen]=\"isOpen\"\n    [content]=\"content\"\n    (isOpenChange)=\"isOpenChangeHandler($event)\"\n  >\n  </stream-modal>\n</ng-template>\n\n<ng-template #formContent>\n  <div\n    class=\"str-chat__dialog str-chat__poll-creation-dialog\"\n    data-testid=\"poll-creation-dialog\"\n  >\n    <div class=\"str-chat__modal-header\">\n      <div class=\"str-chat__modal-header__title\">\n        {{ \"streamChat.Create poll\" | translate }}\n      </div>\n    </div>\n    <div class=\"str-chat__dialog__body\">\n      <form [formGroup]=\"formGroup\" autocomplete=\"off\">\n        <div\n          class=\"str-chat__form__field str-chat__form__input-field str-chat__form__input-field--with-label\"\n          [class.str-chat__form__input-field--has-error]=\"\n            formGroup.get('name')?.errors && formGroup.get('name')?.touched\n          \"\n        >\n          <label class=\"str-chat__form__field-label\" htmlFor=\"name\">\n            {{ \"streamChat.Question\" | translate }}\n          </label>\n          <div class=\"str-chat__form__input-field__value\">\n            <div\n              class=\"str-chat__form-field-error str-chat__form__input-field__error\"\n            >\n              {{\n                (formGroup.get(\"name\")?.errors?.required &&\n                formGroup.get(\"name\")?.touched\n                  ? \"streamChat.Question is required\"\n                  : \"\"\n                ) | translate\n              }}\n            </div>\n            <input\n              id=\"name\"\n              type=\"text\"\n              formControlName=\"name\"\n              placeholder=\"{{ 'streamChat.Ask a question' | translate }}\"\n            />\n          </div>\n        </div>\n        <fieldset class=\"str-chat__form__field str-chat__form__input-fieldset\">\n          <legend class=\"str-chat__form__field-label\">\n            {{ \"streamChat.Options\" | translate }}\n          </legend>\n          <div\n            class=\"str-chat__form-field-error str-chat__form__input-field__error\"\n          >\n            {{\n              (formGroup.get(\"options\")?.errors?.atLeastOne &&\n              formGroup.get(\"options\")?.touched\n                ? \"streamChat.Provide at least one option\"\n                : \"\"\n              ) | translate\n            }}\n          </div>\n          <ng-container formArrayName=\"options\">\n            <div class=\"str-chat__form__input-fieldset__values\">\n              <!-- eslint-disable @angular-eslint/template/use-track-by-function -->\n              <ng-container\n                *ngFor=\"let option of options.controls; let i = index\"\n              >\n                <!-- eslint-enable @angular-eslint/template/use-track-by-function -->\n                <div class=\"str-chat__drag-and-drop-container__item\">\n                  <div class=\"str-chat__form__input-field\">\n                    <input\n                      id=\"option{{ i }}\"\n                      name=\"option{{ i }}\"\n                      type=\"text\"\n                      [formControl]=\"option\"\n                      (input)=\"optionChanged(i)\"\n                      placeholder=\"{{ 'streamChat.Add an option' | translate }}\"\n                    />\n                  </div>\n                </div>\n              </ng-container>\n            </div>\n          </ng-container>\n        </fieldset>\n        <ng-container\n          *ngTemplateOutlet=\"\n            switch;\n            context: {\n              name: 'multiple_answers',\n              control: formGroup.get('multiple_answers'),\n              label: 'streamChat.Multiple answers' | translate\n            }\n          \"\n        ></ng-container>\n        <div\n          *ngIf=\"formGroup.get('multiple_answers')?.value\"\n          class=\"str-chat__form__field str-chat__form__input-field str-chat__form__input-field--with-label\"\n          [class.str-chat__form__input-field--has-error]=\"\n            formGroup.get('maximum_number_of_votes')?.errors &&\n            formGroup.get('maximum_number_of_votes')?.touched\n          \"\n        >\n          <label class=\"str-chat__form__field-label\" htmlFor=\"name\">\n            {{ \"streamChat.Maximum number of votes\" | translate }}\n          </label>\n          <div class=\"str-chat__form__input-field__value\">\n            <div\n              class=\"str-chat__form-field-error str-chat__form__input-field__error\"\n            >\n              {{\n                (formGroup.get(\"maximum_number_of_votes\")?.errors &&\n                formGroup.get(\"maximum_number_of_votes\")?.touched\n                  ? \"streamChat.Provide a value between {{ min }}\n              and {{ max }}\" : \"\" ) | translate: { min: 2, max: 10 } }}\n            </div>\n            <input\n              id=\"maximum_number_of_votes\"\n              type=\"text\"\n              formControlName=\"maximum_number_of_votes\"\n              placeholder=\"{{ 'streamChat.Provide a value between ' \n              + '{{ min }} and {{ max }}' | translate: { min: 2, max: 10 } }}\"\n            />\n          </div>\n        </div>\n        <ng-container\n          *ngTemplateOutlet=\"\n            switch;\n            context: {\n              name: 'is_anonymous',\n              control: formGroup.get('is_anonymous'),\n              label: 'streamChat.Anonymous poll' | translate\n            }\n          \"\n        ></ng-container>\n        <ng-container\n          *ngTemplateOutlet=\"\n            switch;\n            context: {\n              name: 'allow_user_suggested_options',\n              control: formGroup.get('allow_user_suggested_options'),\n              label: 'streamChat.Allow option suggestions' | translate\n            }\n          \"\n        ></ng-container>\n        <ng-container\n          *ngTemplateOutlet=\"\n            switch;\n            context: {\n              name: 'allow_answers',\n              control: formGroup.get('allow_answers'),\n              label: 'streamChat.Allow comments' | translate\n            }\n          \"\n        ></ng-container>\n      </form>\n    </div>\n    <stream-notification-list></stream-notification-list>\n    <div class=\"str-chat__dialog__controls\">\n      <button\n        class=\"str-chat__dialog__controls-button str-chat__dialog__controls-button--cancel\"\n        (click)=\"cancel.emit()\"\n        type=\"button\"\n      >\n        {{ \"streamChat.Cancel\" | translate }}\n      </button>\n      <button\n        class=\"str-chat__dialog__controls-button str-chat__dialog__controls-button--submit\"\n        (click)=\"createPoll()\"\n        [disabled]=\"formGroup.invalid\"\n        type=\"submit\"\n      >\n        {{ \"streamChat.Create\" | translate }}\n      </button>\n    </div>\n  </div>\n</ng-template>\n\n<ng-template #switch let-control=\"control\" let-label=\"label\" let-name=\"name\">\n  <div class=\"str-chat__form__field str-chat__form__switch-field\">\n    <label>\n      <div class=\"str-chat__form__field str-chat__form__switch-field-content\">\n        <div class=\"str-chat__form__field str-chat__form__switch-field__text\">\n          {{ label | translate }}\n        </div>\n      </div>\n      <input\n        type=\"checkbox\"\n        [checked]=\"control.value\"\n        id=\"{{ name }}\"\n        name=\"{{ name }}\"\n      />\n      <div\n        class=\"str-chat__form__switch-field__switch\"\n        [class.str-chat__form__switch-field__switch--on]=\"control.value\"\n        (click)=\"control.setValue(!control.value, { emitEvent: true })\"\n        (keyup.enter)=\"control.setValue(!control.value, { emitEvent: true })\"\n      >\n        <div class=\"str-chat__form__switch-field__switch-handle\"></div>\n      </div>\n    </label>\n  </div>\n</ng-template>\n"]}
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
export function atLeastOneOption() {
|
|
2
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
3
|
+
return (control) => {
|
|
4
|
+
const formArray = control;
|
|
5
|
+
const hasAtLeastOne = formArray.value.some((item) => !!item);
|
|
6
|
+
return hasAtLeastOne ? null : { atLeastOne: true };
|
|
7
|
+
};
|
|
8
|
+
}
|
|
9
|
+
export function maximumNumberOfVotes(canHaveMultipleVotes) {
|
|
10
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
11
|
+
return (control) => {
|
|
12
|
+
const formControl = control;
|
|
13
|
+
return canHaveMultipleVotes.value && !formControl.value
|
|
14
|
+
? { maximumNumberOfVotes: true }
|
|
15
|
+
: null;
|
|
16
|
+
};
|
|
17
|
+
}
|
|
18
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmFsaWRhdG9ycy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3N0cmVhbS1jaGF0LWFuZ3VsYXIvc3JjL2xpYi9wb2xscy9wb2xsLWNvbXBvc2VyL3ZhbGlkYXRvcnMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBT0EsTUFBTSxVQUFVLGdCQUFnQjtJQUM5Qiw4REFBOEQ7SUFDOUQsT0FBTyxDQUFDLE9BQWtDLEVBQUUsRUFBRTtRQUM1QyxNQUFNLFNBQVMsR0FBRyxPQUFnRCxDQUFDO1FBQ25FLE1BQU0sYUFBYSxHQUFHLFNBQVMsQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLENBQUMsSUFBSSxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDN0QsT0FBTyxhQUFhLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsRUFBRSxVQUFVLEVBQUUsSUFBSSxFQUFFLENBQUM7SUFDckQsQ0FBQyxDQUFDO0FBQ0osQ0FBQztBQUVELE1BQU0sVUFBVSxvQkFBb0IsQ0FDbEMsb0JBQTBDO0lBRTFDLDhEQUE4RDtJQUM5RCxPQUFPLENBQUMsT0FBa0MsRUFBRSxFQUFFO1FBQzVDLE1BQU0sV0FBVyxHQUFHLE9BQXFDLENBQUM7UUFDMUQsT0FBTyxvQkFBb0IsQ0FBQyxLQUFLLElBQUksQ0FBQyxXQUFXLENBQUMsS0FBSztZQUNyRCxDQUFDLENBQUMsRUFBRSxvQkFBb0IsRUFBRSxJQUFJLEVBQUU7WUFDaEMsQ0FBQyxDQUFDLElBQUksQ0FBQztJQUNYLENBQUMsQ0FBQztBQUNKLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge1xuICBBYnN0cmFjdENvbnRyb2wsXG4gIEZvcm1BcnJheSxcbiAgRm9ybUNvbnRyb2wsXG4gIFZhbGlkYXRvckZuLFxufSBmcm9tICdAYW5ndWxhci9mb3Jtcyc7XG5cbmV4cG9ydCBmdW5jdGlvbiBhdExlYXN0T25lT3B0aW9uKCk6IFZhbGlkYXRvckZuIHtcbiAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby1leHBsaWNpdC1hbnlcbiAgcmV0dXJuIChjb250cm9sOiBBYnN0cmFjdENvbnRyb2w8YW55LCBhbnk+KSA9PiB7XG4gICAgY29uc3QgZm9ybUFycmF5ID0gY29udHJvbCBhcyBGb3JtQXJyYXk8Rm9ybUNvbnRyb2w8c3RyaW5nIHwgbnVsbD4+O1xuICAgIGNvbnN0IGhhc0F0TGVhc3RPbmUgPSBmb3JtQXJyYXkudmFsdWUuc29tZSgoaXRlbSkgPT4gISFpdGVtKTtcbiAgICByZXR1cm4gaGFzQXRMZWFzdE9uZSA/IG51bGwgOiB7IGF0TGVhc3RPbmU6IHRydWUgfTtcbiAgfTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIG1heGltdW1OdW1iZXJPZlZvdGVzKFxuICBjYW5IYXZlTXVsdGlwbGVWb3RlczogRm9ybUNvbnRyb2w8Ym9vbGVhbj5cbik6IFZhbGlkYXRvckZuIHtcbiAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby1leHBsaWNpdC1hbnlcbiAgcmV0dXJuIChjb250cm9sOiBBYnN0cmFjdENvbnRyb2w8YW55LCBhbnk+KSA9PiB7XG4gICAgY29uc3QgZm9ybUNvbnRyb2wgPSBjb250cm9sIGFzIEZvcm1Db250cm9sPG51bWJlciB8IG51bGw+O1xuICAgIHJldHVybiBjYW5IYXZlTXVsdGlwbGVWb3Rlcy52YWx1ZSAmJiAhZm9ybUNvbnRyb2wudmFsdWVcbiAgICAgID8geyBtYXhpbXVtTnVtYmVyT2ZWb3RlczogdHJ1ZSB9XG4gICAgICA6IG51bGw7XG4gIH07XG59XG4iXX0=
|
|
@@ -0,0 +1,80 @@
|
|
|
1
|
+
import { Component, ChangeDetectionStrategy } from '@angular/core';
|
|
2
|
+
import { BasePollComponent } from '../base-poll.component';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "@ngx-translate/core";
|
|
5
|
+
/**
|
|
6
|
+
*
|
|
7
|
+
*/
|
|
8
|
+
export class PollHeaderComponent extends BasePollComponent {
|
|
9
|
+
constructor() {
|
|
10
|
+
super(...arguments);
|
|
11
|
+
this.name = '';
|
|
12
|
+
this.selectionInstructions = {
|
|
13
|
+
text: '',
|
|
14
|
+
count: undefined,
|
|
15
|
+
};
|
|
16
|
+
}
|
|
17
|
+
stateStoreSelector(poll, markForCheck) {
|
|
18
|
+
const unsubscribe = poll.state.subscribeWithSelector((state) => ({
|
|
19
|
+
name: state.name,
|
|
20
|
+
is_closed: state.is_closed,
|
|
21
|
+
enforce_unique_vote: state.enforce_unique_vote,
|
|
22
|
+
max_votes_allowed: state.max_votes_allowed,
|
|
23
|
+
options: state.options,
|
|
24
|
+
}), (state) => {
|
|
25
|
+
const name = state.name;
|
|
26
|
+
const selectionInstructions = this.getSelectionInstructions(state);
|
|
27
|
+
let changed = false;
|
|
28
|
+
if (name !== this.name) {
|
|
29
|
+
this.name = name;
|
|
30
|
+
changed = true;
|
|
31
|
+
}
|
|
32
|
+
if (selectionInstructions.text !== this.selectionInstructions.text ||
|
|
33
|
+
selectionInstructions.count !== this.selectionInstructions.count) {
|
|
34
|
+
this.selectionInstructions = selectionInstructions;
|
|
35
|
+
changed = true;
|
|
36
|
+
}
|
|
37
|
+
if (changed) {
|
|
38
|
+
markForCheck();
|
|
39
|
+
}
|
|
40
|
+
});
|
|
41
|
+
return unsubscribe;
|
|
42
|
+
}
|
|
43
|
+
getSelectionInstructions(state) {
|
|
44
|
+
if (state.is_closed)
|
|
45
|
+
return {
|
|
46
|
+
text: 'streamChat.Vote ended',
|
|
47
|
+
count: undefined,
|
|
48
|
+
};
|
|
49
|
+
if (state.enforce_unique_vote || state.options.length === 1) {
|
|
50
|
+
return {
|
|
51
|
+
text: 'streamChat.Select one',
|
|
52
|
+
count: undefined,
|
|
53
|
+
};
|
|
54
|
+
}
|
|
55
|
+
if (state.max_votes_allowed)
|
|
56
|
+
return {
|
|
57
|
+
text: 'streamChat.Select up to {{count}}',
|
|
58
|
+
count: state.max_votes_allowed > state.options.length
|
|
59
|
+
? state.options.length
|
|
60
|
+
: state.max_votes_allowed,
|
|
61
|
+
};
|
|
62
|
+
if (state.options.length > 1) {
|
|
63
|
+
return {
|
|
64
|
+
text: 'streamChat.Select one or more',
|
|
65
|
+
count: undefined,
|
|
66
|
+
};
|
|
67
|
+
}
|
|
68
|
+
return {
|
|
69
|
+
text: '',
|
|
70
|
+
count: undefined,
|
|
71
|
+
};
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
PollHeaderComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: PollHeaderComponent, deps: null, target: i0.ɵɵFactoryTarget.Component });
|
|
75
|
+
PollHeaderComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.0.4", type: PollHeaderComponent, selector: "stream-poll-header", usesInheritance: true, ngImport: i0, template: "<div class=\"str-chat__poll-header\">\n <div class=\"str-chat__poll-title\">{{ name }}</div>\n <div class=\"str-chat__poll-subtitle\">\n {{ selectionInstructions.text | translate : selectionInstructions }}\n </div>\n</div>\n", dependencies: [{ kind: "pipe", type: i1.TranslatePipe, name: "translate" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
76
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: PollHeaderComponent, decorators: [{
|
|
77
|
+
type: Component,
|
|
78
|
+
args: [{ selector: 'stream-poll-header', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"str-chat__poll-header\">\n <div class=\"str-chat__poll-title\">{{ name }}</div>\n <div class=\"str-chat__poll-subtitle\">\n {{ selectionInstructions.text | translate : selectionInstructions }}\n </div>\n</div>\n" }]
|
|
79
|
+
}] });
|
|
80
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicG9sbC1oZWFkZXIuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvc3RyZWFtLWNoYXQtYW5ndWxhci9zcmMvbGliL3BvbGxzL3BvbGwtaGVhZGVyL3BvbGwtaGVhZGVyLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3N0cmVhbS1jaGF0LWFuZ3VsYXIvc3JjL2xpYi9wb2xscy9wb2xsLWhlYWRlci9wb2xsLWhlYWRlci5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFFLHVCQUF1QixFQUFFLE1BQU0sZUFBZSxDQUFDO0FBRW5FLE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxNQUFNLHdCQUF3QixDQUFDOzs7QUFPM0Q7O0dBRUc7QUFPSCxNQUFNLE9BQU8sbUJBQW9CLFNBQVEsaUJBQWlCO0lBTjFEOztRQU9FLFNBQUksR0FBRyxFQUFFLENBQUM7UUFDViwwQkFBcUIsR0FBMEI7WUFDN0MsSUFBSSxFQUFFLEVBQUU7WUFDUixLQUFLLEVBQUUsU0FBUztTQUNqQixDQUFDO0tBNEVIO0lBMUVXLGtCQUFrQixDQUMxQixJQUFVLEVBQ1YsWUFBd0I7UUFFeEIsTUFBTSxXQUFXLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxxQkFBcUIsQ0FDbEQsQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDLENBQUM7WUFDVixJQUFJLEVBQUUsS0FBSyxDQUFDLElBQUk7WUFDaEIsU0FBUyxFQUFFLEtBQUssQ0FBQyxTQUFTO1lBQzFCLG1CQUFtQixFQUFFLEtBQUssQ0FBQyxtQkFBbUI7WUFDOUMsaUJBQWlCLEVBQUUsS0FBSyxDQUFDLGlCQUFpQjtZQUMxQyxPQUFPLEVBQUUsS0FBSyxDQUFDLE9BQU87U0FDdkIsQ0FBQyxFQUNGLENBQUMsS0FBSyxFQUFFLEVBQUU7WUFDUixNQUFNLElBQUksR0FBRyxLQUFLLENBQUMsSUFBSSxDQUFDO1lBQ3hCLE1BQU0scUJBQXFCLEdBQUcsSUFBSSxDQUFDLHdCQUF3QixDQUFDLEtBQUssQ0FBQyxDQUFDO1lBRW5FLElBQUksT0FBTyxHQUFHLEtBQUssQ0FBQztZQUNwQixJQUFJLElBQUksS0FBSyxJQUFJLENBQUMsSUFBSSxFQUFFO2dCQUN0QixJQUFJLENBQUMsSUFBSSxHQUFHLElBQUksQ0FBQztnQkFDakIsT0FBTyxHQUFHLElBQUksQ0FBQzthQUNoQjtZQUNELElBQ0UscUJBQXFCLENBQUMsSUFBSSxLQUFLLElBQUksQ0FBQyxxQkFBcUIsQ0FBQyxJQUFJO2dCQUM5RCxxQkFBcUIsQ0FBQyxLQUFLLEtBQUssSUFBSSxDQUFDLHFCQUFxQixDQUFDLEtBQUssRUFDaEU7Z0JBQ0EsSUFBSSxDQUFDLHFCQUFxQixHQUFHLHFCQUFxQixDQUFDO2dCQUNuRCxPQUFPLEdBQUcsSUFBSSxDQUFDO2FBQ2hCO1lBRUQsSUFBSSxPQUFPLEVBQUU7Z0JBQ1gsWUFBWSxFQUFFLENBQUM7YUFDaEI7UUFDSCxDQUFDLENBQ0YsQ0FBQztRQUVGLE9BQU8sV0FBVyxDQUFDO0lBQ3JCLENBQUM7SUFFRCx3QkFBd0IsQ0FDdEIsS0FHQztRQUVELElBQUksS0FBSyxDQUFDLFNBQVM7WUFDakIsT0FBTztnQkFDTCxJQUFJLEVBQUUsdUJBQXVCO2dCQUM3QixLQUFLLEVBQUUsU0FBUzthQUNqQixDQUFDO1FBQ0osSUFBSSxLQUFLLENBQUMsbUJBQW1CLElBQUksS0FBSyxDQUFDLE9BQU8sQ0FBQyxNQUFNLEtBQUssQ0FBQyxFQUFFO1lBQzNELE9BQU87Z0JBQ0wsSUFBSSxFQUFFLHVCQUF1QjtnQkFDN0IsS0FBSyxFQUFFLFNBQVM7YUFDakIsQ0FBQztTQUNIO1FBQ0QsSUFBSSxLQUFLLENBQUMsaUJBQWlCO1lBQ3pCLE9BQU87Z0JBQ0wsSUFBSSxFQUFFLG1DQUFtQztnQkFDekMsS0FBSyxFQUNILEtBQUssQ0FBQyxpQkFBaUIsR0FBRyxLQUFLLENBQUMsT0FBTyxDQUFDLE1BQU07b0JBQzVDLENBQUMsQ0FBQyxLQUFLLENBQUMsT0FBTyxDQUFDLE1BQU07b0JBQ3RCLENBQUMsQ0FBQyxLQUFLLENBQUMsaUJBQWlCO2FBQzlCLENBQUM7UUFDSixJQUFJLEtBQUssQ0FBQyxPQUFPLENBQUMsTUFBTSxHQUFHLENBQUMsRUFBRTtZQUM1QixPQUFPO2dCQUNMLElBQUksRUFBRSwrQkFBK0I7Z0JBQ3JDLEtBQUssRUFBRSxTQUFTO2FBQ2pCLENBQUM7U0FDSDtRQUNELE9BQU87WUFDTCxJQUFJLEVBQUUsRUFBRTtZQUNSLEtBQUssRUFBRSxTQUFTO1NBQ2pCLENBQUM7SUFDSixDQUFDOztnSEFoRlUsbUJBQW1CO29HQUFuQixtQkFBbUIsaUZDbEJoQyx3T0FNQTsyRkRZYSxtQkFBbUI7a0JBTi9CLFNBQVM7K0JBQ0Usb0JBQW9CLG1CQUdiLHVCQUF1QixDQUFDLE1BQU0iLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQsIENoYW5nZURldGVjdGlvblN0cmF0ZWd5IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBQb2xsLCBQb2xsU3RhdGUgfSBmcm9tICdzdHJlYW0tY2hhdCc7XG5pbXBvcnQgeyBCYXNlUG9sbENvbXBvbmVudCB9IGZyb20gJy4uL2Jhc2UtcG9sbC5jb21wb25lbnQnO1xuXG50eXBlIFNlbGVjdGlvbkluc3RydWN0aW9ucyA9IHtcbiAgdGV4dDogc3RyaW5nO1xuICBjb3VudDogbnVtYmVyIHwgdW5kZWZpbmVkO1xufTtcblxuLyoqXG4gKlxuICovXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdzdHJlYW0tcG9sbC1oZWFkZXInLFxuICB0ZW1wbGF0ZVVybDogJy4vcG9sbC1oZWFkZXIuY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZXM6IFtdLFxuICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaCxcbn0pXG5leHBvcnQgY2xhc3MgUG9sbEhlYWRlckNvbXBvbmVudCBleHRlbmRzIEJhc2VQb2xsQ29tcG9uZW50IHtcbiAgbmFtZSA9ICcnO1xuICBzZWxlY3Rpb25JbnN0cnVjdGlvbnM6IFNlbGVjdGlvbkluc3RydWN0aW9ucyA9IHtcbiAgICB0ZXh0OiAnJyxcbiAgICBjb3VudDogdW5kZWZpbmVkLFxuICB9O1xuXG4gIHByb3RlY3RlZCBzdGF0ZVN0b3JlU2VsZWN0b3IoXG4gICAgcG9sbDogUG9sbCxcbiAgICBtYXJrRm9yQ2hlY2s6ICgpID0+IHZvaWRcbiAgKTogKCkgPT4gdm9pZCB7XG4gICAgY29uc3QgdW5zdWJzY3JpYmUgPSBwb2xsLnN0YXRlLnN1YnNjcmliZVdpdGhTZWxlY3RvcihcbiAgICAgIChzdGF0ZSkgPT4gKHtcbiAgICAgICAgbmFtZTogc3RhdGUubmFtZSxcbiAgICAgICAgaXNfY2xvc2VkOiBzdGF0ZS5pc19jbG9zZWQsXG4gICAgICAgIGVuZm9yY2VfdW5pcXVlX3ZvdGU6IHN0YXRlLmVuZm9yY2VfdW5pcXVlX3ZvdGUsXG4gICAgICAgIG1heF92b3Rlc19hbGxvd2VkOiBzdGF0ZS5tYXhfdm90ZXNfYWxsb3dlZCxcbiAgICAgICAgb3B0aW9uczogc3RhdGUub3B0aW9ucyxcbiAgICAgIH0pLFxuICAgICAgKHN0YXRlKSA9PiB7XG4gICAgICAgIGNvbnN0IG5hbWUgPSBzdGF0ZS5uYW1lO1xuICAgICAgICBjb25zdCBzZWxlY3Rpb25JbnN0cnVjdGlvbnMgPSB0aGlzLmdldFNlbGVjdGlvbkluc3RydWN0aW9ucyhzdGF0ZSk7XG5cbiAgICAgICAgbGV0IGNoYW5nZWQgPSBmYWxzZTtcbiAgICAgICAgaWYgKG5hbWUgIT09IHRoaXMubmFtZSkge1xuICAgICAgICAgIHRoaXMubmFtZSA9IG5hbWU7XG4gICAgICAgICAgY2hhbmdlZCA9IHRydWU7XG4gICAgICAgIH1cbiAgICAgICAgaWYgKFxuICAgICAgICAgIHNlbGVjdGlvbkluc3RydWN0aW9ucy50ZXh0ICE9PSB0aGlzLnNlbGVjdGlvbkluc3RydWN0aW9ucy50ZXh0IHx8XG4gICAgICAgICAgc2VsZWN0aW9uSW5zdHJ1Y3Rpb25zLmNvdW50ICE9PSB0aGlzLnNlbGVjdGlvbkluc3RydWN0aW9ucy5jb3VudFxuICAgICAgICApIHtcbiAgICAgICAgICB0aGlzLnNlbGVjdGlvbkluc3RydWN0aW9ucyA9IHNlbGVjdGlvbkluc3RydWN0aW9ucztcbiAgICAgICAgICBjaGFuZ2VkID0gdHJ1ZTtcbiAgICAgICAgfVxuXG4gICAgICAgIGlmIChjaGFuZ2VkKSB7XG4gICAgICAgICAgbWFya0ZvckNoZWNrKCk7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICApO1xuXG4gICAgcmV0dXJuIHVuc3Vic2NyaWJlO1xuICB9XG5cbiAgZ2V0U2VsZWN0aW9uSW5zdHJ1Y3Rpb25zKFxuICAgIHN0YXRlOiBQaWNrPFxuICAgICAgUG9sbFN0YXRlLFxuICAgICAgJ2lzX2Nsb3NlZCcgfCAnZW5mb3JjZV91bmlxdWVfdm90ZScgfCAnbWF4X3ZvdGVzX2FsbG93ZWQnIHwgJ29wdGlvbnMnXG4gICAgPlxuICApOiBTZWxlY3Rpb25JbnN0cnVjdGlvbnMge1xuICAgIGlmIChzdGF0ZS5pc19jbG9zZWQpXG4gICAgICByZXR1cm4ge1xuICAgICAgICB0ZXh0OiAnc3RyZWFtQ2hhdC5Wb3RlIGVuZGVkJyxcbiAgICAgICAgY291bnQ6IHVuZGVmaW5lZCxcbiAgICAgIH07XG4gICAgaWYgKHN0YXRlLmVuZm9yY2VfdW5pcXVlX3ZvdGUgfHwgc3RhdGUub3B0aW9ucy5sZW5ndGggPT09IDEpIHtcbiAgICAgIHJldHVybiB7XG4gICAgICAgIHRleHQ6ICdzdHJlYW1DaGF0LlNlbGVjdCBvbmUnLFxuICAgICAgICBjb3VudDogdW5kZWZpbmVkLFxuICAgICAgfTtcbiAgICB9XG4gICAgaWYgKHN0YXRlLm1heF92b3Rlc19hbGxvd2VkKVxuICAgICAgcmV0dXJuIHtcbiAgICAgICAgdGV4dDogJ3N0cmVhbUNoYXQuU2VsZWN0IHVwIHRvIHt7Y291bnR9fScsXG4gICAgICAgIGNvdW50OlxuICAgICAgICAgIHN0YXRlLm1heF92b3Rlc19hbGxvd2VkID4gc3RhdGUub3B0aW9ucy5sZW5ndGhcbiAgICAgICAgICAgID8gc3RhdGUub3B0aW9ucy5sZW5ndGhcbiAgICAgICAgICAgIDogc3RhdGUubWF4X3ZvdGVzX2FsbG93ZWQsXG4gICAgICB9O1xuICAgIGlmIChzdGF0ZS5vcHRpb25zLmxlbmd0aCA+IDEpIHtcbiAgICAgIHJldHVybiB7XG4gICAgICAgIHRleHQ6ICdzdHJlYW1DaGF0LlNlbGVjdCBvbmUgb3IgbW9yZScsXG4gICAgICAgIGNvdW50OiB1bmRlZmluZWQsXG4gICAgICB9O1xuICAgIH1cbiAgICByZXR1cm4ge1xuICAgICAgdGV4dDogJycsXG4gICAgICBjb3VudDogdW5kZWZpbmVkLFxuICAgIH07XG4gIH1cbn1cbiIsIjxkaXYgY2xhc3M9XCJzdHItY2hhdF9fcG9sbC1oZWFkZXJcIj5cbiAgPGRpdiBjbGFzcz1cInN0ci1jaGF0X19wb2xsLXRpdGxlXCI+e3sgbmFtZSB9fTwvZGl2PlxuICA8ZGl2IGNsYXNzPVwic3RyLWNoYXRfX3BvbGwtc3VidGl0bGVcIj5cbiAgICB7eyBzZWxlY3Rpb25JbnN0cnVjdGlvbnMudGV4dCB8IHRyYW5zbGF0ZSA6IHNlbGVjdGlvbkluc3RydWN0aW9ucyB9fVxuICA8L2Rpdj5cbjwvZGl2PlxuIl19
|
|
@@ -0,0 +1,137 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, Input, } from '@angular/core';
|
|
2
|
+
import { BasePollComponent } from '../base-poll.component';
|
|
3
|
+
import { isVoteAnswer, } from 'stream-chat';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
import * as i1 from "@angular/common";
|
|
6
|
+
import * as i2 from "../../avatar-placeholder/avatar-placeholder.component";
|
|
7
|
+
import * as i3 from "@ngx-translate/core";
|
|
8
|
+
/**
|
|
9
|
+
*
|
|
10
|
+
*/
|
|
11
|
+
export class PollOptionSelectorComponent extends BasePollComponent {
|
|
12
|
+
constructor() {
|
|
13
|
+
super(...arguments);
|
|
14
|
+
this.displayAvatarCount = 3;
|
|
15
|
+
this.voteCountVerbose = false;
|
|
16
|
+
this.isClosed = false;
|
|
17
|
+
this.latestVotes = [];
|
|
18
|
+
this.isWinner = false;
|
|
19
|
+
this.voteCount = 0;
|
|
20
|
+
this.winningOptionCount = 0;
|
|
21
|
+
this.maxVoteAllowedCount = 0;
|
|
22
|
+
this.ownVoteCount = 0;
|
|
23
|
+
}
|
|
24
|
+
ngOnChanges(changes) {
|
|
25
|
+
super.ngOnChanges(changes);
|
|
26
|
+
if (changes['option']) {
|
|
27
|
+
this.setupStateStoreSelector();
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
async toggleVote() {
|
|
31
|
+
if (!this.canVote || !this.option?.id || !this.messageId || this.isClosed)
|
|
32
|
+
return;
|
|
33
|
+
const haveVotedForTheOption = !!this.ownVote;
|
|
34
|
+
if (!haveVotedForTheOption &&
|
|
35
|
+
this.maxVoteAllowedCount > 0 &&
|
|
36
|
+
this.ownVoteCount >= this.maxVoteAllowedCount) {
|
|
37
|
+
this.addNotification('streamChat.You have reached the maximum number of votes allowed');
|
|
38
|
+
return;
|
|
39
|
+
}
|
|
40
|
+
try {
|
|
41
|
+
await (haveVotedForTheOption
|
|
42
|
+
? this.poll?.removeVote(this.ownVote.id, this.messageId)
|
|
43
|
+
: this.poll?.castVote(this.option?.id, this.messageId));
|
|
44
|
+
}
|
|
45
|
+
catch (error) {
|
|
46
|
+
this.notificationService.addTemporaryNotification('streamChat.Failed to cast vote');
|
|
47
|
+
throw error;
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
trackByVoteId(_, vote) {
|
|
51
|
+
return vote.id;
|
|
52
|
+
}
|
|
53
|
+
stateStoreSelector(poll, markForCheck) {
|
|
54
|
+
const unsubscribe = poll.state.subscribeWithSelector((nextValue) => {
|
|
55
|
+
return {
|
|
56
|
+
is_closed: nextValue.is_closed,
|
|
57
|
+
latest_votes_by_option: nextValue.latest_votes_by_option,
|
|
58
|
+
maxVotedOptionIds: nextValue.maxVotedOptionIds,
|
|
59
|
+
ownVotesByOptionId: nextValue.ownVotesByOptionId,
|
|
60
|
+
vote_counts_by_option: nextValue.vote_counts_by_option,
|
|
61
|
+
voting_visibility: nextValue.voting_visibility,
|
|
62
|
+
max_votes_allowed: nextValue.max_votes_allowed,
|
|
63
|
+
};
|
|
64
|
+
}, (state) => {
|
|
65
|
+
const isClosed = state.is_closed;
|
|
66
|
+
const latestVotes = state.latest_votes_by_option[this.option?.id ?? '']?.filter((vote) => !!vote.user && !isVoteAnswer(vote)) ?? [];
|
|
67
|
+
const isWinner = state.maxVotedOptionIds.includes(this.option?.id ?? '') &&
|
|
68
|
+
state.maxVotedOptionIds.length === 1;
|
|
69
|
+
const ownVote = state.ownVotesByOptionId[this.option?.id ?? ''];
|
|
70
|
+
const voteCount = state.vote_counts_by_option[this.option?.id ?? ''];
|
|
71
|
+
const votingVisibility = state.voting_visibility;
|
|
72
|
+
const winningOptionCount = state.vote_counts_by_option[state.maxVotedOptionIds?.[0] ?? ''];
|
|
73
|
+
const maxVoteAllowedCount = state.max_votes_allowed;
|
|
74
|
+
const ownVoteCount = Object.keys(state.ownVotesByOptionId).length;
|
|
75
|
+
let changed = false;
|
|
76
|
+
if (isClosed !== this.isClosed) {
|
|
77
|
+
this.isClosed = isClosed ?? false;
|
|
78
|
+
changed = true;
|
|
79
|
+
}
|
|
80
|
+
if (latestVotes !== this.latestVotes) {
|
|
81
|
+
this.latestVotes = latestVotes ?? [];
|
|
82
|
+
changed = true;
|
|
83
|
+
}
|
|
84
|
+
if (isWinner !== this.isWinner) {
|
|
85
|
+
this.isWinner = isWinner ?? false;
|
|
86
|
+
changed = true;
|
|
87
|
+
}
|
|
88
|
+
if (ownVote !== this.ownVote) {
|
|
89
|
+
this.ownVote = ownVote ?? undefined;
|
|
90
|
+
changed = true;
|
|
91
|
+
}
|
|
92
|
+
if (voteCount !== this.voteCount) {
|
|
93
|
+
this.voteCount = voteCount ?? 0;
|
|
94
|
+
changed = true;
|
|
95
|
+
}
|
|
96
|
+
if (votingVisibility !== this.votingVisibility) {
|
|
97
|
+
this.votingVisibility = votingVisibility ?? undefined;
|
|
98
|
+
changed = true;
|
|
99
|
+
}
|
|
100
|
+
if (winningOptionCount !== this.winningOptionCount) {
|
|
101
|
+
this.winningOptionCount = winningOptionCount ?? 0;
|
|
102
|
+
changed = true;
|
|
103
|
+
}
|
|
104
|
+
if (maxVoteAllowedCount !== this.maxVoteAllowedCount) {
|
|
105
|
+
this.maxVoteAllowedCount = maxVoteAllowedCount ?? 0;
|
|
106
|
+
changed = true;
|
|
107
|
+
}
|
|
108
|
+
if (ownVoteCount !== this.ownVoteCount) {
|
|
109
|
+
this.ownVoteCount = ownVoteCount ?? 0;
|
|
110
|
+
changed = true;
|
|
111
|
+
}
|
|
112
|
+
if (this.dismissNotificationFn &&
|
|
113
|
+
this.maxVoteAllowedCount > 0 &&
|
|
114
|
+
this.ownVoteCount <= this.maxVoteAllowedCount) {
|
|
115
|
+
this.dismissNotificationFn();
|
|
116
|
+
changed = true;
|
|
117
|
+
}
|
|
118
|
+
if (changed) {
|
|
119
|
+
markForCheck();
|
|
120
|
+
}
|
|
121
|
+
});
|
|
122
|
+
return unsubscribe;
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
PollOptionSelectorComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: PollOptionSelectorComponent, deps: null, target: i0.ɵɵFactoryTarget.Component });
|
|
126
|
+
PollOptionSelectorComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.0.4", type: PollOptionSelectorComponent, selector: "stream-poll-option-selector", inputs: { option: "option", displayAvatarCount: "displayAvatarCount", voteCountVerbose: "voteCountVerbose" }, usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<div\n class=\"str-chat__poll-option\"\n [class.str-chat__poll-option--votable]=\"\n !isClosed &&\n canVote &&\n maxVoteAllowedCount &&\n ownVoteCount < maxVoteAllowedCount &&\n !ownVote\n \"\n (click)=\"toggleVote()\"\n (keyup.enter)=\"toggleVote()\"\n>\n <ng-container *ngIf=\"canVote\">\n <ng-container *ngTemplateOutlet=\"checkmark\"></ng-container>\n </ng-container>\n <div class=\"str-chat__poll-option-data\">\n <p class=\"str-chat__poll-option-text\">{{ option?.text }}</p>\n <ng-container *ngIf=\"displayAvatarCount && votingVisibility === 'public'\">\n <div\n *ngFor=\"let vote of latestVotes; trackBy: trackByVoteId\"\n class=\"str-chat__poll-option-voters\"\n >\n <stream-avatar-placeholder\n location=\"poll-voter\"\n [user]=\"vote.user\"\n [imageUrl]=\"vote.user?.image\"\n [name]=\"vote.user?.name\"\n type=\"user\"\n ></stream-avatar-placeholder>\n </div>\n </ng-container>\n <div class=\"str-chat__poll-option-vote-count\">\n <ng-container *ngIf=\"voteCountVerbose; else count\">\n {{ 'streamChat.{{ count }} votes' | translate:{count: voteCount} }}\n </ng-container>\n <ng-template #count>\n {{ voteCount }}\n </ng-template>\n </div>\n </div>\n <ng-container *ngTemplateOutlet=\"amountBar\"></ng-container>\n</div>\n\n<ng-template #checkmark>\n <div\n class=\"str-chat__checkmark\"\n [class.str-chat__checkmark--checked]=\"!!ownVote\"\n ></div>\n</ng-template>\n\n<ng-template #amountBar>\n <div\n class=\"str-chat__amount-bar str-chat__poll-option__votes-bar\"\n [class.str-chat__poll-option__votes-bar--winner]=\"isClosed && isWinner\"\n role=\"progressbar\"\n [ngStyle]=\"{\n '--str-chat__amount-bar-fulfillment':\n (winningOptionCount && voteCount / winningOptionCount) * 100 + '%'\n }\"\n ></div>\n</ng-template>\n", dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "component", type: i2.AvatarPlaceholderComponent, selector: "stream-avatar-placeholder", inputs: ["name", "imageUrl", "location", "channel", "user", "type", "initialsType", "showOnlineIndicator"] }, { kind: "pipe", type: i3.TranslatePipe, name: "translate" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
127
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: PollOptionSelectorComponent, decorators: [{
|
|
128
|
+
type: Component,
|
|
129
|
+
args: [{ selector: 'stream-poll-option-selector', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div\n class=\"str-chat__poll-option\"\n [class.str-chat__poll-option--votable]=\"\n !isClosed &&\n canVote &&\n maxVoteAllowedCount &&\n ownVoteCount < maxVoteAllowedCount &&\n !ownVote\n \"\n (click)=\"toggleVote()\"\n (keyup.enter)=\"toggleVote()\"\n>\n <ng-container *ngIf=\"canVote\">\n <ng-container *ngTemplateOutlet=\"checkmark\"></ng-container>\n </ng-container>\n <div class=\"str-chat__poll-option-data\">\n <p class=\"str-chat__poll-option-text\">{{ option?.text }}</p>\n <ng-container *ngIf=\"displayAvatarCount && votingVisibility === 'public'\">\n <div\n *ngFor=\"let vote of latestVotes; trackBy: trackByVoteId\"\n class=\"str-chat__poll-option-voters\"\n >\n <stream-avatar-placeholder\n location=\"poll-voter\"\n [user]=\"vote.user\"\n [imageUrl]=\"vote.user?.image\"\n [name]=\"vote.user?.name\"\n type=\"user\"\n ></stream-avatar-placeholder>\n </div>\n </ng-container>\n <div class=\"str-chat__poll-option-vote-count\">\n <ng-container *ngIf=\"voteCountVerbose; else count\">\n {{ 'streamChat.{{ count }} votes' | translate:{count: voteCount} }}\n </ng-container>\n <ng-template #count>\n {{ voteCount }}\n </ng-template>\n </div>\n </div>\n <ng-container *ngTemplateOutlet=\"amountBar\"></ng-container>\n</div>\n\n<ng-template #checkmark>\n <div\n class=\"str-chat__checkmark\"\n [class.str-chat__checkmark--checked]=\"!!ownVote\"\n ></div>\n</ng-template>\n\n<ng-template #amountBar>\n <div\n class=\"str-chat__amount-bar str-chat__poll-option__votes-bar\"\n [class.str-chat__poll-option__votes-bar--winner]=\"isClosed && isWinner\"\n role=\"progressbar\"\n [ngStyle]=\"{\n '--str-chat__amount-bar-fulfillment':\n (winningOptionCount && voteCount / winningOptionCount) * 100 + '%'\n }\"\n ></div>\n</ng-template>\n" }]
|
|
130
|
+
}], propDecorators: { option: [{
|
|
131
|
+
type: Input
|
|
132
|
+
}], displayAvatarCount: [{
|
|
133
|
+
type: Input
|
|
134
|
+
}], voteCountVerbose: [{
|
|
135
|
+
type: Input
|
|
136
|
+
}] } });
|
|
137
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"poll-option-selector.component.js","sourceRoot":"","sources":["../../../../../../projects/stream-chat-angular/src/lib/polls/poll-option-selector/poll-option-selector.component.ts","../../../../../../projects/stream-chat-angular/src/lib/polls/poll-option-selector/poll-option-selector.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  Component,\n  Input,\n  OnChanges,\n  SimpleChanges,\n} from '@angular/core';\nimport { BasePollComponent } from '../base-poll.component';\nimport {\n  isVoteAnswer,\n  Poll,\n  PollOption,\n  PollVote,\n  VotingVisibility,\n} from 'stream-chat';\n\n/**\n *\n */\n@Component({\n  selector: 'stream-poll-option-selector',\n  templateUrl: './poll-option-selector.component.html',\n  styles: [],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class PollOptionSelectorComponent\n  extends BasePollComponent\n  implements OnChanges\n{\n  @Input() option: PollOption | undefined;\n  @Input() displayAvatarCount = 3;\n  @Input() voteCountVerbose = false;\n  isClosed = false;\n  latestVotes: PollVote[] = [];\n  isWinner = false;\n  ownVote: PollVote | undefined;\n  voteCount = 0;\n  votingVisibility: VotingVisibility | undefined;\n  winningOptionCount = 0;\n  maxVoteAllowedCount = 0;\n  ownVoteCount = 0;\n\n  ngOnChanges(changes: SimpleChanges): void {\n    super.ngOnChanges(changes);\n    if (changes['option']) {\n      this.setupStateStoreSelector();\n    }\n  }\n\n  async toggleVote() {\n    if (!this.canVote || !this.option?.id || !this.messageId || this.isClosed)\n      return;\n    const haveVotedForTheOption = !!this.ownVote;\n    if (\n      !haveVotedForTheOption &&\n      this.maxVoteAllowedCount > 0 &&\n      this.ownVoteCount >= this.maxVoteAllowedCount\n    ) {\n      this.addNotification(\n        'streamChat.You have reached the maximum number of votes allowed'\n      );\n      return;\n    }\n    try {\n      await (haveVotedForTheOption\n        ? this.poll?.removeVote(this.ownVote!.id, this.messageId)\n        : this.poll?.castVote(this.option?.id, this.messageId));\n    } catch (error) {\n      this.notificationService.addTemporaryNotification(\n        'streamChat.Failed to cast vote'\n      );\n      throw error;\n    }\n  }\n\n  trackByVoteId(_: number, vote: PollVote) {\n    return vote.id;\n  }\n\n  protected stateStoreSelector(\n    poll: Poll,\n    markForCheck: () => void\n  ): () => void {\n    const unsubscribe = poll.state.subscribeWithSelector(\n      (nextValue) => {\n        return {\n          is_closed: nextValue.is_closed,\n          latest_votes_by_option: nextValue.latest_votes_by_option,\n          maxVotedOptionIds: nextValue.maxVotedOptionIds,\n          ownVotesByOptionId: nextValue.ownVotesByOptionId,\n          vote_counts_by_option: nextValue.vote_counts_by_option,\n          voting_visibility: nextValue.voting_visibility,\n          max_votes_allowed: nextValue.max_votes_allowed,\n        };\n      },\n      (state) => {\n        const isClosed = state.is_closed;\n        const latestVotes =\n          state.latest_votes_by_option[this.option?.id ?? '']?.filter(\n            (vote) => !!vote.user && !isVoteAnswer(vote)\n          ) ?? [];\n        const isWinner =\n          state.maxVotedOptionIds.includes(this.option?.id ?? '') &&\n          state.maxVotedOptionIds.length === 1;\n        const ownVote = state.ownVotesByOptionId[this.option?.id ?? ''];\n        const voteCount = state.vote_counts_by_option[this.option?.id ?? ''];\n        const votingVisibility = state.voting_visibility;\n        const winningOptionCount =\n          state.vote_counts_by_option[state.maxVotedOptionIds?.[0] ?? ''];\n        const maxVoteAllowedCount = state.max_votes_allowed;\n        const ownVoteCount = Object.keys(state.ownVotesByOptionId).length;\n\n        let changed = false;\n        if (isClosed !== this.isClosed) {\n          this.isClosed = isClosed ?? false;\n          changed = true;\n        }\n        if (latestVotes !== this.latestVotes) {\n          this.latestVotes = latestVotes ?? [];\n          changed = true;\n        }\n        if (isWinner !== this.isWinner) {\n          this.isWinner = isWinner ?? false;\n          changed = true;\n        }\n        if (ownVote !== this.ownVote) {\n          this.ownVote = ownVote ?? undefined;\n          changed = true;\n        }\n        if (voteCount !== this.voteCount) {\n          this.voteCount = voteCount ?? 0;\n          changed = true;\n        }\n        if (votingVisibility !== this.votingVisibility) {\n          this.votingVisibility = votingVisibility ?? undefined;\n          changed = true;\n        }\n        if (winningOptionCount !== this.winningOptionCount) {\n          this.winningOptionCount = winningOptionCount ?? 0;\n          changed = true;\n        }\n        if (maxVoteAllowedCount !== this.maxVoteAllowedCount) {\n          this.maxVoteAllowedCount = maxVoteAllowedCount ?? 0;\n          changed = true;\n        }\n        if (ownVoteCount !== this.ownVoteCount) {\n          this.ownVoteCount = ownVoteCount ?? 0;\n          changed = true;\n        }\n\n        if (\n          this.dismissNotificationFn &&\n          this.maxVoteAllowedCount > 0 &&\n          this.ownVoteCount <= this.maxVoteAllowedCount\n        ) {\n          this.dismissNotificationFn();\n          changed = true;\n        }\n\n        if (changed) {\n          markForCheck();\n        }\n      }\n    );\n\n    return unsubscribe;\n  }\n}\n","<div\n  class=\"str-chat__poll-option\"\n  [class.str-chat__poll-option--votable]=\"\n    !isClosed &&\n    canVote &&\n    maxVoteAllowedCount &&\n    ownVoteCount < maxVoteAllowedCount &&\n    !ownVote\n  \"\n  (click)=\"toggleVote()\"\n  (keyup.enter)=\"toggleVote()\"\n>\n  <ng-container *ngIf=\"canVote\">\n    <ng-container *ngTemplateOutlet=\"checkmark\"></ng-container>\n  </ng-container>\n  <div class=\"str-chat__poll-option-data\">\n    <p class=\"str-chat__poll-option-text\">{{ option?.text }}</p>\n    <ng-container *ngIf=\"displayAvatarCount && votingVisibility === 'public'\">\n      <div\n        *ngFor=\"let vote of latestVotes; trackBy: trackByVoteId\"\n        class=\"str-chat__poll-option-voters\"\n      >\n        <stream-avatar-placeholder\n          location=\"poll-voter\"\n          [user]=\"vote.user\"\n          [imageUrl]=\"vote.user?.image\"\n          [name]=\"vote.user?.name\"\n          type=\"user\"\n        ></stream-avatar-placeholder>\n      </div>\n    </ng-container>\n    <div class=\"str-chat__poll-option-vote-count\">\n      <ng-container *ngIf=\"voteCountVerbose; else count\">\n        {{ 'streamChat.{{ count }} votes' | translate:{count: voteCount} }}\n      </ng-container>\n      <ng-template #count>\n        {{ voteCount }}\n      </ng-template>\n    </div>\n  </div>\n  <ng-container *ngTemplateOutlet=\"amountBar\"></ng-container>\n</div>\n\n<ng-template #checkmark>\n  <div\n    class=\"str-chat__checkmark\"\n    [class.str-chat__checkmark--checked]=\"!!ownVote\"\n  ></div>\n</ng-template>\n\n<ng-template #amountBar>\n  <div\n    class=\"str-chat__amount-bar str-chat__poll-option__votes-bar\"\n    [class.str-chat__poll-option__votes-bar--winner]=\"isClosed && isWinner\"\n    role=\"progressbar\"\n    [ngStyle]=\"{\n      '--str-chat__amount-bar-fulfillment':\n        (winningOptionCount && voteCount / winningOptionCount) * 100 + '%'\n    }\"\n  ></div>\n</ng-template>\n"]}
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component, Input } from '@angular/core';
|
|
2
|
+
import { BasePollComponent } from '../base-poll.component';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "@angular/common";
|
|
5
|
+
import * as i2 from "../poll-option-selector/poll-option-selector.component";
|
|
6
|
+
/**
|
|
7
|
+
*
|
|
8
|
+
*/
|
|
9
|
+
export class PollOptionsListComponent extends BasePollComponent {
|
|
10
|
+
constructor() {
|
|
11
|
+
super(...arguments);
|
|
12
|
+
/**
|
|
13
|
+
* How many options should be displayed. If there are more options than this number, use the poll actions to display all options
|
|
14
|
+
*/
|
|
15
|
+
this.maxOptionsDisplayed = 10;
|
|
16
|
+
this.options = [];
|
|
17
|
+
}
|
|
18
|
+
stateStoreSelector(poll, markForCheck) {
|
|
19
|
+
const unsubscribe = poll.state.subscribeWithSelector((state) => ({
|
|
20
|
+
options: state.options,
|
|
21
|
+
}), (state) => {
|
|
22
|
+
this.options = state.options;
|
|
23
|
+
markForCheck();
|
|
24
|
+
});
|
|
25
|
+
return unsubscribe;
|
|
26
|
+
}
|
|
27
|
+
trackByOptionId(_, option) {
|
|
28
|
+
return option.id;
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
PollOptionsListComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: PollOptionsListComponent, deps: null, target: i0.ɵɵFactoryTarget.Component });
|
|
32
|
+
PollOptionsListComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.0.4", type: PollOptionsListComponent, selector: "stream-poll-options-list", inputs: { maxOptionsDisplayed: "maxOptionsDisplayed" }, usesInheritance: true, ngImport: i0, template: "<div\n class=\"str-chat__poll-option-list\"\n [class.str-chat__poll-option-list--full]=\"\n !maxOptionsDisplayed || maxOptionsDisplayed === options.length\n \"\n>\n <ng-container\n *ngFor=\"\n let option of options | slice : 0 : maxOptionsDisplayed;\n trackBy: trackByOptionId\n \"\n >\n <stream-poll-option-selector\n [option]=\"option\"\n [messageId]=\"messageId\"\n [pollId]=\"pollId\"\n ></stream-poll-option-selector>\n </ng-container>\n</div>\n", dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "component", type: i2.PollOptionSelectorComponent, selector: "stream-poll-option-selector", inputs: ["option", "displayAvatarCount", "voteCountVerbose"] }, { kind: "pipe", type: i1.SlicePipe, name: "slice" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
33
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: PollOptionsListComponent, decorators: [{
|
|
34
|
+
type: Component,
|
|
35
|
+
args: [{ selector: 'stream-poll-options-list', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div\n class=\"str-chat__poll-option-list\"\n [class.str-chat__poll-option-list--full]=\"\n !maxOptionsDisplayed || maxOptionsDisplayed === options.length\n \"\n>\n <ng-container\n *ngFor=\"\n let option of options | slice : 0 : maxOptionsDisplayed;\n trackBy: trackByOptionId\n \"\n >\n <stream-poll-option-selector\n [option]=\"option\"\n [messageId]=\"messageId\"\n [pollId]=\"pollId\"\n ></stream-poll-option-selector>\n </ng-container>\n</div>\n" }]
|
|
36
|
+
}], propDecorators: { maxOptionsDisplayed: [{
|
|
37
|
+
type: Input
|
|
38
|
+
}] } });
|
|
39
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicG9sbC1vcHRpb25zLWxpc3QuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvc3RyZWFtLWNoYXQtYW5ndWxhci9zcmMvbGliL3BvbGxzL3BvbGwtb3B0aW9ucy1saXN0L3BvbGwtb3B0aW9ucy1saXN0LmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3N0cmVhbS1jaGF0LWFuZ3VsYXIvc3JjL2xpYi9wb2xscy9wb2xsLW9wdGlvbnMtbGlzdC9wb2xsLW9wdGlvbnMtbGlzdC5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsdUJBQXVCLEVBQUUsU0FBUyxFQUFFLEtBQUssRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUMxRSxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQzs7OztBQUczRDs7R0FFRztBQU9ILE1BQU0sT0FBTyx3QkFBeUIsU0FBUSxpQkFBaUI7SUFOL0Q7O1FBT0U7O1dBRUc7UUFDTSx3QkFBbUIsR0FBdUIsRUFBRSxDQUFDO1FBQ3RELFlBQU8sR0FBaUIsRUFBRSxDQUFDO0tBc0I1QjtJQXBCVyxrQkFBa0IsQ0FDMUIsSUFBVSxFQUNWLFlBQXdCO1FBRXhCLE1BQU0sV0FBVyxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMscUJBQXFCLENBQ2xELENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxDQUFDO1lBQ1YsT0FBTyxFQUFFLEtBQUssQ0FBQyxPQUFPO1NBQ3ZCLENBQUMsRUFDRixDQUFDLEtBQUssRUFBRSxFQUFFO1lBQ1IsSUFBSSxDQUFDLE9BQU8sR0FBRyxLQUFLLENBQUMsT0FBTyxDQUFDO1lBQzdCLFlBQVksRUFBRSxDQUFDO1FBQ2pCLENBQUMsQ0FDRixDQUFDO1FBRUYsT0FBTyxXQUFXLENBQUM7SUFDckIsQ0FBQztJQUVELGVBQWUsQ0FBQyxDQUFTLEVBQUUsTUFBa0I7UUFDM0MsT0FBTyxNQUFNLENBQUMsRUFBRSxDQUFDO0lBQ25CLENBQUM7O3FIQTFCVSx3QkFBd0I7eUdBQXhCLHdCQUF3QiwrSUNickMsa2ZBbUJBOzJGRE5hLHdCQUF3QjtrQkFOcEMsU0FBUzsrQkFDRSwwQkFBMEIsbUJBR25CLHVCQUF1QixDQUFDLE1BQU07OEJBTXRDLG1CQUFtQjtzQkFBM0IsS0FBSyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENoYW5nZURldGVjdGlvblN0cmF0ZWd5LCBDb21wb25lbnQsIElucHV0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBCYXNlUG9sbENvbXBvbmVudCB9IGZyb20gJy4uL2Jhc2UtcG9sbC5jb21wb25lbnQnO1xuaW1wb3J0IHsgUG9sbCwgUG9sbE9wdGlvbiB9IGZyb20gJ3N0cmVhbS1jaGF0JztcblxuLyoqXG4gKlxuICovXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdzdHJlYW0tcG9sbC1vcHRpb25zLWxpc3QnLFxuICB0ZW1wbGF0ZVVybDogJy4vcG9sbC1vcHRpb25zLWxpc3QuY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZXM6IFtdLFxuICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaCxcbn0pXG5leHBvcnQgY2xhc3MgUG9sbE9wdGlvbnNMaXN0Q29tcG9uZW50IGV4dGVuZHMgQmFzZVBvbGxDb21wb25lbnQge1xuICAvKipcbiAgICogSG93IG1hbnkgb3B0aW9ucyBzaG91bGQgYmUgZGlzcGxheWVkLiBJZiB0aGVyZSBhcmUgbW9yZSBvcHRpb25zIHRoYW4gdGhpcyBudW1iZXIsIHVzZSB0aGUgcG9sbCBhY3Rpb25zIHRvIGRpc3BsYXkgYWxsIG9wdGlvbnNcbiAgICovXG4gIEBJbnB1dCgpIG1heE9wdGlvbnNEaXNwbGF5ZWQ6IG51bWJlciB8IHVuZGVmaW5lZCA9IDEwO1xuICBvcHRpb25zOiBQb2xsT3B0aW9uW10gPSBbXTtcblxuICBwcm90ZWN0ZWQgc3RhdGVTdG9yZVNlbGVjdG9yKFxuICAgIHBvbGw6IFBvbGwsXG4gICAgbWFya0ZvckNoZWNrOiAoKSA9PiB2b2lkXG4gICk6ICgpID0+IHZvaWQge1xuICAgIGNvbnN0IHVuc3Vic2NyaWJlID0gcG9sbC5zdGF0ZS5zdWJzY3JpYmVXaXRoU2VsZWN0b3IoXG4gICAgICAoc3RhdGUpID0+ICh7XG4gICAgICAgIG9wdGlvbnM6IHN0YXRlLm9wdGlvbnMsXG4gICAgICB9KSxcbiAgICAgIChzdGF0ZSkgPT4ge1xuICAgICAgICB0aGlzLm9wdGlvbnMgPSBzdGF0ZS5vcHRpb25zO1xuICAgICAgICBtYXJrRm9yQ2hlY2soKTtcbiAgICAgIH1cbiAgICApO1xuXG4gICAgcmV0dXJuIHVuc3Vic2NyaWJlO1xuICB9XG5cbiAgdHJhY2tCeU9wdGlvbklkKF86IG51bWJlciwgb3B0aW9uOiBQb2xsT3B0aW9uKSB7XG4gICAgcmV0dXJuIG9wdGlvbi5pZDtcbiAgfVxufVxuIiwiPGRpdlxuICBjbGFzcz1cInN0ci1jaGF0X19wb2xsLW9wdGlvbi1saXN0XCJcbiAgW2NsYXNzLnN0ci1jaGF0X19wb2xsLW9wdGlvbi1saXN0LS1mdWxsXT1cIlxuICAgICFtYXhPcHRpb25zRGlzcGxheWVkIHx8IG1heE9wdGlvbnNEaXNwbGF5ZWQgPT09IG9wdGlvbnMubGVuZ3RoXG4gIFwiXG4+XG4gIDxuZy1jb250YWluZXJcbiAgICAqbmdGb3I9XCJcbiAgICAgIGxldCBvcHRpb24gb2Ygb3B0aW9ucyB8IHNsaWNlIDogMCA6IG1heE9wdGlvbnNEaXNwbGF5ZWQ7XG4gICAgICB0cmFja0J5OiB0cmFja0J5T3B0aW9uSWRcbiAgICBcIlxuICA+XG4gICAgPHN0cmVhbS1wb2xsLW9wdGlvbi1zZWxlY3RvclxuICAgICAgW29wdGlvbl09XCJvcHRpb25cIlxuICAgICAgW21lc3NhZ2VJZF09XCJtZXNzYWdlSWRcIlxuICAgICAgW3BvbGxJZF09XCJwb2xsSWRcIlxuICAgID48L3N0cmVhbS1wb2xsLW9wdGlvbi1zZWxlY3Rvcj5cbiAgPC9uZy1jb250YWluZXI+XG48L2Rpdj5cbiJdfQ==
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
import { ChangeDetectionStrategy, Component } from '@angular/core';
|
|
2
|
+
import { BasePollComponent } from '../base-poll.component';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
/**
|
|
5
|
+
*
|
|
6
|
+
*/
|
|
7
|
+
export class PollPreviewComponent extends BasePollComponent {
|
|
8
|
+
constructor() {
|
|
9
|
+
super(...arguments);
|
|
10
|
+
this.name = '';
|
|
11
|
+
this.isClosed = false;
|
|
12
|
+
}
|
|
13
|
+
stateStoreSelector(poll, markForCheck) {
|
|
14
|
+
const unsubscribe = poll.state.subscribeWithSelector((state) => ({
|
|
15
|
+
name: state.name,
|
|
16
|
+
is_closed: state.is_closed,
|
|
17
|
+
}), (state) => {
|
|
18
|
+
this.name = state.name;
|
|
19
|
+
this.isClosed = state.is_closed ?? false;
|
|
20
|
+
markForCheck();
|
|
21
|
+
});
|
|
22
|
+
return unsubscribe;
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
PollPreviewComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: PollPreviewComponent, deps: null, target: i0.ɵɵFactoryTarget.Component });
|
|
26
|
+
PollPreviewComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.0.4", type: PollPreviewComponent, selector: "stream-poll-preview", usesInheritance: true, ngImport: i0, template: "<div\n class=\"str-chat__quoted-poll-preview\"\n [class.str-chat__quoted-poll-preview--closed]=\"isClosed\"\n>\n <div class=\"str-chat__quoted-poll-preview__icon\">\uD83D\uDCCA</div>\n <div class=\"str-chat__quoted-poll-preview__name\">{{ name }}</div>\n</div>\n", changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
27
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: PollPreviewComponent, decorators: [{
|
|
28
|
+
type: Component,
|
|
29
|
+
args: [{ selector: 'stream-poll-preview', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div\n class=\"str-chat__quoted-poll-preview\"\n [class.str-chat__quoted-poll-preview--closed]=\"isClosed\"\n>\n <div class=\"str-chat__quoted-poll-preview__icon\">\uD83D\uDCCA</div>\n <div class=\"str-chat__quoted-poll-preview__name\">{{ name }}</div>\n</div>\n" }]
|
|
30
|
+
}] });
|
|
31
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicG9sbC1wcmV2aWV3LmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3N0cmVhbS1jaGF0LWFuZ3VsYXIvc3JjL2xpYi9wb2xscy9wb2xsLXByZXZpZXcvcG9sbC1wcmV2aWV3LmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3N0cmVhbS1jaGF0LWFuZ3VsYXIvc3JjL2xpYi9wb2xscy9wb2xsLXByZXZpZXcvcG9sbC1wcmV2aWV3LmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSx1QkFBdUIsRUFBRSxTQUFTLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDbkUsT0FBTyxFQUFFLGlCQUFpQixFQUFFLE1BQU0sd0JBQXdCLENBQUM7O0FBRzNEOztHQUVHO0FBT0gsTUFBTSxPQUFPLG9CQUFxQixTQUFRLGlCQUFpQjtJQU4zRDs7UUFPRSxTQUFJLEdBQUcsRUFBRSxDQUFDO1FBQ1YsYUFBUSxHQUFHLEtBQUssQ0FBQztLQW9CbEI7SUFsQlcsa0JBQWtCLENBQzFCLElBQVUsRUFDVixZQUF3QjtRQUV4QixNQUFNLFdBQVcsR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLHFCQUFxQixDQUNsRCxDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsQ0FBQztZQUNWLElBQUksRUFBRSxLQUFLLENBQUMsSUFBSTtZQUNoQixTQUFTLEVBQUUsS0FBSyxDQUFDLFNBQVM7U0FDM0IsQ0FBQyxFQUNGLENBQUMsS0FBSyxFQUFFLEVBQUU7WUFDUixJQUFJLENBQUMsSUFBSSxHQUFHLEtBQUssQ0FBQyxJQUFJLENBQUM7WUFDdkIsSUFBSSxDQUFDLFFBQVEsR0FBRyxLQUFLLENBQUMsU0FBUyxJQUFJLEtBQUssQ0FBQztZQUN6QyxZQUFZLEVBQUUsQ0FBQztRQUNqQixDQUFDLENBQ0YsQ0FBQztRQUVGLE9BQU8sV0FBVyxDQUFDO0lBQ3JCLENBQUM7O2lIQXJCVSxvQkFBb0I7cUdBQXBCLG9CQUFvQixrRkNiakMsNFFBT0E7MkZETWEsb0JBQW9CO2tCQU5oQyxTQUFTOytCQUNFLHFCQUFxQixtQkFHZCx1QkFBdUIsQ0FBQyxNQUFNIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3ksIENvbXBvbmVudCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQmFzZVBvbGxDb21wb25lbnQgfSBmcm9tICcuLi9iYXNlLXBvbGwuY29tcG9uZW50JztcbmltcG9ydCB7IFBvbGwgfSBmcm9tICdzdHJlYW0tY2hhdCc7XG5cbi8qKlxuICpcbiAqL1xuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnc3RyZWFtLXBvbGwtcHJldmlldycsXG4gIHRlbXBsYXRlVXJsOiAnLi9wb2xsLXByZXZpZXcuY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZXM6IFtdLFxuICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaCxcbn0pXG5leHBvcnQgY2xhc3MgUG9sbFByZXZpZXdDb21wb25lbnQgZXh0ZW5kcyBCYXNlUG9sbENvbXBvbmVudCB7XG4gIG5hbWUgPSAnJztcbiAgaXNDbG9zZWQgPSBmYWxzZTtcblxuICBwcm90ZWN0ZWQgc3RhdGVTdG9yZVNlbGVjdG9yKFxuICAgIHBvbGw6IFBvbGwsXG4gICAgbWFya0ZvckNoZWNrOiAoKSA9PiB2b2lkXG4gICk6ICgpID0+IHZvaWQge1xuICAgIGNvbnN0IHVuc3Vic2NyaWJlID0gcG9sbC5zdGF0ZS5zdWJzY3JpYmVXaXRoU2VsZWN0b3IoXG4gICAgICAoc3RhdGUpID0+ICh7XG4gICAgICAgIG5hbWU6IHN0YXRlLm5hbWUsXG4gICAgICAgIGlzX2Nsb3NlZDogc3RhdGUuaXNfY2xvc2VkLFxuICAgICAgfSksXG4gICAgICAoc3RhdGUpID0+IHtcbiAgICAgICAgdGhpcy5uYW1lID0gc3RhdGUubmFtZTtcbiAgICAgICAgdGhpcy5pc0Nsb3NlZCA9IHN0YXRlLmlzX2Nsb3NlZCA/PyBmYWxzZTtcbiAgICAgICAgbWFya0ZvckNoZWNrKCk7XG4gICAgICB9XG4gICAgKTtcblxuICAgIHJldHVybiB1bnN1YnNjcmliZTtcbiAgfVxufVxuIiwiPGRpdlxuICBjbGFzcz1cInN0ci1jaGF0X19xdW90ZWQtcG9sbC1wcmV2aWV3XCJcbiAgW2NsYXNzLnN0ci1jaGF0X19xdW90ZWQtcG9sbC1wcmV2aWV3LS1jbG9zZWRdPVwiaXNDbG9zZWRcIlxuPlxuICA8ZGl2IGNsYXNzPVwic3RyLWNoYXRfX3F1b3RlZC1wb2xsLXByZXZpZXdfX2ljb25cIj7wn5OKPC9kaXY+XG4gIDxkaXYgY2xhc3M9XCJzdHItY2hhdF9fcXVvdGVkLXBvbGwtcHJldmlld19fbmFtZVwiPnt7IG5hbWUgfX08L2Rpdj5cbjwvZGl2PlxuIl19
|