From 32e357a739915b3c51483971800e9734b80d8ecb Mon Sep 17 00:00:00 2001 From: Eniko Tot <eniko.tot@codebrewery.hu> Date: Thu, 11 Jan 2024 16:12:41 +0100 Subject: [PATCH] [WIP] Start refactoring and restructuring ApplyForm --- .../Bootstrap5/Joblist/ApplyForm.html | 624 ++++++++++++++++++ 1 file changed, 624 insertions(+) create mode 100644 Resources/Private/Templates/Bootstrap5/Joblist/ApplyForm.html diff --git a/Resources/Private/Templates/Bootstrap5/Joblist/ApplyForm.html b/Resources/Private/Templates/Bootstrap5/Joblist/ApplyForm.html new file mode 100644 index 00000000..0e1bd444 --- /dev/null +++ b/Resources/Private/Templates/Bootstrap5/Joblist/ApplyForm.html @@ -0,0 +1,624 @@ +{namespace h=SGalinski\SgJobs\ViewHelpers} +{namespace base=SGalinski\ProjectBase\ViewHelpers} +{namespace sgajax=SGalinski\SgAjax\ViewHelpers} + +<f:layout name="Default" /> +<f:section name="main"> + <f:asset.script identifier="dropzone-js" crossorigin="anonymous" + src="EXT:sg_jobs/Resources/Public/node_modules/dropzone/dist/dropzone-min.js" priority="1" /> + <f:asset.css identifier="dropzone-css" crossorigin="anonymous" + href="EXT:sg_jobs/Resources/Public/node_modules/dropzone/dist/dropzone.css" priority="1" /> + <f:if condition="{job}"> + <f:render partial="ApplyFormSchema" arguments="{_all}" /> + </f:if> + + <div class="default-content-element"> + <h1> + <f:if condition="{job}"> + <f:then> + <f:translate key="frontend.apply.applyAs" /> + <span>{job.title}</span> + </f:then> + <f:else> + <f:translate key="frontend.apply.unsolicitedApplication" /> + </f:else> + </f:if> + </h1> + </div> + <f:if condition="{job}"> + <div class="row"> + <div class="container"> + <div class="row default-content-element sgjobs-description"> + <input id="maxFileSize" type="hidden" data-maxFileSize="{maxFileSize}" /> + <input id="maxFileSizeMessage" type="hidden" data-maxFileSizeMessage="{maxFileSizeMessage}" /> + <div class="col-md-8 col-sm-6 col-xs-12"> + <div class="default-content-element"> + <f:format.html parseFuncTSPath="lib.parseFunc_RTE">{job.task}</f:format.html> + <h3 class="h4"> + <f:translate key="frontend.qualification" /> + </h3> + <f:format.html parseFuncTSPath="lib.parseFunc_RTE">{job.qualification}</f:format.html> + </div> + </div> + <div class="col-md-4 col-sm-6 col-xs-12"> + <div class="highlight-box bg-card sgjobs-meta-box"> + <ul class="default-list"> + <li> + <f:format.raw> + <f:translate key="frontend.jobStart" /> + </f:format.raw> + <f:if condition="{job.alternativeStartDate}"> + <f:then> + {job.alternativeStartDate} + </f:then> + <f:else> + <f:format.date date="{job.startDate}" format="d.m.Y" /> + </f:else> + </f:if> + </li> + <li> + <f:if condition="!{job.telecommutePossible}"> + <f:then> + <f:format.raw> + <f:translate key="frontend.locationLabel" /> + </f:format.raw><br> + <f:for as="company" each="{job.companies}"> + <span class="sgjobs-company-location-wrap"> + {company.name}<br> + {company.street}<br> + <f:if condition="{company.state}"> + {company.state}<br> + </f:if> + {company.zip} {company.city}<br> + </span> + </f:for> + </f:then> + <f:else> + <f:format.raw> + <f:translate key="frontend.jobLocationRemote" /> + </f:format.raw> + </f:else> + </f:if> + </li> + <f:if condition="{job.experienceLevel -> f:count()} > 0"> + <li> + <strong> + <f:translate key="frontend.experienceLevel" />: + </strong> + <f:for each="{job.experienceLevel}" as="experienceLevel" iteration="iterator"> + {experienceLevel.title}{f:if(condition: '!{iterator.isLast}', then: ', ')} + </f:for> + </li> + </f:if> + </ul> + + <hr> + <h3> + <f:format.raw> + <f:translate key="frontend.jobApplyNow" /> + </f:format.raw> + </h3> + + <f:if condition="!{job.hideApplyByPostal}"> + <p> + <f:format.raw> + <f:translate key="frontend.job.via.post" /> + </f:format.raw><br> + {job.company.name}<br> + <f:if condition="{job.contact}"> + <f:then> + {job.contact.title} {job.contact.firstName} {job.contact.lastName}<br> + </f:then> + <f:else> + {job.company.contact.title} {job.company.contact.firstName} + {job.company.contact.lastName}<br> + </f:else> + </f:if> + <f:if condition="{job.contact} && {job.contact.street}"> + <f:then> + {job.contact.street}<br> + <f:if condition="{job.contact.state}"> + {job.contact.state}<br> + </f:if> + {job.contact.zip} {job.contact.city} + <f:if condition="{job.contact.country}"> + <br>{job.contact.country} + </f:if> + </f:then> + <f:else> + <f:if condition="{job.company.contact} && {job.company.contact.street}"> + <f:then> + {job.company.contact.street}<br> + <f:if condition="{job.company.contact.state}"> + {job.company.contact.state}<br> + </f:if> + {job.company.contact.zip} {job.company.contact.city} + <f:if condition="{job.company.contact.country}"> + <br>{job.company.contact.country} + </f:if> + </f:then> + <f:else> + {job.company.street}<br> + <f:if condition="{job.company.state}"> + {job.company.state}<br> + </f:if> + {job.company.zip} {job.company.city} + <f:if condition="{job.company.country}"> + <br>{job.company.country} + </f:if> + </f:else> + </f:if> + </f:else> + </f:if> + </p> + </f:if> + + <f:if condition="!{job.hideApplyByEmail}"> + <p> + <f:format.raw> + <f:translate key="frontend.job.via.email" /> + </f:format.raw> + <br> + <f:comment><!-- Spam Protection (lib.parseFunc encodes adresses) --></f:comment> + <f:if condition="{job.contact}"> + <f:then> + <f:format.html parseFuncTSPath="lib.parseFunc"><a + href="mailto:{job.contact.email}"> + <f:translate key="frontend.emailContact" /> + </a></f:format.html> + </f:then> + <f:else> + <f:format.html parseFuncTSPath="lib.parseFunc"><a + href="mailto:{job.company.contact.email}"> + <f:translate key="frontend.emailContact" /> + </a></f:format.html> + </f:else> + </f:if> + </p> + </f:if> + + <p> + <f:format.raw> + <f:translate key="frontend.job.suggestForm" /> + </f:format.raw> + </p> + + <div class="default-content-element sg-cta sg-cta-with-icon"> + <a href="{f:if(condition: '{job.applyExternalLink}', then: '{job.applyExternalLink}', else: '#apply')}" + class="btn btn-warning btn-lg"> + <f:translate key="frontend.applyNow" /> + </a> + </div> + </div> + <f:if condition="{job.contact}"> + <f:then> + <f:render section="contactBox" + arguments="{contact: job.contact, hideApplyByEmail: job.hideApplyByEmail}" /> + </f:then> + <f:else> + <f:render section="contactBox" + arguments="{contact: job.company.contact, hideApplyByEmail: job.hideApplyByEmail}" /> + </f:else> + </f:if> + <f:if condition="{job.attachment}"> + <div class="highlight-box bg-card sgjobs-meta-box"> + <div class="default-content-element downloads"> + <f:link.typolink target="_blank" + parameter="{job.attachment.0.originalResource.publicUrl}" class="download-link"> + <span class="download-icon download-pdf"> + </span> + <span class="download-link__file-description"> + <f:translate key="frontend.attachment" extensionName="SgJobs" /> + <span class="download-link__file-properties">(PDF, + {job.attachment.0.originalResource.size -> f:format.bytes()})</span> + </span> + </f:link.typolink> + </div> + </div> + </f:if> + </div> + </div> + </div> + </div> + </f:if> + <f:if condition="!{job.applyExternalLink}"> + <div class="row default-content-element"> + <div class="col-md-10 col-sm-10 col-xs-12"> + <f:if condition="{job}"> + <div class="default-content-element"> + <h2> + <f:translate key="frontend.apply.applyAsNow" arguments="{0: '{job.title}'}" /> + </h2> + </div> + </f:if> + <f:form action="apply" class="was-validated" id="apply" controller="Joblist" method="post" + name="applyData" object="{applyData}" enctype="multipart/form-data"> + <f:if condition="{job}"> + <f:then> + <f:form.hidden property="job" value="{job.uid}" /> + <f:form.hidden property="jobId" value="{job.jobId}" /> + <f:form.hidden property="jobTitle" value="{job.title}" /> + </f:then> + </f:if> + <input type="hidden" name="tx_sgjobs_jobapplication[folderName]" value="{folderName}" /> + + <f:if condition="{internalError}"> + <ul class="sg-jobs-validation-error parsley-errors-list filled"> + <li class="parsley-required"> + <f:translate key="frontend.apply.error.general" /> + : {internalError} + </li> + </ul> + </f:if> + <div class="row"> + <f:if condition="!{job}"> + <div class="col"> + <f:render section="formLabel" + arguments="{label-for: 'company', label-text: 'company'}" /> + <f:form.select property="company" multiple="0" size="1" id="apply-company" + class="form-select" options="{companies}" optionLabelField="city" + optionValueField="uid" + prependOptionLabel="{f:translate(key:'frontend.apply.country.empty')}" + required="required" /> + <f:render section="formValidation" arguments="{form-field: 'company'}" /> + </div> + </f:if> + <div class="col"> + <f:render section="formLabel" arguments="{label-for: 'gender', label-text: 'gender'}" /> + <f:form.select property="gender" id="apply-gender" class="form-select" + options="{male: '{f:translate(key: \'frontend.apply.gender.male\')}', female: '{f:translate(key: \'frontend.apply.gender.female\')}', other: '{f:translate(key: \'frontend.apply.gender.other\')}'}" /> + <f:render section="formValidation" arguments="{form-field: 'gender'}" /> + </div> + </div> + <div class="row"> + <div class="col"> + <f:render section="formTextField" + arguments="{field-id: 'firstName', field-text: 'first_name', required: 'required'}" /> + <f:render section="formValidation" arguments="{form-field: 'firstName'}" /> + </div> + + <div class="col"> + <f:render section="formTextField" + arguments="{field-id: 'lastName', field-text: 'last_name', required: 'required'}" /> + <f:render section="formValidation" arguments="{form-field: 'lastName'}" /> + </div> + </div> + <div class="row"> + <div class="col-6"> + <f:render section="formTextField" + arguments="{field-id: 'street', field-text: 'street', required: 'required'}" /> + <f:render section="formValidation" arguments="{form-field: 'street'}" /> + </div> + + <div class="col-4"> + <f:render section="formTextField" + arguments="{field-id: 'city', field-text: 'city', required: 'required'}" /> + <f:render section="formValidation" arguments="{form-field: 'city'}" /> + </div> + + <div class="col-2"> + <f:render section="formTextField" + arguments="{field-id: 'zip', field-text: 'zip', required: 'required'}" /> + <f:render section="formValidation" arguments="{form-field: 'zip'}" /> + </div> + </div> + <div class="row"> + <div class="col"> + <f:render section="formLabel" arguments="{label-for: 'country', label-text: 'country'}" /> + <f:form.countrySelect value="DE" property="country" id="apply-country" class="form-select" + required="required" /> + <f:render section="formValidation" arguments="{form-field: 'county'}" /> + </div> + + <div class="col"> + <f:render section="formLabel" + arguments="{label-for: 'nationality', label-text: 'nationality'}" /> + <f:form.countrySelect value="DE" property="nationality" id="apply-nationality" + class="form-select" required="required" /> + <f:render section="formValidation" arguments="{form-field: 'nationality'}" /> + </div> + </div> + <div class="row"> + <div class="col"> + <f:render section="formTextField" + arguments="{field-id: 'education', field-text: 'education', required: 'required'}" /> + <f:render section="formValidation" arguments="{form-field: 'education'}" /> + </div> + + <div class="col"> + <f:render section="formLabel" + arguments="{label-for: 'birthDate', label-text: 'birthDate'}" /> + <f:form.textfield property="birthDate" id="apply-birthDate" class="form-control" + placeholder="{f:translate(key:'frontend.apply.birthDate')}" required="required" /> + <f:render section="formValidation" arguments="{form-field: 'birthDate'}" /> + </div> + </div> + <div class="row"> + <div class="col"> + <f:render section="formTextField" + arguments="{field-id: 'phone', field-text: 'phone', required: 'required'}" /> + <f:render section="formValidation" arguments="{form-field: 'phone'}" /> + </div> + + <div class="col"> + <f:render section="formTextField" arguments="{field-id: 'mobile', field-text: 'mobile'}" /> + <f:render section="formValidation" arguments="{form-field: 'mobile'}" /> + </div> + </div> + <div class="row"> + <div class="col-6"> + <f:render section="formLabel" arguments="{label-for: 'email', label-text: 'email'}" /> + <f:form.textfield type="email" property="email" id="apply-email" class="form-control" + placeholder="{f:translate(key:'frontend.apply.email')}" required="required" /> + <f:render section="formValidation" arguments="{form-field: 'email'}" /> + </div> + </div> + <div class="row"> + <div class="col"> + <div class="form-group jobs-upload-group"> + <label for="apply-cover-letter" class="form-label filled"> + <f:translate key="frontend.apply.cover_letter" /> + ( + <f:translate key="frontend.apply.allowed_file_extensions" /> + {allowedFileExtensions}) + </label> + <div class="coverLetter-upload jobs-upload" data-max-file-amount="1" + data-valid-file-extensions="{settings.fileUpload.fileTypes}" + data-max-file-size="{maxFileSize}" data-pid="{storagePid}" + data-inner-text="{f:translate(key: 'frontend.DropFiles', extensionName: 'sg_jobs')}" + data-cancel-upload="{f:translate(key: 'frontend.CancelUpload', extensionName: 'sg_jobs')}" + data-remove-file="{f:translate(key: 'frontend.RemoveFile', extensionName: 'sg_jobs')}" + data-file-type-error="{f:translate(key: 'frontend.FileType', extensionName: 'sg_jobs')}" + data-upload-ajax="{sgajax:uri.ajax(extensionName: 'SgJobs', controller: 'Ajax\\Upload', action: 'uploadCoverletter', format: 'json', parameters: '{pageId: storagePid}')}"> + </div> + <f:if condition="{coverLetter.name}"> + <p class="help-block"> + Aktuell: {coverLetter.name} + <input type="hidden" + name="tx_sgjobs_jobapplication[applyData][coverLetter][submittedFile][name]" + value="{coverLetter.name}" /> + <input type="hidden" + name="tx_sgjobs_jobapplication[applyData][coverLetter][submittedFile][type]" + value="{coverLetter.type}" /> + <input type="hidden" + name="tx_sgjobs_jobapplication[applyData][coverLetter][submittedFile][tmp_name]" + value="{coverLetter.tmp_name}" /> + <input type="hidden" + name="tx_sgjobs_jobapplication[applyData][coverLetter][submittedFile][error]" + value="0" /> + <input type="hidden" + name="tx_sgjobs_jobapplication[applyData][coverLetter][submittedFile][size]" + value="{coverLetter.size}" /> + </p> + </f:if> + <f:render section="formValidation" arguments="{form-field: 'coverLetter'}" /> + </div> + </div> + </div> + <div class="col-xs-12"> + <div class="form-group jobs-upload-group"> + <label for="apply-cv" class="smart-label filled"> + <f:translate key="frontend.apply.cv" /> + ( + <f:translate key="frontend.apply.allowed_file_extensions" /> + {allowedFileExtensions}) + </label> + + <div class="cv-upload jobs-upload" data-max-file-amount="1" + data-valid-file-extensions="{settings.fileUpload.fileTypes}" + data-max-file-size="{maxFileSize}" data-pid="{storagePid}" + data-inner-text="{f:translate(key: 'frontend.DropFiles', extensionName: 'sg_jobs')}" + data-cancel-upload="{f:translate(key: 'frontend.CancelUpload', extensionName: 'sg_jobs')}" + data-remove-file="{f:translate(key: 'frontend.RemoveFile', extensionName: 'sg_jobs')}" + data-file-type-error="{f:translate(key: 'frontend.FileType', extensionName: 'sg_jobs')}" + data-upload-ajax="{sgajax:uri.ajax(extensionName: 'SgJobs', controller: 'Ajax\\Upload', action: 'uploadCv', format: 'json', parameters: '{pageId: storagePid}')}"> + </div> + <f:if condition="{cv.name}"> + <p class="help-block"> + Aktuell: {cv.name} + + <f:comment> + <!-- Important, due to a fluid cache issue with the fluid syntax--> + </f:comment> + <input type="hidden" + name="tx_sgjobs_jobapplication[applyData][cv][submittedFile][name]" + value="{cv.name}" /> + <input type="hidden" + name="tx_sgjobs_jobapplication[applyData][cv][submittedFile][type]" + value="{cv.type}" /> + <input type="hidden" + name="tx_sgjobs_jobapplication[applyData][cv][submittedFile][tmp_name]" + value="{cv.tmp_name}" /> + <input type="hidden" + name="tx_sgjobs_jobapplication[applyData][cv][submittedFile][error]" + value="0" /> + <input type="hidden" + name="tx_sgjobs_jobapplication[applyData][cv][submittedFile][size]" + value="{cv.size}" /> + </p> + </f:if> + + <f:render section="formValidation" arguments="{form-field: 'cv'}" /> + </div> + </div> + + <div class="col-xs-12"> + <div class="form-group jobs-upload-group"> + <label for="apply-certificate" class="smart-label filled"> + <f:translate key="frontend.apply.certificate" /> + ( + <f:translate key="frontend.apply.allowed_file_extensions" /> + {allowedFileExtensions}) + </label> + + <div class="certificate-upload jobs-upload" data-max-file-amount="1" + data-valid-file-extensions="{settings.fileUpload.fileTypes}" + data-max-file-size="{maxFileSize}" data-pid="{storagePid}" + data-inner-text="{f:translate(key: 'frontend.DropFiles', extensionName: 'sg_jobs')}" + data-cancel-upload="{f:translate(key: 'frontend.CancelUpload', extensionName: 'sg_jobs')}" + data-remove-file="{f:translate(key: 'frontend.RemoveFile', extensionName: 'sg_jobs')}" + data-file-type-error="{f:translate(key: 'frontend.FileType', extensionName: 'sg_jobs')}" + data-upload-ajax="{sgajax:uri.ajax(extensionName: 'SgJobs', controller: 'Ajax\\Upload', action: 'uploadCv', format: 'json', parameters: '{pageId: storagePid}')}"> + </div> + <f:if condition="{certificate.name}"> + <p class="help-block"> + Aktuell: {certificate.name} + + <f:comment> + <!-- Important, due to a fluid cache issue with the fluid syntax--> + </f:comment> + <input type="hidden" + name="tx_sgjobs_jobapplication[applyData][certificate][submittedFile][name]" + value="{certificate.name}" /> + <input type="hidden" + name="tx_sgjobs_jobapplication[applyData][certificate][submittedFile][type]" + value="{certificate.type}" /> + <input type="hidden" + name="tx_sgjobs_jobapplication[applyData][certificate][submittedFile][tmp_name]" + value="{certificate.tmp_name}" /> + <input type="hidden" + name="tx_sgjobs_jobapplication[applyData][certificate][submittedFile][error]" + value="0" /> + <input type="hidden" + name="tx_sgjobs_jobapplication[applyData][certificate][submittedFile][size]" + value="{certificate.size}" /> + </p> + </f:if> + + <f:render section="formValidation" arguments="{form-field: 'certificate'}" /> + </div> + </div> + + <div class="col-xs-12"> + <div class="form-group"> + <label for="apply-message" class="smart-label"> + <f:translate key="frontend.apply.message" /> + </label> + <f:form.textarea class="form-control" rows="10" property="message" id="apply-message" + placeholder="{f:translate(key:'frontend.apply.message')}" /> + </div> + + <div class="form-group"> + <label> + <f:form.checkbox id="privacy-policy" property="privacyPolicy" value="1" /> + <f:format.raw> + <f:translate key="frontend.apply.privacyPolicy" + arguments="{0: '{f:render(section:\'privacyPolicyCheckboxLink\')}'}" /> + </f:format.raw> + </label> + <f:render section="formValidation" arguments="{form-field: 'privacyPolicy'}" /> + </div> + </div> + <button type="submit" class="btn btn-lg btn-warning">{f:translate(key:'frontend.applyNow')} <i + class="fa fa-paper-plane" aria-hidden="true"></i></button> + </f:form> + </div> + </div> + </f:if> + + <f:if condition="{job.relatedJobs}"> + <f:then> + <div class="row"> + <div class="container"> + <h2> + <f:translate key="frontend.apply.relatedJobs" /> + </h2> + <div class="row default-content-element equal-height-columns stretch-first-child"> + <f:for each="{job.relatedJobs}" as="relatedJob"> + <f:render partial="Teaser" + arguments="{job: relatedJob, applyPageUid: settings.applyPage}" /> + </f:for> + </div> + </div> + </div> + </f:then> + <f:else> + <f:if condition="{settings.enableAutomaticRelatedJobs} && {relatedJobs}"> + <div class="row"> + <div class="container"> + <h2> + <f:translate key="frontend.apply.relatedJobs" /> + </h2> + <div class="row default-content-element equal-height-columns stretch-first-child"> + <f:for each="{relatedJobs}" as="relatedJob"> + <f:render partial="Teaser" + arguments="{job: relatedJob, applyPageUid: settings.applyPage}" /> + </f:for> + </div> + </div> + </div> + </f:if> + </f:else> + </f:if> +</f:section> + +<f:section name="privacyPolicyCheckboxLink"> + <f:link.typolink target="_blank" parameter="{settings.privacyPolicyPage}"> + <f:translate key="frontend.apply.privacyPolicy.link" /> + </f:link.typolink> +</f:section> + +<f:section name="contactBox"> + <div class="highlight-box bg-card sgjobs-meta-box"> + <div class="default-content-element sg-jobs-contact-box"> + <div class="sg-jobs-contact-box__text"> + <p class="h4"> + <f:translate key="frontend.apply.contact" /> + </p> + {contact.title} {contact.firstName} {contact.lastName} + <f:if condition="{contact.phone}"> + <br />{contact.phone} + </f:if> + <f:if condition="{contact.email}"> + <br /> + <f:if condition="!{hideApplyByEmail}"> + <p> + <f:comment> + <!-- Spam Protection (lib.parseFunc encodes adresses) --> + </f:comment> + <f:format.html parseFuncTSPath="lib.parseFunc"><a href="mailto:{contact.email}"> + <f:translate key="frontend.emailContact" /> + </a></f:format.html> + </p> + </f:if> + </f:if> + </div> + <f:if condition="{contact.image}"> + <f:image image="{contact.image}" maxWidth="100" maxHeight="100" + alt="{contact.title} {contact.firstName} {contact.lastName}" /> + </f:if> + </div> + <hr> + <div class="default-content-element sgjobs-social-sharer"> + <p class="h4"> + <f:translate key="frontend.apply.recommend" /> + </p> + <base:sharer /> + </div> + </div> +</f:section> + +<f:section name="formLabel"> + <label for="apply-{label-for}" class="form-label"> + <f:translate key="frontend.apply.{label-text}" /> + </label> +</f:section> + + +<f:section name="formTextField"> + <label for="apply-{field-id}" class="form-label"> + <f:translate key="frontend.apply.{field-text}" /> + </label> + <f:form.textfield property="{field-id}" id="apply-{field-id}" class="form-control" + placeholder="{f:translate(key:'frontend.apply.{field-text}')}" required="{required}" /> +</f:section> + +<f:section name="formValidation"> + <f:form.validationResults for="applyData.{form-field}"> + <ul class="text-danger pt-2"> + <f:for each="{validationResults.errors}" as="error"> + <li>{error.message}</li> + </f:for> + </ul> + </f:form.validationResults> +</f:section> \ No newline at end of file -- GitLab