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,50 +1,147 @@ 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 ($propertyName = $parameterName.split('_0_')[1]) 12 + #if ($propertyName == 'name') 13 + #set ($name = $stringtool.trim($request.get($parameterName))) 14 + #elseif ($propertyName == 'email') 15 + #set ($email = $stringtool.trim($request.get($parameterName))) 16 + #elseif ($propertyName == 'contactWebsite') 17 + #set ($contactWebsite = $stringtool.trim($request.get($parameterName))) 18 + #end 19 + #end 20 + 21 + #if ("$!contactWebsite.trim()" != '') 22 + #set ($statusCode = 400) 23 + #set ($message = 'The request could not be sent. Please try again or contact Agnease by email.') 24 + #elseif ("$!name" == '' && "$!email" == '') 25 + #set ($statusCode = 400) 26 + #set ($message = 'Please enter your name and email.') 27 + #elseif ("$!name" == '') 28 + #set ($statusCode = 400) 29 + #set ($message = 'Please enter your name.') 30 + #elseif ("$!email" == '') 31 + #set ($statusCode = 400) 32 + #set ($message = 'Please enter your email address.') 33 + #else 34 + #try('contactException') 35 + #set ($now = $datetool.get('yyyyMMddHHmm')) 36 + #set ($random = $mathtool.random(100000, 999999)) 37 + #set ($uniqueName = "ContactRequest-${now}-${random}") 38 + #set ($contactRequestDoc = $xwiki.getDocumentAsAuthor('ContactRequests.' + $uniqueName)) 39 + #set ($contactRequestObj = $contactRequestDoc.getObject('Agnease.Code.ContactRequest.ContactRequestClass', true)) 40 + 41 + #foreach ($parameterName in $request.parameterNames) 42 + #set ($propertyName = $parameterName.split('_0_')[1]) 43 + #set ($discard = $contactRequestObj.set($propertyName, $request.get($parameterName))) 44 + #end 45 + 46 + #set ($discard = $contactRequestDoc.saveAsAuthor()) 47 + #set ($statusCode = 200) 48 + #set ($message = 'Your request was successfully sent.') 49 + #end 50 + 51 + #if ("$!contactException" != '') 52 + #set ($statusCode = 400) 53 + #set ($message = 'The request could not be sent. Please try again or contact Agnease by email.') 54 + #end 55 + #end 56 + 57 + #set ($discard = $response.setStatus($statusCode)) 58 + #jsonResponse({'message': $message}) 59 +#end 60 +{{/velocity}} 61 + 1 1 {{velocity}} 2 -#set ($xobject = $doc.getObject('Agnease.Code.ContactForm.ContactFormClass')) 63 +#set ($discard = $xwiki.ssx.use('contact.WebHome')) 64 +#set ($xobject = $doc.getObject('Agnease.Code.ContactRequest.ContactRequestClass')) 65 +#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.68 += Tell Us More About Your Project = 69 +You do not need to have a full specification. A short description is enough to start the conversation. 7 7 {{html clean="false"}} 71 +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> 73 + <div class="xform col-md-7"> 74 + #if ($totalRequests.size() > 50) 75 + ## As a measure to avoid high load on website. 76 + Tell us more about your project at <a href="mailto:alex@agnease.com">alex@agnease.com</a> 77 + #else 78 + <form id="contactForm"> 79 + <dl> 80 + #foreach ($property in $xclass.properties) 81 + #if ($property.name == 'hosting') 82 + <hr> 83 + <h3>Optional project details</h3> 84 + <p>These details help us understand the scope and suggest practical next steps.</p> 85 + #end 86 + <dt #if (!$editing && $hasEdit) 87 + class="editableProperty" 88 + #set ($xobjectPropertyReference = $xobject.getPropertyReference($property.name)) 89 + data-property="$escapetool.xml($services.model.serialize($xobjectPropertyReference))" 90 + data-property-type="object"#end> 91 + ## This must match the id generated by the $doc.display() method below. 92 + #set ($propertyId = "${xclass.name}_${xobject.number}_$property.name") 93 + <label#if ($editing) for="$escapetool.xml($propertyId)"#end> 94 + $escapetool.xml($property.translatedPrettyName) 95 + </label> 96 + ## Support for specifying a translation key as hint in the property definition. 97 + <span class="xHint">$!escapetool.xml($services.localization.render($property.hint))</span> 98 + </dt> 99 + <dd>$doc.display($property.name, 'edit').replace('{{html clean="false" wiki="false"}}', '').replace("{{/html}}", '')</dd> 100 + #end 101 + #if (!$xclass.properties || $xclass.properties.size() == 0) 102 + ## Keep the empty definition term in order to have valid HTML. 103 + <dt></dt> 104 + <dd>$escapetool.xml($services.localization.render('xclass.defaultObjectSheet.noProperties'))</dd> 105 + #end 106 + </dl> 107 + <p class="xHint">* Your information will only be used to respond to this request.</p> 108 + ## Hidden fields to catch requests filled by bots. 109 + <div class="contact-hp-wrapper" aria-hidden="true"> 110 + <label for="contactWebsite">Website</label> 111 + <input 112 + id="contactWebsite" 113 + type="text" 114 + name="contactWebsite" 115 + autocomplete="off" 116 + tabindex="-1" 117 + /> 118 + </div> 119 + <input id="contactSubmit" type="submit" class="btn btn-primary" value="Send my request"> 120 + </form> 121 + #end 122 + {{/html}} 123 + {{html clean="false" wiki="true"}} 124 + <div class="reviewNotifications"> 125 + <div class="hidden reviewNotificationSuccess"> 126 + 127 + {{success}}reviewNotification{{/success}} 128 + 129 + </div> 130 + <div class="hidden reviewNotificationError"> 131 + 132 + {{error}}reviewNotification{{/error}} 133 + 134 + </div> 135 + </div> 136 + {{/html}} 137 + {{html clean="false"}} 37 37 </div> 38 - <div class="col- xs-5">139 + <div class="col-md-5"> 39 39 <div class="widget"> 40 - <div class="icon" aria-hidden="true"> 41 - <i class="fa fa-check-square-o"></i> 42 - <h4>$services.icon.renderHTML('check-square-o') How Agnease can help</h4> 43 - </div> 141 + <h4>$services.icon.renderHTML('check') How Agnease can help</h4> 44 44 <ul> 45 45 <li>XWiki upgrades and long-term maintenance</li> 46 - <li>Knowledge bases and internal portals</li> 47 - <li>SOP, approval, and documentation workflows</li> 144 + <li>Knowledge bases, intranets, SOP and documentation workflows</li> 48 48 <li>Custom XWiki applications and integrations</li> 49 49 <li>LDAP, SSO, OIDC, SAML, and MFA setup</li> 50 50 <li>Migrations from SharePoint, Confluence, MediaWiki, or file-based documentation</li> ... ... @@ -51,12 +51,8 @@ 51 51 <li>Security-aware reviews and platform stabilization</li> 52 52 </ul> 53 53 </div> 54 - 55 55 <div class="widget"> 56 - <div class="icon" aria-hidden="true"> 57 - <i class="fa fa-arrow-right"></i> 58 - <h4> What happens next?</h4> 59 - </div> 152 + <h4>$services.icon.renderHTML('right') What happens next?</h4> 60 60 <ol> 61 61 <li>Your request is reviewed.</li> 62 62 <li>You receive a reply with clarifying questions or suggested next steps.</li> ... ... @@ -63,21 +63,7 @@ 63 63 <li>If useful, we schedule a short call to discuss scope, timeline, and estimated effort.</li> 64 64 </ol> 65 65 </div> 66 - 67 - <div class="widget"> 68 - <div class="icon" aria-hidden="true"> 69 - <i class="fa fa-envelope"></i> 70 - <h4>Prefer email?</h4> 71 - </div> 72 - <p> 73 - You can also contact Agnease directly at 74 - <a href="mailto:alex@agnease.com">alex@agnease.com</a>. 75 - </p> 76 - <p> 77 - Based in Romania, working remotely with international clients. 78 - </p> 79 - </div> 80 80 </div> 81 - </div 160 + </div> 82 82 {{/html}} 83 83 {{/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-wrapperss { 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,36 @@ 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 + successBox.find('.box div p').text(data.message); 25 + successBox.removeClass('hidden'); 26 + form[0].reset(); 27 + }).fail(function (xhr) { 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'); 32 + }).always(function () { 33 + submitButton.prop('disabled', false); 34 + }); 35 + }); 36 +}); - use
-
... ... @@ -1,0 +1,1 @@ 1 +currentPage