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

From version 13.110
edited by Alex Cotiugă
on 2026/05/02 08:30
Change comment: There is no comment for this version
To version 12.36
edited by Alex Cotiugă
on 2026/05/01 16:11
Change comment: There is no comment for this version

Summary

Details

Page properties
Content
... ... @@ -1,75 +14,39 @@
1 -{{velocity wiki="false"}}
2 -#if ($xcontext.action == 'get')
3 - #set ($message = '')
4 - #set ($statusCode = 200)
5 -
6 - #set ($statusCode = 400)
7 - #set ($message = 'error')
8 -
9 - #set ($discard = $response.setStatus($statusCode))
10 - #jsonResponse({'message': $message})
11 -#end
12 -{{/velocity}}
13 -
14 14  {{velocity}}
15 -#set ($discard = $xwiki.ssx.use('contact.WebHome'))
16 16  #set ($xobject = $doc.getObject('Agnease.Code.ContactForm.ContactFormClass'))
17 17  #set ($xclass = $xobject.xWikiClass)
18 18  #set ($editing = true)
19 -= Tell Us More About Your Project =
20 -You do not need to have a full specification. A short description is enough to start the conversation.
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.
21 21  {{html clean="false"}}
22 22   <div class="row">
23 23   <div class="xform col-xs-7">
24 - <form id="contactForm">
10 + <form>
25 25   <dl>
26 - #foreach ($property in $xclass.properties)
27 - #if ($property.name == 'hosting')
28 - <hr>
29 - <h3>Optional project details</h3>
30 - <p>These details help us understand the scope and suggest practical next steps.</p>
31 - #end
32 - <dt #if (!$editing && $hasEdit)
33 - class="editableProperty"
34 - #set ($xobjectPropertyReference = $xobject.getPropertyReference($property.name))
35 - data-property="$escapetool.xml($services.model.serialize($xobjectPropertyReference))"
36 - data-property-type="object"#end>
37 - ## This must match the id generated by the $doc.display() method below.
38 - #set ($propertyId = "${xclass.name}_${xobject.number}_$property.name")
39 - <label#if ($editing) for="$escapetool.xml($propertyId)"#end>
40 - $escapetool.xml($property.translatedPrettyName)
41 - </label>
42 - ## Support for specifying a translation key as hint in the property definition.
43 - <span class="xHint">$!escapetool.xml($services.localization.render($property.hint))</span>
44 - </dt>
45 - <dd>$doc.display($property.name, 'edit').replace('{{html clean="false" wiki="false"}}', '').replace("{{/html}}", '')</dd>
46 - #end
47 - #if (!$xclass.properties || $xclass.properties.size() == 0)
48 - ## Keep the empty definition term in order to have valid HTML.
49 - <dt></dt>
50 - <dd>$escapetool.xml($services.localization.render('xclass.defaultObjectSheet.noProperties'))</dd>
51 - #end
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
52 52   </dl>
53 - <p>Your information will only be used to respond to this request.</p>
54 - ##<p>Your information will only be used to respond to this request. See the Privacy Policy for details.</p>
55 - <input id="contactSubmit" type="submit" class="btn btn-primary" value="Send my request">
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">
56 56   </form>
57 - {{/html}}
58 - {{html clean="false" wiki="true"}}
59 - <div class="reviewNotifications">
60 - <div class="hidden reviewNotificationSuccess">
61 -
62 - {{success}}reviewNotification{{/success}}
63 -
64 - </div>
65 - <div class="hidden reviewNotificationError">
66 -
67 - {{error}}reviewNotification{{/error}}
68 -
69 - </div>
70 - </div>
71 - {{/html}}
72 - {{html clean="false"}}
73 73   </div>
74 74   <div class="col-xs-5">
75 75   <div class="widget">
... ... @@ -76,7 +76,8 @@
76 76   <h4>$services.icon.renderHTML('check') How Agnease can help</h4>
77 77   <ul>
78 78   <li>XWiki upgrades and long-term maintenance</li>
79 - <li>Knowledge bases, intranets, SOP and documentation workflows</li>
43 + <li>Knowledge bases and internal portals</li>
44 + <li>SOP, approval, and documentation workflows</li>
80 80   <li>Custom XWiki applications and integrations</li>
81 81   <li>LDAP, SSO, OIDC, SAML, and MFA setup</li>
82 82   <li>Migrations from SharePoint, Confluence, MediaWiki, or file-based documentation</li>
... ... @@ -83,8 +83,9 @@
83 83   <li>Security-aware reviews and platform stabilization</li>
84 84   </ul>
85 85   </div>
51 +
86 86   <div class="widget">
87 - <h4>$services.icon.renderHTML('right') What happens next?</h4>
53 + <h4>$services.icon.renderHTML('check') What happens next?</h4>
88 88   <ol>
89 89   <li>Your request is reviewed.</li>
90 90   <li>You receive a reply with clarifying questions or suggested next steps.</li>
... ... @@ -91,7 +91,21 @@
91 91   <li>If useful, we schedule a short call to discuss scope, timeline, and estimated effort.</li>
92 92   </ol>
93 93   </div>
60 +
61 + <div class="widget">
62 + <div class="icon" aria-hidden="true">
63 + <i class="fa fa-envelope"></i>
64 + <h4>Prefer email?</h4>
65 + </div>
66 + <p>
67 + You can also contact Agnease directly at
68 + <a href="mailto:alex@agnease.com">alex@agnease.com</a>.
69 + </p>
70 + <p>
71 + Based in Romania, working remotely with international clients.
72 + </p>
73 + </div>
94 94   </div>
95 - </div>
75 + </div
96 96  {{/html}}
97 97  {{/velocity}}
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]
cache
... ... @@ -1,1 +1,0 @@
1 -long
code
... ... @@ -1,54 +1,0 @@
1 -require(['jquery'], function ($) {
2 - var serviceURL = new XWiki.Document('WebHome', 'content').getURL('get', 'xpage=plain');
3 - var form = $('#contactForm');
4 - var submitButton = $('#contactSubmit');
5 -
6 - var successBox = $('.reviewNotificationSuccess');
7 - var errorBox = $('.reviewNotificationError');
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 - form.on('submit', function (event) {
24 - event.preventDefault();
25 -
26 - var data = $.param(form.serializeArray());
27 -
28 - submitButton.prop('disabled', true);
29 -
30 - $.post({
31 - url: serviceURL,
32 - data: data
33 - }).done(function (response) {
34 - // replace with succcess message alert(response.message || 'Your request was sent successfully.');
35 - //var message = 'The request could not be sent. Please try again or contact Agnease by email.';
36 - var successBoxContent = successBox.find('.box div p');
37 - successBoxContent.text(data.message);
38 - successBox.toggleClass('hidden');
39 - if (errorBox.is(':visible')) {
40 - errorBox.toggleClass('hidden');
41 - }
42 - form[0].reset();
43 - }).fail(function (xhr) {
44 - var errorBoxContent = errorBox.find('.box div p');
45 - errorBoxContent.text(xhr.responseJSON.message);
46 - errorBox.toggleClass('hidden');
47 - if (successBox.is(':visible')) {
48 - successBox.toggleClass('hidden');
49 - }
50 - }).always(function () {
51 - submitButton.prop('disabled', false);
52 - });
53 - });
54 -});
use
... ... @@ -1,1 +1,0 @@
1 -currentPage
XWiki.StyleSheetExtension[0]
code
... ... @@ -1,68 +1,0 @@
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,1 +1,0 @@
1 -LESS