Last modified by Alex Cotiugă on 2026/05/04 06:15

From version 12.35
edited by Alex Cotiugă
on 2026/05/01 16:11
Change comment: There is no comment for this version
To version 13.184
edited by Alex Cotiugă
on 2026/05/02 09:18
Change comment: There is no comment for this version

Summary

Details

Page properties
Content
... ... @@ -1,39 +1,92 @@
1 +{{velocity wiki="false"}}
2 +#if ($xcontext.action == 'get')
3 + #set ($message = '')
4 + #set ($now = $datetool.get('yyyyMMddHHmm'))
5 + #set ($random = $mathtool.random(100000, 999999))
6 + #set ($uniqueName = "ContactRequest-${now}-${random}")
7 + #set ($contactRequestDoc = $xwiki.getDocument('ContactRequests.' + $uniqueName))
8 + #set ($contactRequestObj = $contactRequestDoc.getObject('Agnease.Code.ContactForm.ContactFormClass', true))
9 + #set ($statusCode = 200)
10 + #set ($message = 'Your request was successfully sent.')
11 +
12 + ##var message = 'The request could not be sent. Please try again or contact Agnease by email.';
13 +
14 + #foreach ($parameterName in $request.parameterNames)
15 + #set ($discard = $contactRequestObj.set("$parameterName.split('_0_')[1]", $request.get($parameterName)))
16 + #set ($message = "$message $parameterName.split('_0_')[1] : $request.get($parameterName)")
17 + #end
18 +
19 + ##set ($message = "$message ")
20 +
21 + ##set ($discard = $contactRequestDoc.saveAsAuthor())
22 +
23 + ##set ($statusCode = 400)
24 + ##set ($message = 'error')
25 +
26 + #set ($discard = $response.setStatus($statusCode))
27 + #jsonResponse({'message': $message})
28 +#end
29 +{{/velocity}}
30 +
1 1  {{velocity}}
32 +#set ($discard = $xwiki.ssx.use('contact.WebHome'))
2 2  #set ($xobject = $doc.getObject('Agnease.Code.ContactForm.ContactFormClass'))
3 3  #set ($xclass = $xobject.xWikiClass)
4 4  #set ($editing = true)
5 -= Tell Agnease About Your Project =
6 -Share a few details about your documentation, intranet, workflow, migration, upgrade, or XWiki support need. We will review your request and suggest practical next steps.
36 += Tell Us More About Your Project =
37 +You do not need to have a full specification. A short description is enough to start the conversation.
7 7  {{html clean="false"}}
8 8   <div class="row">
9 9   <div class="xform col-xs-7">
10 - <form>
41 + <form id="contactForm">
11 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
43 + #foreach ($property in $xclass.properties)
44 + #if ($property.name == 'hosting')
45 + <hr>
46 + <h3>Optional project details</h3>
47 + <p>These details help us understand the scope and suggest practical next steps.</p>
48 + #end
49 + <dt #if (!$editing && $hasEdit)
50 + class="editableProperty"
51 + #set ($xobjectPropertyReference = $xobject.getPropertyReference($property.name))
52 + data-property="$escapetool.xml($services.model.serialize($xobjectPropertyReference))"
53 + data-property-type="object"#end>
54 + ## This must match the id generated by the $doc.display() method below.
55 + #set ($propertyId = "${xclass.name}_${xobject.number}_$property.name")
56 + <label#if ($editing) for="$escapetool.xml($propertyId)"#end>
57 + $escapetool.xml($property.translatedPrettyName)
58 + </label>
59 + ## Support for specifying a translation key as hint in the property definition.
60 + <span class="xHint">$!escapetool.xml($services.localization.render($property.hint))</span>
61 + </dt>
62 + <dd>$doc.display($property.name, 'edit').replace('{{html clean="false" wiki="false"}}', '').replace("{{/html}}", '')</dd>
63 + #end
64 + #if (!$xclass.properties || $xclass.properties.size() == 0)
65 + ## Keep the empty definition term in order to have valid HTML.
66 + <dt></dt>
67 + <dd>$escapetool.xml($services.localization.render('xclass.defaultObjectSheet.noProperties'))</dd>
68 + #end
33 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">
70 + <p>Your information will only be used to respond to this request.</p>
71 + ##<p>Your information will only be used to respond to this request. See the Privacy Policy for details.</p>
72 + <input id="contactSubmit" type="submit" class="btn btn-primary" value="Send my request">
36 36   </form>
74 + {{/html}}
75 + {{html clean="false" wiki="true"}}
76 + <div class="reviewNotifications">
77 + <div class="hidden reviewNotificationSuccess">
78 +
79 + {{success}}reviewNotification{{/success}}
80 +
81 + </div>
82 + <div class="hidden reviewNotificationError">
83 +
84 + {{error}}reviewNotification{{/error}}
85 +
86 + </div>
87 + </div>
88 + {{/html}}
89 + {{html clean="false"}}
37 37   </div>
38 38   <div class="col-xs-5">
39 39   <div class="widget">
... ... @@ -40,8 +40,7 @@
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>
96 + <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>
104 + <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
112 + </div>
79 79  {{/html}}
80 80  {{/velocity}}
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
XWiki.StyleSheetExtension[0]
code
... ... @@ -1,0 +1,68 @@
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 +}
contentType
... ... @@ -1,0 +1,1 @@
1 +LESS
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 + var data = $.param(form.serializeArray());
13 +
14 + submitButton.prop('disabled', true);
15 +
16 + $.post({
17 + url: serviceURL,
18 + data: data
19 + }).done(function (response) {
20 + var successBoxContent = successBox.find('.box div p');
21 + successBoxContent.text(data.message);
22 + successBox.toggleClass('hidden');
23 + if (errorBox.is(':visible')) {
24 + errorBox.toggleClass('hidden');
25 + }
26 + form[0].reset();
27 + }).fail(function (xhr) {
28 + var errorBoxContent = errorBox.find('.box div p');
29 + errorBoxContent.text(xhr.statusText);
30 + errorBox.toggleClass('hidden');
31 + if (successBox.is(':visible')) {
32 + successBox.toggleClass('hidden');
33 + }
34 + }).always(function () {
35 + submitButton.prop('disabled', false);
36 + });
37 + });
38 +});
use
... ... @@ -1,0 +1,1 @@
1 +currentPage