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 (1 modified, 3 added, 1 removed)
Details
- Page properties
-
- Content
-
... ... @@ -1,47 +1,165 @@ 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 ($discard = $response.setStatus(400)) 36 + #jsonResponse({'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 + 1 1 {{velocity}} 2 -#set ($xobject = $doc.getObject('Agnease.Code.ContactForm.ContactFormClass')) 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 3 #set ($xclass = $xobject.xWikiClass) 4 4 #set ($editing = true) 5 -= Tell Agnease About Your Project =6 - Share a few details aboutyourdocumentation,intranet,workflow,migration,upgrade,or XWikisupportneed. Wewillreview yourrequestand suggestpractical nextsteps.85 += Tell Us More About Your Project = 86 +You do not need to have a full specification. A short description is enough to start the conversation. 7 7 {{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. 8 8 <div class="row"> 9 - <div class="xform col-xs-7"> 10 - <form> 11 - <dl> 12 - #foreach ($property in $xclass.properties) 13 - <dt #if (!$editing && $hasEdit) 14 - class="editableProperty" 15 - #set ($xobjectPropertyReference = $xobject.getPropertyReference($property.name)) 16 - data-property="$escapetool.xml($services.model.serialize($xobjectPropertyReference))" 17 - data-property-type="object"#end> 18 - ## This must match the id generated by the $doc.display() method below. 19 - #set ($propertyId = "${xclass.name}_${xobject.number}_$property.name") 20 - <label#if ($editing) for="$escapetool.xml($propertyId)"#end> 21 - $escapetool.xml($property.translatedPrettyName) 22 - </label> 23 - ## Support for specifying a translation key as hint in the property definition. 24 - <span class="xHint">$!escapetool.xml($services.localization.render($property.hint))</span> 25 - </dt> 26 - <dd>$doc.display($property.name, 'edit').replace('{{html clean="false" wiki="false"}}', '').replace("{{/html}}", '')</dd> 27 - #end 28 - #if (!$xclass.properties || $xclass.properties.size() == 0) 29 - ## Keep the empty definition term in order to have valid HTML. 30 - <dt></dt> 31 - <dd>$escapetool.xml($services.localization.render('xclass.defaultObjectSheet.noProperties'))</dd> 32 - #end 33 - </dl> 34 - <p>Your information will only be used to respond to this request. See the Privacy Policy for details.</p> 35 - <input type="submit" class="btn btn-primary" value="Send my request"> 36 - </form> 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> 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.getsystemDate.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"}} 37 37 </div> 38 - <div class="col- xs-5">157 + <div class="col-md-5"> 39 39 <div class="widget"> 40 40 <h4>$services.icon.renderHTML('check') How Agnease can help</h4> 41 41 <ul> 42 42 <li>XWiki upgrades and long-term maintenance</li> 43 - <li>Knowledge bases and internal portals</li> 44 - <li>SOP, approval, and documentation workflows</li> 162 + <li>Knowledge bases, intranets, SOP and documentation workflows</li> 45 45 <li>Custom XWiki applications and integrations</li> 46 46 <li>LDAP, SSO, OIDC, SAML, and MFA setup</li> 47 47 <li>Migrations from SharePoint, Confluence, MediaWiki, or file-based documentation</li> ... ... @@ -48,12 +48,8 @@ 48 48 <li>Security-aware reviews and platform stabilization</li> 49 49 </ul> 50 50 </div> 51 - 52 52 <div class="widget"> 53 - <div class="icon" aria-hidden="true"> 54 - <i class="fa fa-arrow-right"></i> 55 - <h4> What happens next?</h4> 56 - </div> 170 + <h4>$services.icon.renderHTML('right') What happens next?</h4> 57 57 <ol> 58 58 <li>Your request is reviewed.</li> 59 59 <li>You receive a reply with clarifying questions or suggested next steps.</li> ... ... @@ -60,21 +60,7 @@ 60 60 <li>If useful, we schedule a short call to discuss scope, timeline, and estimated effort.</li> 61 61 </ol> 62 62 </div> 63 - 64 - <div class="widget"> 65 - <div class="icon" aria-hidden="true"> 66 - <i class="fa fa-envelope"></i> 67 - <h4>Prefer email?</h4> 68 - </div> 69 - <p> 70 - You can also contact Agnease directly at 71 - <a href="mailto:alex@agnease.com">alex@agnease.com</a>. 72 - </p> 73 - <p> 74 - Based in Romania, working remotely with international clients. 75 - </p> 76 - </div> 77 77 </div> 78 - </div 178 + </div> 79 79 {{/html}} 80 80 {{/velocity}}
- Agnease.Code.ContactForm.ContactFormClass[0]
-
- XWiki.StyleSheetExtension[0]
-
- code
-
... ... @@ -1,0 +1,76 @@ 1 +@brand: #00937D; 2 +@brand-strong: #007B6A; 3 +@text: #2D3A34; 4 +@muted: #5B6B64; 5 +@line: #E4ECE9; 6 +@radius: 16px; 7 +@shadow-sm: 0 6px 20px rgba(0,0,0,.06); 8 +@shadow: 0 12px 36px rgba(0,0,0,.08); 9 +@maxw: 1140px; 10 + 11 +/* ===== Contact page widgets ===== */ 12 +#mainContentArea { 13 + padding: 0; 14 +} 15 + 16 +.col-xs-5 { 17 + padding-top: 45px; 18 +} 19 + 20 +.col-xs-5 .widget { 21 + margin-bottom: 16px; 22 + padding: 16px; 23 +} 24 + 25 +.col-xs-5 .widget h4 { 26 + display: flex; 27 + align-items: center; 28 + gap: 8px; 29 + margin: 0 0 10px; 30 + padding-bottom: 8px; 31 + border-bottom: 1px solid fade(@line, 60%); 32 + color: @text; 33 + font-size: 18px; 34 + line-height: 1.25; 35 +} 36 + 37 +.col-xs-5 .widget h4 .fa, 38 +.col-xs-5 .widget h4 .glyphicon { 39 + color: @brand; 40 + font-size: 15px; 41 +} 42 + 43 +.col-xs-5 .widget ul, 44 +.col-xs-5 .widget ol { 45 + margin: 0; 46 + padding-left: 1.2rem; 47 + color: @muted; 48 +} 49 + 50 +.col-xs-5 .widget li { 51 + margin: 5px 0; 52 + line-height: 1.4; 53 +} 54 + 55 +.col-xs-5 .widget p { 56 + margin: 0 0 8px; 57 + color: @muted; 58 + line-height: 1.45; 59 +} 60 + 61 +.col-xs-5 .widget p:last-child { 62 + margin-bottom: 0; 63 +} 64 + 65 +.col-xs-5 .widget a { 66 + color: @brand; 67 + font-weight: 700; 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 +} - contentType
-
... ... @@ -1,0 +1,1 @@ 1 +LESS
- 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
- XWiki.JavaScriptExtension[0]
-
- cache
-
... ... @@ -1,0 +1,1 @@ 1 +long - code
-
... ... @@ -1,0 +1,38 @@ 1 +require(['jquery'], function ($) { 2 + var serviceURL = new XWiki.Document('WebHome', 'contact').getURL('get'); 3 + var form = $('#contactForm'); 4 + var submitButton = $('#contactSubmit'); 5 + 6 + var successBox = $('.reviewNotificationSuccess'); 7 + var errorBox = $('.reviewNotificationError'); 8 + 9 + form.on('submit', function (event) { 10 + event.preventDefault(); 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(''); 17 + 18 + submitButton.prop('disabled', true); 19 + 20 + $.post({ 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'); 27 + form[0].reset(); 28 + }).fail(function (xhr) { 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 () { 35 + submitButton.prop('disabled', false); 36 + }); 37 + }); 38 +}); - use
-
... ... @@ -1,0 +1,1 @@ 1 +currentPage