Skip to content

Commit

Permalink
Merge branch 'master' of github.com:surveyjs/survey-library into issu…
Browse files Browse the repository at this point in the history
…e/6655-survey-backgroundImage

# Conflicts:
#	src/knockout/templates/index.html
  • Loading branch information
OlgaLarina committed Sep 4, 2023
2 parents 855cfcf + bb894ed commit 926604c
Show file tree
Hide file tree
Showing 151 changed files with 2,145 additions and 766 deletions.
23 changes: 11 additions & 12 deletions docs/handle-survey-results-continue-incomplete.md
Original file line number Diff line number Diff line change
@@ -1,17 +1,15 @@
# Continue an Incomplete Survey

Your respondents may not complete your survey in a single session. In this case, you can restore their answers from the previous session next time they get to the survey. Incomplete results can be loaded from your database or the browser's [localStorage](https://developer.mozilla.org/en-US/docs/Web/API/Window/localStorage).
Respondents may not complete your survey in a single session. In this case, you can restore their answers from the previous session next time they get to the survey. Incomplete results can be loaded from your database or the browser's [`localStorage`](https://developer.mozilla.org/en-US/docs/Web/API/Window/localStorage).

To save incomplete results, enable the Survey's [`sendResultOnPageNext`](/Documentation/Library?id=surveymodel#sendResultOnPageNext) property. With this setting, the Survey raises the [`onPartialSend`](/Documentation/Library?id=surveymodel#onPartialSend) event each time a respondent navigates to the next survey page. Handle this event to send incomplete results to your database or `localStorage`:
To save incomplete results, implement a function that sends them to your server or saves them in the `localStorage`. Call this function within `SurveyModel`'s [`onValueChanged`](https://surveyjs.io/form-library/documentation/api-reference/survey-data-model#onValueChanged) and [`onCurrentPageChanged`](https://surveyjs.io/form-library/documentation/api-reference/survey-data-model#onCurrentPageChanged) event handlers to save survey results when users change a question value or switch between pages. If you use the `localStorage`, you also need to delete survey results from it when the survey is completed. Handle the [`onComplete`](https://surveyjs.io/form-library/documentation/api-reference/survey-data-model#onComplete) event for this purpose.

```js
import { Model } from "survey-core";

const surveyJson = { ... };
const survey = new Model(surveyJson);

survey.sendResultOnPageNext = true;

const storageItemKey = "my-survey";

function saveSurveyData (survey) {
Expand All @@ -20,13 +18,9 @@ function saveSurveyData (survey) {
window.localStorage.setItem(storageItemKey, JSON.stringify(data));
}

// Save survey results
survey.onPartialSend.add((survey) => {
saveSurveyData(survey);
});
survey.onComplete.add((survey) => {
saveSurveyData(survey);
});
// Save survey results to the local storage
survey.onValueChanged.add(saveSurveyData);
survey.onCurrentPageChanged.add(saveSurveyData);

// Restore survey results
const prevData = window.localStorage.getItem(storageItemKey) || null;
Expand All @@ -37,9 +31,14 @@ if (prevData) {
survey.currentPageNo = data.pageNo;
}
}

// Empty the local storage after the survey is completed
survey.onComplete.add(() => {
window.localStorage.setItem(storageItemKey, "");
});
```

[View Demo](https://surveyjs.io/Examples/Library?id=real-patient-history (linkStyle))
[View Demo](/form-library/examples/survey-editprevious/ (linkStyle))

## See Also

Expand Down
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -107,7 +107,7 @@
"@typescript-eslint/eslint-plugin": "^4.29.0",
"@typescript-eslint/parser": "^4.29.0",
"ace-builds": "1.2.2",
"ajv": "4.11.2",
"ajv": "6.12.3",
"axe-core": "^3.5.6",
"axe-testcafe": "^3.0.0",
"babel-loader": "8.2.5",
Expand Down
19 changes: 14 additions & 5 deletions packages/survey-angular-ui/example/src/testCafe/countriesMock.json
Original file line number Diff line number Diff line change
@@ -1,9 +1,18 @@
{
"RestResponse": {
"result": [
{ "alpha2_code": "US", "name": "Unated States" },
{ "alpha2_code": "CU", "name": "Cuba" },
{ "alpha2_code": "RO","name":"Romania" }
]
"result": [
{
"alpha2_code": "US",
"name": "United States"
},
{
"alpha2_code": "CU",
"name": "Cuba"
},
{
"alpha2_code": "RO",
"name": "Romania"
}
]
}
}
5 changes: 3 additions & 2 deletions packages/survey-angular-ui/src/angular-ui.module.ts
Original file line number Diff line number Diff line change
Expand Up @@ -110,6 +110,7 @@ import { PaneldynamicRemoveButtonComponent } from "./components/paneldynamic-act
import { TimerPanelComponent } from "./components/timer-panel/timer-panel.component";
import { NotifierComponent } from "./components/notifier/notifier.component";
import { ComponentsContainerComponent } from "./components-container.component";
import { MultipleTextRowComponent } from "./questions/multipletextrow.component";
@NgModule({
declarations: [
VisibleDirective, Key2ClickDirective, PanelDynamicAddBtn, PanelDynamicNextBtn, PanelDynamicPrevBtn, PanelDynamicProgressText, ElementComponent, TemplateRendererComponent,
Expand All @@ -126,7 +127,7 @@ import { ComponentsContainerComponent } from "./components-container.component";
MultipleTextComponent, MultipleTextItemComponent, DynamicComponentDirective, RankingQuestionComponent, RankingItemComponent, PanelDynamicQuestionComponent, EmbeddedViewContentComponent, CustomWidgetComponent, MatrixCellComponent, MatrixTableComponent, MatrixDropdownComponent,
MatrixDynamicComponent, MatrixDetailButtonComponent, MatrixDynamicRemoveButtonComponent, MatrixDynamicDragDropIconComponent, MatrixRequiredHeader, ExpressionComponent, SafeResourceUrlPipe, BrandInfoComponent,
CustomQuestionComponent, CompositeQuestionComponent, ButtonGroupItemComponent, ButtonGroupQuestionComponent, MatrixRowComponent, ModalComponent, LogoImageComponent, SkeletonComponent, TimerPanelComponent, PaneldynamicRemoveButtonComponent,
NotifierComponent, ComponentsContainerComponent
NotifierComponent, ComponentsContainerComponent, MultipleTextRowComponent
],
imports: [
CommonModule, FormsModule
Expand All @@ -147,7 +148,7 @@ import { ComponentsContainerComponent } from "./components-container.component";
MultipleTextComponent, MultipleTextItemComponent, DynamicComponentDirective, RankingQuestionComponent, RankingItemComponent, PanelDynamicQuestionComponent, EmbeddedViewContentComponent, CustomWidgetComponent, MatrixCellComponent, MatrixTableComponent, MatrixDropdownComponent,
MatrixDynamicComponent, MatrixDetailButtonComponent, MatrixDynamicRemoveButtonComponent, MatrixDynamicDragDropIconComponent, MatrixRequiredHeader, ExpressionComponent, SafeResourceUrlPipe,
CustomQuestionComponent, CompositeQuestionComponent, ButtonGroupQuestionComponent, ModalComponent, LogoImageComponent, SkeletonComponent, TimerPanelComponent, PaneldynamicRemoveButtonComponent,
NotifierComponent, ComponentsContainerComponent
NotifierComponent, ComponentsContainerComponent, MultipleTextRowComponent
],
providers: [PopupService],
})
Expand Down
3 changes: 2 additions & 1 deletion packages/survey-angular-ui/src/angular-ui.ts
Original file line number Diff line number Diff line change
Expand Up @@ -75,8 +75,9 @@ export * from "./utils/safe-html.pipe";
export * from "./questions/comment.component";
export * from "./questions/signature.component";
export * from "./questions/multipletext.component";
export * from "./errors.component";
export * from "./questions/multipletextitem.component";
export * from "./questions/multipletextrow.component";
export * from "./errors.component";
export * from "./utils/dynamic.directive";
export * from "./questions/ranking.component";
export * from "./questions/ranking-item.component";
Expand Down
20 changes: 4 additions & 16 deletions packages/survey-angular-ui/src/errors.component.ts
Original file line number Diff line number Diff line change
@@ -1,26 +1,14 @@
import { Component, DoCheck, ElementRef, HostBinding, Input, OnDestroy, OnInit, ViewChild, ViewContainerRef } from "@angular/core";
import { Base, SurveyElement, TooltipManager } from "survey-core";
import { Component, HostBinding, Input } from "@angular/core";
import { SurveyElement } from "survey-core";

@Component({
templateUrl: "./errors.component.html",
selector: "'[sv-ng-errors]'"
})
export class ErrorsComponent implements OnDestroy, OnInit {
export class ErrorsComponent {
@Input() element!: SurveyElement | any;
@Input() location?: String;
@ViewChild("errorsContainer", { static: true }) errorsContainerRef!: ElementRef<HTMLDivElement>;
private tooltipManager!: TooltipManager;
constructor(private viewContainerRef: ViewContainerRef) {}
ngOnInit(): void {
if (this.location == "tooltip") {
this.tooltipManager = new TooltipManager(this.viewContainerRef.element.nativeElement);
}
}
ngOnDestroy() {
if (!!this.tooltipManager) {
this.tooltipManager.dispose();
}
}
constructor() {}
@HostBinding("attr.role") get role (): string {
return "alert";
}
Expand Down
1 change: 0 additions & 1 deletion packages/survey-angular-ui/src/question.component.html
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,6 @@
<sv-ng-comment [question]="model"></sv-ng-comment>
</div>
<div *ngIf="model.showErrorOnBottom && model.hasVisibleErrors" [element]="model" sv-ng-errors></div>
<div *ngIf="model.isErrorsModeTooltip && model.hasVisibleErrors" [element]="model" [location]="'tooltip'" sv-ng-errors></div>
<div *ngIf="model.hasDescriptionUnderInput" [class]="model.cssDescription" [model]="model.locDescription" sv-ng-string></div>
</div>
<div [element]="model" *ngIf="model.hasTitleOnBottom" sv-ng-element-header></div>
Expand Down
Original file line number Diff line number Diff line change
@@ -1,12 +1,7 @@
<table [class]="model.cssClasses.root" #contentElement>
<tbody>
<tr
*ngFor="let row of model.getRows(); index as rowIndex; trackBy: trackRowBy"
[class]="model.cssClasses.row"
>
<ng-container *ngFor="let item of row; trackBy: trackItemBy" >
<td [class]="model.cssClasses.cell" [question]="model" [model]="item" sv-ng-multipletext-item></td>
</ng-container>
</tr>
<ng-container *ngFor="let row of model.getRows(); index as rowIndex; trackBy: trackRowBy">
<sv-ng-multipletext-row [model]="row" [question]="model"></sv-ng-multipletext-row>
</ng-container>
</tbody>
</table>
</table>
Original file line number Diff line number Diff line change
Expand Up @@ -11,9 +11,6 @@ export class MultipleTextComponent extends QuestionAngular<QuestionMultipleTextM
trackRowBy = (index: number): string => {
return this.model.inputId + "rowkey" + index;
}
trackItemBy (_: number, item: MultipleTextItemModel): string {
return "item" + item.editor.id;
}
}

AngularComponentFactory.Instance.registerComponent("multipletext-question", MultipleTextComponent);
Original file line number Diff line number Diff line change
@@ -1,15 +1,38 @@
import { MultipleTextItemModel, QuestionMultipleTextModel, QuestionTextModel } from "survey-core";
import { Component, Input } from "@angular/core";
import { MultipleTextCell, MultipleTextItemModel, QuestionMultipleTextModel, QuestionTextModel } from "survey-core";
import { Component, DoCheck, Input, OnDestroy } from "@angular/core";
import { BaseAngular } from "../base-angular";

@Component({
selector: "'[sv-ng-multipletext-item]'",
templateUrl: "./mutlipletextitem.component.html"
})
export class MultipleTextItemComponent extends BaseAngular<QuestionTextModel> {
export class MultipleTextItemComponent extends BaseAngular<QuestionTextModel> implements DoCheck, OnDestroy {
@Input() question!: QuestionMultipleTextModel;
@Input() model!: MultipleTextItemModel;
@Input() model!: MultipleTextCell;
protected getModel(): QuestionTextModel {
return this.model.editor;
if(!this.model.isErrorsCell) {
return this.model.item.editor;
}
return null as any;
}
public get item(): MultipleTextItemModel {
return this.model.item;
}
public get editor(): QuestionTextModel {
return this.model.item.editor;
}
override ngDoCheck(): void {
super.ngDoCheck();
if(this.model.isErrorsCell) {
this.editor.registerFunctionOnPropertyValueChanged("errors", () => {
this.update();
}, "__ngSubscription")
}
}
override ngOnDestroy(): void {
super.ngOnDestroy();
if(this.model.isErrorsCell) {
this.editor.unRegisterFunctionOnPropertyValueChanged("errors", "__ngSubscription")
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
<ng-template #template>
<tr [class]="question.cssClasses.row" *ngIf="model.isVisible">
<ng-container *ngFor="let cell of model.cells; trackBy: trackItemBy">
<td [class]="cell.className" [question]="question" [model]="cell" sv-ng-multipletext-item></td>
</ng-container>
</tr>
</ng-template>
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
import { MultipleTextCell, MultipleTextItemModel, MutlipleTextRow, QuestionMultipleTextModel, QuestionTextModel } from "survey-core";
import { Component, Input } from "@angular/core";
import { BaseAngular } from "../base-angular";

@Component({
selector: "sv-ng-multipletext-row",
templateUrl: "./multipletextrow.component.html",
styleUrls: ["../hide-host.scss"]
})
export class MultipleTextRowComponent extends BaseAngular<MutlipleTextRow> {
@Input() question!: QuestionMultipleTextModel;
@Input() model!: MutlipleTextRow;
protected getModel(): MutlipleTextRow {
return this.model;
}
trackItemBy (_: number, cell: MultipleTextCell): string {
return "item" + cell.item.editor.id;
}
}
Original file line number Diff line number Diff line change
@@ -1,18 +1,18 @@
<label [class]="question.getItemLabelCss(model)">
<span [class]="question.getItemTitleCss()">
<span *ngIf=" model.editor.isRequireTextBeforeTitle || model.editor.isRequireTextOnStart"
[class]="question.cssClasses.requiredText">{{ model.editor.requiredText }}</span>
<sv-ng-string [model]="model.locTitle"></sv-ng-string>
<span
*ngIf="model.editor.isRequireTextAfterTitle"
[class]="question.cssClasses.requiredText"
>{{ model.editor.requiredText }}</span
>
</span>
<div [class]="question.getItemCss()" (focusin)="model.focusIn()">
<div *ngIf="model.editor.showErrorOnTop && model.editor.hasVisibleErrors" [element]="model.editor" sv-ng-errors></div>
<sv-ng-text-question [model]="model.editor"></sv-ng-text-question>
<div *ngIf="model.editor.showErrorOnBottom && model.editor.hasVisibleErrors" [element]="model.editor" sv-ng-errors></div>
</div>
<div *ngIf="model.editor.isErrorsModeTooltip && model.editor.hasVisibleErrors" [element]="model.editor" [location]="'tooltip'" sv-ng-errors></div>
</label>
<ng-container *ngIf="!model.isErrorsCell">
<label [class]="question.getItemLabelCss(item)">
<span [class]="question.getItemTitleCss()">
<span *ngIf=" item.editor.isRequireTextBeforeTitle || item.editor.isRequireTextOnStart"
[class]="question.cssClasses.requiredText">{{ item.editor.requiredText }}</span>
<sv-ng-string [model]="item.locTitle"></sv-ng-string>
<span *ngIf="item.editor.isRequireTextAfterTitle">&nbsp;</span>
<span *ngIf="item.editor.isRequireTextAfterTitle" [class]="question.cssClasses.requiredText" aria-hidden="true">{{
item.editor.requiredText }}</span>
</span>
<div [class]="question.getItemCss()" (focusin)="item.focusIn()">
<sv-ng-text-question [model]="item.editor"></sv-ng-text-question>
</div>
</label>
</ng-container>
<ng-container *ngIf="model.isErrorsCell">
<div *ngIf="item.editor.hasVisibleErrors" [element]="item.editor" sv-ng-errors></div>
</ng-container>
4 changes: 2 additions & 2 deletions packages/survey-angular-ui/src/survey-content.component.html
Original file line number Diff line number Diff line change
Expand Up @@ -26,9 +26,9 @@
<ng-template [component]="{ name: 'sv-components-container', data: { survey: model, container: 'footer', needRenderWrapper: false } }"></ng-template>
<div *ngIf="model.state === 'completed' && model.showCompletedPage" [class]="model.completedCss"
[innerHtml]="model.processedCompletedHtml"></div>
<div *ngIf="model.state === 'completedbefore'" [class]="model.css.body"
<div *ngIf="model.state === 'completedbefore'" [class]="model.completedBeforeCss"
[innerHtml]="model.processedCompletedBeforeHtml"></div>
<div *ngIf="model.state === 'loading'" [class]="model.css.body" [innerHtml]="model.processedLoadingHtml"></div>
<div *ngIf="model.state === 'loading'" [class]="model.loadingBodyCss" [innerHtml]="model.processedLoadingHtml"></div>
<div *ngIf="model.state === 'empty'" [class]="model.css.bodyEmpty">{{model.emptySurveyText}}</div>
</div>
</form>
Expand Down
19 changes: 14 additions & 5 deletions packages/survey-vue3-ui/example/public/testCafe/countriesMock.json
Original file line number Diff line number Diff line change
@@ -1,9 +1,18 @@
{
"RestResponse": {
"result": [
{ "alpha2_code": "US", "name": "Unated States" },
{ "alpha2_code": "CU", "name": "Cuba" },
{ "alpha2_code": "RO","name":"Romania" }
]
"result": [
{
"alpha2_code": "US",
"name": "United States"
},
{
"alpha2_code": "CU",
"name": "Cuba"
},
{
"alpha2_code": "RO",
"name": "Romania"
}
]
}
}
5 changes: 0 additions & 5 deletions packages/survey-vue3-ui/src/Element.vue
Original file line number Diff line number Diff line change
Expand Up @@ -53,11 +53,6 @@
:element="element"
:location="'bottom'"
/>
<survey-errors
v-if="!element.isPanel && element.isErrorsModeTooltip"
:element="element"
:location="'tooltip'"
/>
<div
v-if="!element.isPanel && (element as Question).hasDescriptionUnderInput"
:class="element.cssClasses.descriptionUnderInput"
Expand Down
21 changes: 2 additions & 19 deletions packages/survey-vue3-ui/src/Errors.vue
Original file line number Diff line number Diff line change
Expand Up @@ -30,29 +30,12 @@
</template>

<script lang="ts" setup>
import { type PanelModel, type Question, TooltipManager } from "survey-core";
import { onUnmounted, onUpdated, ref } from "vue";
import type { PanelModel, Question } from "survey-core";
import { ref } from "vue";
const props = defineProps<{
element: Question | PanelModel;
location?: string;
}>();
const root = ref<HTMLElement>();
let tooltipManager: TooltipManager;
onUpdated(() => {
if (props.location == "tooltip" && root.value instanceof HTMLElement) {
if (!tooltipManager || root.value !== tooltipManager.tooltipElement) {
tooltipManager = new TooltipManager(root.value as HTMLElement);
}
}
if (!(root.value instanceof HTMLElement) && !!tooltipManager) {
tooltipManager.dispose();
}
});
onUnmounted(() => {
if (tooltipManager) {
tooltipManager.dispose();
}
});
</script>
Loading

0 comments on commit 926604c

Please sign in to comment.