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,48 +1,145 @@ 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 + #end 21 + #end 22 + #end 23 + 24 + #if ("$!contactWebsite.trim()" != '') 25 + #set ($statusCode = 400) 26 + #set ($message = 'The request could not be sent. Please try again or contact Agnease by email.') 27 + #elseif ("$!name" == '' && "$!email" == '') 28 + #set ($statusCode = 400) 29 + #set ($message = 'Please enter your name and email.') 30 + #elseif ("$!name" == '') 31 + #set ($statusCode = 400) 32 + #set ($message = 'Please enter your name.') 33 + #elseif ("$!email" == '') 34 + #set ($statusCode = 400) 35 + #set ($message = 'Please enter your email address.') 36 + #else 37 + #try('contactException') 38 + #set ($now = $datetool.get('yyyyMMddHHmm')) 39 + #set ($random = $mathtool.random(100000, 999999)) 40 + #set ($uniqueName = "ContactRequest-${now}-${random}") 41 + #set ($contactRequestDoc = $xwiki.getDocumentAsAuthor('ContactRequests.' + $uniqueName)) 42 + #set ($contactRequestObj = $contactRequestDoc.getObject('Agnease.Code.ContactRequest.ContactRequestClass', true)) 43 + 44 + #foreach ($parameterName in $request.parameterNames) 45 + #set ($propertyName = $parameterName.split('_0_')[1]) 46 + #set ($discard = $contactRequestObj.set($propertyName, $request.get($parameterName))) 47 + #end 48 + 49 + #set ($discard = $contactRequestDoc.saveAsAuthor()) 50 + #set ($statusCode = 200) 51 + #set ($message = 'Your request was successfully sent.') 52 + #end 53 + 54 + #if ("$!contactException" != '') 55 + #set ($statusCode = 400) 56 + #set ($message = 'The request could not be sent. Please try again or contact Agnease by email.') 57 + #end 58 + #end 59 + 60 + #set ($discard = $response.setStatus($statusCode)) 61 + #jsonResponse({'message': $message}) 62 +#end 63 +{{/velocity}} 64 + 1 1 {{velocity}} 2 2 #set ($discard = $xwiki.ssx.use('contact.WebHome')) 3 -#set ($xobject = $doc.getObject('Agnease.Code.ContactForm.ContactFormClass')) 67 +#set ($xobject = $doc.getObject('Agnease.Code.ContactRequest.ContactRequestClass')) 68 +#set ($totalRequests = $services.query.xwql('from doc.object(Agnease.Code.ContactRequest.ContactRequestClass) contact').execute()) 4 4 #set ($xclass = $xobject.xWikiClass) 5 5 #set ($editing = true) 6 6 = Tell Us More About Your Project = 7 7 You do not need to have a full specification. A short description is enough to start the conversation. 8 8 {{html clean="false"}} 74 +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. 9 9 <div class="row"> 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> 76 + <div class="xform col-md-7"> 77 + #if ($totalRequests.size() > 50) 78 + ## As a measure to avoid high load on website. 79 + Tell us more about your project at <a href="mailto:alex@agnease.com">alex@agnease.com</a> 80 + #else 81 + <form id="contactForm"> 82 + <dl> 83 + #foreach ($property in $xclass.properties) 84 + #if ($property.name == 'hosting') 85 + <hr> 86 + <h3>Optional project details</h3> 87 + <p>These details help us understand the scope and suggest practical next steps.</p> 88 + #end 89 + <dt #if (!$editing && $hasEdit) 90 + class="editableProperty" 91 + #set ($xobjectPropertyReference = $xobject.getPropertyReference($property.name)) 92 + data-property="$escapetool.xml($services.model.serialize($xobjectPropertyReference))" 93 + data-property-type="object"#end> 94 + ## This must match the id generated by the $doc.display() method below. 95 + #set ($propertyId = "${xclass.name}_${xobject.number}_$property.name") 96 + <label#if ($editing) for="$escapetool.xml($propertyId)"#end> 97 + $escapetool.xml($property.translatedPrettyName) 98 + </label> 99 + ## Support for specifying a translation key as hint in the property definition. 100 + <span class="xHint">$!escapetool.xml($services.localization.render($property.hint))</span> 101 + </dt> 102 + <dd>$doc.display($property.name, 'edit').replace('{{html clean="false" wiki="false"}}', '').replace("{{/html}}", '')</dd> 18 18 #end 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> 104 + #if (!$xclass.properties || $xclass.properties.size() == 0) 105 + ## Keep the empty definition term in order to have valid HTML. 106 + <dt></dt> 107 + <dd>$escapetool.xml($services.localization.render('xclass.defaultObjectSheet.noProperties'))</dd> 108 + #end 109 + </dl> 110 + <p class="xHint">* Your information will only be used to respond to this request.</p> 111 + ## Hidden fields to catch requests filled by bots. 112 + <div class="contact-hp-wrapper" aria-hidden="true"> 113 + <label for="Agnease.Code.ContactRequest.ContactRequestClass_0_contactWebsite">Website</label> 114 + <input 115 + id="contactWebsite" 116 + type="text" 117 + name="Agnease.Code.ContactRequest.ContactRequestClass_0_contactWebsite" 118 + autocomplete="off" 119 + tabindex="-1" 120 + /> 121 + </div> 122 + <input id="contactSubmit" type="submit" class="btn btn-primary" value="Send my request"> 123 + </form> 124 + #end 125 + {{/html}} 126 + {{html clean="false" wiki="true"}} 127 + <div class="reviewNotifications"> 128 + <div class="hidden reviewNotificationSuccess"> 129 + 130 + {{success}}reviewNotification{{/success}} 131 + 132 + </div> 133 + <div class="hidden reviewNotificationError"> 134 + 135 + {{error}}reviewNotification{{/error}} 136 + 137 + </div> 138 + </div> 139 + {{/html}} 140 + {{html clean="false"}} 44 44 </div> 45 - <div class="col- xs-5">142 + <div class="col-md-5"> 46 46 <div class="widget"> 47 47 <h4>$services.icon.renderHTML('check') How Agnease can help</h4> 48 48 <ul> ... ... @@ -62,18 +62,6 @@ 62 62 <li>If useful, we schedule a short call to discuss scope, timeline, and estimated effort.</li> 63 63 </ol> 64 64 </div> 65 - <div class="reviewNotifications"> 66 - <div class="hidden reviewNotificationSuccess"> 67 - 68 - {{success}}reviewNotification{{/success}} 69 - 70 - </div> 71 - <div class="hidden reviewNotificationError"> 72 - 73 - {{error}}reviewNotification{{/error}} 74 - 75 - </div> 76 - </div> 77 77 </div> 78 78 </div> 79 79 {{/html}}
- 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,62 +6,32 @@ 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 - 38 38 form.on('submit', function (event) { 39 39 event.preventDefault(); 40 40 41 - var data = getFormData(); 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(''); 42 42 43 43 submitButton.prop('disabled', true); 44 44 45 45 $.post({ 46 46 url: serviceURL, 47 - data: data 48 - }).done(function (response) { 49 - // replace with succcess message alert(response.message || 'Your request was sent successfully.'); 50 - //var message = 'The request could not be sent. Please try again or contact Agnease by email.'; 51 - var successBoxContent = successBox.find('.box div p'); 52 - successBoxContent.text(data.message); 53 - successBox.toggleClass('hidden'); 54 - if (errorBox.is(':visible')) { 55 - errorBox.toggleClass('hidden'); 56 - } 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'); 57 57 form[0].reset(); 58 58 }).fail(function (xhr) { 59 - var errorBoxContent = errorBox.find('.boxdivp');60 - error BoxContent.text(xhr.responseJSON.message);61 - er rorBox.toggleClass('hidden');62 - if (successBox.is(':visible')) {63 - successBox.toggleClass('hidden');64 - }29 + console.log('fail' + xhr) 30 + var message = xhr.responseJSON && xhr.responseJSON.message ? xhr.responseJSON.message 31 + : 'The request could not be sent. Please try again or contact Agnease by email at alex@agnease.com'; 32 + errorBox.find('.box div p').text(message); 33 + errorBox.removeClass('hidden'); 34 + }).always(function () { 65 65 submitButton.prop('disabled', false); 66 66 }); 67 67 });
- 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