Changes for page Get in Touch About Your XWiki Project
Last modified by Agnease on 2026/06/16 17:18
Summary
-
Page properties (1 modified, 0 added, 0 removed)
-
Objects (2 modified, 1 added, 2 removed)
Details
- Page properties
-
- Content
-
... ... @@ -1,148 +67,48 @@ 1 -{{velocity wiki="false"}} 2 -#if ($xcontext.action == 'get') 3 - #set ($statusCode = 400) 4 - #set ($message = 'The request could not be sent. Please try again or contact Agnease by email at alex@agnease.com.') 5 - 6 - #set ($name = '') 7 - #set ($email = '') 8 - #set ($contactWebsite = '') 9 - 10 - #foreach ($parameterName in $request.parameterNames) 11 - #set ($propertyParts = $parameterName.split('_0_')) 12 - #if ($propertyParts.size() > 1) 13 - #set ($propertyName = $parameterName.split('_0_')[1]) 14 - #if ($propertyName == 'name') 15 - #set ($name = $stringtool.trim($request.get($parameterName))) 16 - #elseif ($propertyName == 'email') 17 - #set ($email = $stringtool.trim($request.get($parameterName))) 18 - #elseif ($propertyName == 'contactWebsite') 19 - #set ($contactWebsite = $stringtool.trim($request.get($parameterName))) 20 - #elseif ($propertyName == 'contactStartedAt') 21 - #set ($startedAtRaw = $stringtool.trim($request.get($parameterName))) 22 - #end 23 - #end 24 - #end 25 - 26 - #if ("$!contactWebsite.trim()" != '') 27 - #set ($statusCode = 400) 28 - #set ($message = 'The request could not be sent. Please try again or contact Agnease by email.') 29 - #elseif ("$!name" == '' && "$!email" == '') 30 - #set ($statusCode = 400) 31 - #set ($message = 'Please enter your name and email.') 32 - #elseif ("$!name" == '') 33 - #set ($statusCode = 400) 34 - #set ($message = 'Please enter your name.') 35 - #elseif ("$!email" == '') 36 - #set ($statusCode = 400) 37 - #set ($message = 'Please enter your email address.') 38 - #else 39 - #try('contactException') 40 - #set ($now = $datetool.get('yyyyMMddHHmm')) 41 - #set ($random = $mathtool.random(100000, 999999)) 42 - #set ($uniqueName = "ContactRequest-${now}-${random}") 43 - #set ($contactRequestDoc = $xwiki.getDocumentAsAuthor('ContactRequests.' + $uniqueName)) 44 - #set ($contactRequestObj = $contactRequestDoc.getObject('Agnease.Code.ContactRequest.ContactRequestClass', true)) 45 - 46 - #foreach ($parameterName in $request.parameterNames) 47 - #set ($propertyName = $parameterName.split('_0_')[1]) 48 - #set ($discard = $contactRequestObj.set($propertyName, $request.get($parameterName))) 49 - #end 50 - 51 - #set ($discard = $contactRequestDoc.saveAsAuthor()) 52 - #set ($statusCode = 200) 53 - #set ($message = 'Your request was successfully sent.') 54 - #end 55 - 56 - #if ("$!contactException" != '') 57 - #set ($statusCode = 400) 58 - #set ($message = 'The request could not be sent. Please try again or contact Agnease by email.') 59 - #end 60 - #end 61 - 62 - #set ($discard = $response.setStatus($statusCode)) 63 - #jsonResponse({'message': $message}) 64 -#end 65 -{{/velocity}} 66 - 67 67 {{velocity}} 68 68 #set ($discard = $xwiki.ssx.use('contact.WebHome')) 69 -#set ($xobject = $doc.getObject('Agnease.Code.ContactRequest.ContactRequestClass')) 70 -#set ($totalRequests = $services.query.xwql('from doc.object(Agnease.Code.ContactRequest.ContactRequestClass) contact').execute()) 3 +#set ($xobject = $doc.getObject('Agnease.Code.ContactForm.ContactFormClass')) 71 71 #set ($xclass = $xobject.xWikiClass) 72 72 #set ($editing = true) 73 73 = Tell Us More About Your Project = 74 74 You do not need to have a full specification. A short description is enough to start the conversation. 75 75 {{html clean="false"}} 76 -You can also <a href="https://calendly.com/alex-agnease/30min?back=1&month=2026-06" target="_blank">book a free XWiki review call</a> to discuss your current setup. 77 77 <div class="row"> 78 - <div class="xform col-md-7"> 79 - #if ($totalRequests.size() > 50) 80 - ## As a measure to avoid high load on website. 81 - Tell us more about your project at <a href="mailto:alex@agnease.com">alex@agnease.com</a> 82 - #else 83 - <form id="contactForm"> 84 - <dl> 85 - #foreach ($property in $xclass.properties) 86 - #if ($property.name == 'hosting') 87 - <hr> 88 - <h3>Optional project details</h3> 89 - <p>These details help us understand the scope and suggest practical next steps.</p> 90 - #end 91 - <dt #if (!$editing && $hasEdit) 92 - class="editableProperty" 93 - #set ($xobjectPropertyReference = $xobject.getPropertyReference($property.name)) 94 - data-property="$escapetool.xml($services.model.serialize($xobjectPropertyReference))" 95 - data-property-type="object"#end> 96 - ## This must match the id generated by the $doc.display() method below. 97 - #set ($propertyId = "${xclass.name}_${xobject.number}_$property.name") 98 - <label#if ($editing) for="$escapetool.xml($propertyId)"#end> 99 - $escapetool.xml($property.translatedPrettyName) 100 - </label> 101 - ## Support for specifying a translation key as hint in the property definition. 102 - <span class="xHint">$!escapetool.xml($services.localization.render($property.hint))</span> 103 - </dt> 104 - <dd>$doc.display($property.name, 'edit').replace('{{html clean="false" wiki="false"}}', '').replace("{{/html}}", '')</dd> 10 + <div class="xform col-xs-7"> 11 + <form id="contactForm"> 12 + <dl> 13 + #foreach ($property in $xclass.properties) 14 + #if ($property.name == 'hosting') 15 + <hr> 16 + <h3>Optional project details</h3> 17 + <p>These details help us understand the scope and suggest practical next steps.</p> 105 105 #end 106 - #if (!$xclass.properties || $xclass.properties.size() == 0) 107 - ## Keep the empty definition term in order to have valid HTML. 108 - <dt></dt> 109 - <dd>$escapetool.xml($services.localization.render('xclass.defaultObjectSheet.noProperties'))</dd> 110 - #end 111 - </dl> 112 - <p class="xHint">* Your information will only be used to respond to this request.</p> 113 - ## Hidden fields to catch requests filled by bots. 114 - <div class="contact-hp-wrapper" aria-hidden="true"> 115 - <label for="Agnease.Code.ContactRequest.ContactRequestClass_0_contactWebsite">Website</label> 116 - <input 117 - id="contactWebsite" 118 - type="text" 119 - name="Agnease.Code.ContactRequest.ContactRequestClass_0_contactWebsite" 120 - autocomplete="off" 121 - tabindex="-1" 122 - /> 123 - </div> 124 - <input type="hidden" name="Agnease.Code.ContactRequest.ContactRequestClass_0_contactStartedAt" value="$datetool.getsystemDate.time" /> 125 - <input id="contactSubmit" type="submit" class="btn btn-primary" value="Send my request"> 126 - </form> 127 - #end 128 - {{/html}} 129 - {{html clean="false" wiki="true"}} 130 - <div class="reviewNotifications"> 131 - <div class="hidden reviewNotificationSuccess"> 132 - 133 - {{success}}reviewNotification{{/success}} 134 - 135 - </div> 136 - <div class="hidden reviewNotificationError"> 137 - 138 - {{error}}reviewNotification{{/error}} 139 - 140 - </div> 141 - </div> 142 - {{/html}} 143 - {{html clean="false"}} 19 + <dt #if (!$editing && $hasEdit) 20 + class="editableProperty" 21 + #set ($xobjectPropertyReference = $xobject.getPropertyReference($property.name)) 22 + data-property="$escapetool.xml($services.model.serialize($xobjectPropertyReference))" 23 + data-property-type="object"#end> 24 + ## This must match the id generated by the $doc.display() method below. 25 + #set ($propertyId = "${xclass.name}_${xobject.number}_$property.name") 26 + <label#if ($editing) for="$escapetool.xml($propertyId)"#end> 27 + $escapetool.xml($property.translatedPrettyName) 28 + </label> 29 + ## Support for specifying a translation key as hint in the property definition. 30 + <span class="xHint">$!escapetool.xml($services.localization.render($property.hint))</span> 31 + </dt> 32 + <dd>$doc.display($property.name, 'edit').replace('{{html clean="false" wiki="false"}}', '').replace("{{/html}}", '')</dd> 33 + #end 34 + #if (!$xclass.properties || $xclass.properties.size() == 0) 35 + ## Keep the empty definition term in order to have valid HTML. 36 + <dt></dt> 37 + <dd>$escapetool.xml($services.localization.render('xclass.defaultObjectSheet.noProperties'))</dd> 38 + #end 39 + </dl> 40 + <p>Your information will only be used to respond to this request.</p> 41 + ##<p>Your information will only be used to respond to this request. See the Privacy Policy for details.</p> 42 + <input id="contactSubmit" type="submit" class="btn btn-primary" value="Send my request"> 43 + </form> 144 144 </div> 145 - <div class="col- md-5">45 + <div class="col-xs-5"> 146 146 <div class="widget"> 147 147 <h4>$services.icon.renderHTML('check') How Agnease can help</h4> 148 148 <ul> ... ... @@ -162,6 +162,20 @@ 162 162 <li>If useful, we schedule a short call to discuss scope, timeline, and estimated effort.</li> 163 163 </ol> 164 164 </div> 65 + {{/html}} 66 + {{html clean="false" wiki="true"}} 67 + <div class="reviewNotifications"> 68 + <div class="hidden reviewNotificationSuccess"> 69 + 70 + {{success}}reviewNotification{{/success}} 71 + 72 + </div> 73 + <div class="hidden reviewNotificationError"> 74 + 75 + {{error}}reviewNotification{{/error}} 76 + 77 + </div> 78 + </div> 165 165 </div> 166 166 </div> 167 167 {{/html}}
- Agnease.Code.ContactRequest.ContactRequestClass[0]
-
- alreadyUseXWiki
-
... ... @@ -1,1 +1,0 @@ 1 -1 - hosting
-
... ... @@ -1,1 +1,0 @@ 1 -1
- Agnease.Code.SEODetailsClass[0]
-
- metaDescription
-
... ... @@ -1,1 +1,0 @@ 1 -Contact Agnease for XWiki consulting, upgrades, support, custom development, integrations, migrations, authentication, security reviews and long-term platform care. - metaTitle
-
... ... @@ -1,1 +1,0 @@ 1 -Contact Agnease | XWiki Consulting and Support
- XWiki.JavaScriptExtension[0]
-
- code
-
... ... @@ -1,5 +1,5 @@ 1 1 require(['jquery'], function ($) { 2 - var serviceURL = new XWiki.Document('WebHome', 'cont act').getURL('get');2 + var serviceURL = new XWiki.Document('WebHome', 'content').getURL('get', 'xpage=plain'); 3 3 var form = $('#contactForm'); 4 4 var submitButton = $('#contactSubmit'); 5 5 ... ... @@ -6,33 +6,65 @@ 6 6 var successBox = $('.reviewNotificationSuccess'); 7 7 var errorBox = $('.reviewNotificationError'); 8 8 9 + var nameFieldName = 'Agnease.Code.ContactForm.ContactFormClass_0_name'; 10 + var emailFieldName = 'Agnease.Code.ContactForm.ContactFormClass_0_email'; 11 + 12 + function getFormData() { 13 + var data = {}; 14 + 15 + $.each(form.serializeArray(), function (_, field) { 16 + data[field.name] = field.value; 17 + }); 18 + console.log(data) 19 + 20 + return data; 21 + } 22 + 23 + function isValidEmail(value) { 24 + return /^[^\s@]+@[^\s@]+\.[^\s@]+$/.test(value); 25 + } 26 + 27 + function updateSubmitState() { 28 + var data = getFormData(); 29 + var name = $.trim(data[nameFieldName] || ''); 30 + var email = $.trim(data[emailFieldName] || ''); 31 + 32 + submitButton.prop('disabled', !(name.length > 0 && isValidEmail(email))); 33 + } 34 + 35 + form.on('input change keyup', 'input, textarea, select', updateSubmitState); 36 + updateSubmitState(); 37 + 9 9 form.on('submit', function (event) { 10 10 event.preventDefault(); 11 11 12 - // Always reset notifications before starting a new request. 13 - successBox.addClass('hidden'); 14 - errorBox.addClass('hidden'); 15 - successBox.find('.box div p').text(''); 16 - errorBox.find('.box div p').text(''); 41 + var data = getFormData(); 42 + 43 + console.log(data) 17 17 18 18 submitButton.prop('disabled', true); 19 - 46 +/* 20 20 $.post({ 21 21 url: serviceURL, 22 - data: $.param(form.serializeArray()) 23 - }).done(function (data) { 24 - console.log(data) 25 - successBox.find('.box div p').text(data.message); 26 - successBox.removeClass('hidden'); 49 + data: data 50 + }).done(function (response) { 51 + // replace with succcess message alert(response.message || 'Your request was sent successfully.'); 52 + //var message = 'The request could not be sent. Please try again or contact Agnease by email.'; 53 + var successBoxContent = successBox.find('.box div p'); 54 + successBoxContent.text(data.message); 55 + successBox.toggleClass('hidden'); 56 + if (errorBox.is(':visible')) { 57 + errorBox.toggleClass('hidden'); 58 + } 27 27 form[0].reset(); 28 28 }).fail(function (xhr) { 29 - console.log('fail'+xhr)30 - var message = xhr.responseJSON && xhr.responseJSON.message ?xhr.responseJSON.message31 - : 'Therequest could not be sent. Please try againorcontact Agnease by email at alex@agnease.com';32 - e rrorBox.find('.box divp').text(message);33 - e rrorBox.removeClass('hidden');34 - }).always(function(){61 + var errorBoxContent = errorBox.find('.box div p'); 62 + errorBoxContent.text(xhr.responseJSON.message); 63 + errorBox.toggleClass('hidden'); 64 + if (successBox.is(':visible')) { 65 + successBox.toggleClass('hidden'); 66 + } 35 35 submitButton.prop('disabled', false); 36 - }); 68 + });*/ 37 37 }); 38 38 });
- XWiki.StyleSheetExtension[0]
-
- code
-
... ... @@ -66,11 +66,3 @@ 66 66 color: @brand; 67 67 font-weight: 700; 68 68 } 69 -/* CSS for hidden field to identify requests filled by bots. */ 70 -.contact-hp-wrapper { 71 - position: absolute; 72 - left: -9999px; 73 - width: 1px; 74 - height: 1px; 75 - overflow: hidden; 76 -}
- Agnease.Code.ContactForm.ContactFormClass[0]
-
- alreadyUseXWiki
-
... ... @@ -1,0 +1,1 @@ 1 +1 - customDevelopment
-
... ... @@ -1,0 +1,1 @@ 1 +1 - hosting
-
... ... @@ -1,0 +1,1 @@ 1 +1 - users
-
... ... @@ -1,0 +1,1 @@ 1 +1