[IMP] survey_extra_fields, survey_base : modifications to make them work in v18
This commit is contained in:
@@ -13,7 +13,7 @@ Add fields used by several survey addons
|
|||||||
* Implementation of theses fields should be in another module
|
* Implementation of theses fields should be in another module
|
||||||
|
|
||||||
""",
|
""",
|
||||||
"version": "16.0.1.0.0",
|
"version": "18.0.1.0.0",
|
||||||
"license": "AGPL-3",
|
"license": "AGPL-3",
|
||||||
"author": "Elabore",
|
"author": "Elabore",
|
||||||
"website": "https://www.elabore.coop",
|
"website": "https://www.elabore.coop",
|
||||||
|
|||||||
@@ -13,7 +13,7 @@ Add extra question types to surveys:
|
|||||||
- Client-side validation (size and extension) before form submission
|
- Client-side validation (size and extension) before form submission
|
||||||
- Server-side validation on save to enforce constraints
|
- Server-side validation on save to enforce constraints
|
||||||
""",
|
""",
|
||||||
"version": "16.0.1.0.0",
|
"version": "18.0.1.0.0",
|
||||||
"license": "AGPL-3",
|
"license": "AGPL-3",
|
||||||
"author": "Elabore",
|
"author": "Elabore",
|
||||||
"website": "https://www.elabore.coop",
|
"website": "https://www.elabore.coop",
|
||||||
|
|||||||
@@ -31,7 +31,7 @@ class SurveyExtraFieldsController(Survey):
|
|||||||
[("access_token", "=", survey_token)], limit=1
|
[("access_token", "=", survey_token)], limit=1
|
||||||
)
|
)
|
||||||
if not survey:
|
if not survey:
|
||||||
return request.not_found()
|
raise request.not_found()
|
||||||
|
|
||||||
answer = request.env["survey.user_input"].sudo().search(
|
answer = request.env["survey.user_input"].sudo().search(
|
||||||
[
|
[
|
||||||
@@ -41,14 +41,14 @@ class SurveyExtraFieldsController(Survey):
|
|||||||
limit=1,
|
limit=1,
|
||||||
)
|
)
|
||||||
if not answer:
|
if not answer:
|
||||||
return request.not_found()
|
raise request.not_found()
|
||||||
|
|
||||||
line = request.env["survey.user_input.line"].sudo().browse(line_id)
|
line = request.env["survey.user_input.line"].sudo().browse(line_id)
|
||||||
if not line.exists() or line.user_input_id != answer:
|
if not line.exists() or line.user_input_id != answer:
|
||||||
return request.not_found()
|
raise request.not_found()
|
||||||
|
|
||||||
if not line.value_file:
|
if not line.value_file:
|
||||||
return request.not_found()
|
raise request.not_found()
|
||||||
|
|
||||||
file_content = base64.b64decode(line.value_file)
|
file_content = base64.b64decode(line.value_file)
|
||||||
filename = line.value_file_fname or "file"
|
filename = line.value_file_fname or "file"
|
||||||
|
|||||||
@@ -4,10 +4,10 @@
|
|||||||
#
|
#
|
||||||
msgid ""
|
msgid ""
|
||||||
msgstr ""
|
msgstr ""
|
||||||
"Project-Id-Version: Odoo Server 16.0\n"
|
"Project-Id-Version: Odoo Server 18.0\n"
|
||||||
"Report-Msgid-Bugs-To: \n"
|
"Report-Msgid-Bugs-To: \n"
|
||||||
"POT-Creation-Date: 2026-02-18 16:38+0000\n"
|
"POT-Creation-Date: 2026-06-10 14:31+0000\n"
|
||||||
"PO-Revision-Date: 2026-02-18 16:38+0000\n"
|
"PO-Revision-Date: 2026-06-10 14:31+0000\n"
|
||||||
"Last-Translator: \n"
|
"Last-Translator: \n"
|
||||||
"Language-Team: \n"
|
"Language-Team: \n"
|
||||||
"MIME-Version: 1.0\n"
|
"MIME-Version: 1.0\n"
|
||||||
@@ -36,8 +36,8 @@ msgid ""
|
|||||||
"Comma-separated list of allowed extensions (e.g. .pdf,.docx). Leave empty to"
|
"Comma-separated list of allowed extensions (e.g. .pdf,.docx). Leave empty to"
|
||||||
" allow all types."
|
" allow all types."
|
||||||
msgstr ""
|
msgstr ""
|
||||||
"Liste d'extensions autorisées séparées par une virgule (e.g. .pdf,.docx). Laisser vide pour"
|
"Liste d'extensions autorisées séparées par une virgule (e.g. .pdf,.docx). "
|
||||||
" autoriser tous les types de fichier."
|
"Laisser vide pour autoriser tous les types de fichier."
|
||||||
|
|
||||||
#. module: survey_extra_fields
|
#. module: survey_extra_fields
|
||||||
#: model:ir.model.fields.selection,name:survey_extra_fields.selection__survey_question__question_type__file
|
#: model:ir.model.fields.selection,name:survey_extra_fields.selection__survey_question__question_type__file
|
||||||
@@ -54,7 +54,9 @@ msgstr "Taille maximale du fichier"
|
|||||||
#. module: survey_extra_fields
|
#. module: survey_extra_fields
|
||||||
#: model:ir.model.fields,help:survey_extra_fields.field_survey_question__max_file_size
|
#: model:ir.model.fields,help:survey_extra_fields.field_survey_question__max_file_size
|
||||||
msgid "Maximum file size in MB. Leave 0 for no limit."
|
msgid "Maximum file size in MB. Leave 0 for no limit."
|
||||||
msgstr "Taille maximale du fichier en MB. Laisser à 0 pour ne pas restreindre la taille. La valeur par défaut est 10 MB."
|
msgstr ""
|
||||||
|
"Taille maximale du fichier en MB. Laisser à 0 pour ne pas restreindre la "
|
||||||
|
"taille. La valeur par défaut est 10 MB."
|
||||||
|
|
||||||
#. module: survey_extra_fields
|
#. module: survey_extra_fields
|
||||||
#: model:ir.model.fields,field_description:survey_extra_fields.field_survey_question__question_type
|
#: model:ir.model.fields,field_description:survey_extra_fields.field_survey_question__question_type
|
||||||
@@ -66,13 +68,6 @@ msgstr "Type de question"
|
|||||||
msgid "Skipped"
|
msgid "Skipped"
|
||||||
msgstr "Ignoré"
|
msgstr "Ignoré"
|
||||||
|
|
||||||
#. module: survey_extra_fields
|
|
||||||
#: model:ir.model.fields,field_description:survey_extra_fields.field_survey_question__smart_search
|
|
||||||
#: model:ir.model.fields,field_description:survey_extra_fields.field_survey_user_input__smart_search
|
|
||||||
#: model:ir.model.fields,field_description:survey_extra_fields.field_survey_user_input_line__smart_search
|
|
||||||
msgid "Smart Search"
|
|
||||||
msgstr "Recherche intelligente"
|
|
||||||
|
|
||||||
#. module: survey_extra_fields
|
#. module: survey_extra_fields
|
||||||
#: model:ir.model,name:survey_extra_fields.model_survey_question
|
#: model:ir.model,name:survey_extra_fields.model_survey_question
|
||||||
msgid "Survey Question"
|
msgid "Survey Question"
|
||||||
@@ -91,34 +86,39 @@ msgstr "Ligne d'entrée pour l'utilisateur du sondage"
|
|||||||
#. module: survey_extra_fields
|
#. module: survey_extra_fields
|
||||||
#. odoo-python
|
#. odoo-python
|
||||||
#: code:addons/survey_extra_fields/models/survey_user_input.py:0
|
#: code:addons/survey_extra_fields/models/survey_user_input.py:0
|
||||||
#, python-format
|
|
||||||
msgid "The file '%(name)s' exceeds the maximum allowed size of %(size)s MB."
|
msgid "The file '%(name)s' exceeds the maximum allowed size of %(size)s MB."
|
||||||
msgstr "Le fichier '%(name)s' dépasse la taille maximale autorisée de %(size)s MB."
|
msgstr ""
|
||||||
|
"Le fichier '%(name)s' dépasse la taille maximale autorisée de %(size)s MB."
|
||||||
|
|
||||||
#. module: survey_extra_fields
|
#. module: survey_extra_fields
|
||||||
#. odoo-python
|
#. odoo-python
|
||||||
#: code:addons/survey_extra_fields/models/survey_user_input.py:0
|
#: code:addons/survey_extra_fields/models/survey_user_input.py:0
|
||||||
#, python-format
|
|
||||||
msgid "The file '%(name)s' is not allowed. Accepted formats: %(exts)s."
|
msgid "The file '%(name)s' is not allowed. Accepted formats: %(exts)s."
|
||||||
msgstr "Le fichier '%(name)s' n'est pas autorisé. Les formats de fichier autorisés sont : %(exts)s."
|
msgstr ""
|
||||||
|
"Le fichier '%(name)s' n'est pas autorisé. Les formats de fichier autorisés "
|
||||||
|
"sont : %(exts)s."
|
||||||
|
|
||||||
#. module: survey_extra_fields
|
#. module: survey_extra_fields
|
||||||
#. odoo-javascript
|
#. odoo-javascript
|
||||||
#: code:addons/survey_extra_fields/static/src/js/survey_form.js:0
|
#: code:addons/survey_extra_fields/static/src/js/survey_form.js:0
|
||||||
#, python-format
|
|
||||||
msgid "The file exceeds the maximum allowed size of %s MB."
|
msgid "The file exceeds the maximum allowed size of %s MB."
|
||||||
msgstr "Le fichier dépasse la taille maximale autorisée de %s MB."
|
msgstr "Le fichier dépasse la taille maximale autorisée de %s MB."
|
||||||
|
|
||||||
#. module: survey_extra_fields
|
#. module: survey_extra_fields
|
||||||
#. odoo-javascript
|
#. odoo-python
|
||||||
#: code:addons/survey_extra_fields/static/src/js/survey_form.js:0
|
#: code:addons/survey_extra_fields/models/survey_user_input.py:0
|
||||||
#, python-format
|
msgid "This answer cannot be overwritten."
|
||||||
msgid "This file type is not allowed. Accepted formats: %s."
|
msgstr "Cette réponse ne peut pas être remplacée."
|
||||||
msgstr "Le fichier n'est pas autorisé. Les formats de fichier autorisés sont : %s."
|
|
||||||
|
#. module: survey_extra_fields
|
||||||
|
#. odoo-javascript
|
||||||
|
#: code:addons/survey_extra_fields/static/src/js/survey_form.js:0
|
||||||
|
msgid "This file type is not allowed. Accepted formats: %s."
|
||||||
|
msgstr ""
|
||||||
|
"Le fichier n'est pas autorisé. Les formats de fichier autorisés sont : %s."
|
||||||
|
|
||||||
#. module: survey_extra_fields
|
#. module: survey_extra_fields
|
||||||
#. odoo-javascript
|
#. odoo-javascript
|
||||||
#: code:addons/survey_extra_fields/static/src/js/survey_form.js:0
|
#: code:addons/survey_extra_fields/static/src/js/survey_form.js:0
|
||||||
#, python-format
|
|
||||||
msgid "This question requires an answer."
|
msgid "This question requires an answer."
|
||||||
msgstr "Cette question requiert une réponse."
|
msgstr "Cette question requiert une réponse."
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ import os
|
|||||||
from typing import TYPE_CHECKING
|
from typing import TYPE_CHECKING
|
||||||
|
|
||||||
from odoo import _, models
|
from odoo import _, models
|
||||||
from odoo.exceptions import ValidationError
|
from odoo.exceptions import UserError, ValidationError
|
||||||
|
|
||||||
if TYPE_CHECKING:
|
if TYPE_CHECKING:
|
||||||
from odoo.addons.survey_extra_fields.models.survey_question import SurveyQuestion
|
from odoo.addons.survey_extra_fields.models.survey_question import SurveyQuestion
|
||||||
@@ -17,12 +17,20 @@ if TYPE_CHECKING:
|
|||||||
class SurveyUserInput(models.Model):
|
class SurveyUserInput(models.Model):
|
||||||
_inherit = "survey.user_input"
|
_inherit = "survey.user_input"
|
||||||
|
|
||||||
def save_lines(self, question: SurveyQuestion, answer: str | None, comment: str | None = None) -> None:
|
def _save_lines(
|
||||||
|
self,
|
||||||
|
question: SurveyQuestion,
|
||||||
|
answer: str | None,
|
||||||
|
comment: str | None = None,
|
||||||
|
overwrite_existing: bool = True,
|
||||||
|
) -> None:
|
||||||
if question.question_type == "file":
|
if question.question_type == "file":
|
||||||
old_answers = self.env["survey.user_input.line"].search([
|
old_answers = self.env["survey.user_input.line"].search([
|
||||||
("user_input_id", "=", self.id),
|
("user_input_id", "=", self.id),
|
||||||
("question_id", "=", question.id),
|
("question_id", "=", question.id),
|
||||||
])
|
])
|
||||||
|
if old_answers and not overwrite_existing:
|
||||||
|
raise UserError(_("This answer cannot be overwritten."))
|
||||||
vals = {
|
vals = {
|
||||||
"user_input_id": self.id,
|
"user_input_id": self.id,
|
||||||
"question_id": question.id,
|
"question_id": question.id,
|
||||||
@@ -43,7 +51,9 @@ class SurveyUserInput(models.Model):
|
|||||||
else:
|
else:
|
||||||
self.env["survey.user_input.line"].create(vals)
|
self.env["survey.user_input.line"].create(vals)
|
||||||
else:
|
else:
|
||||||
return super().save_lines(question, answer, comment=comment)
|
return super()._save_lines(
|
||||||
|
question, answer, comment=comment, overwrite_existing=overwrite_existing
|
||||||
|
)
|
||||||
|
|
||||||
def _check_file_constraints(
|
def _check_file_constraints(
|
||||||
self,
|
self,
|
||||||
|
|||||||
@@ -1,146 +1,148 @@
|
|||||||
odoo.define("survey_extra_fields.survey_form", function (require) {
|
/** @odoo-module **/
|
||||||
"use strict";
|
|
||||||
|
|
||||||
var core = require("web.core");
|
import publicWidget from "@web/legacy/js/public/public_widget";
|
||||||
var _t = core._t;
|
import { _t } from "@web/core/l10n/translation";
|
||||||
var survey_form = require("survey.form");
|
import { rpc } from "@web/core/network/rpc";
|
||||||
|
|
||||||
survey_form.include({
|
publicWidget.registry.SurveyFormWidget.include({
|
||||||
_readFileAsDataURL: function (file) {
|
_readFileAsDataURL: function (file) {
|
||||||
return new Promise(function (resolve, reject) {
|
return new Promise(function (resolve, reject) {
|
||||||
var reader = new FileReader();
|
const reader = new FileReader();
|
||||||
reader.onload = function (e) {
|
reader.onload = function (e) {
|
||||||
resolve(e.target.result);
|
resolve(e.target.result);
|
||||||
};
|
};
|
||||||
reader.onerror = function () {
|
reader.onerror = function () {
|
||||||
reject(reader.error);
|
reject(reader.error);
|
||||||
};
|
};
|
||||||
reader.readAsDataURL(file);
|
reader.readAsDataURL(file);
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
_submitForm: function (options) {
|
/**
|
||||||
var self = this;
|
* @override
|
||||||
var $fileInputs = this.$('input[data-question-type="file"]');
|
* The base implementation builds the submit params synchronously and fires
|
||||||
var hasFiles = false;
|
* the RPC immediately. File inputs need to be read asynchronously (FileReader),
|
||||||
$fileInputs.each(function () {
|
* so when the current page contains file answers we replicate the submit flow
|
||||||
if (this.files && this.files.length > 0) {
|
* here, injecting the base64 file payload before submitting.
|
||||||
hasFiles = true;
|
*/
|
||||||
return false;
|
_submitForm: async function (options) {
|
||||||
}
|
const fileInputs = this.el.querySelectorAll('input[data-question-type="file"]');
|
||||||
});
|
const hasFiles = Array.from(fileInputs).some(
|
||||||
|
(input) => input.files && input.files.length > 0
|
||||||
|
);
|
||||||
|
|
||||||
if (!hasFiles || this.options.isStartScreen) {
|
if (!hasFiles || this.options.isStartScreen) {
|
||||||
return this._super(options);
|
return this._super(options);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Async flow: read files then submit
|
const params = {};
|
||||||
var params = {};
|
if (options.previousPageId) {
|
||||||
if (options.previousPageId) {
|
params.previous_page_id = options.previousPageId;
|
||||||
params.previous_page_id = options.previousPageId;
|
}
|
||||||
}
|
if (options.nextSkipped) {
|
||||||
|
params.next_skipped_page_or_question = true;
|
||||||
|
}
|
||||||
|
|
||||||
var $form = this.$("form");
|
const $form = this.$("form");
|
||||||
var formData = new FormData($form[0]);
|
const formData = new FormData($form[0]);
|
||||||
|
|
||||||
if (!options.skipValidation) {
|
if (!options.skipValidation && !this._validateForm($form, formData)) {
|
||||||
if (!this._validateForm($form, formData)) {
|
return;
|
||||||
return;
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
this._prepareSubmitValues(formData, params);
|
this._prepareSubmitValues(formData, params);
|
||||||
|
|
||||||
// Read all selected files as base64
|
// Read all selected files as base64 and add them to the submit params.
|
||||||
var filePromises = [];
|
const filePromises = [];
|
||||||
$fileInputs.each(function () {
|
for (const input of fileInputs) {
|
||||||
if (this.files && this.files.length > 0) {
|
if (input.files && input.files.length > 0) {
|
||||||
var file = this.files[0];
|
const file = input.files[0];
|
||||||
var name = this.name;
|
const name = input.name;
|
||||||
filePromises.push(
|
filePromises.push(
|
||||||
self._readFileAsDataURL(file).then(function (dataURL) {
|
this._readFileAsDataURL(file).then((dataURL) => {
|
||||||
params[name] = JSON.stringify({
|
params[name] = JSON.stringify({
|
||||||
data: dataURL.split(",")[1],
|
data: dataURL.split(",")[1],
|
||||||
name: file.name,
|
name: file.name,
|
||||||
});
|
|
||||||
})
|
|
||||||
);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
this.preventEnterSubmit = true;
|
|
||||||
|
|
||||||
if (this.options.sessionInProgress) {
|
|
||||||
this.fadeInOutDelay = 400;
|
|
||||||
this.readonly = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
Promise.all(filePromises).then(function () {
|
|
||||||
var submitPromise = self._rpc({
|
|
||||||
route: _.str.sprintf(
|
|
||||||
"%s/%s/%s",
|
|
||||||
"/survey/submit",
|
|
||||||
self.options.surveyToken,
|
|
||||||
self.options.answerToken
|
|
||||||
),
|
|
||||||
params: params,
|
|
||||||
});
|
|
||||||
self._nextScreen(submitPromise, options);
|
|
||||||
});
|
|
||||||
},
|
|
||||||
|
|
||||||
_validateForm: function ($form, formData) {
|
|
||||||
var result = this._super.apply(this, arguments);
|
|
||||||
var errors = {};
|
|
||||||
var inactiveQuestionIds = this.options.sessionInProgress
|
|
||||||
? []
|
|
||||||
: this._getInactiveConditionalQuestionIds();
|
|
||||||
|
|
||||||
$form.find('input[data-question-type="file"]').each(function () {
|
|
||||||
var $questionWrapper = $(this).closest(".js_question-wrapper");
|
|
||||||
var questionId = $questionWrapper.attr("id");
|
|
||||||
if (inactiveQuestionIds.includes(parseInt(questionId))) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
var questionRequired = $questionWrapper.data("required");
|
|
||||||
var constrErrorMsg =
|
|
||||||
$questionWrapper.data("constrErrorMsg") ||
|
|
||||||
_t("This question requires an answer.");
|
|
||||||
if (questionRequired && !(this.files && this.files.length > 0)) {
|
|
||||||
errors[questionId] = constrErrorMsg;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (this.files && this.files.length > 0) {
|
|
||||||
var file = this.files[0];
|
|
||||||
var maxSizeMB = parseInt($(this).data("maxFileSize"));
|
|
||||||
if (maxSizeMB && file.size > maxSizeMB * 1024 * 1024) {
|
|
||||||
errors[questionId] = _.str.sprintf(
|
|
||||||
_t("The file exceeds the maximum allowed size of %s MB."),
|
|
||||||
maxSizeMB
|
|
||||||
);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
var allowedExtensions = $(this).data("allowedExtensions");
|
|
||||||
if (allowedExtensions) {
|
|
||||||
var allowed = allowedExtensions.split(",").map(function (e) {
|
|
||||||
return e.trim().toLowerCase();
|
|
||||||
});
|
});
|
||||||
var ext = "." + file.name.split(".").pop().toLowerCase();
|
})
|
||||||
if (!allowed.includes(ext)) {
|
);
|
||||||
errors[questionId] = _.str.sprintf(
|
}
|
||||||
_t("This file type is not allowed. Accepted formats: %s."),
|
}
|
||||||
allowedExtensions
|
|
||||||
);
|
// Prevent user from submitting more times using enter key.
|
||||||
}
|
this.preventEnterSubmit = true;
|
||||||
|
if (this.options.sessionInProgress) {
|
||||||
|
this.fadeInOutDelay = 400;
|
||||||
|
this.readonly = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
await Promise.all(filePromises);
|
||||||
|
|
||||||
|
const submitPromise = rpc(
|
||||||
|
`/survey/submit/${this.options.surveyToken}/${this.options.answerToken}`,
|
||||||
|
params
|
||||||
|
);
|
||||||
|
this._nextScreen(submitPromise, options);
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @override
|
||||||
|
* Add client-side validation (required, max size, allowed extensions) for
|
||||||
|
* file questions, which the base implementation does not know about.
|
||||||
|
*/
|
||||||
|
_validateForm: function ($form, formData) {
|
||||||
|
const result = this._super.apply(this, arguments);
|
||||||
|
const errors = {};
|
||||||
|
const inactiveQuestionIds = this.options.sessionInProgress
|
||||||
|
? []
|
||||||
|
: this._getInactiveConditionalQuestionIds();
|
||||||
|
|
||||||
|
$form.find('input[data-question-type="file"]').each(function () {
|
||||||
|
const $questionWrapper = $(this).closest(".js_question-wrapper");
|
||||||
|
const questionId = $questionWrapper.attr("id");
|
||||||
|
if (inactiveQuestionIds.includes(parseInt(questionId))) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const questionRequired = $questionWrapper.data("required");
|
||||||
|
const constrErrorMsg =
|
||||||
|
$questionWrapper.data("constrErrorMsg") ||
|
||||||
|
_t("This question requires an answer.");
|
||||||
|
if (questionRequired && !(this.files && this.files.length > 0)) {
|
||||||
|
errors[questionId] = constrErrorMsg;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (this.files && this.files.length > 0) {
|
||||||
|
const file = this.files[0];
|
||||||
|
const maxSizeMB = parseInt($(this).data("maxFileSize"));
|
||||||
|
if (maxSizeMB && file.size > maxSizeMB * 1024 * 1024) {
|
||||||
|
errors[questionId] = _t(
|
||||||
|
"The file exceeds the maximum allowed size of %s MB.",
|
||||||
|
maxSizeMB
|
||||||
|
);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const allowedExtensions = $(this).data("allowedExtensions");
|
||||||
|
if (allowedExtensions) {
|
||||||
|
const allowed = allowedExtensions
|
||||||
|
.split(",")
|
||||||
|
.map((e) => e.trim().toLowerCase());
|
||||||
|
const ext = "." + file.name.split(".").pop().toLowerCase();
|
||||||
|
if (!allowed.includes(ext)) {
|
||||||
|
errors[questionId] = _t(
|
||||||
|
"This file type is not allowed. Accepted formats: %s.",
|
||||||
|
allowedExtensions
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
|
||||||
|
|
||||||
if (_.keys(errors).length > 0) {
|
|
||||||
this._showErrors(errors);
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
return result;
|
});
|
||||||
},
|
|
||||||
});
|
if (Object.keys(errors).length > 0) {
|
||||||
|
this._showErrors(errors);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
export default publicWidget.registry.SurveyFormWidget;
|
||||||
|
|||||||
@@ -46,14 +46,14 @@ class TestSurveyFileCommon(common.TestSurveyCommon):
|
|||||||
|
|
||||||
|
|
||||||
class TestSurveyFileSaveLines(TestSurveyFileCommon):
|
class TestSurveyFileSaveLines(TestSurveyFileCommon):
|
||||||
"""Test the save_lines method for file question type."""
|
"""Test the _save_lines method for file question type."""
|
||||||
|
|
||||||
def test_save_file_answer(self):
|
def test_save_file_answer(self):
|
||||||
"""Submitting a file stores base64 data and filename."""
|
"""Submitting a file stores base64 data and filename."""
|
||||||
answer = self._add_answer(self.survey, self.survey_manager.partner_id)
|
answer = self._add_answer(self.survey, self.survey_manager.partner_id)
|
||||||
file_json = json.dumps({"data": self.file_b64, "name": self.file_name})
|
file_json = json.dumps({"data": self.file_b64, "name": self.file_name})
|
||||||
|
|
||||||
answer.save_lines(self.question_file, file_json)
|
answer._save_lines(self.question_file, file_json)
|
||||||
|
|
||||||
line = answer.user_input_line_ids.filtered(
|
line = answer.user_input_line_ids.filtered(
|
||||||
lambda l: l.question_id == self.question_file
|
lambda l: l.question_id == self.question_file
|
||||||
@@ -68,7 +68,7 @@ class TestSurveyFileSaveLines(TestSurveyFileCommon):
|
|||||||
"""Submitting empty answer marks the line as skipped."""
|
"""Submitting empty answer marks the line as skipped."""
|
||||||
answer = self._add_answer(self.survey, self.survey_manager.partner_id)
|
answer = self._add_answer(self.survey, self.survey_manager.partner_id)
|
||||||
|
|
||||||
answer.save_lines(self.question_file, "")
|
answer._save_lines(self.question_file, "")
|
||||||
|
|
||||||
line = answer.user_input_line_ids.filtered(
|
line = answer.user_input_line_ids.filtered(
|
||||||
lambda l: l.question_id == self.question_file
|
lambda l: l.question_id == self.question_file
|
||||||
@@ -81,11 +81,11 @@ class TestSurveyFileSaveLines(TestSurveyFileCommon):
|
|||||||
"""Submitting a new file updates the existing answer line."""
|
"""Submitting a new file updates the existing answer line."""
|
||||||
answer = self._add_answer(self.survey, self.survey_manager.partner_id)
|
answer = self._add_answer(self.survey, self.survey_manager.partner_id)
|
||||||
file_json_1 = json.dumps({"data": self.file_b64, "name": "first.pdf"})
|
file_json_1 = json.dumps({"data": self.file_b64, "name": "first.pdf"})
|
||||||
answer.save_lines(self.question_file, file_json_1)
|
answer._save_lines(self.question_file, file_json_1)
|
||||||
|
|
||||||
new_b64 = base64.b64encode(b"Updated content").decode()
|
new_b64 = base64.b64encode(b"Updated content").decode()
|
||||||
file_json_2 = json.dumps({"data": new_b64, "name": "second.pdf"})
|
file_json_2 = json.dumps({"data": new_b64, "name": "second.pdf"})
|
||||||
answer.save_lines(self.question_file, file_json_2)
|
answer._save_lines(self.question_file, file_json_2)
|
||||||
|
|
||||||
lines = answer.user_input_line_ids.filtered(
|
lines = answer.user_input_line_ids.filtered(
|
||||||
lambda l: l.question_id == self.question_file
|
lambda l: l.question_id == self.question_file
|
||||||
@@ -98,9 +98,9 @@ class TestSurveyFileSaveLines(TestSurveyFileCommon):
|
|||||||
"""Uploading a file then submitting empty marks line as skipped."""
|
"""Uploading a file then submitting empty marks line as skipped."""
|
||||||
answer = self._add_answer(self.survey, self.survey_manager.partner_id)
|
answer = self._add_answer(self.survey, self.survey_manager.partner_id)
|
||||||
file_json = json.dumps({"data": self.file_b64, "name": self.file_name})
|
file_json = json.dumps({"data": self.file_b64, "name": self.file_name})
|
||||||
answer.save_lines(self.question_file, file_json)
|
answer._save_lines(self.question_file, file_json)
|
||||||
|
|
||||||
answer.save_lines(self.question_file, "")
|
answer._save_lines(self.question_file, "")
|
||||||
|
|
||||||
line = answer.user_input_line_ids.filtered(
|
line = answer.user_input_line_ids.filtered(
|
||||||
lambda l: l.question_id == self.question_file
|
lambda l: l.question_id == self.question_file
|
||||||
@@ -117,7 +117,7 @@ class TestSurveyFileConstraints(TestSurveyFileCommon):
|
|||||||
self.question_file.write({"max_file_size": 0, "allowed_extensions": False})
|
self.question_file.write({"max_file_size": 0, "allowed_extensions": False})
|
||||||
answer = self._add_answer(self.survey, self.survey_manager.partner_id)
|
answer = self._add_answer(self.survey, self.survey_manager.partner_id)
|
||||||
file_json = json.dumps({"data": self.file_b64, "name": "anything.exe"})
|
file_json = json.dumps({"data": self.file_b64, "name": "anything.exe"})
|
||||||
answer.save_lines(self.question_file, file_json)
|
answer._save_lines(self.question_file, file_json)
|
||||||
line = answer.user_input_line_ids.filtered(
|
line = answer.user_input_line_ids.filtered(
|
||||||
lambda l: l.question_id == self.question_file
|
lambda l: l.question_id == self.question_file
|
||||||
)
|
)
|
||||||
@@ -128,7 +128,7 @@ class TestSurveyFileConstraints(TestSurveyFileCommon):
|
|||||||
self.question_file.write({"max_file_size": 10})
|
self.question_file.write({"max_file_size": 10})
|
||||||
answer = self._add_answer(self.survey, self.survey_manager.partner_id)
|
answer = self._add_answer(self.survey, self.survey_manager.partner_id)
|
||||||
file_json = json.dumps({"data": self.file_b64, "name": self.file_name})
|
file_json = json.dumps({"data": self.file_b64, "name": self.file_name})
|
||||||
answer.save_lines(self.question_file, file_json)
|
answer._save_lines(self.question_file, file_json)
|
||||||
line = answer.user_input_line_ids.filtered(
|
line = answer.user_input_line_ids.filtered(
|
||||||
lambda l: l.question_id == self.question_file
|
lambda l: l.question_id == self.question_file
|
||||||
)
|
)
|
||||||
@@ -142,14 +142,14 @@ class TestSurveyFileConstraints(TestSurveyFileCommon):
|
|||||||
answer = self._add_answer(self.survey, self.survey_manager.partner_id)
|
answer = self._add_answer(self.survey, self.survey_manager.partner_id)
|
||||||
file_json = json.dumps({"data": large_b64, "name": self.file_name})
|
file_json = json.dumps({"data": large_b64, "name": self.file_name})
|
||||||
with self.assertRaises(Exception):
|
with self.assertRaises(Exception):
|
||||||
answer.save_lines(self.question_file, file_json)
|
answer._save_lines(self.question_file, file_json)
|
||||||
|
|
||||||
def test_allowed_extension_passes(self):
|
def test_allowed_extension_passes(self):
|
||||||
"""File with an allowed extension passes."""
|
"""File with an allowed extension passes."""
|
||||||
self.question_file.write({"allowed_extensions": ".pdf,.docx"})
|
self.question_file.write({"allowed_extensions": ".pdf,.docx"})
|
||||||
answer = self._add_answer(self.survey, self.survey_manager.partner_id)
|
answer = self._add_answer(self.survey, self.survey_manager.partner_id)
|
||||||
file_json = json.dumps({"data": self.file_b64, "name": "report.docx"})
|
file_json = json.dumps({"data": self.file_b64, "name": "report.docx"})
|
||||||
answer.save_lines(self.question_file, file_json)
|
answer._save_lines(self.question_file, file_json)
|
||||||
line = answer.user_input_line_ids.filtered(
|
line = answer.user_input_line_ids.filtered(
|
||||||
lambda l: l.question_id == self.question_file
|
lambda l: l.question_id == self.question_file
|
||||||
)
|
)
|
||||||
@@ -161,14 +161,14 @@ class TestSurveyFileConstraints(TestSurveyFileCommon):
|
|||||||
answer = self._add_answer(self.survey, self.survey_manager.partner_id)
|
answer = self._add_answer(self.survey, self.survey_manager.partner_id)
|
||||||
file_json = json.dumps({"data": self.file_b64, "name": "script.exe"})
|
file_json = json.dumps({"data": self.file_b64, "name": "script.exe"})
|
||||||
with self.assertRaises(Exception):
|
with self.assertRaises(Exception):
|
||||||
answer.save_lines(self.question_file, file_json)
|
answer._save_lines(self.question_file, file_json)
|
||||||
|
|
||||||
def test_both_constraints_valid(self):
|
def test_both_constraints_valid(self):
|
||||||
"""File respecting both size and extension constraints passes."""
|
"""File respecting both size and extension constraints passes."""
|
||||||
self.question_file.write({"max_file_size": 10, "allowed_extensions": ".pdf"})
|
self.question_file.write({"max_file_size": 10, "allowed_extensions": ".pdf"})
|
||||||
answer = self._add_answer(self.survey, self.survey_manager.partner_id)
|
answer = self._add_answer(self.survey, self.survey_manager.partner_id)
|
||||||
file_json = json.dumps({"data": self.file_b64, "name": self.file_name})
|
file_json = json.dumps({"data": self.file_b64, "name": self.file_name})
|
||||||
answer.save_lines(self.question_file, file_json)
|
answer._save_lines(self.question_file, file_json)
|
||||||
line = answer.user_input_line_ids.filtered(
|
line = answer.user_input_line_ids.filtered(
|
||||||
lambda l: l.question_id == self.question_file
|
lambda l: l.question_id == self.question_file
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -8,10 +8,10 @@
|
|||||||
<field name="arch" type="xml">
|
<field name="arch" type="xml">
|
||||||
<xpath expr="//page[@name='options']//field[@name='matrix_subtype']" position="before">
|
<xpath expr="//page[@name='options']//field[@name='matrix_subtype']" position="before">
|
||||||
<field name="max_file_size"
|
<field name="max_file_size"
|
||||||
attrs="{'invisible': [('question_type', '!=', 'file')]}"
|
invisible="question_type != 'file'"
|
||||||
string="Max File Size (MB)"/>
|
string="Max File Size (MB)"/>
|
||||||
<field name="allowed_extensions"
|
<field name="allowed_extensions"
|
||||||
attrs="{'invisible': [('question_type', '!=', 'file')]}"
|
invisible="question_type != 'file'"
|
||||||
placeholder=".pdf,.docx,.xlsx"/>
|
placeholder=".pdf,.docx,.xlsx"/>
|
||||||
</xpath>
|
</xpath>
|
||||||
</field>
|
</field>
|
||||||
|
|||||||
@@ -6,7 +6,7 @@
|
|||||||
id="question_container_inh_type_file"
|
id="question_container_inh_type_file"
|
||||||
inherit_id="survey.question_container"
|
inherit_id="survey.question_container"
|
||||||
>
|
>
|
||||||
<xpath expr="//t[@t-call='survey.question_matrix']/.." position="after">
|
<xpath expr="//t[@t-call='survey.question_matrix']" position="after">
|
||||||
<t t-if="question.question_type == 'file'">
|
<t t-if="question.question_type == 'file'">
|
||||||
<t t-call="survey_extra_fields.question_file"/>
|
<t t-call="survey_extra_fields.question_file"/>
|
||||||
</t>
|
</t>
|
||||||
|
|||||||
@@ -8,7 +8,7 @@
|
|||||||
<field name="arch" type="xml">
|
<field name="arch" type="xml">
|
||||||
<xpath expr="//field[@name='suggested_answer_id']" position="after">
|
<xpath expr="//field[@name='suggested_answer_id']" position="after">
|
||||||
<field name="value_file" filename="value_file_fname" colspan="2"
|
<field name="value_file" filename="value_file_fname" colspan="2"
|
||||||
attrs="{'invisible': [('answer_type', '!=', 'file')]}"/>
|
invisible="answer_type != 'file'"/>
|
||||||
<field name="value_file_fname" invisible="1"/>
|
<field name="value_file_fname" invisible="1"/>
|
||||||
</xpath>
|
</xpath>
|
||||||
</field>
|
</field>
|
||||||
|
|||||||
Reference in New Issue
Block a user