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