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, 2 added, 1 removed)
Details
- Page properties
-
- Content
-
... ... @@ -1,12 +1,56 @@ 1 1 {{velocity wiki="false"}} 2 2 #if ($xcontext.action == 'get') 3 -test 4 - #set ($message = '') 5 - #set ($statusCode = 200) 6 - 7 7 #set ($statusCode = 400) 8 - #set ($message = 'error') 4 + #set ($message = 'The request could not be sent. Please try again or contact Agnease by email at alex@agnease.com.') 9 9 6 + #set ($name = '') 7 + #set ($email = '') 8 + 9 + #foreach ($parameterName in $request.parameterNames) 10 + #set ($propertyName = $parameterName.split('_0_')[1]) 11 + #if ($propertyName == 'name') 12 + #set ($name = $stringtool.trim($request.get($parameterName))) 13 + #elseif ($propertyName == 'email') 14 + #set ($email = $stringtool.trim($request.get($parameterName))) 15 + #end 16 + #end 17 + 18 + #if ("$!request.contactWebsite".trim() != '') 19 + #set ($statusCode = 400) 20 + #set ($message = 'The request could not be sent. Please try again or contact Agnease by email.') 21 + #elseif ("$!name" == '' && "$!email" == '') 22 + #set ($statusCode = 400) 23 + #set ($message = 'Please enter your name and email.') 24 + #elseif ("$!name" == '') 25 + #set ($statusCode = 400) 26 + #set ($message = 'Please enter your name.') 27 + #elseif ("$!email" == '') 28 + #set ($statusCode = 400) 29 + #set ($message = 'Please enter your email address.') 30 + #else 31 + #try('contactException') 32 + #set ($now = $datetool.get('yyyyMMddHHmm')) 33 + #set ($random = $mathtool.random(100000, 999999)) 34 + #set ($uniqueName = "ContactRequest-${now}-${random}") 35 + #set ($contactRequestDoc = $xwiki.getDocumentAsAuthor('ContactRequests.' + $uniqueName)) 36 + #set ($contactRequestObj = $contactRequestDoc.getObject('Agnease.Code.ContactRequest.ContactRequestClass', true)) 37 + 38 + #foreach ($parameterName in $request.parameterNames) 39 + #set ($propertyName = $parameterName.split('_0_')[1]) 40 + #set ($discard = $contactRequestObj.set($propertyName, $request.get($parameterName))) 41 + #end 42 + 43 + #set ($discard = $contactRequestDoc.saveAsAuthor()) 44 + #set ($statusCode = 200) 45 + #set ($message = 'Your request was successfully sent.') 46 + #end 47 + 48 + #if ("$!contactException" != '') 49 + #set ($statusCode = 400) 50 + #set ($message = 'The request could not be sent. Please try again or contact Agnease by email.') 51 + #end 52 + #end 53 + 10 10 #set ($discard = $response.setStatus($statusCode)) 11 11 #jsonResponse({'message': $message}) 12 12 #end ... ... @@ -14,47 +14,64 @@ 14 14 15 15 {{velocity}} 16 16 #set ($discard = $xwiki.ssx.use('contact.WebHome')) 17 -#set ($xobject = $doc.getObject('Agnease.Code.ContactForm.ContactFormClass')) 61 +#set ($xobject = $doc.getObject('Agnease.Code.ContactRequest.ContactRequestClass')) 62 +#set ($totalRequests = $services.query.xwql('from doc.object(Agnease.Code.ContactRequest.ContactRequestClass) contact').execute()) 18 18 #set ($xclass = $xobject.xWikiClass) 19 19 #set ($editing = true) 20 20 = Tell Us More About Your Project = 21 21 You do not need to have a full specification. A short description is enough to start the conversation. 22 22 {{html clean="false"}} 68 +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. 23 23 <div class="row"> 24 - <div class="xform col-xs-7"> 25 - <form id="contactForm"> 26 - <dl> 27 - #foreach ($property in $xclass.properties) 28 - #if ($property.name == 'hosting') 29 - <hr> 30 - <h3>Optional project details</h3> 31 - <p>These details help us understand the scope and suggest practical next steps.</p> 70 + <div class="xform col-md-7"> 71 + #if ($totalRequests.size() > 50) 72 + ## As a measure to avoid high load on website. 73 + Tell us more about your project at <a href="mailto:alex@agnease.com">alex@agnease.com</a> 74 + #else 75 + <form id="contactForm"> 76 + <dl> 77 + #foreach ($property in $xclass.properties) 78 + #if ($property.name == 'hosting') 79 + <hr> 80 + <h3>Optional project details</h3> 81 + <p>These details help us understand the scope and suggest practical next steps.</p> 82 + #end 83 + <dt #if (!$editing && $hasEdit) 84 + class="editableProperty" 85 + #set ($xobjectPropertyReference = $xobject.getPropertyReference($property.name)) 86 + data-property="$escapetool.xml($services.model.serialize($xobjectPropertyReference))" 87 + data-property-type="object"#end> 88 + ## This must match the id generated by the $doc.display() method below. 89 + #set ($propertyId = "${xclass.name}_${xobject.number}_$property.name") 90 + <label#if ($editing) for="$escapetool.xml($propertyId)"#end> 91 + $escapetool.xml($property.translatedPrettyName) 92 + </label> 93 + ## Support for specifying a translation key as hint in the property definition. 94 + <span class="xHint">$!escapetool.xml($services.localization.render($property.hint))</span> 95 + </dt> 96 + <dd>$doc.display($property.name, 'edit').replace('{{html clean="false" wiki="false"}}', '').replace("{{/html}}", '')</dd> 32 32 #end 33 - <dt #if (!$editing && $hasEdit) 34 - class="editableProperty" 35 - #set ($xobjectPropertyReference = $xobject.getPropertyReference($property.name)) 36 - data-property="$escapetool.xml($services.model.serialize($xobjectPropertyReference))" 37 - data-property-type="object"#end> 38 - ## This must match the id generated by the $doc.display() method below. 39 - #set ($propertyId = "${xclass.name}_${xobject.number}_$property.name") 40 - <label#if ($editing) for="$escapetool.xml($propertyId)"#end> 41 - $escapetool.xml($property.translatedPrettyName) 42 - </label> 43 - ## Support for specifying a translation key as hint in the property definition. 44 - <span class="xHint">$!escapetool.xml($services.localization.render($property.hint))</span> 45 - </dt> 46 - <dd>$doc.display($property.name, 'edit').replace('{{html clean="false" wiki="false"}}', '').replace("{{/html}}", '')</dd> 47 - #end 48 - #if (!$xclass.properties || $xclass.properties.size() == 0) 49 - ## Keep the empty definition term in order to have valid HTML. 50 - <dt></dt> 51 - <dd>$escapetool.xml($services.localization.render('xclass.defaultObjectSheet.noProperties'))</dd> 52 - #end 53 - </dl> 54 - <p>Your information will only be used to respond to this request.</p> 55 - ##<p>Your information will only be used to respond to this request. See the Privacy Policy for details.</p> 56 - <input id="contactSubmit" type="submit" class="btn btn-primary" value="Send my request"> 57 - </form> 98 + #if (!$xclass.properties || $xclass.properties.size() == 0) 99 + ## Keep the empty definition term in order to have valid HTML. 100 + <dt></dt> 101 + <dd>$escapetool.xml($services.localization.render('xclass.defaultObjectSheet.noProperties'))</dd> 102 + #end 103 + </dl> 104 + <p class="xHint">* Your information will only be used to respond to this request.</p> 105 + ## Hidden fields to catch requests filled by bots. 106 + <div class="contact-hp-wrapper" aria-hidden="true"> 107 + <label for="contactWebsite">Website</label> 108 + <input 109 + id="contactWebsite" 110 + type="text" 111 + name="contactWebsite" 112 + autocomplete="off" 113 + tabindex="-1" 114 + /> 115 + </div> 116 + <input id="contactSubmit" type="submit" class="btn btn-primary" value="Send my request"> 117 + </form> 118 + #end 58 58 {{/html}} 59 59 {{html clean="false" wiki="true"}} 60 60 <div class="reviewNotifications"> ... ... @@ -72,7 +72,7 @@ 72 72 {{/html}} 73 73 {{html clean="false"}} 74 74 </div> 75 - <div class="col- xs-5">136 + <div class="col-md-5"> 76 76 <div class="widget"> 77 77 <h4>$services.icon.renderHTML('check') How Agnease can help</h4> 78 78 <ul>
- Agnease.Code.ContactForm.ContactFormClass[0]
-
- alreadyUseXWiki
-
... ... @@ -1,1 +1,0 @@ 1 -1 - customDevelopment
-
... ... @@ -1,1 +1,0 @@ 1 -1 - hosting
-
... ... @@ -1,1 +1,0 @@ 1 -1 - users
-
... ... @@ -1,1 +1,0 @@ 1 -1
- XWiki.JavaScriptExtension[0]
-
- code
-
... ... @@ -1,5 +1,5 @@ 1 1 require(['jquery'], function ($) { 2 - var serviceURL = new XWiki.Document('WebHome', 'cont ent').getURL('get', 'xpage=plain');2 + var serviceURL = new XWiki.Document('WebHome', 'contact').getURL('get'); 3 3 var form = $('#contactForm'); 4 4 var submitButton = $('#contactSubmit'); 5 5 ... ... @@ -6,47 +6,29 @@ 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 23 form.on('submit', function (event) { 24 24 event.preventDefault(); 25 25 26 - var data = $.param(form.serializeArray()); 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(''); 27 27 28 28 submitButton.prop('disabled', true); 29 29 30 30 $.post({ 31 31 url: serviceURL, 32 - data: data 33 - }).done(function (response) { 34 - // replace with succcess message alert(response.message || 'Your request was sent successfully.'); 35 - //var message = 'The request could not be sent. Please try again or contact Agnease by email.'; 36 - var successBoxContent = successBox.find('.box div p'); 37 - successBoxContent.text(data.message); 38 - successBox.toggleClass('hidden'); 39 - if (errorBox.is(':visible')) { 40 - errorBox.toggleClass('hidden'); 41 - } 22 + data: $.param(form.serializeArray()) 23 + }).done(function (data) { 24 + successBox.find('.box div p').text(data.message); 25 + successBox.removeClass('hidden'); 42 42 form[0].reset(); 43 43 }).fail(function (xhr) { 44 - var errorBoxContent = errorBox.find('.box div p'); 45 - errorBoxContent.text(xhr.responseJSON.message); 46 - errorBox.toggleClass('hidden'); 47 - if (successBox.is(':visible')) { 48 - successBox.toggleClass('hidden'); 49 - } 28 + var message = xhr.responseJSON && xhr.responseJSON.message ? xhr.responseJSON.message 29 + : 'The request could not be sent. Please try again or contact Agnease by email at alex@agnease.com'; 30 + errorBox.find('.box div p').text(message); 31 + errorBox.removeClass('hidden'); 50 50 }).always(function () { 51 51 submitButton.prop('disabled', false); 52 52 });
- XWiki.StyleSheetExtension[0]
-
- code
-
... ... @@ -66,3 +66,11 @@ 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.ContactRequest.ContactRequestClass[0]
-
- alreadyUseXWiki
-
... ... @@ -1,0 +1,1 @@ 1 +1 - hosting
-
... ... @@ -1,0 +1,1 @@ 1 +1
- Agnease.Code.SEODetailsClass[0]
-
- metaDescription
-
... ... @@ -1,0 +1,1 @@ 1 +Contact Agnease for XWiki consulting, upgrades, support, custom development, integrations, migrations, authentication, security reviews and long-term platform care. - metaTitle
-
... ... @@ -1,0 +1,1 @@ 1 +Contact Agnease | XWiki Consulting and Support