Changes for page Home
                  Last modified by Jesse Veentjer on 26-5-2025 10:13
              
      
      From version  167.1 
    
    
              edited by Jesse Veentjer
        
on 08-11-2022 01:37
     on 08-11-2022 01:37
      Change comment:
              There is no comment for this version
          
         
      To version  219.32 
    
    
              edited by Jesse Veentjer
        
on 19-2-2024 11:41
     on 19-2-2024 11:41
      Change comment:
              There is no comment for this version
          
         Summary
- 
          Page properties (1 modified, 0 added, 0 removed)
- 
          Attachments (0 modified, 2 added, 2 removed)
- 
          Objects (5 modified, 1 added, 0 removed)
Details
- Page properties
- 
      - Content
-   ... ... @@ -1,66 +1,16988 @@ 1 1 (% class="floatinginfobox wbrechts" %) 2 2 ((( 3 3 (% class="wbrb" %) 4 -[[image:cheer.gif||data-xwiki-image-style-alignment="center"]] 4 +(% aria-label="aanhetwerk.gif image widget" contenteditable="false" data-xwiki-image-style-alignment="center" role="region" tabindex="-1" data-widget="image" %)[[image:attach:aanhetwerk.gif||data-xwiki-image-style-alignment="center"]](% title="Click and drag to resize" %)(% aria-label="aanhetwerk.gif image widget" contenteditable="false" data-xwiki-image-style-alignment="center" role="region" tabindex="-1" style="background-color:rgba(220,220,220,0.5)" %)[[image:data:image/gif;base64,R0lGODlhAQABAPABAP///wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==||draggable="true" height="15" role="presentation" title="Click and drag to move" width="15"]] 5 5 6 6 (% class="wbtoc" %) 7 7 ((( 8 8 (% class="wbtocheader" %)Inhoud 9 9 10 -{{toc/}} 10 +(% aria-label="macro:toc widget" contenteditable="false" role="region" tabindex="-1" %) 11 +((( 12 +(% class="macro" data-macro="startmacro:toc|-|" data-widget="xwiki-macro" data-xwiki-dom-updated="true" %) 13 +((( 14 +(% class="macro-placeholder hidden" %) 15 +((( 16 +macro:toc 11 11 ))) 18 + 19 +(% class="wikitoc" %) 20 +* [[Welkom op de WB Kennisbank>>doc:null||anchor="HWelkomopdeWBKennisbank"]] 21 +* [[Nieuws>>doc:null||anchor="HNieuws"]] 12 12 ))) 13 13 24 +(% style="background-color:rgba(220,220,220,0.5)" %)[[image:data:image/gif;base64,R0lGODlhAQABAPABAP///wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==||height="15" role="presentation" title="Click and drag to move" width="15"]] 25 +))) 26 +))) 27 +))) 28 + 14 14 (% class="wblinks" %) 15 15 ((( 16 -= **Welkom op de WB Kennisbank !** =31 += **Welkom op de WB Kennisbank** = 17 17 18 -In deze omgeving kunnen we kennis, de laatste ontwikkelingen (nieuws), bestanden en ideeën plaatsen en delen. Klik[[ hier >>doc:.Werkwijze.Applicaties.Kennisbank.WebHome]]voor uitleg hoe de kennisbank werkt (bv. hoe een pagina aan te maken). 33 +(% aria-label="macro:velocity widget" contenteditable="false" role="region" tabindex="-1" %) 34 +((( 35 +(% class="macro" data-macro="startmacro:velocity|-||-|#if(!$isGuest) 36 +##gebruiker is ingelogd 37 +(% style=~"font-size:18px;font-style: italic;~" ~%)**[[Zorg dat je op de hoogte bent van de kennisbank huisregels!>>doc:.Werkwijze.Kennisbank.Huisregels kennisbank.WebHome||target=_blank]]** 38 +#end" data-widget="xwiki-macro" data-xwiki-dom-updated="true" %) 39 +((( 40 +(% class="macro-placeholder hidden" %) 41 +((( 42 +macro:velocity 43 +))) 19 19 20 -{{warning}} 21 -[[Zorg dat je op de hoogte bent van dekennisbank huisregels>>doc:.Werkwijze.Applicaties.Kennisbank.Huisregels kennisbank.WebHome]]! 22 -{{/warning}} 45 +(% style="font-size:18px; font-style:italic" %)**//[[Zorg dat je op de hoogte bent van de kennisbank huisregels!>>doc:.Werkwijze.Kennisbank.Huisregels kennisbank.WebHome||target="_blank"]]//** 46 +))) 23 23 48 +(% style="background:url(~"https://kennis.wardenburg.nl/webjars/wiki%3Axwiki/xwiki-platform-ckeditor-webjar/15.8/plugins/widget/images/handle.png~") rgba(220, 220, 220, 0.5); left:0px; top:-15px" %)[[image:data:image/gif;base64,R0lGODlhAQABAPABAP///wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==||height="15" role="presentation" title="Click and drag to move" width="15"]] 49 +))) 24 24 51 +Toegang tot (kwalitatief) goede informatie is een uitdaging waar iedereen mee worsteld, met het lanceren van dit digitale archief waarin we kennis, de laatste ontwikkelingen (nieuws), bestanden en ideeën kunnen plaatsen en delen gaan we het vinden van informatie eenvoudiger maken. In de kern maakt kennisbank het mogelijk om informatie te genereren organiseren en delen 25 25 26 -{{velocity}} 53 +(% aria-label="macro:showhide widget" contenteditable="false" role="region" tabindex="-1" %) 54 +((( 55 +(% class="macro" data-macro="startmacro:showhide|-|showmessage=~"We kunnen dit op vele manieren gebruiken, klik hier om enkele voorbeelden te zien:~" hidemessage=~"Verstop voorbeelden~" effectduration=~"1~"|-|✅ Toegang tot informatie , verbreed je kennis naar nieuwe onderwerpen. 56 +* Gedeelde kennis is makkelijk doorzoekbaar, je kan naar specifieke delen van pagina’s linken met een weblink! 57 +* Kennis van collega's wordt met elkaar gedeeld. Het voorkomt dat je zelf het wiel moet uitvinden terwijl er al informatie van een collega over een bepaald onderwerp is. 58 +* //Help collega's en klanten door hen in staat te stellen zichzelf te helpen//[[(Nb. zonder in te loggen zien bezoekers ook de 'Openbaar' folder)>>doc:.Openbaar.WebHome]] 27 27 60 +✅ Dankzij de nieuwssectie kunnen ook de laatste ontwikkelingen per 'nieuwsonderwerp' op de kennisbank geplaatst worden. 61 + 62 +✅ Deel je kennis door het vast te leggen in de kennisbank 63 +* Of het nou gaat om een bekende fout in een apparaat/softwareversie of een handige manier om een installatie te configureren, als je tijd hebt geïnvesteerd om een probleem oplossen is het jammer wanneer bijvoorbeeld een collega moet zwoegen om tot dezelfde oplossing te komen. 64 + 65 +✅ Zorgt voor transparantie en duidelijkheid op kennisgebied 66 +* Uitleg en handleidingen van installaties maar ook onze werkprocessen kunnen vastgelegd en gedeeld worden met de kennisbank. 67 +* Ervaringen en discussies kunnen per onderwerp geplaatst en besproken worden." data-widget="xwiki-macro" data-xwiki-dom-updated="true" %) 68 +((( 69 +(% class="macro-placeholder hidden" %) 70 +((( 71 +macro:showhide 72 +))) 73 + 74 +(% class="macro" data-macro="startmacro:velocity|-||-|#set($discard = $xwiki.jsx.use(~"Macros.ShowHideMacro~")) 75 +#set($mparams = $wikimacro.parameters) 76 +#if(!$mparams) 77 + #set($mparams = $xcontext.macro.params) 78 +#end 79 +#if($mparams.id) 80 + #set($id = $util.convertToAlphaNumeric($mparams.id)) 81 +#end 82 +#set($showmessage = $escapetool.xml($mparams.showmessage)) 83 +#set($hidemessage = $escapetool.xml($mparams.hidemessage)) 84 +#set($divstyle = $escapetool.xml($mparams.style)) 85 +#set($effect = $escapetool.xml($mparams.effect)) 86 +#if($effect == ~"appear~" || $effect == ~"fade~") 87 + #set($effect = ~"fadeToggle~") 88 +#elseif ($effect == ~"blind~" || $effect == ~"slide~") 89 + #set($effect = ~"slideToggle~") 90 +#end 91 +#set($effectduration = $mathtool.mul(1000,$mparams.effectduration)) 92 +(% #if($divstyle && $divstyle!=~"~") style=~"$\{divstyle}~" #end ~%) 93 +((( 94 +(% class=~"showhidebutton~" ~%) 95 +((( 96 +\{\{html clean=false}} 97 +<a href=~"javascript:void(0)~" #if($id && $id!=~"~")id=~"showhidebuttontext$\{id}~" #end data-show-duration=~"$effectduration~" data-show-effect=~"$effect~" data-show-message=~"$showmessage~" data-hide-message=~"$hidemessage~">$mparams.showmessage</a> 98 +\{\{/html}} 99 +))) 100 +(% class=~"showhidecontent~" #if($id && $id!=~"~")id=~"showhidecontent$\{id}~" #end style=~"display: none;~" ~%) 101 +((( 102 +((( 103 +$xcontext.macro.content 104 +))) 105 +))) 106 +)))" %) 107 +((( 108 +(% class="macro-placeholder hidden" %) 109 +((( 110 +macro:velocity 111 +))) 112 + 113 +((( 114 +(% class="showhidebutton" %) 115 +((( 116 +(% data-macro="startmacro:html|-|clean=~"false~"|-|<a href=~"javascript:void(0)~" data-show-duration=~"1000~" data-show-effect=~"toggle~" data-show-message=~"We kunnen dit op vele manieren gebruiken, klik hier om enkele voorbeelden te zien:~" data-hide-message=~"Verstop voorbeelden~">We kunnen dit op vele manieren gebruiken, klik hier om enkele voorbeelden te zien:</a>" class="macro hidden macro-placeholder" %)macro:html(% class="macro" data-macro="startmacro:html|-|clean=~"false~"|-|<a href=~"javascript:void(0)~" data-show-duration=~"1000~" data-show-effect=~"toggle~" data-show-message=~"We kunnen dit op vele manieren gebruiken, klik hier om enkele voorbeelden te zien:~" data-hide-message=~"Verstop voorbeelden~">We kunnen dit op vele manieren gebruiken, klik hier om enkele voorbeelden te zien:</a>" %)[[We kunnen dit op vele manieren gebruiken, klik hier om enkele voorbeelden te zien:>>path:javascript:void(0)||data-hide-message="Verstop voorbeelden" data-show-duration="1000" data-show-effect="toggle" data-show-message="We kunnen dit op vele manieren gebruiken, klik hier om enkele voorbeelden te zien:"]] 117 +))) 118 + 119 +(% class="showhidecontent" style="display: none;" %) 120 +((( 121 +((( 122 +✅ Toegang tot informatie , verbreed je kennis naar nieuwe onderwerpen. 123 + 124 +* Gedeelde kennis is makkelijk doorzoekbaar, je kan naar specifieke delen van pagina’s linken met een weblink! 125 +* Kennis van collega's wordt met elkaar gedeeld. Het voorkomt dat je zelf het wiel moet uitvinden terwijl er al informatie van een collega over een bepaald onderwerp is. 126 +* //Help collega's en klanten door hen in staat te stellen zichzelf te helpen//[[(Nb. zonder in te loggen zien bezoekers ook de 'Openbaar' folder)>>doc:.Openbaar.WebHome]] 127 + 128 +✅ Dankzij de nieuwssectie kunnen ook de laatste ontwikkelingen per 'nieuwsonderwerp' op de kennisbank geplaatst worden. 129 + 130 +✅ Deel je kennis door het vast te leggen in de kennisbank 131 + 132 +* Of het nou gaat om een bekende fout in een apparaat/softwareversie of een handige manier om een installatie te configureren, als je tijd hebt geïnvesteerd om een probleem oplossen is het jammer wanneer bijvoorbeeld een collega moet zwoegen om tot dezelfde oplossing te komen. 133 + 134 +✅ Zorgt voor transparantie en duidelijkheid op kennisgebied 135 + 136 +* Uitleg en handleidingen van installaties maar ook onze werkprocessen kunnen vastgelegd en gedeeld worden met de kennisbank. 137 +* Ervaringen en discussies kunnen per onderwerp geplaatst en besproken worden. 138 +))) 139 +))) 140 +))) 141 +))) 142 +))) 143 + 144 +(% style="background:url(~"https://kennis.wardenburg.nl/webjars/wiki%3Axwiki/xwiki-platform-ckeditor-webjar/15.8/plugins/widget/images/handle.png~") rgba(220, 220, 220, 0.5); left:0px; top:-15px" %)[[image:data:image/gif;base64,R0lGODlhAQABAPABAP///wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==||height="15" role="presentation" title="Click and drag to move" width="15"]] 145 +))) 146 + 147 +---- 148 + 149 +* [[Klik hier voor uitleg hoe de kennisbank werkt>>doc:Main.Werkwijze.Kennisbank.WebHome]]. //Als je verdwaald raakt kan je altijd linksbovenin op het logo klikken, dan ga je terug naar deze homepagina. // 150 +* [[Voor discussies,vragen en/of opmerkingen over de kennisbank kan je aan dit teamkanaal meedoen>>https://teams.microsoft.com/l/team/19%3aAxo-k8j1CVRGXTb0t3Dc6tjNDNudeTZjcP_KZVFR5381%40thread.tacv2/conversations?groupId=4da292cd-ff72-4d4c-adc5-8aa3dfc4ae3f&tenantId=42b413cc-67db-4507-a4a6-ce30ce798022||rel="noopener noreferrer" target="_blank"]]. 151 + 152 +(% aria-label="macro:velocity widget" contenteditable="false" role="region" tabindex="-1" %) 153 +((( 154 +(% class="macro" data-macro="startmacro:velocity|-||-| 28 28 #if($isGuest) 29 29 #set($loginscherm = 'https://kennis.wardenburg.nl/login/XWiki/XWikiLogin') 30 30 ==**Inloggen**== 31 -Door in te loggen krijg je meer functies op de kennisbank.[[Log hier in op de wiki >>$loginscherm ||target="_blank"]] met je voornaam.achternaam (hetzelfde als je e-mail zonder het @wardenburg.nl gedeelte). 158 +Door in te loggen krijg je meer functies op de kennisbank. [[Log hier in op de wiki >>$loginscherm ||target=~"_blank~"]] met je voornaam.achternaam (hetzelfde als je e-mail zonder het @wardenburg.nl gedeelte). 159 + 160 +Ondervind je problemen met het inloggen? [[Neem dan contact op met Automatisering.>>https://teams.microsoft.com/l/chat/0/0?users=jesse.veentjer@wardenburg.nl&topicname=Xwiki|target=_blank]] 32 32 #end 33 33 #if(!$isGuest) 34 34 ##gebruiker is ingelogd 164 +(% class=~"nieuws~" ~%) 165 += **Nieuws** = 166 +Hieronder vind je de laatste (10) nieuwsberichten, [[een overzicht van alle nieuwsberichten en een onderverdeling van de categorieën vindt je op de nieuws homepagina>>doc:Nieuws.WebHome]]. 35 35 36 -= **Inhoud Kennisbank** = 37 -Hieronder vind je de navigatieboom van de kennisbank (zie ook het tabje 'Navigatie' links). Let op, de zoekfunctie kijkt alleen naar paginatitels; voor de volledige zoekfunctie gebruik je de zoekbalk linksbovenin je scherm. 38 - 39 -(% class="wbtree" %) 168 +(% class=~"nieuwstabel~" ~%) 40 40 ((( 41 - 42 -{{documentTree root="document:WebHome" showTranslations="false" showAttachments="false" finder="true"/}} 170 +\{\{blogpostlist blog=~"Nieuws.WebHome~" limit=~"10~" layout=~"cards~" layoutParams=~"displayTitle=true|useSummary=true~"/}} 43 43 ))) 172 +#end" data-widget="xwiki-macro" data-xwiki-dom-updated="true" %) 173 +((( 174 +(% class="macro-placeholder hidden" %) 175 +((( 176 +macro:velocity 177 +))) 44 44 45 -(% class="nieuws" %) 179 +(% id="HNieuws" class="nieuws" %) 46 46 = **Nieuws** = 47 47 182 +Hieronder vind je de laatste (10) nieuwsberichten, [[een overzicht van alle nieuwsberichten en een onderverdeling van de categorieën vindt je op de nieuws homepagina>>doc:Nieuws.WebHome]]. 183 + 48 48 (% class="nieuwstabel" %) 49 49 ((( 50 -{{blogpostlist blog="Nieuws.WebHome" limit="10" layout="cards" layoutParams="displayTitle=true|useSummary=true"/}} 186 +(% class="macro" data-macro="startmacro:blogpostlist|-|blog=~"Nieuws.WebHome~" limit=~"10~" layout=~"cards~" layoutParams=~"displayTitle=true|useSummary=true~"" %) 187 +((( 188 +(% class="macro-placeholder hidden" %) 189 +((( 190 +macro:blogpostlist 51 51 ))) 192 + 193 +(% class="macro" data-macro="startmacro:blogPostLayoutCards|-|reference=~"Nieuws.Nieuwsbrief januari 2024~" params=~"displayTitle=true|useSummary=true|list-limit=10|list-layout=cards|list-blog=Nieuws.WebHome|list-paginated=no|list-results-count=10|postIndex=0|previousPostReference=~"" %) 194 +((( 195 +(% class="macro-placeholder hidden" %) 196 +((( 197 +macro:blogPostLayoutCards 198 +))) 199 + 200 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogCode~"" %) 201 +((( 202 +(% class="macro-placeholder hidden" %) 203 +((( 204 +macro:include 205 +))) 206 + 207 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogParameters~"" %) 208 +((( 209 +(% class="macro-placeholder hidden" %) 210 +((( 211 +macro:include 212 +))) 213 + 214 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|## Blog 215 +#set($blogClassname = 'Blog.BlogClass') 216 +#set($blogTemplate = 'Blog.BlogTemplate') 217 +#set($blogSheet = 'Blog.BlogSheet') 218 +## Blog entries 219 +#set($blogPostClassname = 'Blog.BlogPostClass') 220 +#set($blogPostTemplate = 'Blog.BlogPostTemplate') 221 +#set($blogPostSheet = 'Blog.BlogPostSheet') 222 +#set($blogPostObjectNumber = $xwiki.getDocument($blogPostTemplate).getObject($blogPostClassname).number) 223 +#set($oldArticleClassname = 'XWiki.ArticleClass') 224 +## Categories 225 +#set($blogCategoryClassname = 'Blog.CategoryClass') 226 +#set($blogCategoryTemplate = 'Blog.CategoryTemplate') 227 +#set($blogCategorySheet = 'Blog.CategorySheet') 228 +#set($blogCategoriesSheet = 'Blog.CategoriesSheet') 229 +#set($oldBlogCategoryClassname = 'Blog.Categories') 230 +#set($defaultCategoryParent = 'Blog.Categories') 231 +## Style 232 +#set($blogStyleDocumentName = 'Blog.BlogStyle') 233 +#set($blogStyleDocument = $xwiki.getDocument($blogStyleDocumentName)) 234 +## Clientside scripts 235 +#set($blogScriptsDocumentName = 'Blog.BlogScripts') 236 +#set($blogPublisher = $xwiki.getDocument('Blog.Publisher')) 237 +## Misc 238 +#set($thisURL = $doc.getURL($xcontext.action, $request.queryString)) 239 +#set($isBlogPost = $doc.getObject($blogPostClassname)) 240 +#set($defaultBlogSpace = 'Blog') 241 +#set($isCategoryPage = $doc.getObject($blogCategoryClassname)) 242 +#set($isCategoriesHomePage = $doc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 243 +## 244 +## 245 +## 246 +#** 247 + * Displays an image, taken from the blog style document. 248 + * 249 + * @param $imgName The name of the icon from icons set to use. 250 + *# 251 +#macro(toolImage $imgName)(% class=~"icon-manage~"~%)$services.icon.render($imgName)(%~%)#end" %) 252 +((( 253 +(% class="macro-placeholder hidden" %) 254 +((( 255 +macro:velocity 256 +))) 257 +))) 258 +))) 259 + 260 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|## 261 +## 262 +## 263 +## Import the blog skin and javascripts. 264 +$!xwiki.ssx.use($blogStyleDocumentName)## 265 +$!xwiki.jsx.use($blogScriptsDocumentName)## 266 +## 267 +## import the hierarchy for the path the blog post footer (in displayEntryBlogLocation) 268 +#template('hierarchy_macros.vm')## 269 +## 270 +## 271 +#** 272 + * Prints a blog. This is the main macro used in the BlogSheet. 273 + * 274 + * @param blogDoc the XDocument holding the blog definition object. 275 + *### 276 +#macro(printBlog $blogDoc) 277 + \{\{include reference='Blog.CreatePost'/}} 278 + 279 + ## Use the blogPostList macro to display the blogposts 280 + ## The blogPostList is used only in case of 'all' or 'paged' blog display type because the blogPostList macro 281 + ## do not support FTM the monthly and weekly blog display types 282 + #getBlogDisplayType($blogDoc $displayType) 283 + #if ($displayType == 'weekly' || $displayType == 'monthly') 284 + #getBlogEntries($blogDoc $entries) 285 + #displayBlog($entries 'index' true true) 286 + #displayNavigationLinks($blogDoc) 287 + #else 288 + #getBlogDisplayType($blogDoc $displayType) 289 + #set ($paginated = 'no') 290 + #if ($displayType == 'paginated') 291 + #set ($paginated = 'yes') 292 + #end 293 + #getBlogPostsLayout($blogDoc $postsLayout) 294 + (% class=~"hfeed index~" ~%)(((\{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$paginated~" layout=~"$!postsLayout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" /}}))) 295 + #end 52 52 #end 53 -{{/velocity}} 297 +## 298 +## 299 +## 300 +#** 301 + * Shows blog information. In view mode, the description is printed. In edit mode, allows changing blog settings: title, 302 + * description, blog type (global or in-space), index display type (fixed size pagination, weekly index, monthly index, 303 + * all entries). 304 + * 305 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 306 + *### 307 +#macro(showBlogInfo $blogDoc) 308 + #if($blogDoc.getObject($blogClassname)) 309 + ## Keep testing for inline action for backward compatibility with older blogs. 310 + #if($xcontext.action == 'edit' || $xcontext.action == 'inline') 311 + #macro(displayProperty $blogDoc $propname) 312 + ; #displayPropName($xwiki.getClass($blogClassname).get($propname)): 313 + : $blogDoc.display($propname) 314 + #end 315 + #displayProperty($blogDoc 'title') 316 + #displayProperty($blogDoc 'description') 317 + #displayProperty($blogDoc 'displayType') 318 + #displayProperty($blogDoc 'itemsPerPage') 319 + #displayProperty($blogDoc 'postsLayout') 320 + #displayProperty($blogDoc 'postsLayoutParameters') 321 + #else 322 + $blogDoc.display('description') 323 + #end 324 + #elseif($doc.fullName == $blogSheet) 325 += $services.localization.render('blog.code.blogsheet') = 326 + \{\{translation key='blog.code.sheetexplanation'/}} 327 + #else 328 + \{\{warning}}\{\{translation key='blog.code.notblog'/}}\{\{/warning}} 329 + #end 330 +#end 331 +## 332 +## 333 +## 334 +#** 335 + * Retrieve the blog document, which usually is either <tt><Space>.WebHome</tt> for whole-spaces blogs, or 336 + * <tt><Space>.Blog</tt> for in-space blogs. If none of these documents contains a blog object, then the first 337 + * (alphabetically) document in the target space that contains one is returned. Finally, if no document in the current 338 + * space contains a blog object, then <tt>Blog.WebHome</tt> is returned as the default blog. 339 + * 340 + * @param space A <tt>String</tt>, the name of the space where to search. 341 + * @param blogDoc The resulting XDocument. 342 + *### 343 +#macro(getBlogDocument $space $blogDoc) 344 + #set ($result = $NULL) 345 + ## Check if the current space has a WebPreferences page that contains a Blog.EnablePanelsConfigurationClass object. 346 + ## It is possible to display a blog's panels info in a page that is not inside a blog and in that case we need to 347 + ## identify the right blog based on a configuration object in a WebPreferences page. 348 + #set ($spaceReference = $services.model.resolveSpace($space)) 349 + #set ($preferencesDocRef = $services.model.createDocumentReference('WebPreferences', $spaceReference)) 350 + #set ($preferencesDoc = $xwiki.getDocument($preferencesDocRef)) 351 + #set ($preferencesObj = $preferencesDoc.getObject('Blog.EnablePanelsConfigurationClass')) 352 + #if ($preferencesObj) 353 + #set ($result = $xwiki.getDocument( $preferencesObj.getValue('blog'))) 354 + #end 355 + ## If no special case occurs (like the previous one), identify a blog by checking the current location from its hierarchy. 356 + #if (~"$!result~" == '') 357 + ## First, try the Space.WebHome, for a whole-space blog 358 + #set($result = $xwiki.getDocument(~"$\{space}.WebHome~")) 359 + #if(!$result.getObject($blogClassname)) 360 + ## Second, try the Space.Blog document 361 + #set($result = $xwiki.getDocument(~"$\{space}.Blog~")) 362 + #if(!$result.getObject($blogClassname)) 363 + ## Third, try searching for a blog document in the current space 364 + ## Prevent the query fail when the space contains dots '.' 365 + #set($blogDocs = $services.query.hql(~", BaseObject obj where doc.space = :space and obj.name = doc.fullName and obj.className = '$blogClassname' order by doc.name~").setLimit(1).setOffset(0).bindValue('space', $space).execute()) 366 + #if($blogDocs.size() > 0) 367 + #set($result = $xwiki.getDocument($blogDocs.get(0))) 368 + #else 369 + ## Fourth, try searching for a blog document that have the its 'postsLocation' set to the current space 370 + #set($blogDocs = $services.query.hql(~", BaseObject obj, StringProperty as postsLocationProp where obj.name = doc.fullName and obj.className = '$blogClassname' and obj.id=postsLocationProp.id.id and postsLocationProp.id.name='postsLocation' and postsLocationProp.value=:postsLocation order by doc.name~").setLimit(1).setOffset(0).bindValue('postsLocation', $space).execute()) 371 + #if($blogDocs.size() > 0) 372 + #set($result = $xwiki.getDocument($blogDocs.get(0))) 373 + #else 374 + ## Last, fallback to Blog.WebHome, the default blog 375 + #set($result = $xwiki.getDocument('Blog.WebHome')) 376 + #end 377 + #end 378 + #end 379 + #end 380 + #end 381 + #set ($blogDoc = $NULL) 382 + #setVariable (~"$blogDoc~" $result) 383 +#end 384 +## 385 +## 386 +## 387 +#** 388 + * Retrieve the blog title. 389 + * 390 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass<tt> object with the <tt>title</tt> property set. 391 + * @param title The resulting title. 392 + *### 393 +#macro(getBlogTitle $blogDoc $title) 394 + ## Titles can contain velocity code (most commonly translations), so we should evaluate them. 395 + #set ($title = $NULL) 396 + #setVariable (~"$title~" $!blogDoc.displayTitle) 397 +#end 398 +## 399 +## 400 +## 401 +#** 402 + * Retrieve the blog description. 403 + * 404 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>description</tt> 405 + * property set. 406 + * @param description The resulting description. 407 + *### 408 +#macro(getBlogDescription $blogDoc $description) 409 + #getBlogProperty($blogDoc 'description' '' $result) 410 + #set ($description = $NULL) 411 + #setVariable (~"$description~" $result) 412 +#end 413 +## 414 +## 415 +## 416 +#** 417 + * Retrieves a list of entries to be displayed. The entries are either part of the blog's space, or have the blog 418 + * document set as a parent. The number and range of entries returned (from all those belonging to this blog) depends on 419 + * the blog display type: paginated (fixed number of entries), weekly (all entries in a week), monthly (all entries in a 420 + * month), or all. 421 + * 422 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 423 + * @param entries The resulting list of entries to display, a list of XDocument names. 424 + *### 425 +#macro(getBlogEntries $blogDoc $entries) 426 + #if (!$entries) 427 + #setVariable (~"$entries~" []) 428 + #end 429 + #getAllBlogPostsQuery($query) 430 + #isDefaultBlog($blogDoc $isDefault) 431 + #set($queryParams = \{}) 432 + #if ($isDefault) 433 + #getCategoryAllBlogPostsQuery($query) 434 + #set($query = ~"$\{query} and (doc.creator = :creator or (isPublished.value = 1 and hidden.value = 0)) and (doc.space = :space or catList like :catList escape '!')~") 435 + #set($discard = $queryParams.put('creator', $xcontext.user)) 436 + #set($discard = $queryParams.put('space', $blogDoc.space)) 437 + #set($sanitizedBlogSpace = $blogDoc.space.replaceAll('([%_!])', '!$1').concat('.%')) 438 + #set($discard = $queryParams.put('catList', $sanitizedBlogSpace)) 439 + #else 440 + #set($query = ~"$\{query} and (doc.space = :space or doc.parent = :parent)~") 441 + #getBlogPostsLocation($blogDoc.space $blogPostsLocation) 442 + #set($discard = $queryParams.put('space', $blogPostsLocation)) 443 + #set($discard = $queryParams.put('parent', $blogDoc.space)) 444 + #end 445 + #getBlogDisplayType($blogDoc $displayType) 446 + #if($displayType == 'weekly') 447 + #getWeeklyBlogEntries($blogDoc $query $entries $queryParams) 448 + #elseif($displayType == 'monthly') 449 + #getMonthlyBlogEntries($blogDoc $query $entries $queryParams) 450 + #elseif($displayType == 'all') 451 + #getAllBlogEntries($blogDoc $query $entries $queryParams) 452 + #else 453 + #getPagedBlogEntries($blogDoc $query $entries $queryParams) 454 + #end 455 +#end 456 +## 457 +## 458 +## 459 +#** 460 + * Retrieves a list of entries to be displayed. The entries are taken from a ~"page~" of the blog, a sequence of documents 461 + * defined by the request parameters <tt>ipp</tt> (items per page) and <tt>page</tt> (the current page). Initially the 462 + * first page is displayed, with the number of entries defined in the blog object in the <tt>itemsPerPage</tt> property 463 + * (10 if not defined). 464 + * 465 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 466 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 467 + * refined to restrict to a given space, or to a given search criteria, etc. 468 + * @param entries The resulting list of entries to display, a list of XDocument names. 469 + * @param queryParams The parameters to bind with the query. 470 + *### 471 +#macro(getPagedBlogEntries $blogDoc $query $entries $queryParams) 472 + #if (!$entries) 473 + #setVariable (~"$entries~" []) 474 + #end 475 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 476 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~")) 477 + #bindQueryParameters($countQueryObj $queryParams) 478 + #bindQueryParameters($queryObj $queryParams) 479 + #set($totalEntries = $countQueryObj.count()) 480 + #getBlogProperty($blogDoc 'itemsPerPage' '10' $defaultItemsPerPage) 481 + #set($defaultItemsPerPage = $numbertool.toNumber($defaultItemsPerPage).intValue()) 482 + ## This macro is defined in the default macros.vm library. It also sets $itemsPerPage and $startAt. 483 + #preparePagedViewParams($totalEntries $defaultItemsPerPage) 484 + #set($discard = $entries.addAll($queryObj.setLimit($itemsPerPage).setOffset($startAt).addFilter(~"unique~").execute())) 485 +#end 486 +## 487 +## 488 +## 489 +#** 490 + * Retrieves a list of entries to be displayed. The entries are taken from a week of the blog. The target week is 491 + * defined by the request parameters <tt>week</tt> (the week number in the year, from 1 to 52) and <tt>year</tt> (4 492 + * digit year). Initially the current week is displayed. 493 + * 494 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 495 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 496 + * refined to restrict to a given space, or to a given search criteria, etc. 497 + * @param entries The resulting list of entries to display, a list of XDocument names. 498 + * @param queryParams The parameters to bind with the query. 499 + *### 500 +#macro(getWeeklyBlogEntries $blogDoc $query $entries $queryParams) 501 + #if (!$entries) 502 + #setVariable (~"$entries~" []) 503 + #end 504 + #getRequestedWeek($weekDate) 505 + #set($dateFormatter = $xwiki.jodatime.getDateTimeFormatterForPattern('yyyy-MM-dd')) 506 + #set($minDay = $dateFormatter.print($weekDate.toMutableDateTime().weekOfWeekyear().roundFloor())) 507 + #set($maxDay = $dateFormatter.print($weekDate.toMutableDateTime().weekOfWeekyear().roundCeiling())) 508 + #set($query = ~"$\{query} and publishDate.value >= '$minDay' and publishDate.value < '$maxDay'~") 509 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 510 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 511 + #bindQueryParameters($countQueryObj $queryParams) 512 + #bindQueryParameters($queryObj $queryParams) 513 + #set($totalEntries = $countQueryObj.count()) 514 + #set($discard = $entries.addAll($queryObj.execute())) 515 +#end 516 +## 517 +## 518 +## 519 +#** 520 + * Retrieves a list of entries to be displayed. The entries are taken from a month of the blog. The target month is 521 + * defined by the request parameters <tt>month</tt> (the month number, from 1 to 12) and <tt>year</tt> (4 522 + * digit year). Initially the current month is displayed. 523 + * 524 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 525 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 526 + * refined to restrict to a given space, or to a given search criteria, etc. 527 + * @param entries The resulting list of entries to display, a list of XDocument names. 528 + * @param queryParams The parameters to bind with the query. 529 + *### 530 +#macro(getMonthlyBlogEntries $blogDoc $query $entries) 531 + #if (!$entries) 532 + #setVariable (~"$entries~" []) 533 + #end 534 + #getRequestedMonth($monthDate) 535 + #set($dateFormatter = $xwiki.jodatime.getDateTimeFormatterForPattern('yyyy-MM-dd')) 536 + #set($minDay = $dateFormatter.print($monthDate.toMutableDateTime().monthOfYear().roundFloor())) 537 + #set($maxDay = $dateFormatter.print($monthDate.toMutableDateTime().monthOfYear().roundCeiling())) 538 + #set($query = ~"$\{query} and publishDate.value >= '$minDay' and publishDate.value < '$maxDay'~") 539 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 540 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 541 + #bindQueryParameters($countQueryObj $queryParams) 542 + #bindQueryParameters($queryObj $queryParams) 543 + #set($totalEntries = $countQueryObj.count()) 544 + #set($discard = $entries.addAll($queryObj.execute())) 545 +#end 546 +## 547 +## 548 +## 549 +#** 550 + * Retrieves a list of entries to be displayed. All entries belonging to the current blog are returned. 551 + * 552 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 553 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 554 + * refined to restrict to a given space, or to a given search criteria, etc. 555 + * @param entries The resulting list of entries to display, a list of XDocument names. 556 + * @param queryParams The parameters to bind with the query. 557 + *### 558 +#macro(getAllBlogEntries $blogDoc $query $entries $queryParams) 559 + #if (!$entries) 560 + #setVariable (~"$entries~" []) 561 + #end 562 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 563 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 564 + #bindQueryParameters($countQueryObj $queryParams) 565 + #bindQueryParameters($queryObj $queryParams) 566 + #set($totalEntries = $countQueryObj.count()) 567 + #set($discard = $entries.addAll($queryObj.execute())) 568 +#end 569 +## 570 +## 571 +## 572 +#** 573 + * Retrieves a list of entries to be displayed. Only (and all) unpublished entries are returned. 574 + * 575 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 576 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 577 + * refined to restrict to a given space, or to a given search criteria, etc. 578 + * @param queryParams The parameters to bind with the query. 579 + * @param entries The resulting list of entries to display, a list of XDocument names. 580 + *### 581 +#macro(getUnpublishedBlogEntries $blogDoc $query $queryParams $entries) 582 + #if (!$entries) 583 + #setVariable (~"$entries~" []) 584 + #end 585 + #set($query = ~"$\{query} and isPublished.value = 0~") 586 + #set ($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 587 + #set ($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 588 + #bindQueryParameters($countQueryObj $queryParams) 589 + #bindQueryParameters($queryObj $queryParams) 590 + #set($totalEntries = $countQueryObj.count()) 591 + #set($discard = $entries.addAll($queryObj.execute())) 592 +#end 593 +## 594 +## 595 +## 596 +#** 597 + * Retrieves a list of entries to be displayed. The entries are taken from all the wiki, and not from a specific blog. 598 + * 599 + * @param entries The resulting list of entries to display, a list of XDocument names. 600 + *### 601 +#macro(getGlobalBlogEntries $entries) 602 + #if (!$entries) 603 + #setVariable (~"$entries~" []) 604 + #end 605 + #getAllBlogPostsQuery($query) 606 + #set($totalEntries = $services.query.hql($query).count()) 607 + #set($defaultItemsPerPage = 20) 608 + ## This macro is defined in the default macros.vm library. It also sets $itemsPerPage and $startAt. 609 + #preparePagedViewParams($totalEntries $defaultItemsPerPage) 610 + #set($discard = $entries.addAll($services.query.hql(~"$\{query} order by publishDate.value desc~").setLimit($itemsPerPage).setOffset($startAt).execute())) 611 +#end 612 +#** 613 + * Return the base query for selecting blog entries. It filters only visible entries, but does not bind to a specific 614 + * blog, nor specify a range or an ordering criteria. 615 + * 616 + * This macro is a duplicate of #getAllBlogPostsQuery($query). Keep it in case it's used outside, in custom code. 617 + * 618 + * @param query The basic query for selecting blog entries. 619 + *# 620 +#macro(getBlogEntriesBaseQuery $query) 621 + #getAllBlogPostsQuery($query) 622 +#end 623 +#** 624 + * Return the Query for selecting the all wiki blog posts without filtering 625 + * 626 + * @param query The basic query for selecting blog entries. 627 + *# 628 +#macro(getAllBlogPostsQuery $query) 629 + #set ($query = $NULL) 630 + #setVariable(~"$query~" ~", BaseObject as obj, IntegerProperty isPublished, 631 + IntegerProperty hidden, DateProperty publishDate 632 + where doc.fullName <> '$blogPostTemplate' and 633 + obj.name=doc.fullName and obj.className='$blogPostClassname' and 634 + isPublished.id.id = obj.id and isPublished.id.name = 'published' and 635 + hidden.id.id = obj.id and hidden.id.name='hidden' and 636 + publishDate.id.id = obj.id and publishDate.id.name='publishDate' and 637 + (doc.creator = '$escapetool.sql($xcontext.user)' or (isPublished.value = 1 and hidden.value = 0))~") 638 +#end 639 +## 640 +## 641 +## 642 +###** 643 + * Return the Query for selecting the all wiki blog posts with categories filtering 644 + * 645 + * @param query The basic query for selecting blog entries. 646 + *### 647 +#macro(getCategoryAllBlogPostsQuery $query) 648 + #set ($query = $NULL) 649 + #getAllBlogPostsQuery($baseQuery) 650 + #set ($baseQuery = $baseQuery.replace('DateProperty publishDate', 'DateProperty publishDate, DBStringListProperty as category left join category.list catList')) 651 + #setVariable(~"$query~" ~"$\{baseQuery} and obj.id=category.id.id and category.id.name='category'~") 652 +#end 653 +## 654 +## 655 +## 656 +#** 657 + * Determines how is the blog index split into pages: paginated (fixed number of entries), weekly (all entries in a 658 + * week), monthly (all entries in a month), or all. 659 + * 660 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>displayType</tt> 661 + * property set. 662 + * @param displayType The resulting string. If the blog object does not define anything, it is considered paginated. 663 + *### 664 +#macro(getBlogDisplayType $blogDoc $displayType) 665 + #getBlogProperty($blogDoc 'displayType' 'paginated' $result) 666 + #set ($displayType = $NULL) 667 + #setVariable (~"$displayType~" $result) 668 +#end 669 +## 670 +## 671 +## 672 +#** 673 + * Displays a list of entries. 674 + * 675 + * @param entries The entries to display, a list of XDocument names. 676 + * @param displaying What exactly is displayed: blog index, a single blog entry, a blog category, search results, 677 + * unpublished entries, etc. This will be used as the classname(s) for the container div (hfeed). Currently 678 + * used values: index, single, category, search, unpublished, hidden. 679 + * @param onlyExtract If <tt>true</tt>, only display the extract of articles where available, otherwise display the full content. 680 + * @param shouldDisplayTitles If <tt>true</tt>, display the blog title (blog posts shouldn't display the title when they're 681 + * displayed alone on their page since it's the page title which is used in this case) 682 + *### 683 +#macro(displayBlog $entries $displaying $onlyExtract $shouldDisplayTitles) 684 + #set($blogDay = '') 685 + (% class=~"hfeed $!\{displaying}~" ~%)((( 686 + (% class=~"blogDay~" ~%)((( 687 + #foreach ($entryDoc in $xwiki.wrapDocs($entries)) 688 + #getEntryObject($entryDoc $entryObj) 689 + ## Although all entries should have one of the two objects, better check to be sure. 690 + #if(~"$!\{entryObj}~" != '') 691 + #getEntryDate($entryDoc $entryObj $entryDate) 692 + ## Display a ~"calendar sheet~" for each day. All entries posted on the same day share one such sheet. 693 + #set($entryDateStr = $xwiki.formatDate($entryDate, 'yyyyMMMMdd')) 694 + #if($blogDay != $entryDateStr) 695 + #if($blogDay != '') 696 + ))) 697 + (% class=~"blogDay~" ~%)((( 698 + #end 699 + #displayBlogDate($entryDate) 700 + #set ($blogDay = $entryDateStr) 701 + #end 702 + ## Finally, display the entry. 703 + #displayEntry($entryDoc $entryObj $onlyExtract $shouldDisplayTitles true) 704 + #end 705 + #end 706 + )))## blogDay 707 + )))## hfeed 708 +#end 709 +## 710 +## 711 +## 712 +#** 713 + * Get the entry object, either a new BlogPost or an old Article. 714 + * 715 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 716 + * @param entryObj The resulting xobject of the blog post. 717 + *### 718 +#macro(getEntryObject $entryDoc $__entryObj) 719 + #set($result = $entryDoc.getObject(~"$\{blogPostClassname}~")) 720 + #if(!$result) 721 + #set($result = $entryDoc.getObject(~"$\{oldArticleClassname}~")) 722 + #end 723 + ## NOTE: The reason we put an underscore in front of the variable name is to prevent the following line from 724 + ## overwriting the $entryObj variable that may be defined before this macro is called. Of course, $__entryObj may be 725 + ## overwritten in this case but it's less likely to have such a variable defined before. 726 + #set ($__entryObj = $NULL) 727 + #setVariable (~"$__entryObj~" $result) 728 +#end 729 +## 730 +## 731 +## 732 +#** 733 + * Gets the date associated with a blog entry. This is the publication date. For unpublished entries, initially this is 734 + * the document creation date, but can be edited by the user. 735 + * 736 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 737 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 738 + * @param result The resulting date, an instance of <tt>java.util.Date</tt>. 739 + *### 740 +#macro(getEntryDate $entryDoc $entryObj $result) 741 + #set ($result = $NULL) 742 + #setVariable (~"$result~" $entryObj.getProperty('publishDate').value) 743 +#end 744 +## 745 +## 746 +## 747 +#** 748 + * Displays a date, nicely formatted as a calendar page. 749 + * 750 + * @param date The date to display, an instance of <tt>java.util.Date</tt>. 751 + *### 752 +#macro(displayBlogDate $date) 753 + #set($year = $xwiki.formatDate($date, 'yyyy')) 754 + ## 3 letter month name, like Jan, Dec. 755 + #set($month = $xwiki.formatDate($date, 'MMM')) 756 + ## Uncomment to get a full length month name, like January, December. 757 + ## TODO: this could be defined somewhere in the blog style. 758 + ## #set($month = $xwiki.formatDate($date, 'MMMM')) 759 + #set($day = $xwiki.formatDate($date, 'dd')) 760 + (% class=~"blogdate~" ~%) 761 + == (% class=~"month~" ~%)$month(%~%) (% class=~"day~" ~%)$day(%~%) (% class=~"year~" ~%)$year(%~%) == 762 +#end 763 +## 764 +## 765 +## 766 +#** 767 + * Displays a blog article: management tools, header, content, footer. 768 + * 769 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 770 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 771 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 772 + * @param shouldDisplayTitle If <tt>true</tt>, display the blog title (blog posts shouldn't display the title 773 + * when they're displayed alone on their page since it's the page title which is used in this case) 774 + * @param shouldDisplayActions If <tt>true</tt>, display the blog post actions buttons 775 + *### 776 +#macro(displayEntry $entryDoc $entryObj $onlyExtract $shouldDisplayTitle $shouldDisplayActions) 777 + ## Only articles with an explicit hidden setting or an explicit unpublished setting are hidden 778 + #isPublished($entryObj $isPublished) 779 + #isHidden($entryObj $isHidden) 780 + #if($doc.fullName == $entryDoc.fullName) 781 + (% class=~"hentry single-article~" ~%)((( 782 + #else 783 + (% class=~"hentry#if(!$isPublished) unpublished-article#elseif($isHidden) hidden-article#end~" ~%)((( 784 + #end 785 + #if ($shouldDisplayActions) 786 + #displayEntryTools($entryDoc $entryObj) 787 + #end 788 + #if($shouldDisplayTitle) 789 + #displayEntryTitle($entryDoc $entryObj) 790 + #end 791 + #if($doc.fullName == $entryDoc.fullName) 792 + #if(!$isPublished) 793 + \{\{warning}}\{\{translation key='blog.code.published'/}}\{\{/warning}} 794 + #elseif($isHidden) 795 + \{\{warning}}\{\{translation key='blog.code.hidden'/}}\{\{/warning}} 796 + #end 797 + #end 798 + #displayEntryContent($entryDoc $entryObj $onlyExtract) 799 + #displayEntryFooter($entryDoc $entryObj) 800 + )))## hentry 801 +#end 802 +## 803 +## 804 +## 805 +#** 806 + * Checks if the provided blog is published or not. 807 + * 808 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 809 + * @param isPublished The resulting boolean, true if the entry is considered published. 810 + *### 811 +#macro(isPublished $entryObj $isPublished) 812 + #set ($isPublished = $NULL) 813 + ## This should work for both old articles, which don't have the 'published' property at all, and 814 + ## are considered published by default, and new entries, that should have 1 if published. 815 + #if (~"$!\{entryObj.getProperty('published').value}~" != '0') 816 + #setVariable (~"$isPublished~" true) 817 + #else 818 + #setVariable (~"$isPublished~" false) 819 + #end 820 +#end 821 +## 822 +## 823 +## 824 +#** 825 + * Checks if the provided blog is hidden or not. 826 + * 827 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass<tt> xclass. 828 + * @param isHiddel The resulting boolean, true if the entry is considered hidden. 829 + *### 830 +#macro(isHidden $entryObj $isHidden) 831 + #set ($isHidden = $NULL) 832 + ## This should work for both old articles, which don't have the 'hidden' property at all, and 833 + ## are considered visible by default, and new entries, that should have 1 if hidden. 834 + #if (~"$!\{entryObj.getProperty('hidden').value}~" == '1') 835 + #setVariable (~"$isHidden~" true) 836 + #else 837 + #setVariable (~"$isHidden~" false) 838 + #end 839 +#end 840 +## 841 +## 842 +## 843 +#** 844 + * Displays several ~"tools~" for manipulating blog posts: hide/show, publish, edit. 845 + * 846 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 847 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 848 + *### 849 +#macro(displayEntryTools $entryDoc $entryObj) 850 + #if($xcontext.action == 'view') 851 + (% class=~"blog-entry-toolbox~" ~%)((( 852 + #displayPublishButton($entryDoc $entryObj) 853 + #displayHideShowButton($entryDoc $entryObj) 854 + #displayEditButton($entryDoc $entryObj) 855 + #displayDeleteButton($entryDoc $entryObj) 856 + ))) 857 + #end 858 +#end 859 +## 860 +## 861 +## 862 +#** 863 + * Displays the publish button to the entry <strong>creator</strong>, if the article is not published yet. 864 + * 865 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 866 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 867 + * @todo AJAX calls. 868 + *### 869 +#macro(displayPublishButton $entryDoc $entryObj) 870 + #isPublished($entryObj $isPublished) 871 + #if(!$isPublished && $entryDoc.creator == $xcontext.user && $xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 872 + [[#toolImage('world')>>path:$blogPublisher.getURL('view', ~"entryName=$\{escapetool.url($entryDoc.fullName)}&xredirect=$\{escapetool.url($thisURL)}&form_token=$!\{services.csrf.getToken()}~")||title=~"$services.localization.render('blog.code.notpublished')~"]]## 873 + #end 874 +#end 875 +## 876 +## 877 +## 878 +#** 879 + * Displays the hide or show button to the entry <strong>creator</strong>, if the article is already published. 880 + * 881 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 882 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 883 + *### 884 +#macro(displayHideShowButton $entryDoc $entryObj) 885 + #isPublished($entryObj $isPublished) 886 + #isHidden($entryObj $isHidden) 887 + ## Only published articles can be hidden. Unpublished articles are considered already hidden. 888 + #if($isPublished && $entryDoc.creator == $xcontext.user && $xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 889 + #set ($queryString = \{ 890 + 'xredirect' : $thisURL, 891 + 'form_token' : $services.csrf.getToken() 892 + }) 893 + #if ($isHidden) 894 + #set ($discard = $queryString.putAll(\{ 895 + ~"$\{entryObj.getxWikiClass().getName()}_$\{entryObj.number}_hidden~" : 0, 896 + 'comment' : $services.localization.render('blog.code.madevisible') 897 + })) 898 + #set ($lockURL = $entryDoc.getURL('save', $escapetool.url($queryString))) 899 + [[#toolImage('unlock')>>path:$lockURL||class=~"blog-tool-show~" title=~"$services.localization.render('blog.code.makevisible')~"]]## 900 + #else 901 + #set ($discard = $queryString.putAll(\{ 902 + ~"$\{entryObj.getxWikiClass().getName()}_$\{entryObj.number}_hidden~" : 1, 903 + 'comment' : $services.localization.render('blog.code.hid') 904 + })) 905 + #set ($lockURL = $entryDoc.getURL('save', $escapetool.url($queryString))) 906 + [[#toolImage('lock')>>path:$lockURL||class=~"blog-tool-hide~" title=~"$services.localization.render('blog.code.hide')~"]]## 907 + #end 908 + #end 909 +#end 910 +## 911 +## 912 +## 913 +#** 914 + * Displays the edit button to those that can edit the article. 915 + * 916 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 917 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 918 + *### 919 +#macro(displayEditButton $entryDoc $entryObj) 920 + #if($xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 921 + ## Call getDefaultEditMode() for backward compatibility with older blog posts. 922 + [[#toolImage('pencil')>>path:$entryDoc.getURL('edit')||title=~"$services.localization.render('blog.code.editpost')~"]]## 923 + #end 924 +#end 925 +## 926 +## 927 +## 928 +#** 929 + * Displays the delete button to those that can edit the article. 930 + * 931 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 932 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 933 + * @todo AJAX calls. 934 + *### 935 +#macro(displayDeleteButton $entryDoc $entryObj) 936 + #if($xwiki.hasAccessLevel('delete', $xcontext.user, $entryDoc.fullName)) 937 + [[#toolImage('cross')>>path:$entryDoc.getURL('delete')||title=~"$services.localization.render('blog.code.deletepost')~"]]## 938 + #end 939 +#end 940 +## 941 +## 942 +## 943 +#** 944 + * Displays the title of the entry. 945 + * 946 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 947 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 948 + *### 949 +#macro(displayEntryTitle $entryDoc $entryObj) 950 + #if($doc.fullName == $entryDoc.fullName) 951 + (% class=~"entry-title~" ~%) 952 + = $services.rendering.escape($entryDoc.display('title', 'view', $entryObj), $xwiki.getCurrentContentSyntaxId()) = 953 + #else 954 + (% class=~"entry-title~" ~%) 955 + === [[$services.rendering.escape($entryDoc.display('title', 'view', $entryObj),$xwiki.getCurrentContentSyntaxId())>>doc:$services.rendering.escape($services.model.serialize($entryDoc.getDocumentReference(),'default'),$xwiki.getCurrentContentSyntaxId())]] === 956 + #end 957 +#end 958 +## 959 +## 960 +## 961 +#** 962 + * Displays the body of the entry. 963 + * 964 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 965 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 966 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 967 + *### 968 +#macro(displayEntryContent $entryDoc $entryObj $onlyExtract) 969 + (% class=~"#if($onlyExtract)entry-summary#\{else}entry-content#end~" ~%)((( 970 + #getEntryContent($entryDoc $entryObj $onlyExtract $entryContent) 971 + ## FIXME: This causes the blog's content to not be annotatable. See http://jira.xwiki.org/browse/XWIKI-6328 972 + ## Should probably be replaced by a display macro call with a reference to the object property holding the post's content 973 + \{\{html wiki=~"false~"}}$entryDoc.getRenderedContent($entryContent, $entryDoc.syntax.toIdString())\{\{/html}} 974 + ))) ## entry-content 975 + (% class=~"clearfloats~" ~%)((())) 976 +#end 977 +## 978 +## 979 +## 980 +#** 981 + * Extracts the body of the entry that should be displayed. If <tt>onlyExtract</tt> is <tt>true</tt>, display the content 982 + * of the <tt>extract</tt> field (if not empty). 983 + * 984 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 985 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 986 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 987 + * @param entryContent The resulting content. 988 + * @param removeEllipsis If <tt>true</tt>, then don't display an ellipsis at the end of the content (only used when 989 + * <tt>onlyExtract</tt> is <tt>true</tt>) 990 + *### 991 +#macro(getEntryContent $entryDoc $entryObj $onlyExtract $entryContent $removeEllipsis) 992 + #if ($onlyExtract) 993 + ## Note: We trim the summary so that if there's some white space it won't cause the summary to be used instead 994 + ## of the content. 995 + #set ($macro.result = $entryObj.getProperty('extract').value.trim()) 996 + #end 997 + #if(~"$!macro.result~" == '') 998 + #set($macro.result = $entryObj.getProperty('content').value) 999 +#* Disabled until the content can be cleanly cut. 1000 +* #if($onlyExtract && $result.length()>$maxchars) 1001 +* #set($i = $macro.result.lastIndexOf(~" ~", $maxchars)) 1002 +* #set($i = $i + 1) 1003 +* #set($macro.result = ~"$\{macro.result.substring(0,$i)} *[...>$\{entryDoc.fullName}]*~") 1004 +* #end 1005 +## *### 1006 + #elseif (!$removeEllipsis) 1007 + #if($entryDoc.syntax.toIdString() == 'xwiki/1.0') 1008 + #set($macro.result = ~"$\{macro.result} <a href='$\{entryDoc.getExternalURL()}' title='$services.localization.render('blog.code.readpost')'>...</a>~") 1009 + #elseif($entryDoc.syntax.toIdString() == 'xwiki/2.0' || $entryDoc.syntax.toIdString() == 'xwiki/2.1') 1010 + #set($macro.result = ~"$\{macro.result} [[...>>$\{services.rendering.escape($entryDoc.getFullName(), $entryDoc.getSyntax())}||title='$services.localization.render('blog.code.readpost')']]~") 1011 + #end 1012 + #end 1013 + #set ($entryContent = $NULL) 1014 + #setVariable (~"$entryContent~" $macro.result) 1015 +#end 1016 +## 1017 +## 1018 +## 1019 +#** 1020 + * Displays the footer of the entry. 1021 + * 1022 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 1023 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 1024 + *### 1025 +#macro(displayEntryFooter $entryDoc $entryObj) 1026 + (% class=~"entry-footer~" ~%)((( 1027 + #isPublished($entryObj $isPublished) 1028 + (% class='entry-author-label' ~%) 1029 + #if($isPublished) 1030 + \{\{translation key='blog.code.postedby'/}} ## 1031 + #else 1032 + \{\{translation key='blog.code.createdby'/}} ## 1033 + #end 1034 + \{\{html wiki=~"false~" clean=~"false~"}}<span class='author vcard'>#userfn($entryDoc.creator)</span>\{\{/html}} ## 1035 + #getEntryDate($entryDoc $entryObj $entryDate) 1036 + #listCategories($entryObj) #* 1037 + ## Since the publish date and update date are not set at the exact same time, there could be a small difference that 1038 + ## we assume cannot be more than 3 seconds. 1039 + *#(% class=~"separator~" ~%)·(%~%) [[\{\{translation key='blog.code.permalink'/}}>>$services.rendering.escape($entryDoc,'xwiki/2.0')||rel=~"bookmark~"]] ## 1040 + #if ($showcomments) 1041 + (% class=~"separator~" ~%)·(%~%) [[\{\{translation key='blog.code.comments'/}}>>$services.rendering.escape($entryDoc,'xwiki/2.0')||anchor=~"Comments~"]] (% class=~"itemCount~" ~%)($entryDoc.comments.size())(%~%) ## 1042 + #end ## 1043 + #if($entryDoc != $doc) ## 1044 + #displayEntryBlogLocation($entryDoc $entryObj) ## 1045 + #end 1046 + )))## entry-footer 1047 +#end 1048 +## 1049 +## 1050 +#** 1051 + * Display the blog for the entry (if it is not the currently displayed blog) 1052 + * 1053 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 1054 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 1055 + *### 1056 +#macro(displayEntryBlogLocation $entryDoc $entryObj) 1057 + #getBlogPostsLocation($entryDoc.getSpace() $blogPostsLocation) ## 1058 + #if(~"$!blogPostsLocation~" != ~"~") ## 1059 + #set($blogSpaceRef = $services.model.resolveSpace($blogPostsLocation)) ## 1060 + #set($blogDocRef = $services.model.createDocumentReference($services.model.getEntityReference('DOCUMENT', 'default').getName(), $blogSpaceRef)) ## 1061 + #if($doc.documentReference != $blogDocRef) ## 1062 + #set($blogDoc = $xwiki.getDocument($blogDocRef)) ## 1063 + #set($blogObj = $blogDoc.getObject('Blog.BlogClass')) ## 1064 + #if($blogObj) (% class=~"blogrefpath~" ~%)((( ## 1065 + $services.localization.render('blog.code.postedin') \{\{html clean=~"false~" wiki=~"false~"}} #hierarchy($blogDocRef, \{'selfIsActive' : true, 'local': true}) \{\{/html}} ## 1066 + )))(%~%)## 1067 + #end 1068 + #end 1069 + #end 1070 +#end 1071 +## 1072 +## 1073 +## 1074 +## 1075 +#** 1076 + * List the categories an entry belongs to. Used in the footer. The categories are instances of <tt>Blog.CategoryClass</tt>. 1077 + * 1078 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 1079 + *### 1080 +#macro(listCategories $entryObj) 1081 + #if($entryObj.getxWikiClass().getName() == $blogPostClassname) 1082 + #set($categories = $entryObj.getProperty('category').value) 1083 + #set($first = true) 1084 + #if($categories.size() > 0) 1085 + #foreach($category in $categories) 1086 + #set($categoryDoc = $!xwiki.getDocument($category)) 1087 + #if(!$categoryDoc.isNew() && $categoryDoc.getObject($\{blogCategoryClassname})) 1088 + #if($foreach.count == 1) 1089 + (% class='separator' ~%)·(%~%) $services.localization.render('blog.code.categories') ## 1090 + #else 1091 + , ## 1092 + #end## 1093 + [[$!\{categoryDoc.getObject($blogCategoryClassname).getValue('name')}>>$\{category}||rel='tag']]## 1094 + #end## 1095 + #end## 1096 + #end 1097 + #end 1098 +#end 1099 +## 1100 +## 1101 +## 1102 +#** 1103 + * Displays blog pagination links (older and newer entries). 1104 + * 1105 + * @param blogDoc the XDocument holding the blog definition object. 1106 + *### 1107 +#macro(displayNavigationLinks $blogDoc) 1108 + (% class=~"clearfloats~" ~%)((())) 1109 + #getBlogDisplayType($blogDoc $displayType) 1110 + #if($displayType == 'weekly') 1111 + (% class=~"pagingLinks~" ~%)((( 1112 + #getRequestedWeek($weekDate) 1113 + $weekDate.addWeeks(-1)## 1114 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.previousweek'/}}>>$doc.documentReference.name||queryString=~"year=$weekDate.weekyear&week=$weekDate.weekOfWeekyear~" class=~"button~"]]**(%~%) 1115 + #sep() 1116 + $weekDate.addWeeks(2)## 2 because we already subtracted 1 above 1117 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.nextweek'/}} »>>$doc.documentReference.name||queryString=~"year=$weekDate.weekyear&week=$weekDate.weekOfWeekyear~" class=~"button~"]]**(%~%) 1118 + ))) 1119 + #elseif($displayType == 'monthly') 1120 + (% class=~"pagingLinks~" ~%)((( 1121 + #getRequestedMonth($monthDate) 1122 + $monthDate.addMonths(-1)## 1123 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.previousmonth'/}}>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"year=$monthDate.year&month=$monthDate.monthOfYear~" class=~"button~"]]**(%~%) 1124 + #sep() 1125 + $monthDate.addMonths(2)## 2 because we already subtracted 1 above 1126 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.nextmonth'/}} »>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"year=$monthDate.year&month=$monthDate.monthOfYear~" class=~"button~"]]**(%~%) 1127 + ))) 1128 + #elseif($displayType == 'all') 1129 + #else 1130 + ## Paginated 1131 + #if(($totalPages > 1)) 1132 + #set($queryString = '') 1133 + #foreach($p in $request.getParameterNames()) 1134 + #if($p != 'page' && $p != 'ipp') 1135 + #foreach($v in $request.getParameterValues($p)) 1136 + #set($queryString = ~"$\{queryString}&$\{escapetool.url($p)}=$\{escapetool.url($v)}~") 1137 + #end 1138 + #end 1139 + #end 1140 + (% class=~"pagingLinks~" ~%)((( 1141 + #if ($currentPageNumber < $totalPages) 1142 + #set($currentPageNumber = $currentPageNumber + 1) 1143 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.olderposts'/}}>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"page=$\{currentPageNumber}&ipp=$\{itemsPerPage}$queryString~" class=~"button~"]]**(%~%) 1144 + #set($currentPageNumber = $currentPageNumber - 1) 1145 + #end 1146 + #if ($currentPageNumber > 1) 1147 + #if ($currentPageNumber < $totalPages) 1148 + #sep() 1149 + #end 1150 + #set($currentPageNumber = $currentPageNumber - 1) 1151 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.newerposts'/}} »>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"page=$\{currentPageNumber}&ipp=$\{itemsPerPage}$queryString~" class=~"button~"]]**(%~%) 1152 + #set($currentPageNumber = $currentPageNumber + 1) 1153 + #end 1154 + (% class=~"clear~" ~%)(%~%) 1155 + )))## pagingLinks 1156 + #end 1157 + #end 1158 +#end 1159 +## 1160 +## 1161 +## 1162 +#** 1163 + * Displays a message box with ~"publish~" icon. 1164 + * 1165 + * @param message A text message concerning blog article publishing 1166 + *### 1167 +#macro(publishMessageBox $message) 1168 +(% class=~"plainmessage publish-message~" ~%)((($services.icon.render('world') $message))) 1169 +#end 1170 +#** 1171 + * Displays a message box with ~"show/hide~" icon. 1172 + * 1173 + * @param message A text message concerning blog article hiding 1174 + *### 1175 +#macro(hideMessageBox $message) 1176 +(% class=~"plainmessage hide-message~" ~%)((($services.icon.render('unlock') $message))) 1177 +#end 1178 +## 1179 +## 1180 +## 1181 +#** 1182 + * Determine the requested week, for using in a weekly-indexed blog. The relevant request parameters are 1183 + * <tt>year</tt> and <tt>week</tt>. By default, the current week is used. 1184 + * 1185 + * @param monthDate The resulting week, a JODATime MutableDateTime. 1186 + *### 1187 +#macro(getRequestedWeek $weekDate) 1188 + #set ($weekDate = $NULL) 1189 + #setVariable (~"$weekDate~" $xwiki.jodatime.mutableDateTime) 1190 + #if(~"$!\{request.year}~" != '') 1191 + #set ($discard = $weekDate.setYear($numbertool.toNumber($request.year).intValue())) 1192 + #end 1193 + #if(~"$!\{request.week}~" != '') 1194 + #set ($discard = $weekDate.setWeekOfWeekyear($numbertool.toNumber($request.week).intValue())) 1195 + #end 1196 +#end 1197 +## 1198 +## 1199 +## 1200 +#** 1201 + * Determine the requested month, for using in a monthly-indexed blog. The relevant request parameters are 1202 + * <tt>year</tt> and <tt>month</tt>. By default, the current month is used. 1203 + * 1204 + * @param monthDate The resulting month, a JODATime MutableDateTime. 1205 + *### 1206 +#macro(getRequestedMonth $monthDate) 1207 + #set ($monthDate = $NULL) 1208 + #setVariable (~"$monthDate~" $xwiki.jodatime.mutableDateTime) 1209 + #if(~"$!\{request.year}~" != '') 1210 + #set ($discard = $monthDate.setYear($numbertool.toNumber($request.year).intValue())) 1211 + #end 1212 + #if(~"$!\{request.month}~" != '') 1213 + #set ($discard = $monthDate.setMonthOfYear($numbertool.toNumber($request.month).intValue())) 1214 + #end 1215 +#end 1216 +## 1217 +## 1218 +## 1219 +#** 1220 + * Retrieve a blog property (title, display type, etc). 1221 + * 1222 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 1223 + * @param propertyName The name of the property to be retrieved. One of the <tt>Blog.BlogClass</tt>'s properties. 1224 + * @param defaultValue The default value to use in case the blog object does not define one. 1225 + * @param propertyValue The resulting value. 1226 + *### 1227 +#macro(getBlogProperty $blogDoc $propertyName $defaultValue $propertyValue) 1228 + #set($result = ~"$!\{blogDoc.getObject($\{blogClassname}).getProperty($propertyName).value}~") 1229 + #if($result == '') 1230 + #set($result = $defaultValue) 1231 + #end 1232 + #set ($propertyValue = $NULL) 1233 + #setVariable (~"$propertyValue~" $result) 1234 +#end 54 54 1236 +#** 1237 + * If an error occurs when executing an action, set a specific response status and display an error message. 1238 + * 1239 + * @param status The response status. 1240 + * @param text The user readable error to be displayed. Can be a translation key. 1241 + * @param parameters The parameters to use when decoding the translation key. 1242 + *### 1243 +#macro(blog__actionResponseError $status $text $parameters) 1244 + $response.setStatus($status) 1245 + #if($request.ajax) 1246 + $services.localization.render($text, $!parameters) 1247 + #else 1248 + \{\{error}}$services.localization.render($text, $!parameters)\{\{/error}} 1249 + #end 1250 +#end 1251 +## 1252 +## 1253 +## 1254 +#** 1255 + * Check if a blog is the Default blog (The one in the 'Blog' space). 1256 + * 1257 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 1258 + * @param isDefault The resulting boolean. 1259 + *### 1260 +#macro(isDefaultBlog $blogDoc $isDefault) 1261 + #set ($result = false) 1262 + #if ($blogDoc.space == 'Blog') 1263 + #set ($result = true) 1264 + #end 1265 + #setVariable(~"$isDefault~" $result) 1266 +#end 1267 +## 1268 +## 1269 +## 1270 +#** 1271 + * Retrieve the blog posts location (space). 1272 + * 1273 + * @param blogSpace The blog space. It should contain a document with a <tt>Blog.BlogClass</tt> object 1274 + * @param postsLocation The resulting location. 1275 + *### 1276 +#macro(getBlogPostsLocation $blogSpace $postsLocation) 1277 + #getBlogDocument($blogSpace $blogDoc) 1278 + #getBlogProperty($blogDoc 'postsLocation' $blogDoc.space $result) 1279 + #set ($postsLocation = $NULL) 1280 + #setVariable (~"$postsLocation~" $result) 1281 +#end 1282 +## 1283 +## 1284 +## 1285 +#** 1286 + * Retrieve the blog categories location (space). 1287 + * 1288 + * @param blogSpace The blog space. It should contain a document with a <tt>Blog.BlogClass</tt> object 1289 + * @param categoriesLocation The resulting location. 1290 + *### 1291 +#macro(getBlogCategoriesLocation $blogSpace $categoriesLocation) 1292 + #getBlogDocument($blogSpace $blogDoc) 1293 + #getBlogProperty($blogDoc 'categoriesLocation' 'Blog' $result) 1294 + #set ($postsLocation = $NULL) 1295 + #setVariable (~"$categoriesLocation~" $result) 1296 +#end 1297 +###** 1298 + * Return the Query for selecting the blog posts based on the context where the posts are displayed. 1299 + * for example there is 4 different panel contexts: 1300 + * aBlog.aPost or aBlog.WebHome 1301 + * aCategorySpace.aCategory 1302 + * aCategorySpace.WebHome 1303 + * Blog.aPost or Blog.WebHome 1304 + * 1305 + * @param query The query for selecting blog entries. 1306 + * @param queryParams The parameters to bind with the generated query. 1307 + * @param targetDoc The document in which the articles will be displayed. 1308 + *### 1309 +#macro(getAllBlogPostsQueryBasedOnDisplayContext $targetDoc $query $queryParams) 1310 + #set ($query = $NULL) 1311 + #set ($queryParams = $NULL) 1312 + #getCategoryAllBlogPostsQuery($resultQuery) 1313 + #set ($resultQueryParams = \{}) 1314 + #set($sanitizedSpace = $targetDoc.space.replaceAll('([%_!])', '!$1').concat('.%')) 1315 + #if ($targetDoc.space == $defaultBlogSpace && !$targetDoc.getObject($blogCategoryClassname)) 1316 + ## Get all posts that are in the default blog space('Blog') or in category 'Blog.%' 1317 + #set ($discard = $resultQueryParams.put('space', $targetDoc.space)) 1318 + #set ($discard = $resultQueryParams.put('catList', $sanitizedSpace)) 1319 + #set($resultQuery = ~"$\{resultQuery} and (doc.space = :space or catList like :catList escape '!')~") 1320 + #elseif($targetDoc.getObject($blogCategoryClassname)) 1321 + ## Get all posts that are in a category aCategorySpace.aCategory 1322 + #set ($discard = $resultQueryParams.put('catList', $targetDoc.fullName)) 1323 + #set($resultQuery = ~"$\{resultQuery} and (catList=:catList)~") 1324 + #elseif($targetDoc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 1325 + ## Get all posts that are in a category aCategorySpace.% 1326 + #set ($discard = $resultQueryParams.put('catList', $sanitizedSpace)) 1327 + ## Exclude incategorized posts 1328 + #set ($excludedCategory = ~"$\{targetDoc.space}.WebHome~") 1329 + #if ($targetDoc.space == $defaultBlogSpace) 1330 + #set ($excludedCategory = ~"Blog.Categories.WebHome~") 1331 + #end 1332 + #set ($discard = $resultQueryParams.put('excludedCategory', $excludedCategory)) 1333 + #set($resultQuery = ~"$\{resultQuery} and (catList<>:excludedCategory) and (catList like :catList escape '!')~") 1334 + #else 1335 + ## Get all posts in blog space aBlog 1336 + #getAllBlogPostsQuery($resultQuery) 1337 + #getBlogPostsLocation($targetDoc.space $postsLocation) 1338 + #set ($discard = $resultQueryParams.put('space', $postsLocation)) 1339 + #set($resultQuery = ~"$\{resultQuery} and doc.space = :space~") 1340 + #end 1341 + #setVariable(~"$query~" $resultQuery) 1342 + #setVariable(~"$queryParams~" $resultQueryParams) 1343 +#end 1344 +## 1345 +## 1346 +## 1347 +###** 1348 + * Display blog posts based on the context where the posts are displayed. 1349 + * for example there is 4 different panel contexts: 1350 + * aBlog.aPost or aBlog.WebHome 1351 + * aCategorySpace.aCategory 1352 + * aCategorySpace.WebHome 1353 + * Blog.aPost or Blog.WebHome 1354 + * 1355 + * @param targetDoc The document in which the articles will be displayed. 1356 + * @param postsVisiblity The visibility of the posts to display: recent, unpublished, ... 1357 + * @param layout Layout of the the posts to display 1358 + * @param layoutParams additional layout parameters, it is a string like 'param1=val1|...|paramN=valueN' 1359 + * @param limit the number of posts to display 1360 + *### 1361 +#macro(displayBlogPostsBasedOnContext $targetDoc $postsVisiblity $layout $layoutParams $limit) 1362 + #if ($postLayout == 'full') 1363 + #set ($macro.paginated = 'yes') 1364 + #end 1365 + #if ($targetDoc.space == $defaultBlogSpace && !$targetDoc.getObject($blogCategoryClassname)) 1366 + ## Display all posts that are in the default blog space('Blog') or in category 'Blog.%' 1367 + #getBlogPostsLayout($xwiki.getDocument(~"$\{defaultBlogSpace}.WebHome~") $postsLayout) 1368 + #if (~"$!layout~" == '') 1369 + #set ($layout = $postsLayout) 1370 + #end 1371 + #if ($postsVisiblity == 'recent') 1372 + \{\{blogpostlist blog=~"$\{defaultBlogSpace.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')}.WebHome~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 1373 + #elseif($postsVisiblity == 'unpublished') 1374 + \{\{blogpostlist blog=~"$\{defaultBlogSpace.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')}.WebHome~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 1375 + #end 1376 + #elseif($targetDoc.getObject($blogCategoryClassname)) 1377 + ## Display all posts that are in a category aCategorySpace.aCategory 1378 + #getBlogDocumentForCategoriesSpace($targetDoc.space $blogDoc) 1379 + #getBlogPostsLayout($blogDoc $postsLayout) 1380 + #if (~"$!layout~" == '') 1381 + #set ($layout = $postsLayout) 1382 + #end 1383 + #if ($postsVisiblity == 'recent') 1384 + \{\{blogpostlist category=~"$targetDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 1385 + #elseif($postsVisiblity == 'unpublished') 1386 + \{\{blogpostlist category=~"$targetDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 1387 + #end 1388 + #elseif($targetDoc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 1389 + ## Display all posts that are in a category aCategorySpace.% 1390 + #getBlogDocumentForCategoriesSpace($targetDoc.space $blogDoc) 1391 + #getBlogPostsLayout($blogDoc $postsLayout) 1392 + #if (~"$!layout~" == '') 1393 + #set ($layout = $postsLayout) 1394 + #end 1395 + #if ($postsVisiblity == 'recent') 1396 + \{\{blogpostlist category=~"$targetDoc.space.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 1397 + #elseif($postsVisiblity == 'unpublished') 1398 + \{\{blogpostlist category=~"$targetDoc.space.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 1399 + #end 1400 + #else 1401 + ## Display all posts in blog space aBlog 1402 + #getBlogDocument($targetDoc.space $blogDoc) 1403 + #getBlogPostsLayout($blogDoc $postsLayout) 1404 + #if (~"$!layout~" == '') 1405 + #set ($layout = $postsLayout) 1406 + #end 1407 + #if ($postsVisiblity == 'recent') 1408 + \{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 1409 + #elseif($postsVisiblity == 'unpublished') 1410 + \{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 1411 + #end 1412 + #end 1413 +#end 1414 +## 1415 +## 1416 +## 1417 +#** 1418 + * Bind parameters to a query object. 1419 + * 1420 + * @param queryObj The query object (org.xwiki.query.internal.ScriptQuery object) 1421 + * @param queryParams the query parameters. 1422 + *### 1423 +#macro(bindQueryParameters $queryObj $queryParams) 1424 + #set ($output = $queryObj) 1425 + #foreach( $key in $queryParams.keySet() ) 1426 + #set($output = $queryObj.bindValue($key, $queryParams.get($key))) 1427 + #end 1428 + #setVariable(~"$queryObj~" $output) 1429 +#end 1430 +## 1431 +## 1432 +## 1433 +#** 1434 + * Determines the blogposts layout: Default layout (Calendar), Image thumbnail 1435 + * 1436 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>postsLayout</tt> 1437 + * property set. 1438 + * @param postsLayout The resulting string. If the blog object does not define anything, it is considered full (default). 1439 + *### 1440 +#macro(getBlogPostsLayout $blogDoc $postsLayout) 1441 + #getBlogProperty($blogDoc 'postsLayout' 'full' $res) 1442 + #set ($postsLayout = $NULL) 1443 + #setVariable (~"$postsLayout~" $res) 1444 +#end 1445 +## 1446 +## 1447 +## 1448 +#** 1449 + * Retrieve the blog document of a categories space, which is the blog descriptor that have its categoriesLocation property set to the categories space 1450 + * 1451 + * @param categoriesSpace A <tt>String</tt>, the name of the space. 1452 + * @param blogDoc The resulting XDocument. 1453 + *### 1454 +#macro(getBlogDocumentForCategoriesSpace $categoriesSpace $blogDoc) 1455 + #set($macro.blogDocs = $services.query.hql(~", BaseObject obj, StringProperty as categoriesLocationProp where obj.name = doc.fullName and obj.className = '$blogClassname' and obj.id=categoriesLocationProp.id.id and categoriesLocationProp.id.name='categoriesLocation' and categoriesLocationProp.value=:categoriesLocation order by doc.creationDate~").setLimit(1).setOffset(0).bindValue('categoriesLocation', $categoriesSpace).execute()) 1456 + #if($macro.blogDocs.size() > 0 && $categoriesSpace != $defaultBlogSpace) 1457 + #set($macro.result = $xwiki.getDocument($macro.blogDocs.get(0))) 1458 + #else 1459 + ## Fallback to Blog.WebHome, the default blog 1460 + #set($macro.result = $xwiki.getDocument('Blog.WebHome')) 1461 + #end 1462 + #set ($blogDoc = $NULL) 1463 + #setVariable (~"$blogDoc~" $macro.result) 1464 +#end" %) 1465 +((( 1466 +(% class="macro-placeholder hidden" %) 1467 +((( 1468 +macro:velocity 1469 +))) 1470 +))) 1471 +))) 55 55 1473 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogPostLayoutMacros~"" %) 1474 +((( 1475 +(% class="macro-placeholder hidden" %) 1476 +((( 1477 +macro:include 1478 +))) 1479 + 1480 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|#** 1481 + * Extract the layout parameters from a string. 1482 + * 1483 + * @param layoutParamsString The string representation of the layout parameters. 1484 + * It should contain a String following this format ~"paramName1=Value1|paramName2=Value2|...|paramNameK=ValueK~" 1485 + * @param layoutsParameters The resulting layout parameters Map. 1486 + *### 1487 +#macro(extractLayoutParametersFromString $layoutParamsString $layoutsParameters) 1488 + #set ($layoutsParameters = $NULL) 1489 + #set ($macro.layoutParams = \{}) 1490 + #if (~"$!layoutParamsString~" != '') 1491 + #set ($macro.paramsArr = $layoutParamsString.split('\\|')) 1492 + #foreach ($item in $macro.paramsArr) 1493 + #set ($itemSplit = $item.split('=')) 1494 + #if ($itemSplit.size() == 2) 1495 + #set ($discard = $macro.layoutParams.put($itemSplit[0].trim(), $itemSplit[1].trim())) 1496 + #end 1497 + #end 1498 + #end 1499 + #setVariable(~"$layoutsParameters~" $macro.layoutParams) 1500 +#end" %) 1501 +((( 1502 +(% class="macro-placeholder hidden" %) 1503 +((( 1504 +macro:velocity 1505 +))) 1506 +))) 1507 +))) 1508 + 1509 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-| #macro(displaySmallPost $pDoc $pObj) 1510 + #initLayoutVars($pDoc $pObj) 1511 + <div class=~"col-xs-12 col-sm-6 next_blog~"> 1512 + <a href=~"$pDoc.uRL~" class=~"thumbnail~"> 1513 + <div class=~"row~"> 1514 + <div class=~"col-xs-4~"> 1515 + #if ($imageAtt) 1516 + <img src=~"$escapetool.xml($pDoc.getAttachmentURL($pObj.getValue('image'),'download',$imgQs))~" /> 1517 + #end 1518 + </div> 1519 + <div class=~"col-xs-8 art_det~"> 1520 + <p class=~"text-left~"> 1521 + #if($displayTitle)$!postTitle #end 1522 + <br/><span class=~"date_info~"> $!dateStr </span> 1523 + </p> 1524 + #displayPostDetails($pDoc) 1525 + </div> 1526 + </div> 1527 + </a> 1528 + </div> 1529 + #end 1530 + ## 1531 + ## 1532 + ## 1533 + #macro(displayPinnedPost $pDoc $pObj) 1534 + #initLayoutVars($pDoc $pObj) 1535 + <div class=~"col-xs-12 col-sm-4 col-md-4 blog_pin~"> 1536 + <a href=~"$pDoc.uRL~" class=~"all-post-link~"> 1537 + <div class=~"thumbnail~"> 1538 + #if ($displayTitle && $displayTitleFirstOnPinnedPosts) 1539 + <h5 class=~"title-top~">$!postTitle <br/> <span class=~"date_info~"> $!dateStr </span> </h5> 1540 + #end 1541 + #if ($imageAtt) 1542 + <img src=~"$escapetool.xml($pDoc.getAttachmentURL($pObj.getValue('image'),'download',$imgQs))~" /> 1543 + #end 1544 + <div class=~"caption~"> 1545 + #if ($displayTitle && !$displayTitleFirstOnPinnedPosts) 1546 + <h5 class=~"text-left~">$!postTitle <br/> <span class=~"date_info~"> $!dateStr </span> </h5> 1547 + #end 1548 + #if ($displaySummaryOnPinnedPosts) 1549 + <div class=~"text-left post-summary~"> 1550 + #set ($postContent = $pObj.getProperty('extract').value) 1551 + $!pDoc.getRenderedContent($postContent, $pDoc.syntax.toIdString()) 1552 + </div> 1553 + #end 1554 + #displayPostDetails($pDoc) 1555 + </div> 1556 + </div> 1557 + </a> 1558 + </div> 1559 + #end 1560 + ## 1561 + ## 1562 + ## 1563 + #macro(formatPostDate $pDoc $pObj) 1564 + #set ($formattedDate = $NULL) 1565 + #set($dateStr = $datetool.format('medium', $postDate, $xcontext.locale)) 1566 + #if (~"$!dateStr~" != '') 1567 + #set ($dateArr = $dateStr.split(' ')) 1568 + #if ($dateArr.size() > 3) 1569 + #set ($dateStr = ~"~") 1570 + #foreach($s in $dateArr.subList(0, 3)) 1571 + #set ($dateStr = ~"$\{dateStr}$\{s} ~") 1572 + #end 1573 + #end 1574 + #end 1575 + #setVariable(~"$formattedDate~" $dateStr) 1576 + #end 1577 + ## 1578 + ## 1579 + ## 1580 + #macro(displayPostDetails $pDoc) 1581 + <div class=~"row post-details~"> 1582 + <div class=~"col-xs-8 detail~"> 1583 + #if ($services.like.displayButton($pDoc.documentReference)) 1584 + #set ($optLikeRecord = $services.like.getLikes($pDoc.documentReference)) 1585 + ## Retrieve the likes number in XWiki 12.9+ 1586 + #set ($likeNumber = $optLikeRecord.get()) 1587 + #if (!$stringtool.isNumeric($likeNumber.toString())) 1588 + ## Retrieve the likes number in XWiki versions greater than 12.7 and lower than 12.9. 1589 + #set ($likeNumber = $optLikeRecord.get().likeNumber) 1590 + #end 1591 + #if ($stringtool.isNumeric($likeNumber.toString())) 1592 + <div class=~"post-likes btn btn-default disabled badge~" 1593 + title=~"$escapetool.xml($services.localization.render('like.button.title', [$likeNumber]))~"> 1594 + $services.icon.renderHTML('heart') <span class=~"like-number~">$likeNumber</span> 1595 + </div> 1596 + #end 1597 + #elseif ($services.ratings) 1598 + #set ($ratingsConfigDoc = $services.ratings.getConfigurationDocument($pDoc.documentReference)) 1599 + #if ($ratingsConfigDoc.getValue('displayRatings') == 1) 1600 + <ul class=~"pull-left note list-inline~"> 1601 + #foreach ($x in [1..5]) 1602 + #set ($cls = ~"~") 1603 + #if ($x > $averageVote) 1604 + #set ($cls = ~"-o~") 1605 + #end 1606 + <li><span class=~"fa fa-star$\{cls}~"></span></li> 1607 + #end 1608 + </ul> 1609 + #end 1610 + #end 1611 + </div> 1612 + #if ($showPostComments) 1613 + <div class=~"col-xs-4 com_det~"> 1614 + <p class=~"text-right~">$services.icon.renderHTML('comments') ($!nbComments)</p> 1615 + </div> 1616 + #end 1617 + </div> 1618 + #end 1619 + ## 1620 + ## 1621 + ## 1622 + #macro(initLayoutVars $pDoc $pObj) 1623 + #set ($postTitle = $pDoc.display('title', 'view', $pObj)) 1624 + #set ($imageAtt = $pDoc.getAttachment($pObj.getValue('image'))) 1625 + #isPublished($pObj $isPublished) 1626 + #isHidden($pObj $isHidden) 1627 + #getEntryDate($pDoc $pObj $postDate) 1628 + #formatPostDate($postDate $dateStr) 1629 + #set ($averageVote = $mathtool.round($services.ratings.getAverageRating($pDoc.documentReference).getAverageVote())) 1630 + #set ($nbComments = $pDoc.getComments().size()) 1631 + #set ($showPostComments = true) 1632 + #if ($xwiki.getSpacePreferenceFor('showcomments', $pDoc.documentReference.lastSpaceReference, '1') == '0') 1633 + #set ($showPostComments = false) 1634 + #end 1635 + #end 1636 + ## 1637 + ## 1638 + ## 1639 + #macro(displayEditPinnedPostsButton) 1640 + #if (~"$!macroAdditionalParams.get('postIndex')~" == '0' && $services.security.authorization.hasAccess('edit', $pinnedPostsSourceDoc.fullName)) 1641 + #set ($discard = $xwiki.ssx.use('Blog.PinnedPostsSheet')) 1642 + #set ($discard = $xwiki.jsx.use('Blog.PinnedPostsSheet')) 1643 + #set ($editPinnedPostsURL = $xwiki.getURL($pinnedPostsSourceDoc.fullName, 'get', 'sheet=Blog.PinnedPostsSheet')) 1644 + #set ($newPinnedPostsObjectURL = $xwiki.getURL($pinnedPostsSourceDoc.fullName, 'objectadd', ~"classname=Blog.PinnedPostsClass&form_token=$!\{services.csrf.getToken()}~")) 1645 + <div class=~"edit-pinned-posts-container~"> 1646 + <div class=~"col-xs-12 col-sm-12 col-md-12 text-right edit-pinned-posts~"> 1647 + <a href=~"javascript:;~" class=~"editPinnedPosts~"> 1648 + <span class=~"glyphicon glyphicon-edit~" aria-hidden=~"true~"></span> $services.localization.render('blog.post.layout.cards.pinnedposts.edit') 1649 + </a> 1650 + <input type=~"hidden~" id=~"editPinnedPostsURL~" value=~"$escapetool.xml($editPinnedPostsURL)~"/> 1651 + <input type=~"hidden~" id=~"newPinnedPostsObjectURL~" value=~"$escapetool.xml($newPinnedPostsObjectURL)~"/> 1652 + </div> 1653 + </div> 1654 + #if (~"$!pinnedPostsObj~" == '') 1655 + <input type=~"hidden~" name=~"createPinnedPostsObject~" value=~"1~"/> 1656 + #end 1657 + #end 1658 + #end" %) 1659 +((( 1660 +(% class="macro-placeholder hidden" %) 1661 +((( 1662 +macro:velocity 1663 +))) 1664 +))) 1665 + 1666 +(% class="macro" data-macro="startmacro:velocity|-||-| #if ($services.security.authorization.hasAccess('view', $xcontext.macro.params.reference)) 1667 + #set ($postDoc = $xwiki.getDocument($xcontext.macro.params.reference)) 1668 + #getEntryObject($postDoc $postObj) 1669 + #if (~"$!postObj~" != '') 1670 + #extractLayoutParametersFromString($xcontext.macro.params.params $macroAdditionalParams) 1671 + ## 1672 + #set ($displayTitle = true) 1673 + #if (~"$!macroAdditionalParams.get('displayTitle')~" == 'false') 1674 + #set ($displayTitle = false) 1675 + #end 1676 + ## 1677 + #set ($displayTitleFirstOnPinnedPosts = false) 1678 + #if (~"$!macroAdditionalParams.get('displayTitleFirstOnPinnedPosts')~" == 'true') 1679 + #set ($displayTitleFirstOnPinnedPosts = true) 1680 + #end 1681 + ## 1682 + #set ($displaySummaryOnPinnedPosts = true) 1683 + #if (~"$!macroAdditionalParams.get('displaySummaryOnPinnedPosts')~" == 'false') 1684 + #set ($displaySummaryOnPinnedPosts = false) 1685 + #end 1686 + ## 1687 + #set ($postIndex = $numbertool.toNumber($macroAdditionalParams.get('postIndex')).intValue()) 1688 + #if ($postIndex == 0) 1689 + #set ($stopBlogPostsDisplay = false) 1690 + #end 1691 + #set ($listResultsCount = $numbertool.toNumber($macroAdditionalParams.get('list-results-count')).intValue()) 1692 + #set($discard = $xwiki.ssx.use('Blog.BlogPostLayoutCards')) 1693 + #set($scaleWidth = 600) 1694 + #set($imgQs=~"width=$scaleWidth~") 1695 + ## Display pinned posts 1696 + ## Get the list of pinned posts 1697 + #set ($pinnedPosts = []) 1698 + #set ($pinnedPostsSourceDoc = $NULL) 1699 + #if (~"$!macroAdditionalParams.get('forceDisplayPinnedPosts')~" == 'true' && ~"$!macroAdditionalParams.get('list-blog')~" != '') 1700 + #set ($pinnedPostsSourceDoc = $xwiki.getDocument($macroAdditionalParams.get('list-blog'))) 1701 + ## Note that the 'list-blog' parameter contains the value of the 'blog' parameter of the blogpostlist macro passed to the current layout macro 1702 + #end 1703 + #if (~"$!pinnedPostsSourceDoc~" != '') 1704 + #set ($pinnedPostsObj = $pinnedPostsSourceDoc.getObject('Blog.PinnedPostsClass')) 1705 + #if (~"$!pinnedPostsObj~" != '') 1706 + #set ($orderedPinnedPostsJSON = $pinnedPostsObj.getProperty('orderedPosts').value) 1707 + #if (~"$!orderedPinnedPostsJSON~" != '') 1708 + #set ($pinnedPosts = $jsontool.parse($orderedPinnedPostsJSON)) 1709 + #else 1710 + #set ($pinnedPosts = $pinnedPostsObj.getProperty('posts').value) 1711 + #end 1712 + #end 1713 + #if ($postIndex == 0 && $pinnedPosts.size() > 0) 1714 + \{\{html clean=~"false~"}} 1715 + #set ($x = 0) 1716 + #set ($showPinnedPostsButton = true) 1717 + #foreach ($pinnedPost in $pinnedPosts) 1718 + #if ($x == 0) 1719 + <div class=~"row flex-container~"> 1720 + #if ($showPinnedPostsButton) 1721 + #displayEditPinnedPostsButton() 1722 + #set($showPinnedPostsButton = false) 1723 + #end 1724 + #end 1725 + #set ($pinnedPostDoc = $xwiki.getDocument($pinnedPost)) 1726 + #getEntryObject($pinnedPostDoc $pinnedPostObj) 1727 + #if (~"$!pinnedPostObj~" != '') 1728 + #displayPinnedPost($pinnedPostDoc $pinnedPostObj) 1729 + #end 1730 + #set ($x = $mathtool.add($x, 1)) 1731 + #if ($x == 3) 1732 + #set ($x = 0) 1733 + </div> 1734 + #end 1735 + #end 1736 + #if ($mathtool.mod($x, 3) != 0) 1737 + </div> 1738 + #end 1739 + \{\{/html}} 1740 + 1741 + ## If the first post is a pinned post : this means that all the posts are pinned 1742 + ## In this case, avoid displaying the posts again after the pinned posts section. 1743 + #if ($pinnedPosts.contains($xcontext.macro.params.reference)) 1744 + #set ($stopBlogPostsDisplay = true) 1745 + #end 1746 + #elseif($postIndex == 0 && $pinnedPosts.size() == 0) 1747 + 1748 + \{\{html}} 1749 + <div class=~"row no-pinnded-posts~"> 1750 + #displayEditPinnedPostsButton() 1751 + </div> 1752 + \{\{/html}} 1753 + 1754 + #end 1755 + #end 1756 + #if (!$stopBlogPostsDisplay) 1757 + \{\{html clean=~"false~"}} 1758 + #if (~"$!nbDisplayedPosts~" == '' || $postIndex == 0) 1759 + #set ($nbDisplayedPosts = 0) 1760 + #end 1761 + #if($mathtool.mod($nbDisplayedPosts, 2) == 0) 1762 + #if ($nbDisplayedPosts != 0) 1763 + </div> 1764 + #set ($lastHtmlTag = 'c') 1765 + #end 1766 + #if ($nbDisplayedPosts == $mathtool.sub($listResultsCount, 1)) 1767 + <div class=~"row~"> 1768 + #set ($lastHtmlTag = 'o') 1769 + #set ($lastRowClass = ~"~") 1770 + #else 1771 + <div class=~"row flex-container~"> 1772 + #set ($lastHtmlTag = 'o') 1773 + #set ($lastRowClass = ~"flex~") 1774 + #end 1775 + #end 1776 + #displaySmallPost($postDoc $postObj) 1777 + #if ($nbDisplayedPosts == $mathtool.sub($listResultsCount, 1)) 1778 + </div> 1779 + #set ($lastHtmlTag = 'c') 1780 + #end 1781 + #set ($nbDisplayedPosts = $mathtool.add($nbDisplayedPosts, 1)) 1782 + #if ($postIndex == $mathtool.sub($listResultsCount, 1)) 1783 + #if (~"$!lastHtmlTag~" == 'o') 1784 + #if ($mathtool.mod($nbDisplayedPosts, 2) == 1 && $lastRowClass == ~"flex~") 1785 + <div class=~"col-xs-12 col-sm-6~"></div> 1786 + #end 1787 + </div> 1788 + #end 1789 + #end 1790 + \{\{/html}} 1791 + #end 1792 + #else 1793 + \{\{error}}$services.localization.render('blog.blogpostlayout.notpost', [$xcontext.macro.params.reference])\{\{/error}} 1794 + #end 1795 + #else 1796 + \{\{error}}$services.localization.render('blog.blogpostlayout.post_view_not_allowed', [$xcontext.macro.params.reference])\{\{/error}} 1797 + #end" %) 1798 +((( 1799 +(% class="macro-placeholder hidden" %) 1800 +((( 1801 +macro:velocity 1802 +))) 1803 + 1804 +(% class="macro" data-macro="startmacro:html|-|clean=~"false~"|-|<div class=~"row flex-container~"> 1805 +<div class=~"col-xs-12 col-sm-6 next_blog~"> 1806 +<a href=~"/Nieuws/Nieuwsbrief%20januari%202024~" class=~"thumbnail~"> 1807 +<div class=~"row~"> 1808 +<div class=~"col-xs-4~"> 1809 +<img src=~"/download/Nieuws/Nieuwsbrief%20januari%202024/man-outdoor-light-architecture-people-street-1194699-pxhere.com.jpg?width=600&rev=1.1~" /> 1810 +</div> 1811 +<div class=~"col-xs-8 art_det~"> 1812 +<p class=~"text-left~"> 1813 +Nieuwsbrief januari 2024 <br/><span class=~"date_info~"> Feb 5, 2024, </span> 1814 +</p> 1815 +<div class=~"row post-details~"> 1816 +<div class=~"col-xs-8 detail~"> 1817 +<div class=~"post-likes btn btn-default disabled badge~" 1818 +title=~"Number of likes on this page: 3~"> 1819 +<span class=~"fa fa-heart~"></span> <span class=~"like-number~">3</span> 1820 +</div> 1821 +</div> 1822 +<div class=~"col-xs-4 com_det~"> 1823 +<p class=~"text-right~"><span class=~"fa fa-comments~"></span> (0)</p> 1824 +</div> 1825 +</div> 1826 +</div> 1827 +</div> 1828 +</a> 1829 +</div>" %) 1830 +((( 1831 +(% class="macro-placeholder hidden" %) 1832 +((( 1833 +macro:html 1834 +))) 1835 + 1836 +(% class="row flex-container" %) 1837 +((( 1838 +(% class="col-xs-12 col-sm-6 next_blog" %) 1839 +((( 1840 +(% class="row" %) 1841 +((( 1842 +(% class="col-xs-4" %) 1843 +((( 1844 +[[~[~[image:/download/Nieuws/Nieuwsbrief%20januari%202024/man-outdoor-light-architecture-people-street-1194699-pxhere.com.jpg?width=600&rev=1.1~]~]>>path:/Nieuws/Nieuwsbrief%20januari%202024||class="thumbnail"]] 1845 +))) 1846 + 1847 +(% class="col-xs-8 art_det" %) 1848 +((( 1849 +(% class="text-left" %) 1850 +[[Nieuwsbrief januari 2024 1851 +(% class="date_info" %) Feb 5, 2024,>>path:/Nieuws/Nieuwsbrief%20januari%202024||class="thumbnail"]] 1852 + 1853 +(% class="row post-details" %) 1854 +((( 1855 +(% class="col-xs-8 detail" %) 1856 +((( 1857 +(% class="post-likes btn btn-default disabled badge" title="Number of likes on this page: 3" %) 1858 +((( 1859 +[[(% class="like-number" %)3>>path:/Nieuws/Nieuwsbrief%20januari%202024||class="thumbnail"]] 1860 +))) 1861 +))) 1862 + 1863 +(% class="col-xs-4 com_det" %) 1864 +((( 1865 +(% class="text-right" %) 1866 +[[(0)>>path:/Nieuws/Nieuwsbrief%20januari%202024||class="thumbnail"]] 1867 +))) 1868 +))) 1869 +))) 1870 +))) 1871 +))) 1872 + 1873 +(% class="macro" data-macro="startmacro:blogPostLayoutCards|-|reference=~"Nieuws.Audit BMI en BORG~" params=~"displayTitle=true|useSummary=true|list-limit=10|list-layout=cards|list-blog=Nieuws.WebHome|list-paginated=no|list-results-count=10|postIndex=0|previousPostReference=|postIndex=1|previousPostReference=Nieuws.Nieuwsbrief januari 2024~"" %) 1874 +((( 1875 +(% class="macro-placeholder hidden" %) 1876 +((( 1877 +macro:blogPostLayoutCards 1878 +))) 1879 + 1880 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogCode~"" %) 1881 +((( 1882 +(% class="macro-placeholder hidden" %) 1883 +((( 1884 +macro:include 1885 +))) 1886 + 1887 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogParameters~"" %) 1888 +((( 1889 +(% class="macro-placeholder hidden" %) 1890 +((( 1891 +macro:include 1892 +))) 1893 + 1894 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|## Blog 1895 +#set($blogClassname = 'Blog.BlogClass') 1896 +#set($blogTemplate = 'Blog.BlogTemplate') 1897 +#set($blogSheet = 'Blog.BlogSheet') 1898 +## Blog entries 1899 +#set($blogPostClassname = 'Blog.BlogPostClass') 1900 +#set($blogPostTemplate = 'Blog.BlogPostTemplate') 1901 +#set($blogPostSheet = 'Blog.BlogPostSheet') 1902 +#set($blogPostObjectNumber = $xwiki.getDocument($blogPostTemplate).getObject($blogPostClassname).number) 1903 +#set($oldArticleClassname = 'XWiki.ArticleClass') 1904 +## Categories 1905 +#set($blogCategoryClassname = 'Blog.CategoryClass') 1906 +#set($blogCategoryTemplate = 'Blog.CategoryTemplate') 1907 +#set($blogCategorySheet = 'Blog.CategorySheet') 1908 +#set($blogCategoriesSheet = 'Blog.CategoriesSheet') 1909 +#set($oldBlogCategoryClassname = 'Blog.Categories') 1910 +#set($defaultCategoryParent = 'Blog.Categories') 1911 +## Style 1912 +#set($blogStyleDocumentName = 'Blog.BlogStyle') 1913 +#set($blogStyleDocument = $xwiki.getDocument($blogStyleDocumentName)) 1914 +## Clientside scripts 1915 +#set($blogScriptsDocumentName = 'Blog.BlogScripts') 1916 +#set($blogPublisher = $xwiki.getDocument('Blog.Publisher')) 1917 +## Misc 1918 +#set($thisURL = $doc.getURL($xcontext.action, $request.queryString)) 1919 +#set($isBlogPost = $doc.getObject($blogPostClassname)) 1920 +#set($defaultBlogSpace = 'Blog') 1921 +#set($isCategoryPage = $doc.getObject($blogCategoryClassname)) 1922 +#set($isCategoriesHomePage = $doc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 1923 +## 1924 +## 1925 +## 1926 +#** 1927 + * Displays an image, taken from the blog style document. 1928 + * 1929 + * @param $imgName The name of the icon from icons set to use. 1930 + *# 1931 +#macro(toolImage $imgName)(% class=~"icon-manage~"~%)$services.icon.render($imgName)(%~%)#end" %) 1932 +((( 1933 +(% class="macro-placeholder hidden" %) 1934 +((( 1935 +macro:velocity 1936 +))) 1937 +))) 1938 +))) 1939 + 1940 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|## 1941 +## 1942 +## 1943 +## Import the blog skin and javascripts. 1944 +$!xwiki.ssx.use($blogStyleDocumentName)## 1945 +$!xwiki.jsx.use($blogScriptsDocumentName)## 1946 +## 1947 +## import the hierarchy for the path the blog post footer (in displayEntryBlogLocation) 1948 +#template('hierarchy_macros.vm')## 1949 +## 1950 +## 1951 +#** 1952 + * Prints a blog. This is the main macro used in the BlogSheet. 1953 + * 1954 + * @param blogDoc the XDocument holding the blog definition object. 1955 + *### 1956 +#macro(printBlog $blogDoc) 1957 + \{\{include reference='Blog.CreatePost'/}} 1958 + 1959 + ## Use the blogPostList macro to display the blogposts 1960 + ## The blogPostList is used only in case of 'all' or 'paged' blog display type because the blogPostList macro 1961 + ## do not support FTM the monthly and weekly blog display types 1962 + #getBlogDisplayType($blogDoc $displayType) 1963 + #if ($displayType == 'weekly' || $displayType == 'monthly') 1964 + #getBlogEntries($blogDoc $entries) 1965 + #displayBlog($entries 'index' true true) 1966 + #displayNavigationLinks($blogDoc) 1967 + #else 1968 + #getBlogDisplayType($blogDoc $displayType) 1969 + #set ($paginated = 'no') 1970 + #if ($displayType == 'paginated') 1971 + #set ($paginated = 'yes') 1972 + #end 1973 + #getBlogPostsLayout($blogDoc $postsLayout) 1974 + (% class=~"hfeed index~" ~%)(((\{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$paginated~" layout=~"$!postsLayout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" /}}))) 1975 + #end 1976 +#end 1977 +## 1978 +## 1979 +## 1980 +#** 1981 + * Shows blog information. In view mode, the description is printed. In edit mode, allows changing blog settings: title, 1982 + * description, blog type (global or in-space), index display type (fixed size pagination, weekly index, monthly index, 1983 + * all entries). 1984 + * 1985 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 1986 + *### 1987 +#macro(showBlogInfo $blogDoc) 1988 + #if($blogDoc.getObject($blogClassname)) 1989 + ## Keep testing for inline action for backward compatibility with older blogs. 1990 + #if($xcontext.action == 'edit' || $xcontext.action == 'inline') 1991 + #macro(displayProperty $blogDoc $propname) 1992 + ; #displayPropName($xwiki.getClass($blogClassname).get($propname)): 1993 + : $blogDoc.display($propname) 1994 + #end 1995 + #displayProperty($blogDoc 'title') 1996 + #displayProperty($blogDoc 'description') 1997 + #displayProperty($blogDoc 'displayType') 1998 + #displayProperty($blogDoc 'itemsPerPage') 1999 + #displayProperty($blogDoc 'postsLayout') 2000 + #displayProperty($blogDoc 'postsLayoutParameters') 2001 + #else 2002 + $blogDoc.display('description') 2003 + #end 2004 + #elseif($doc.fullName == $blogSheet) 2005 += $services.localization.render('blog.code.blogsheet') = 2006 + \{\{translation key='blog.code.sheetexplanation'/}} 2007 + #else 2008 + \{\{warning}}\{\{translation key='blog.code.notblog'/}}\{\{/warning}} 2009 + #end 2010 +#end 2011 +## 2012 +## 2013 +## 2014 +#** 2015 + * Retrieve the blog document, which usually is either <tt><Space>.WebHome</tt> for whole-spaces blogs, or 2016 + * <tt><Space>.Blog</tt> for in-space blogs. If none of these documents contains a blog object, then the first 2017 + * (alphabetically) document in the target space that contains one is returned. Finally, if no document in the current 2018 + * space contains a blog object, then <tt>Blog.WebHome</tt> is returned as the default blog. 2019 + * 2020 + * @param space A <tt>String</tt>, the name of the space where to search. 2021 + * @param blogDoc The resulting XDocument. 2022 + *### 2023 +#macro(getBlogDocument $space $blogDoc) 2024 + #set ($result = $NULL) 2025 + ## Check if the current space has a WebPreferences page that contains a Blog.EnablePanelsConfigurationClass object. 2026 + ## It is possible to display a blog's panels info in a page that is not inside a blog and in that case we need to 2027 + ## identify the right blog based on a configuration object in a WebPreferences page. 2028 + #set ($spaceReference = $services.model.resolveSpace($space)) 2029 + #set ($preferencesDocRef = $services.model.createDocumentReference('WebPreferences', $spaceReference)) 2030 + #set ($preferencesDoc = $xwiki.getDocument($preferencesDocRef)) 2031 + #set ($preferencesObj = $preferencesDoc.getObject('Blog.EnablePanelsConfigurationClass')) 2032 + #if ($preferencesObj) 2033 + #set ($result = $xwiki.getDocument( $preferencesObj.getValue('blog'))) 2034 + #end 2035 + ## If no special case occurs (like the previous one), identify a blog by checking the current location from its hierarchy. 2036 + #if (~"$!result~" == '') 2037 + ## First, try the Space.WebHome, for a whole-space blog 2038 + #set($result = $xwiki.getDocument(~"$\{space}.WebHome~")) 2039 + #if(!$result.getObject($blogClassname)) 2040 + ## Second, try the Space.Blog document 2041 + #set($result = $xwiki.getDocument(~"$\{space}.Blog~")) 2042 + #if(!$result.getObject($blogClassname)) 2043 + ## Third, try searching for a blog document in the current space 2044 + ## Prevent the query fail when the space contains dots '.' 2045 + #set($blogDocs = $services.query.hql(~", BaseObject obj where doc.space = :space and obj.name = doc.fullName and obj.className = '$blogClassname' order by doc.name~").setLimit(1).setOffset(0).bindValue('space', $space).execute()) 2046 + #if($blogDocs.size() > 0) 2047 + #set($result = $xwiki.getDocument($blogDocs.get(0))) 2048 + #else 2049 + ## Fourth, try searching for a blog document that have the its 'postsLocation' set to the current space 2050 + #set($blogDocs = $services.query.hql(~", BaseObject obj, StringProperty as postsLocationProp where obj.name = doc.fullName and obj.className = '$blogClassname' and obj.id=postsLocationProp.id.id and postsLocationProp.id.name='postsLocation' and postsLocationProp.value=:postsLocation order by doc.name~").setLimit(1).setOffset(0).bindValue('postsLocation', $space).execute()) 2051 + #if($blogDocs.size() > 0) 2052 + #set($result = $xwiki.getDocument($blogDocs.get(0))) 2053 + #else 2054 + ## Last, fallback to Blog.WebHome, the default blog 2055 + #set($result = $xwiki.getDocument('Blog.WebHome')) 2056 + #end 2057 + #end 2058 + #end 2059 + #end 2060 + #end 2061 + #set ($blogDoc = $NULL) 2062 + #setVariable (~"$blogDoc~" $result) 2063 +#end 2064 +## 2065 +## 2066 +## 2067 +#** 2068 + * Retrieve the blog title. 2069 + * 2070 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass<tt> object with the <tt>title</tt> property set. 2071 + * @param title The resulting title. 2072 + *### 2073 +#macro(getBlogTitle $blogDoc $title) 2074 + ## Titles can contain velocity code (most commonly translations), so we should evaluate them. 2075 + #set ($title = $NULL) 2076 + #setVariable (~"$title~" $!blogDoc.displayTitle) 2077 +#end 2078 +## 2079 +## 2080 +## 2081 +#** 2082 + * Retrieve the blog description. 2083 + * 2084 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>description</tt> 2085 + * property set. 2086 + * @param description The resulting description. 2087 + *### 2088 +#macro(getBlogDescription $blogDoc $description) 2089 + #getBlogProperty($blogDoc 'description' '' $result) 2090 + #set ($description = $NULL) 2091 + #setVariable (~"$description~" $result) 2092 +#end 2093 +## 2094 +## 2095 +## 2096 +#** 2097 + * Retrieves a list of entries to be displayed. The entries are either part of the blog's space, or have the blog 2098 + * document set as a parent. The number and range of entries returned (from all those belonging to this blog) depends on 2099 + * the blog display type: paginated (fixed number of entries), weekly (all entries in a week), monthly (all entries in a 2100 + * month), or all. 2101 + * 2102 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 2103 + * @param entries The resulting list of entries to display, a list of XDocument names. 2104 + *### 2105 +#macro(getBlogEntries $blogDoc $entries) 2106 + #if (!$entries) 2107 + #setVariable (~"$entries~" []) 2108 + #end 2109 + #getAllBlogPostsQuery($query) 2110 + #isDefaultBlog($blogDoc $isDefault) 2111 + #set($queryParams = \{}) 2112 + #if ($isDefault) 2113 + #getCategoryAllBlogPostsQuery($query) 2114 + #set($query = ~"$\{query} and (doc.creator = :creator or (isPublished.value = 1 and hidden.value = 0)) and (doc.space = :space or catList like :catList escape '!')~") 2115 + #set($discard = $queryParams.put('creator', $xcontext.user)) 2116 + #set($discard = $queryParams.put('space', $blogDoc.space)) 2117 + #set($sanitizedBlogSpace = $blogDoc.space.replaceAll('([%_!])', '!$1').concat('.%')) 2118 + #set($discard = $queryParams.put('catList', $sanitizedBlogSpace)) 2119 + #else 2120 + #set($query = ~"$\{query} and (doc.space = :space or doc.parent = :parent)~") 2121 + #getBlogPostsLocation($blogDoc.space $blogPostsLocation) 2122 + #set($discard = $queryParams.put('space', $blogPostsLocation)) 2123 + #set($discard = $queryParams.put('parent', $blogDoc.space)) 2124 + #end 2125 + #getBlogDisplayType($blogDoc $displayType) 2126 + #if($displayType == 'weekly') 2127 + #getWeeklyBlogEntries($blogDoc $query $entries $queryParams) 2128 + #elseif($displayType == 'monthly') 2129 + #getMonthlyBlogEntries($blogDoc $query $entries $queryParams) 2130 + #elseif($displayType == 'all') 2131 + #getAllBlogEntries($blogDoc $query $entries $queryParams) 2132 + #else 2133 + #getPagedBlogEntries($blogDoc $query $entries $queryParams) 2134 + #end 2135 +#end 2136 +## 2137 +## 2138 +## 2139 +#** 2140 + * Retrieves a list of entries to be displayed. The entries are taken from a ~"page~" of the blog, a sequence of documents 2141 + * defined by the request parameters <tt>ipp</tt> (items per page) and <tt>page</tt> (the current page). Initially the 2142 + * first page is displayed, with the number of entries defined in the blog object in the <tt>itemsPerPage</tt> property 2143 + * (10 if not defined). 2144 + * 2145 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 2146 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 2147 + * refined to restrict to a given space, or to a given search criteria, etc. 2148 + * @param entries The resulting list of entries to display, a list of XDocument names. 2149 + * @param queryParams The parameters to bind with the query. 2150 + *### 2151 +#macro(getPagedBlogEntries $blogDoc $query $entries $queryParams) 2152 + #if (!$entries) 2153 + #setVariable (~"$entries~" []) 2154 + #end 2155 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 2156 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~")) 2157 + #bindQueryParameters($countQueryObj $queryParams) 2158 + #bindQueryParameters($queryObj $queryParams) 2159 + #set($totalEntries = $countQueryObj.count()) 2160 + #getBlogProperty($blogDoc 'itemsPerPage' '10' $defaultItemsPerPage) 2161 + #set($defaultItemsPerPage = $numbertool.toNumber($defaultItemsPerPage).intValue()) 2162 + ## This macro is defined in the default macros.vm library. It also sets $itemsPerPage and $startAt. 2163 + #preparePagedViewParams($totalEntries $defaultItemsPerPage) 2164 + #set($discard = $entries.addAll($queryObj.setLimit($itemsPerPage).setOffset($startAt).addFilter(~"unique~").execute())) 2165 +#end 2166 +## 2167 +## 2168 +## 2169 +#** 2170 + * Retrieves a list of entries to be displayed. The entries are taken from a week of the blog. The target week is 2171 + * defined by the request parameters <tt>week</tt> (the week number in the year, from 1 to 52) and <tt>year</tt> (4 2172 + * digit year). Initially the current week is displayed. 2173 + * 2174 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 2175 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 2176 + * refined to restrict to a given space, or to a given search criteria, etc. 2177 + * @param entries The resulting list of entries to display, a list of XDocument names. 2178 + * @param queryParams The parameters to bind with the query. 2179 + *### 2180 +#macro(getWeeklyBlogEntries $blogDoc $query $entries $queryParams) 2181 + #if (!$entries) 2182 + #setVariable (~"$entries~" []) 2183 + #end 2184 + #getRequestedWeek($weekDate) 2185 + #set($dateFormatter = $xwiki.jodatime.getDateTimeFormatterForPattern('yyyy-MM-dd')) 2186 + #set($minDay = $dateFormatter.print($weekDate.toMutableDateTime().weekOfWeekyear().roundFloor())) 2187 + #set($maxDay = $dateFormatter.print($weekDate.toMutableDateTime().weekOfWeekyear().roundCeiling())) 2188 + #set($query = ~"$\{query} and publishDate.value >= '$minDay' and publishDate.value < '$maxDay'~") 2189 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 2190 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 2191 + #bindQueryParameters($countQueryObj $queryParams) 2192 + #bindQueryParameters($queryObj $queryParams) 2193 + #set($totalEntries = $countQueryObj.count()) 2194 + #set($discard = $entries.addAll($queryObj.execute())) 2195 +#end 2196 +## 2197 +## 2198 +## 2199 +#** 2200 + * Retrieves a list of entries to be displayed. The entries are taken from a month of the blog. The target month is 2201 + * defined by the request parameters <tt>month</tt> (the month number, from 1 to 12) and <tt>year</tt> (4 2202 + * digit year). Initially the current month is displayed. 2203 + * 2204 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 2205 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 2206 + * refined to restrict to a given space, or to a given search criteria, etc. 2207 + * @param entries The resulting list of entries to display, a list of XDocument names. 2208 + * @param queryParams The parameters to bind with the query. 2209 + *### 2210 +#macro(getMonthlyBlogEntries $blogDoc $query $entries) 2211 + #if (!$entries) 2212 + #setVariable (~"$entries~" []) 2213 + #end 2214 + #getRequestedMonth($monthDate) 2215 + #set($dateFormatter = $xwiki.jodatime.getDateTimeFormatterForPattern('yyyy-MM-dd')) 2216 + #set($minDay = $dateFormatter.print($monthDate.toMutableDateTime().monthOfYear().roundFloor())) 2217 + #set($maxDay = $dateFormatter.print($monthDate.toMutableDateTime().monthOfYear().roundCeiling())) 2218 + #set($query = ~"$\{query} and publishDate.value >= '$minDay' and publishDate.value < '$maxDay'~") 2219 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 2220 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 2221 + #bindQueryParameters($countQueryObj $queryParams) 2222 + #bindQueryParameters($queryObj $queryParams) 2223 + #set($totalEntries = $countQueryObj.count()) 2224 + #set($discard = $entries.addAll($queryObj.execute())) 2225 +#end 2226 +## 2227 +## 2228 +## 2229 +#** 2230 + * Retrieves a list of entries to be displayed. All entries belonging to the current blog are returned. 2231 + * 2232 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 2233 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 2234 + * refined to restrict to a given space, or to a given search criteria, etc. 2235 + * @param entries The resulting list of entries to display, a list of XDocument names. 2236 + * @param queryParams The parameters to bind with the query. 2237 + *### 2238 +#macro(getAllBlogEntries $blogDoc $query $entries $queryParams) 2239 + #if (!$entries) 2240 + #setVariable (~"$entries~" []) 2241 + #end 2242 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 2243 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 2244 + #bindQueryParameters($countQueryObj $queryParams) 2245 + #bindQueryParameters($queryObj $queryParams) 2246 + #set($totalEntries = $countQueryObj.count()) 2247 + #set($discard = $entries.addAll($queryObj.execute())) 2248 +#end 2249 +## 2250 +## 2251 +## 2252 +#** 2253 + * Retrieves a list of entries to be displayed. Only (and all) unpublished entries are returned. 2254 + * 2255 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 2256 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 2257 + * refined to restrict to a given space, or to a given search criteria, etc. 2258 + * @param queryParams The parameters to bind with the query. 2259 + * @param entries The resulting list of entries to display, a list of XDocument names. 2260 + *### 2261 +#macro(getUnpublishedBlogEntries $blogDoc $query $queryParams $entries) 2262 + #if (!$entries) 2263 + #setVariable (~"$entries~" []) 2264 + #end 2265 + #set($query = ~"$\{query} and isPublished.value = 0~") 2266 + #set ($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 2267 + #set ($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 2268 + #bindQueryParameters($countQueryObj $queryParams) 2269 + #bindQueryParameters($queryObj $queryParams) 2270 + #set($totalEntries = $countQueryObj.count()) 2271 + #set($discard = $entries.addAll($queryObj.execute())) 2272 +#end 2273 +## 2274 +## 2275 +## 2276 +#** 2277 + * Retrieves a list of entries to be displayed. The entries are taken from all the wiki, and not from a specific blog. 2278 + * 2279 + * @param entries The resulting list of entries to display, a list of XDocument names. 2280 + *### 2281 +#macro(getGlobalBlogEntries $entries) 2282 + #if (!$entries) 2283 + #setVariable (~"$entries~" []) 2284 + #end 2285 + #getAllBlogPostsQuery($query) 2286 + #set($totalEntries = $services.query.hql($query).count()) 2287 + #set($defaultItemsPerPage = 20) 2288 + ## This macro is defined in the default macros.vm library. It also sets $itemsPerPage and $startAt. 2289 + #preparePagedViewParams($totalEntries $defaultItemsPerPage) 2290 + #set($discard = $entries.addAll($services.query.hql(~"$\{query} order by publishDate.value desc~").setLimit($itemsPerPage).setOffset($startAt).execute())) 2291 +#end 2292 +#** 2293 + * Return the base query for selecting blog entries. It filters only visible entries, but does not bind to a specific 2294 + * blog, nor specify a range or an ordering criteria. 2295 + * 2296 + * This macro is a duplicate of #getAllBlogPostsQuery($query). Keep it in case it's used outside, in custom code. 2297 + * 2298 + * @param query The basic query for selecting blog entries. 2299 + *# 2300 +#macro(getBlogEntriesBaseQuery $query) 2301 + #getAllBlogPostsQuery($query) 2302 +#end 2303 +#** 2304 + * Return the Query for selecting the all wiki blog posts without filtering 2305 + * 2306 + * @param query The basic query for selecting blog entries. 2307 + *# 2308 +#macro(getAllBlogPostsQuery $query) 2309 + #set ($query = $NULL) 2310 + #setVariable(~"$query~" ~", BaseObject as obj, IntegerProperty isPublished, 2311 + IntegerProperty hidden, DateProperty publishDate 2312 + where doc.fullName <> '$blogPostTemplate' and 2313 + obj.name=doc.fullName and obj.className='$blogPostClassname' and 2314 + isPublished.id.id = obj.id and isPublished.id.name = 'published' and 2315 + hidden.id.id = obj.id and hidden.id.name='hidden' and 2316 + publishDate.id.id = obj.id and publishDate.id.name='publishDate' and 2317 + (doc.creator = '$escapetool.sql($xcontext.user)' or (isPublished.value = 1 and hidden.value = 0))~") 2318 +#end 2319 +## 2320 +## 2321 +## 2322 +###** 2323 + * Return the Query for selecting the all wiki blog posts with categories filtering 2324 + * 2325 + * @param query The basic query for selecting blog entries. 2326 + *### 2327 +#macro(getCategoryAllBlogPostsQuery $query) 2328 + #set ($query = $NULL) 2329 + #getAllBlogPostsQuery($baseQuery) 2330 + #set ($baseQuery = $baseQuery.replace('DateProperty publishDate', 'DateProperty publishDate, DBStringListProperty as category left join category.list catList')) 2331 + #setVariable(~"$query~" ~"$\{baseQuery} and obj.id=category.id.id and category.id.name='category'~") 2332 +#end 2333 +## 2334 +## 2335 +## 2336 +#** 2337 + * Determines how is the blog index split into pages: paginated (fixed number of entries), weekly (all entries in a 2338 + * week), monthly (all entries in a month), or all. 2339 + * 2340 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>displayType</tt> 2341 + * property set. 2342 + * @param displayType The resulting string. If the blog object does not define anything, it is considered paginated. 2343 + *### 2344 +#macro(getBlogDisplayType $blogDoc $displayType) 2345 + #getBlogProperty($blogDoc 'displayType' 'paginated' $result) 2346 + #set ($displayType = $NULL) 2347 + #setVariable (~"$displayType~" $result) 2348 +#end 2349 +## 2350 +## 2351 +## 2352 +#** 2353 + * Displays a list of entries. 2354 + * 2355 + * @param entries The entries to display, a list of XDocument names. 2356 + * @param displaying What exactly is displayed: blog index, a single blog entry, a blog category, search results, 2357 + * unpublished entries, etc. This will be used as the classname(s) for the container div (hfeed). Currently 2358 + * used values: index, single, category, search, unpublished, hidden. 2359 + * @param onlyExtract If <tt>true</tt>, only display the extract of articles where available, otherwise display the full content. 2360 + * @param shouldDisplayTitles If <tt>true</tt>, display the blog title (blog posts shouldn't display the title when they're 2361 + * displayed alone on their page since it's the page title which is used in this case) 2362 + *### 2363 +#macro(displayBlog $entries $displaying $onlyExtract $shouldDisplayTitles) 2364 + #set($blogDay = '') 2365 + (% class=~"hfeed $!\{displaying}~" ~%)((( 2366 + (% class=~"blogDay~" ~%)((( 2367 + #foreach ($entryDoc in $xwiki.wrapDocs($entries)) 2368 + #getEntryObject($entryDoc $entryObj) 2369 + ## Although all entries should have one of the two objects, better check to be sure. 2370 + #if(~"$!\{entryObj}~" != '') 2371 + #getEntryDate($entryDoc $entryObj $entryDate) 2372 + ## Display a ~"calendar sheet~" for each day. All entries posted on the same day share one such sheet. 2373 + #set($entryDateStr = $xwiki.formatDate($entryDate, 'yyyyMMMMdd')) 2374 + #if($blogDay != $entryDateStr) 2375 + #if($blogDay != '') 2376 + ))) 2377 + (% class=~"blogDay~" ~%)((( 2378 + #end 2379 + #displayBlogDate($entryDate) 2380 + #set ($blogDay = $entryDateStr) 2381 + #end 2382 + ## Finally, display the entry. 2383 + #displayEntry($entryDoc $entryObj $onlyExtract $shouldDisplayTitles true) 2384 + #end 2385 + #end 2386 + )))## blogDay 2387 + )))## hfeed 2388 +#end 2389 +## 2390 +## 2391 +## 2392 +#** 2393 + * Get the entry object, either a new BlogPost or an old Article. 2394 + * 2395 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 2396 + * @param entryObj The resulting xobject of the blog post. 2397 + *### 2398 +#macro(getEntryObject $entryDoc $__entryObj) 2399 + #set($result = $entryDoc.getObject(~"$\{blogPostClassname}~")) 2400 + #if(!$result) 2401 + #set($result = $entryDoc.getObject(~"$\{oldArticleClassname}~")) 2402 + #end 2403 + ## NOTE: The reason we put an underscore in front of the variable name is to prevent the following line from 2404 + ## overwriting the $entryObj variable that may be defined before this macro is called. Of course, $__entryObj may be 2405 + ## overwritten in this case but it's less likely to have such a variable defined before. 2406 + #set ($__entryObj = $NULL) 2407 + #setVariable (~"$__entryObj~" $result) 2408 +#end 2409 +## 2410 +## 2411 +## 2412 +#** 2413 + * Gets the date associated with a blog entry. This is the publication date. For unpublished entries, initially this is 2414 + * the document creation date, but can be edited by the user. 2415 + * 2416 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 2417 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 2418 + * @param result The resulting date, an instance of <tt>java.util.Date</tt>. 2419 + *### 2420 +#macro(getEntryDate $entryDoc $entryObj $result) 2421 + #set ($result = $NULL) 2422 + #setVariable (~"$result~" $entryObj.getProperty('publishDate').value) 2423 +#end 2424 +## 2425 +## 2426 +## 2427 +#** 2428 + * Displays a date, nicely formatted as a calendar page. 2429 + * 2430 + * @param date The date to display, an instance of <tt>java.util.Date</tt>. 2431 + *### 2432 +#macro(displayBlogDate $date) 2433 + #set($year = $xwiki.formatDate($date, 'yyyy')) 2434 + ## 3 letter month name, like Jan, Dec. 2435 + #set($month = $xwiki.formatDate($date, 'MMM')) 2436 + ## Uncomment to get a full length month name, like January, December. 2437 + ## TODO: this could be defined somewhere in the blog style. 2438 + ## #set($month = $xwiki.formatDate($date, 'MMMM')) 2439 + #set($day = $xwiki.formatDate($date, 'dd')) 2440 + (% class=~"blogdate~" ~%) 2441 + == (% class=~"month~" ~%)$month(%~%) (% class=~"day~" ~%)$day(%~%) (% class=~"year~" ~%)$year(%~%) == 2442 +#end 2443 +## 2444 +## 2445 +## 2446 +#** 2447 + * Displays a blog article: management tools, header, content, footer. 2448 + * 2449 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 2450 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 2451 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 2452 + * @param shouldDisplayTitle If <tt>true</tt>, display the blog title (blog posts shouldn't display the title 2453 + * when they're displayed alone on their page since it's the page title which is used in this case) 2454 + * @param shouldDisplayActions If <tt>true</tt>, display the blog post actions buttons 2455 + *### 2456 +#macro(displayEntry $entryDoc $entryObj $onlyExtract $shouldDisplayTitle $shouldDisplayActions) 2457 + ## Only articles with an explicit hidden setting or an explicit unpublished setting are hidden 2458 + #isPublished($entryObj $isPublished) 2459 + #isHidden($entryObj $isHidden) 2460 + #if($doc.fullName == $entryDoc.fullName) 2461 + (% class=~"hentry single-article~" ~%)((( 2462 + #else 2463 + (% class=~"hentry#if(!$isPublished) unpublished-article#elseif($isHidden) hidden-article#end~" ~%)((( 2464 + #end 2465 + #if ($shouldDisplayActions) 2466 + #displayEntryTools($entryDoc $entryObj) 2467 + #end 2468 + #if($shouldDisplayTitle) 2469 + #displayEntryTitle($entryDoc $entryObj) 2470 + #end 2471 + #if($doc.fullName == $entryDoc.fullName) 2472 + #if(!$isPublished) 2473 + \{\{warning}}\{\{translation key='blog.code.published'/}}\{\{/warning}} 2474 + #elseif($isHidden) 2475 + \{\{warning}}\{\{translation key='blog.code.hidden'/}}\{\{/warning}} 2476 + #end 2477 + #end 2478 + #displayEntryContent($entryDoc $entryObj $onlyExtract) 2479 + #displayEntryFooter($entryDoc $entryObj) 2480 + )))## hentry 2481 +#end 2482 +## 2483 +## 2484 +## 2485 +#** 2486 + * Checks if the provided blog is published or not. 2487 + * 2488 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 2489 + * @param isPublished The resulting boolean, true if the entry is considered published. 2490 + *### 2491 +#macro(isPublished $entryObj $isPublished) 2492 + #set ($isPublished = $NULL) 2493 + ## This should work for both old articles, which don't have the 'published' property at all, and 2494 + ## are considered published by default, and new entries, that should have 1 if published. 2495 + #if (~"$!\{entryObj.getProperty('published').value}~" != '0') 2496 + #setVariable (~"$isPublished~" true) 2497 + #else 2498 + #setVariable (~"$isPublished~" false) 2499 + #end 2500 +#end 2501 +## 2502 +## 2503 +## 2504 +#** 2505 + * Checks if the provided blog is hidden or not. 2506 + * 2507 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass<tt> xclass. 2508 + * @param isHiddel The resulting boolean, true if the entry is considered hidden. 2509 + *### 2510 +#macro(isHidden $entryObj $isHidden) 2511 + #set ($isHidden = $NULL) 2512 + ## This should work for both old articles, which don't have the 'hidden' property at all, and 2513 + ## are considered visible by default, and new entries, that should have 1 if hidden. 2514 + #if (~"$!\{entryObj.getProperty('hidden').value}~" == '1') 2515 + #setVariable (~"$isHidden~" true) 2516 + #else 2517 + #setVariable (~"$isHidden~" false) 2518 + #end 2519 +#end 2520 +## 2521 +## 2522 +## 2523 +#** 2524 + * Displays several ~"tools~" for manipulating blog posts: hide/show, publish, edit. 2525 + * 2526 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 2527 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 2528 + *### 2529 +#macro(displayEntryTools $entryDoc $entryObj) 2530 + #if($xcontext.action == 'view') 2531 + (% class=~"blog-entry-toolbox~" ~%)((( 2532 + #displayPublishButton($entryDoc $entryObj) 2533 + #displayHideShowButton($entryDoc $entryObj) 2534 + #displayEditButton($entryDoc $entryObj) 2535 + #displayDeleteButton($entryDoc $entryObj) 2536 + ))) 2537 + #end 2538 +#end 2539 +## 2540 +## 2541 +## 2542 +#** 2543 + * Displays the publish button to the entry <strong>creator</strong>, if the article is not published yet. 2544 + * 2545 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 2546 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 2547 + * @todo AJAX calls. 2548 + *### 2549 +#macro(displayPublishButton $entryDoc $entryObj) 2550 + #isPublished($entryObj $isPublished) 2551 + #if(!$isPublished && $entryDoc.creator == $xcontext.user && $xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 2552 + [[#toolImage('world')>>path:$blogPublisher.getURL('view', ~"entryName=$\{escapetool.url($entryDoc.fullName)}&xredirect=$\{escapetool.url($thisURL)}&form_token=$!\{services.csrf.getToken()}~")||title=~"$services.localization.render('blog.code.notpublished')~"]]## 2553 + #end 2554 +#end 2555 +## 2556 +## 2557 +## 2558 +#** 2559 + * Displays the hide or show button to the entry <strong>creator</strong>, if the article is already published. 2560 + * 2561 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 2562 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 2563 + *### 2564 +#macro(displayHideShowButton $entryDoc $entryObj) 2565 + #isPublished($entryObj $isPublished) 2566 + #isHidden($entryObj $isHidden) 2567 + ## Only published articles can be hidden. Unpublished articles are considered already hidden. 2568 + #if($isPublished && $entryDoc.creator == $xcontext.user && $xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 2569 + #set ($queryString = \{ 2570 + 'xredirect' : $thisURL, 2571 + 'form_token' : $services.csrf.getToken() 2572 + }) 2573 + #if ($isHidden) 2574 + #set ($discard = $queryString.putAll(\{ 2575 + ~"$\{entryObj.getxWikiClass().getName()}_$\{entryObj.number}_hidden~" : 0, 2576 + 'comment' : $services.localization.render('blog.code.madevisible') 2577 + })) 2578 + #set ($lockURL = $entryDoc.getURL('save', $escapetool.url($queryString))) 2579 + [[#toolImage('unlock')>>path:$lockURL||class=~"blog-tool-show~" title=~"$services.localization.render('blog.code.makevisible')~"]]## 2580 + #else 2581 + #set ($discard = $queryString.putAll(\{ 2582 + ~"$\{entryObj.getxWikiClass().getName()}_$\{entryObj.number}_hidden~" : 1, 2583 + 'comment' : $services.localization.render('blog.code.hid') 2584 + })) 2585 + #set ($lockURL = $entryDoc.getURL('save', $escapetool.url($queryString))) 2586 + [[#toolImage('lock')>>path:$lockURL||class=~"blog-tool-hide~" title=~"$services.localization.render('blog.code.hide')~"]]## 2587 + #end 2588 + #end 2589 +#end 2590 +## 2591 +## 2592 +## 2593 +#** 2594 + * Displays the edit button to those that can edit the article. 2595 + * 2596 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 2597 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 2598 + *### 2599 +#macro(displayEditButton $entryDoc $entryObj) 2600 + #if($xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 2601 + ## Call getDefaultEditMode() for backward compatibility with older blog posts. 2602 + [[#toolImage('pencil')>>path:$entryDoc.getURL('edit')||title=~"$services.localization.render('blog.code.editpost')~"]]## 2603 + #end 2604 +#end 2605 +## 2606 +## 2607 +## 2608 +#** 2609 + * Displays the delete button to those that can edit the article. 2610 + * 2611 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 2612 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 2613 + * @todo AJAX calls. 2614 + *### 2615 +#macro(displayDeleteButton $entryDoc $entryObj) 2616 + #if($xwiki.hasAccessLevel('delete', $xcontext.user, $entryDoc.fullName)) 2617 + [[#toolImage('cross')>>path:$entryDoc.getURL('delete')||title=~"$services.localization.render('blog.code.deletepost')~"]]## 2618 + #end 2619 +#end 2620 +## 2621 +## 2622 +## 2623 +#** 2624 + * Displays the title of the entry. 2625 + * 2626 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 2627 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 2628 + *### 2629 +#macro(displayEntryTitle $entryDoc $entryObj) 2630 + #if($doc.fullName == $entryDoc.fullName) 2631 + (% class=~"entry-title~" ~%) 2632 + = $services.rendering.escape($entryDoc.display('title', 'view', $entryObj), $xwiki.getCurrentContentSyntaxId()) = 2633 + #else 2634 + (% class=~"entry-title~" ~%) 2635 + === [[$services.rendering.escape($entryDoc.display('title', 'view', $entryObj),$xwiki.getCurrentContentSyntaxId())>>doc:$services.rendering.escape($services.model.serialize($entryDoc.getDocumentReference(),'default'),$xwiki.getCurrentContentSyntaxId())]] === 2636 + #end 2637 +#end 2638 +## 2639 +## 2640 +## 2641 +#** 2642 + * Displays the body of the entry. 2643 + * 2644 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 2645 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 2646 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 2647 + *### 2648 +#macro(displayEntryContent $entryDoc $entryObj $onlyExtract) 2649 + (% class=~"#if($onlyExtract)entry-summary#\{else}entry-content#end~" ~%)((( 2650 + #getEntryContent($entryDoc $entryObj $onlyExtract $entryContent) 2651 + ## FIXME: This causes the blog's content to not be annotatable. See http://jira.xwiki.org/browse/XWIKI-6328 2652 + ## Should probably be replaced by a display macro call with a reference to the object property holding the post's content 2653 + \{\{html wiki=~"false~"}}$entryDoc.getRenderedContent($entryContent, $entryDoc.syntax.toIdString())\{\{/html}} 2654 + ))) ## entry-content 2655 + (% class=~"clearfloats~" ~%)((())) 2656 +#end 2657 +## 2658 +## 2659 +## 2660 +#** 2661 + * Extracts the body of the entry that should be displayed. If <tt>onlyExtract</tt> is <tt>true</tt>, display the content 2662 + * of the <tt>extract</tt> field (if not empty). 2663 + * 2664 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 2665 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 2666 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 2667 + * @param entryContent The resulting content. 2668 + * @param removeEllipsis If <tt>true</tt>, then don't display an ellipsis at the end of the content (only used when 2669 + * <tt>onlyExtract</tt> is <tt>true</tt>) 2670 + *### 2671 +#macro(getEntryContent $entryDoc $entryObj $onlyExtract $entryContent $removeEllipsis) 2672 + #if ($onlyExtract) 2673 + ## Note: We trim the summary so that if there's some white space it won't cause the summary to be used instead 2674 + ## of the content. 2675 + #set ($macro.result = $entryObj.getProperty('extract').value.trim()) 2676 + #end 2677 + #if(~"$!macro.result~" == '') 2678 + #set($macro.result = $entryObj.getProperty('content').value) 2679 +#* Disabled until the content can be cleanly cut. 2680 +* #if($onlyExtract && $result.length()>$maxchars) 2681 +* #set($i = $macro.result.lastIndexOf(~" ~", $maxchars)) 2682 +* #set($i = $i + 1) 2683 +* #set($macro.result = ~"$\{macro.result.substring(0,$i)} *[...>$\{entryDoc.fullName}]*~") 2684 +* #end 2685 +## *### 2686 + #elseif (!$removeEllipsis) 2687 + #if($entryDoc.syntax.toIdString() == 'xwiki/1.0') 2688 + #set($macro.result = ~"$\{macro.result} <a href='$\{entryDoc.getExternalURL()}' title='$services.localization.render('blog.code.readpost')'>...</a>~") 2689 + #elseif($entryDoc.syntax.toIdString() == 'xwiki/2.0' || $entryDoc.syntax.toIdString() == 'xwiki/2.1') 2690 + #set($macro.result = ~"$\{macro.result} [[...>>$\{services.rendering.escape($entryDoc.getFullName(), $entryDoc.getSyntax())}||title='$services.localization.render('blog.code.readpost')']]~") 2691 + #end 2692 + #end 2693 + #set ($entryContent = $NULL) 2694 + #setVariable (~"$entryContent~" $macro.result) 2695 +#end 2696 +## 2697 +## 2698 +## 2699 +#** 2700 + * Displays the footer of the entry. 2701 + * 2702 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 2703 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 2704 + *### 2705 +#macro(displayEntryFooter $entryDoc $entryObj) 2706 + (% class=~"entry-footer~" ~%)((( 2707 + #isPublished($entryObj $isPublished) 2708 + (% class='entry-author-label' ~%) 2709 + #if($isPublished) 2710 + \{\{translation key='blog.code.postedby'/}} ## 2711 + #else 2712 + \{\{translation key='blog.code.createdby'/}} ## 2713 + #end 2714 + \{\{html wiki=~"false~" clean=~"false~"}}<span class='author vcard'>#userfn($entryDoc.creator)</span>\{\{/html}} ## 2715 + #getEntryDate($entryDoc $entryObj $entryDate) 2716 + #listCategories($entryObj) #* 2717 + ## Since the publish date and update date are not set at the exact same time, there could be a small difference that 2718 + ## we assume cannot be more than 3 seconds. 2719 + *#(% class=~"separator~" ~%)·(%~%) [[\{\{translation key='blog.code.permalink'/}}>>$services.rendering.escape($entryDoc,'xwiki/2.0')||rel=~"bookmark~"]] ## 2720 + #if ($showcomments) 2721 + (% class=~"separator~" ~%)·(%~%) [[\{\{translation key='blog.code.comments'/}}>>$services.rendering.escape($entryDoc,'xwiki/2.0')||anchor=~"Comments~"]] (% class=~"itemCount~" ~%)($entryDoc.comments.size())(%~%) ## 2722 + #end ## 2723 + #if($entryDoc != $doc) ## 2724 + #displayEntryBlogLocation($entryDoc $entryObj) ## 2725 + #end 2726 + )))## entry-footer 2727 +#end 2728 +## 2729 +## 2730 +#** 2731 + * Display the blog for the entry (if it is not the currently displayed blog) 2732 + * 2733 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 2734 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 2735 + *### 2736 +#macro(displayEntryBlogLocation $entryDoc $entryObj) 2737 + #getBlogPostsLocation($entryDoc.getSpace() $blogPostsLocation) ## 2738 + #if(~"$!blogPostsLocation~" != ~"~") ## 2739 + #set($blogSpaceRef = $services.model.resolveSpace($blogPostsLocation)) ## 2740 + #set($blogDocRef = $services.model.createDocumentReference($services.model.getEntityReference('DOCUMENT', 'default').getName(), $blogSpaceRef)) ## 2741 + #if($doc.documentReference != $blogDocRef) ## 2742 + #set($blogDoc = $xwiki.getDocument($blogDocRef)) ## 2743 + #set($blogObj = $blogDoc.getObject('Blog.BlogClass')) ## 2744 + #if($blogObj) (% class=~"blogrefpath~" ~%)((( ## 2745 + $services.localization.render('blog.code.postedin') \{\{html clean=~"false~" wiki=~"false~"}} #hierarchy($blogDocRef, \{'selfIsActive' : true, 'local': true}) \{\{/html}} ## 2746 + )))(%~%)## 2747 + #end 2748 + #end 2749 + #end 2750 +#end 2751 +## 2752 +## 2753 +## 2754 +## 2755 +#** 2756 + * List the categories an entry belongs to. Used in the footer. The categories are instances of <tt>Blog.CategoryClass</tt>. 2757 + * 2758 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 2759 + *### 2760 +#macro(listCategories $entryObj) 2761 + #if($entryObj.getxWikiClass().getName() == $blogPostClassname) 2762 + #set($categories = $entryObj.getProperty('category').value) 2763 + #set($first = true) 2764 + #if($categories.size() > 0) 2765 + #foreach($category in $categories) 2766 + #set($categoryDoc = $!xwiki.getDocument($category)) 2767 + #if(!$categoryDoc.isNew() && $categoryDoc.getObject($\{blogCategoryClassname})) 2768 + #if($foreach.count == 1) 2769 + (% class='separator' ~%)·(%~%) $services.localization.render('blog.code.categories') ## 2770 + #else 2771 + , ## 2772 + #end## 2773 + [[$!\{categoryDoc.getObject($blogCategoryClassname).getValue('name')}>>$\{category}||rel='tag']]## 2774 + #end## 2775 + #end## 2776 + #end 2777 + #end 2778 +#end 2779 +## 2780 +## 2781 +## 2782 +#** 2783 + * Displays blog pagination links (older and newer entries). 2784 + * 2785 + * @param blogDoc the XDocument holding the blog definition object. 2786 + *### 2787 +#macro(displayNavigationLinks $blogDoc) 2788 + (% class=~"clearfloats~" ~%)((())) 2789 + #getBlogDisplayType($blogDoc $displayType) 2790 + #if($displayType == 'weekly') 2791 + (% class=~"pagingLinks~" ~%)((( 2792 + #getRequestedWeek($weekDate) 2793 + $weekDate.addWeeks(-1)## 2794 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.previousweek'/}}>>$doc.documentReference.name||queryString=~"year=$weekDate.weekyear&week=$weekDate.weekOfWeekyear~" class=~"button~"]]**(%~%) 2795 + #sep() 2796 + $weekDate.addWeeks(2)## 2 because we already subtracted 1 above 2797 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.nextweek'/}} »>>$doc.documentReference.name||queryString=~"year=$weekDate.weekyear&week=$weekDate.weekOfWeekyear~" class=~"button~"]]**(%~%) 2798 + ))) 2799 + #elseif($displayType == 'monthly') 2800 + (% class=~"pagingLinks~" ~%)((( 2801 + #getRequestedMonth($monthDate) 2802 + $monthDate.addMonths(-1)## 2803 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.previousmonth'/}}>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"year=$monthDate.year&month=$monthDate.monthOfYear~" class=~"button~"]]**(%~%) 2804 + #sep() 2805 + $monthDate.addMonths(2)## 2 because we already subtracted 1 above 2806 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.nextmonth'/}} »>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"year=$monthDate.year&month=$monthDate.monthOfYear~" class=~"button~"]]**(%~%) 2807 + ))) 2808 + #elseif($displayType == 'all') 2809 + #else 2810 + ## Paginated 2811 + #if(($totalPages > 1)) 2812 + #set($queryString = '') 2813 + #foreach($p in $request.getParameterNames()) 2814 + #if($p != 'page' && $p != 'ipp') 2815 + #foreach($v in $request.getParameterValues($p)) 2816 + #set($queryString = ~"$\{queryString}&$\{escapetool.url($p)}=$\{escapetool.url($v)}~") 2817 + #end 2818 + #end 2819 + #end 2820 + (% class=~"pagingLinks~" ~%)((( 2821 + #if ($currentPageNumber < $totalPages) 2822 + #set($currentPageNumber = $currentPageNumber + 1) 2823 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.olderposts'/}}>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"page=$\{currentPageNumber}&ipp=$\{itemsPerPage}$queryString~" class=~"button~"]]**(%~%) 2824 + #set($currentPageNumber = $currentPageNumber - 1) 2825 + #end 2826 + #if ($currentPageNumber > 1) 2827 + #if ($currentPageNumber < $totalPages) 2828 + #sep() 2829 + #end 2830 + #set($currentPageNumber = $currentPageNumber - 1) 2831 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.newerposts'/}} »>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"page=$\{currentPageNumber}&ipp=$\{itemsPerPage}$queryString~" class=~"button~"]]**(%~%) 2832 + #set($currentPageNumber = $currentPageNumber + 1) 2833 + #end 2834 + (% class=~"clear~" ~%)(%~%) 2835 + )))## pagingLinks 2836 + #end 2837 + #end 2838 +#end 2839 +## 2840 +## 2841 +## 2842 +#** 2843 + * Displays a message box with ~"publish~" icon. 2844 + * 2845 + * @param message A text message concerning blog article publishing 2846 + *### 2847 +#macro(publishMessageBox $message) 2848 +(% class=~"plainmessage publish-message~" ~%)((($services.icon.render('world') $message))) 2849 +#end 2850 +#** 2851 + * Displays a message box with ~"show/hide~" icon. 2852 + * 2853 + * @param message A text message concerning blog article hiding 2854 + *### 2855 +#macro(hideMessageBox $message) 2856 +(% class=~"plainmessage hide-message~" ~%)((($services.icon.render('unlock') $message))) 2857 +#end 2858 +## 2859 +## 2860 +## 2861 +#** 2862 + * Determine the requested week, for using in a weekly-indexed blog. The relevant request parameters are 2863 + * <tt>year</tt> and <tt>week</tt>. By default, the current week is used. 2864 + * 2865 + * @param monthDate The resulting week, a JODATime MutableDateTime. 2866 + *### 2867 +#macro(getRequestedWeek $weekDate) 2868 + #set ($weekDate = $NULL) 2869 + #setVariable (~"$weekDate~" $xwiki.jodatime.mutableDateTime) 2870 + #if(~"$!\{request.year}~" != '') 2871 + #set ($discard = $weekDate.setYear($numbertool.toNumber($request.year).intValue())) 2872 + #end 2873 + #if(~"$!\{request.week}~" != '') 2874 + #set ($discard = $weekDate.setWeekOfWeekyear($numbertool.toNumber($request.week).intValue())) 2875 + #end 2876 +#end 2877 +## 2878 +## 2879 +## 2880 +#** 2881 + * Determine the requested month, for using in a monthly-indexed blog. The relevant request parameters are 2882 + * <tt>year</tt> and <tt>month</tt>. By default, the current month is used. 2883 + * 2884 + * @param monthDate The resulting month, a JODATime MutableDateTime. 2885 + *### 2886 +#macro(getRequestedMonth $monthDate) 2887 + #set ($monthDate = $NULL) 2888 + #setVariable (~"$monthDate~" $xwiki.jodatime.mutableDateTime) 2889 + #if(~"$!\{request.year}~" != '') 2890 + #set ($discard = $monthDate.setYear($numbertool.toNumber($request.year).intValue())) 2891 + #end 2892 + #if(~"$!\{request.month}~" != '') 2893 + #set ($discard = $monthDate.setMonthOfYear($numbertool.toNumber($request.month).intValue())) 2894 + #end 2895 +#end 2896 +## 2897 +## 2898 +## 2899 +#** 2900 + * Retrieve a blog property (title, display type, etc). 2901 + * 2902 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 2903 + * @param propertyName The name of the property to be retrieved. One of the <tt>Blog.BlogClass</tt>'s properties. 2904 + * @param defaultValue The default value to use in case the blog object does not define one. 2905 + * @param propertyValue The resulting value. 2906 + *### 2907 +#macro(getBlogProperty $blogDoc $propertyName $defaultValue $propertyValue) 2908 + #set($result = ~"$!\{blogDoc.getObject($\{blogClassname}).getProperty($propertyName).value}~") 2909 + #if($result == '') 2910 + #set($result = $defaultValue) 2911 + #end 2912 + #set ($propertyValue = $NULL) 2913 + #setVariable (~"$propertyValue~" $result) 2914 +#end 2915 + 2916 +#** 2917 + * If an error occurs when executing an action, set a specific response status and display an error message. 2918 + * 2919 + * @param status The response status. 2920 + * @param text The user readable error to be displayed. Can be a translation key. 2921 + * @param parameters The parameters to use when decoding the translation key. 2922 + *### 2923 +#macro(blog__actionResponseError $status $text $parameters) 2924 + $response.setStatus($status) 2925 + #if($request.ajax) 2926 + $services.localization.render($text, $!parameters) 2927 + #else 2928 + \{\{error}}$services.localization.render($text, $!parameters)\{\{/error}} 2929 + #end 2930 +#end 2931 +## 2932 +## 2933 +## 2934 +#** 2935 + * Check if a blog is the Default blog (The one in the 'Blog' space). 2936 + * 2937 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 2938 + * @param isDefault The resulting boolean. 2939 + *### 2940 +#macro(isDefaultBlog $blogDoc $isDefault) 2941 + #set ($result = false) 2942 + #if ($blogDoc.space == 'Blog') 2943 + #set ($result = true) 2944 + #end 2945 + #setVariable(~"$isDefault~" $result) 2946 +#end 2947 +## 2948 +## 2949 +## 2950 +#** 2951 + * Retrieve the blog posts location (space). 2952 + * 2953 + * @param blogSpace The blog space. It should contain a document with a <tt>Blog.BlogClass</tt> object 2954 + * @param postsLocation The resulting location. 2955 + *### 2956 +#macro(getBlogPostsLocation $blogSpace $postsLocation) 2957 + #getBlogDocument($blogSpace $blogDoc) 2958 + #getBlogProperty($blogDoc 'postsLocation' $blogDoc.space $result) 2959 + #set ($postsLocation = $NULL) 2960 + #setVariable (~"$postsLocation~" $result) 2961 +#end 2962 +## 2963 +## 2964 +## 2965 +#** 2966 + * Retrieve the blog categories location (space). 2967 + * 2968 + * @param blogSpace The blog space. It should contain a document with a <tt>Blog.BlogClass</tt> object 2969 + * @param categoriesLocation The resulting location. 2970 + *### 2971 +#macro(getBlogCategoriesLocation $blogSpace $categoriesLocation) 2972 + #getBlogDocument($blogSpace $blogDoc) 2973 + #getBlogProperty($blogDoc 'categoriesLocation' 'Blog' $result) 2974 + #set ($postsLocation = $NULL) 2975 + #setVariable (~"$categoriesLocation~" $result) 2976 +#end 2977 +###** 2978 + * Return the Query for selecting the blog posts based on the context where the posts are displayed. 2979 + * for example there is 4 different panel contexts: 2980 + * aBlog.aPost or aBlog.WebHome 2981 + * aCategorySpace.aCategory 2982 + * aCategorySpace.WebHome 2983 + * Blog.aPost or Blog.WebHome 2984 + * 2985 + * @param query The query for selecting blog entries. 2986 + * @param queryParams The parameters to bind with the generated query. 2987 + * @param targetDoc The document in which the articles will be displayed. 2988 + *### 2989 +#macro(getAllBlogPostsQueryBasedOnDisplayContext $targetDoc $query $queryParams) 2990 + #set ($query = $NULL) 2991 + #set ($queryParams = $NULL) 2992 + #getCategoryAllBlogPostsQuery($resultQuery) 2993 + #set ($resultQueryParams = \{}) 2994 + #set($sanitizedSpace = $targetDoc.space.replaceAll('([%_!])', '!$1').concat('.%')) 2995 + #if ($targetDoc.space == $defaultBlogSpace && !$targetDoc.getObject($blogCategoryClassname)) 2996 + ## Get all posts that are in the default blog space('Blog') or in category 'Blog.%' 2997 + #set ($discard = $resultQueryParams.put('space', $targetDoc.space)) 2998 + #set ($discard = $resultQueryParams.put('catList', $sanitizedSpace)) 2999 + #set($resultQuery = ~"$\{resultQuery} and (doc.space = :space or catList like :catList escape '!')~") 3000 + #elseif($targetDoc.getObject($blogCategoryClassname)) 3001 + ## Get all posts that are in a category aCategorySpace.aCategory 3002 + #set ($discard = $resultQueryParams.put('catList', $targetDoc.fullName)) 3003 + #set($resultQuery = ~"$\{resultQuery} and (catList=:catList)~") 3004 + #elseif($targetDoc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 3005 + ## Get all posts that are in a category aCategorySpace.% 3006 + #set ($discard = $resultQueryParams.put('catList', $sanitizedSpace)) 3007 + ## Exclude incategorized posts 3008 + #set ($excludedCategory = ~"$\{targetDoc.space}.WebHome~") 3009 + #if ($targetDoc.space == $defaultBlogSpace) 3010 + #set ($excludedCategory = ~"Blog.Categories.WebHome~") 3011 + #end 3012 + #set ($discard = $resultQueryParams.put('excludedCategory', $excludedCategory)) 3013 + #set($resultQuery = ~"$\{resultQuery} and (catList<>:excludedCategory) and (catList like :catList escape '!')~") 3014 + #else 3015 + ## Get all posts in blog space aBlog 3016 + #getAllBlogPostsQuery($resultQuery) 3017 + #getBlogPostsLocation($targetDoc.space $postsLocation) 3018 + #set ($discard = $resultQueryParams.put('space', $postsLocation)) 3019 + #set($resultQuery = ~"$\{resultQuery} and doc.space = :space~") 3020 + #end 3021 + #setVariable(~"$query~" $resultQuery) 3022 + #setVariable(~"$queryParams~" $resultQueryParams) 3023 +#end 3024 +## 3025 +## 3026 +## 3027 +###** 3028 + * Display blog posts based on the context where the posts are displayed. 3029 + * for example there is 4 different panel contexts: 3030 + * aBlog.aPost or aBlog.WebHome 3031 + * aCategorySpace.aCategory 3032 + * aCategorySpace.WebHome 3033 + * Blog.aPost or Blog.WebHome 3034 + * 3035 + * @param targetDoc The document in which the articles will be displayed. 3036 + * @param postsVisiblity The visibility of the posts to display: recent, unpublished, ... 3037 + * @param layout Layout of the the posts to display 3038 + * @param layoutParams additional layout parameters, it is a string like 'param1=val1|...|paramN=valueN' 3039 + * @param limit the number of posts to display 3040 + *### 3041 +#macro(displayBlogPostsBasedOnContext $targetDoc $postsVisiblity $layout $layoutParams $limit) 3042 + #if ($postLayout == 'full') 3043 + #set ($macro.paginated = 'yes') 3044 + #end 3045 + #if ($targetDoc.space == $defaultBlogSpace && !$targetDoc.getObject($blogCategoryClassname)) 3046 + ## Display all posts that are in the default blog space('Blog') or in category 'Blog.%' 3047 + #getBlogPostsLayout($xwiki.getDocument(~"$\{defaultBlogSpace}.WebHome~") $postsLayout) 3048 + #if (~"$!layout~" == '') 3049 + #set ($layout = $postsLayout) 3050 + #end 3051 + #if ($postsVisiblity == 'recent') 3052 + \{\{blogpostlist blog=~"$\{defaultBlogSpace.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')}.WebHome~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 3053 + #elseif($postsVisiblity == 'unpublished') 3054 + \{\{blogpostlist blog=~"$\{defaultBlogSpace.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')}.WebHome~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 3055 + #end 3056 + #elseif($targetDoc.getObject($blogCategoryClassname)) 3057 + ## Display all posts that are in a category aCategorySpace.aCategory 3058 + #getBlogDocumentForCategoriesSpace($targetDoc.space $blogDoc) 3059 + #getBlogPostsLayout($blogDoc $postsLayout) 3060 + #if (~"$!layout~" == '') 3061 + #set ($layout = $postsLayout) 3062 + #end 3063 + #if ($postsVisiblity == 'recent') 3064 + \{\{blogpostlist category=~"$targetDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 3065 + #elseif($postsVisiblity == 'unpublished') 3066 + \{\{blogpostlist category=~"$targetDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 3067 + #end 3068 + #elseif($targetDoc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 3069 + ## Display all posts that are in a category aCategorySpace.% 3070 + #getBlogDocumentForCategoriesSpace($targetDoc.space $blogDoc) 3071 + #getBlogPostsLayout($blogDoc $postsLayout) 3072 + #if (~"$!layout~" == '') 3073 + #set ($layout = $postsLayout) 3074 + #end 3075 + #if ($postsVisiblity == 'recent') 3076 + \{\{blogpostlist category=~"$targetDoc.space.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 3077 + #elseif($postsVisiblity == 'unpublished') 3078 + \{\{blogpostlist category=~"$targetDoc.space.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 3079 + #end 3080 + #else 3081 + ## Display all posts in blog space aBlog 3082 + #getBlogDocument($targetDoc.space $blogDoc) 3083 + #getBlogPostsLayout($blogDoc $postsLayout) 3084 + #if (~"$!layout~" == '') 3085 + #set ($layout = $postsLayout) 3086 + #end 3087 + #if ($postsVisiblity == 'recent') 3088 + \{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 3089 + #elseif($postsVisiblity == 'unpublished') 3090 + \{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 3091 + #end 3092 + #end 3093 +#end 3094 +## 3095 +## 3096 +## 3097 +#** 3098 + * Bind parameters to a query object. 3099 + * 3100 + * @param queryObj The query object (org.xwiki.query.internal.ScriptQuery object) 3101 + * @param queryParams the query parameters. 3102 + *### 3103 +#macro(bindQueryParameters $queryObj $queryParams) 3104 + #set ($output = $queryObj) 3105 + #foreach( $key in $queryParams.keySet() ) 3106 + #set($output = $queryObj.bindValue($key, $queryParams.get($key))) 3107 + #end 3108 + #setVariable(~"$queryObj~" $output) 3109 +#end 3110 +## 3111 +## 3112 +## 3113 +#** 3114 + * Determines the blogposts layout: Default layout (Calendar), Image thumbnail 3115 + * 3116 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>postsLayout</tt> 3117 + * property set. 3118 + * @param postsLayout The resulting string. If the blog object does not define anything, it is considered full (default). 3119 + *### 3120 +#macro(getBlogPostsLayout $blogDoc $postsLayout) 3121 + #getBlogProperty($blogDoc 'postsLayout' 'full' $res) 3122 + #set ($postsLayout = $NULL) 3123 + #setVariable (~"$postsLayout~" $res) 3124 +#end 3125 +## 3126 +## 3127 +## 3128 +#** 3129 + * Retrieve the blog document of a categories space, which is the blog descriptor that have its categoriesLocation property set to the categories space 3130 + * 3131 + * @param categoriesSpace A <tt>String</tt>, the name of the space. 3132 + * @param blogDoc The resulting XDocument. 3133 + *### 3134 +#macro(getBlogDocumentForCategoriesSpace $categoriesSpace $blogDoc) 3135 + #set($macro.blogDocs = $services.query.hql(~", BaseObject obj, StringProperty as categoriesLocationProp where obj.name = doc.fullName and obj.className = '$blogClassname' and obj.id=categoriesLocationProp.id.id and categoriesLocationProp.id.name='categoriesLocation' and categoriesLocationProp.value=:categoriesLocation order by doc.creationDate~").setLimit(1).setOffset(0).bindValue('categoriesLocation', $categoriesSpace).execute()) 3136 + #if($macro.blogDocs.size() > 0 && $categoriesSpace != $defaultBlogSpace) 3137 + #set($macro.result = $xwiki.getDocument($macro.blogDocs.get(0))) 3138 + #else 3139 + ## Fallback to Blog.WebHome, the default blog 3140 + #set($macro.result = $xwiki.getDocument('Blog.WebHome')) 3141 + #end 3142 + #set ($blogDoc = $NULL) 3143 + #setVariable (~"$blogDoc~" $macro.result) 3144 +#end" %) 3145 +((( 3146 +(% class="macro-placeholder hidden" %) 3147 +((( 3148 +macro:velocity 3149 +))) 3150 +))) 3151 +))) 3152 + 3153 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogPostLayoutMacros~"" %) 3154 +((( 3155 +(% class="macro-placeholder hidden" %) 3156 +((( 3157 +macro:include 3158 +))) 3159 + 3160 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|#** 3161 + * Extract the layout parameters from a string. 3162 + * 3163 + * @param layoutParamsString The string representation of the layout parameters. 3164 + * It should contain a String following this format ~"paramName1=Value1|paramName2=Value2|...|paramNameK=ValueK~" 3165 + * @param layoutsParameters The resulting layout parameters Map. 3166 + *### 3167 +#macro(extractLayoutParametersFromString $layoutParamsString $layoutsParameters) 3168 + #set ($layoutsParameters = $NULL) 3169 + #set ($macro.layoutParams = \{}) 3170 + #if (~"$!layoutParamsString~" != '') 3171 + #set ($macro.paramsArr = $layoutParamsString.split('\\|')) 3172 + #foreach ($item in $macro.paramsArr) 3173 + #set ($itemSplit = $item.split('=')) 3174 + #if ($itemSplit.size() == 2) 3175 + #set ($discard = $macro.layoutParams.put($itemSplit[0].trim(), $itemSplit[1].trim())) 3176 + #end 3177 + #end 3178 + #end 3179 + #setVariable(~"$layoutsParameters~" $macro.layoutParams) 3180 +#end" %) 3181 +((( 3182 +(% class="macro-placeholder hidden" %) 3183 +((( 3184 +macro:velocity 3185 +))) 3186 +))) 3187 +))) 3188 + 3189 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-| #macro(displaySmallPost $pDoc $pObj) 3190 + #initLayoutVars($pDoc $pObj) 3191 + <div class=~"col-xs-12 col-sm-6 next_blog~"> 3192 + <a href=~"$pDoc.uRL~" class=~"thumbnail~"> 3193 + <div class=~"row~"> 3194 + <div class=~"col-xs-4~"> 3195 + #if ($imageAtt) 3196 + <img src=~"$escapetool.xml($pDoc.getAttachmentURL($pObj.getValue('image'),'download',$imgQs))~" /> 3197 + #end 3198 + </div> 3199 + <div class=~"col-xs-8 art_det~"> 3200 + <p class=~"text-left~"> 3201 + #if($displayTitle)$!postTitle #end 3202 + <br/><span class=~"date_info~"> $!dateStr </span> 3203 + </p> 3204 + #displayPostDetails($pDoc) 3205 + </div> 3206 + </div> 3207 + </a> 3208 + </div> 3209 + #end 3210 + ## 3211 + ## 3212 + ## 3213 + #macro(displayPinnedPost $pDoc $pObj) 3214 + #initLayoutVars($pDoc $pObj) 3215 + <div class=~"col-xs-12 col-sm-4 col-md-4 blog_pin~"> 3216 + <a href=~"$pDoc.uRL~" class=~"all-post-link~"> 3217 + <div class=~"thumbnail~"> 3218 + #if ($displayTitle && $displayTitleFirstOnPinnedPosts) 3219 + <h5 class=~"title-top~">$!postTitle <br/> <span class=~"date_info~"> $!dateStr </span> </h5> 3220 + #end 3221 + #if ($imageAtt) 3222 + <img src=~"$escapetool.xml($pDoc.getAttachmentURL($pObj.getValue('image'),'download',$imgQs))~" /> 3223 + #end 3224 + <div class=~"caption~"> 3225 + #if ($displayTitle && !$displayTitleFirstOnPinnedPosts) 3226 + <h5 class=~"text-left~">$!postTitle <br/> <span class=~"date_info~"> $!dateStr </span> </h5> 3227 + #end 3228 + #if ($displaySummaryOnPinnedPosts) 3229 + <div class=~"text-left post-summary~"> 3230 + #set ($postContent = $pObj.getProperty('extract').value) 3231 + $!pDoc.getRenderedContent($postContent, $pDoc.syntax.toIdString()) 3232 + </div> 3233 + #end 3234 + #displayPostDetails($pDoc) 3235 + </div> 3236 + </div> 3237 + </a> 3238 + </div> 3239 + #end 3240 + ## 3241 + ## 3242 + ## 3243 + #macro(formatPostDate $pDoc $pObj) 3244 + #set ($formattedDate = $NULL) 3245 + #set($dateStr = $datetool.format('medium', $postDate, $xcontext.locale)) 3246 + #if (~"$!dateStr~" != '') 3247 + #set ($dateArr = $dateStr.split(' ')) 3248 + #if ($dateArr.size() > 3) 3249 + #set ($dateStr = ~"~") 3250 + #foreach($s in $dateArr.subList(0, 3)) 3251 + #set ($dateStr = ~"$\{dateStr}$\{s} ~") 3252 + #end 3253 + #end 3254 + #end 3255 + #setVariable(~"$formattedDate~" $dateStr) 3256 + #end 3257 + ## 3258 + ## 3259 + ## 3260 + #macro(displayPostDetails $pDoc) 3261 + <div class=~"row post-details~"> 3262 + <div class=~"col-xs-8 detail~"> 3263 + #if ($services.like.displayButton($pDoc.documentReference)) 3264 + #set ($optLikeRecord = $services.like.getLikes($pDoc.documentReference)) 3265 + ## Retrieve the likes number in XWiki 12.9+ 3266 + #set ($likeNumber = $optLikeRecord.get()) 3267 + #if (!$stringtool.isNumeric($likeNumber.toString())) 3268 + ## Retrieve the likes number in XWiki versions greater than 12.7 and lower than 12.9. 3269 + #set ($likeNumber = $optLikeRecord.get().likeNumber) 3270 + #end 3271 + #if ($stringtool.isNumeric($likeNumber.toString())) 3272 + <div class=~"post-likes btn btn-default disabled badge~" 3273 + title=~"$escapetool.xml($services.localization.render('like.button.title', [$likeNumber]))~"> 3274 + $services.icon.renderHTML('heart') <span class=~"like-number~">$likeNumber</span> 3275 + </div> 3276 + #end 3277 + #elseif ($services.ratings) 3278 + #set ($ratingsConfigDoc = $services.ratings.getConfigurationDocument($pDoc.documentReference)) 3279 + #if ($ratingsConfigDoc.getValue('displayRatings') == 1) 3280 + <ul class=~"pull-left note list-inline~"> 3281 + #foreach ($x in [1..5]) 3282 + #set ($cls = ~"~") 3283 + #if ($x > $averageVote) 3284 + #set ($cls = ~"-o~") 3285 + #end 3286 + <li><span class=~"fa fa-star$\{cls}~"></span></li> 3287 + #end 3288 + </ul> 3289 + #end 3290 + #end 3291 + </div> 3292 + #if ($showPostComments) 3293 + <div class=~"col-xs-4 com_det~"> 3294 + <p class=~"text-right~">$services.icon.renderHTML('comments') ($!nbComments)</p> 3295 + </div> 3296 + #end 3297 + </div> 3298 + #end 3299 + ## 3300 + ## 3301 + ## 3302 + #macro(initLayoutVars $pDoc $pObj) 3303 + #set ($postTitle = $pDoc.display('title', 'view', $pObj)) 3304 + #set ($imageAtt = $pDoc.getAttachment($pObj.getValue('image'))) 3305 + #isPublished($pObj $isPublished) 3306 + #isHidden($pObj $isHidden) 3307 + #getEntryDate($pDoc $pObj $postDate) 3308 + #formatPostDate($postDate $dateStr) 3309 + #set ($averageVote = $mathtool.round($services.ratings.getAverageRating($pDoc.documentReference).getAverageVote())) 3310 + #set ($nbComments = $pDoc.getComments().size()) 3311 + #set ($showPostComments = true) 3312 + #if ($xwiki.getSpacePreferenceFor('showcomments', $pDoc.documentReference.lastSpaceReference, '1') == '0') 3313 + #set ($showPostComments = false) 3314 + #end 3315 + #end 3316 + ## 3317 + ## 3318 + ## 3319 + #macro(displayEditPinnedPostsButton) 3320 + #if (~"$!macroAdditionalParams.get('postIndex')~" == '0' && $services.security.authorization.hasAccess('edit', $pinnedPostsSourceDoc.fullName)) 3321 + #set ($discard = $xwiki.ssx.use('Blog.PinnedPostsSheet')) 3322 + #set ($discard = $xwiki.jsx.use('Blog.PinnedPostsSheet')) 3323 + #set ($editPinnedPostsURL = $xwiki.getURL($pinnedPostsSourceDoc.fullName, 'get', 'sheet=Blog.PinnedPostsSheet')) 3324 + #set ($newPinnedPostsObjectURL = $xwiki.getURL($pinnedPostsSourceDoc.fullName, 'objectadd', ~"classname=Blog.PinnedPostsClass&form_token=$!\{services.csrf.getToken()}~")) 3325 + <div class=~"edit-pinned-posts-container~"> 3326 + <div class=~"col-xs-12 col-sm-12 col-md-12 text-right edit-pinned-posts~"> 3327 + <a href=~"javascript:;~" class=~"editPinnedPosts~"> 3328 + <span class=~"glyphicon glyphicon-edit~" aria-hidden=~"true~"></span> $services.localization.render('blog.post.layout.cards.pinnedposts.edit') 3329 + </a> 3330 + <input type=~"hidden~" id=~"editPinnedPostsURL~" value=~"$escapetool.xml($editPinnedPostsURL)~"/> 3331 + <input type=~"hidden~" id=~"newPinnedPostsObjectURL~" value=~"$escapetool.xml($newPinnedPostsObjectURL)~"/> 3332 + </div> 3333 + </div> 3334 + #if (~"$!pinnedPostsObj~" == '') 3335 + <input type=~"hidden~" name=~"createPinnedPostsObject~" value=~"1~"/> 3336 + #end 3337 + #end 3338 + #end" %) 3339 +((( 3340 +(% class="macro-placeholder hidden" %) 3341 +((( 3342 +macro:velocity 3343 +))) 3344 +))) 3345 + 3346 +(% class="macro" data-macro="startmacro:velocity|-||-| #if ($services.security.authorization.hasAccess('view', $xcontext.macro.params.reference)) 3347 + #set ($postDoc = $xwiki.getDocument($xcontext.macro.params.reference)) 3348 + #getEntryObject($postDoc $postObj) 3349 + #if (~"$!postObj~" != '') 3350 + #extractLayoutParametersFromString($xcontext.macro.params.params $macroAdditionalParams) 3351 + ## 3352 + #set ($displayTitle = true) 3353 + #if (~"$!macroAdditionalParams.get('displayTitle')~" == 'false') 3354 + #set ($displayTitle = false) 3355 + #end 3356 + ## 3357 + #set ($displayTitleFirstOnPinnedPosts = false) 3358 + #if (~"$!macroAdditionalParams.get('displayTitleFirstOnPinnedPosts')~" == 'true') 3359 + #set ($displayTitleFirstOnPinnedPosts = true) 3360 + #end 3361 + ## 3362 + #set ($displaySummaryOnPinnedPosts = true) 3363 + #if (~"$!macroAdditionalParams.get('displaySummaryOnPinnedPosts')~" == 'false') 3364 + #set ($displaySummaryOnPinnedPosts = false) 3365 + #end 3366 + ## 3367 + #set ($postIndex = $numbertool.toNumber($macroAdditionalParams.get('postIndex')).intValue()) 3368 + #if ($postIndex == 0) 3369 + #set ($stopBlogPostsDisplay = false) 3370 + #end 3371 + #set ($listResultsCount = $numbertool.toNumber($macroAdditionalParams.get('list-results-count')).intValue()) 3372 + #set($discard = $xwiki.ssx.use('Blog.BlogPostLayoutCards')) 3373 + #set($scaleWidth = 600) 3374 + #set($imgQs=~"width=$scaleWidth~") 3375 + ## Display pinned posts 3376 + ## Get the list of pinned posts 3377 + #set ($pinnedPosts = []) 3378 + #set ($pinnedPostsSourceDoc = $NULL) 3379 + #if (~"$!macroAdditionalParams.get('forceDisplayPinnedPosts')~" == 'true' && ~"$!macroAdditionalParams.get('list-blog')~" != '') 3380 + #set ($pinnedPostsSourceDoc = $xwiki.getDocument($macroAdditionalParams.get('list-blog'))) 3381 + ## Note that the 'list-blog' parameter contains the value of the 'blog' parameter of the blogpostlist macro passed to the current layout macro 3382 + #end 3383 + #if (~"$!pinnedPostsSourceDoc~" != '') 3384 + #set ($pinnedPostsObj = $pinnedPostsSourceDoc.getObject('Blog.PinnedPostsClass')) 3385 + #if (~"$!pinnedPostsObj~" != '') 3386 + #set ($orderedPinnedPostsJSON = $pinnedPostsObj.getProperty('orderedPosts').value) 3387 + #if (~"$!orderedPinnedPostsJSON~" != '') 3388 + #set ($pinnedPosts = $jsontool.parse($orderedPinnedPostsJSON)) 3389 + #else 3390 + #set ($pinnedPosts = $pinnedPostsObj.getProperty('posts').value) 3391 + #end 3392 + #end 3393 + #if ($postIndex == 0 && $pinnedPosts.size() > 0) 3394 + \{\{html clean=~"false~"}} 3395 + #set ($x = 0) 3396 + #set ($showPinnedPostsButton = true) 3397 + #foreach ($pinnedPost in $pinnedPosts) 3398 + #if ($x == 0) 3399 + <div class=~"row flex-container~"> 3400 + #if ($showPinnedPostsButton) 3401 + #displayEditPinnedPostsButton() 3402 + #set($showPinnedPostsButton = false) 3403 + #end 3404 + #end 3405 + #set ($pinnedPostDoc = $xwiki.getDocument($pinnedPost)) 3406 + #getEntryObject($pinnedPostDoc $pinnedPostObj) 3407 + #if (~"$!pinnedPostObj~" != '') 3408 + #displayPinnedPost($pinnedPostDoc $pinnedPostObj) 3409 + #end 3410 + #set ($x = $mathtool.add($x, 1)) 3411 + #if ($x == 3) 3412 + #set ($x = 0) 3413 + </div> 3414 + #end 3415 + #end 3416 + #if ($mathtool.mod($x, 3) != 0) 3417 + </div> 3418 + #end 3419 + \{\{/html}} 3420 + 3421 + ## If the first post is a pinned post : this means that all the posts are pinned 3422 + ## In this case, avoid displaying the posts again after the pinned posts section. 3423 + #if ($pinnedPosts.contains($xcontext.macro.params.reference)) 3424 + #set ($stopBlogPostsDisplay = true) 3425 + #end 3426 + #elseif($postIndex == 0 && $pinnedPosts.size() == 0) 3427 + 3428 + \{\{html}} 3429 + <div class=~"row no-pinnded-posts~"> 3430 + #displayEditPinnedPostsButton() 3431 + </div> 3432 + \{\{/html}} 3433 + 3434 + #end 3435 + #end 3436 + #if (!$stopBlogPostsDisplay) 3437 + \{\{html clean=~"false~"}} 3438 + #if (~"$!nbDisplayedPosts~" == '' || $postIndex == 0) 3439 + #set ($nbDisplayedPosts = 0) 3440 + #end 3441 + #if($mathtool.mod($nbDisplayedPosts, 2) == 0) 3442 + #if ($nbDisplayedPosts != 0) 3443 + </div> 3444 + #set ($lastHtmlTag = 'c') 3445 + #end 3446 + #if ($nbDisplayedPosts == $mathtool.sub($listResultsCount, 1)) 3447 + <div class=~"row~"> 3448 + #set ($lastHtmlTag = 'o') 3449 + #set ($lastRowClass = ~"~") 3450 + #else 3451 + <div class=~"row flex-container~"> 3452 + #set ($lastHtmlTag = 'o') 3453 + #set ($lastRowClass = ~"flex~") 3454 + #end 3455 + #end 3456 + #displaySmallPost($postDoc $postObj) 3457 + #if ($nbDisplayedPosts == $mathtool.sub($listResultsCount, 1)) 3458 + </div> 3459 + #set ($lastHtmlTag = 'c') 3460 + #end 3461 + #set ($nbDisplayedPosts = $mathtool.add($nbDisplayedPosts, 1)) 3462 + #if ($postIndex == $mathtool.sub($listResultsCount, 1)) 3463 + #if (~"$!lastHtmlTag~" == 'o') 3464 + #if ($mathtool.mod($nbDisplayedPosts, 2) == 1 && $lastRowClass == ~"flex~") 3465 + <div class=~"col-xs-12 col-sm-6~"></div> 3466 + #end 3467 + </div> 3468 + #end 3469 + #end 3470 + \{\{/html}} 3471 + #end 3472 + #else 3473 + \{\{error}}$services.localization.render('blog.blogpostlayout.notpost', [$xcontext.macro.params.reference])\{\{/error}} 3474 + #end 3475 + #else 3476 + \{\{error}}$services.localization.render('blog.blogpostlayout.post_view_not_allowed', [$xcontext.macro.params.reference])\{\{/error}} 3477 + #end" %) 3478 +((( 3479 +(% class="macro-placeholder hidden" %) 3480 +((( 3481 +macro:velocity 3482 +))) 3483 + 3484 +(% class="macro" data-macro="startmacro:html|-|clean=~"false~"|-|<div class=~"col-xs-12 col-sm-6 next_blog~"> 3485 +<a href=~"/Nieuws/Audit%20BMI%20en%20BORG~" class=~"thumbnail~"> 3486 +<div class=~"row~"> 3487 +<div class=~"col-xs-4~"> 3488 +<img src=~"/download/Nieuws/Audit%20BMI%20en%20BORG/giphy.gif?width=600&rev=1.1~" /> 3489 +</div> 3490 +<div class=~"col-xs-8 art_det~"> 3491 +<p class=~"text-left~"> 3492 +Audit BMI en BORG <br/><span class=~"date_info~"> Jan 15, 2024, </span> 3493 +</p> 3494 +<div class=~"row post-details~"> 3495 +<div class=~"col-xs-8 detail~"> 3496 +<div class=~"post-likes btn btn-default disabled badge~" 3497 +title=~"Number of likes on this page: 0~"> 3498 +<span class=~"fa fa-heart~"></span> <span class=~"like-number~">0</span> 3499 +</div> 3500 +</div> 3501 +<div class=~"col-xs-4 com_det~"> 3502 +<p class=~"text-right~"><span class=~"fa fa-comments~"></span> (0)</p> 3503 +</div> 3504 +</div> 3505 +</div> 3506 +</div> 3507 +</a> 3508 +</div>" %) 3509 +((( 3510 +(% class="macro-placeholder hidden" %) 3511 +((( 3512 +macro:html 3513 +))) 3514 + 3515 +(% class="col-xs-12 col-sm-6 next_blog" %) 3516 +((( 3517 +(% class="row" %) 3518 +((( 3519 +(% class="col-xs-4" %) 3520 +((( 3521 +[[~[~[image:/download/Nieuws/Audit%20BMI%20en%20BORG/giphy.gif?width=600&rev=1.1~]~]>>path:/Nieuws/Audit%20BMI%20en%20BORG||class="thumbnail"]] 3522 +))) 3523 + 3524 +(% class="col-xs-8 art_det" %) 3525 +((( 3526 +(% class="text-left" %) 3527 +[[Audit BMI en BORG 3528 +(% class="date_info" %) Jan 15, 2024,>>path:/Nieuws/Audit%20BMI%20en%20BORG||class="thumbnail"]] 3529 + 3530 +(% class="row post-details" %) 3531 +((( 3532 +(% class="col-xs-8 detail" %) 3533 +((( 3534 +(% class="post-likes btn btn-default disabled badge" title="Number of likes on this page: 0" %) 3535 +((( 3536 +[[(% class="like-number" %)0>>path:/Nieuws/Audit%20BMI%20en%20BORG||class="thumbnail"]] 3537 +))) 3538 +))) 3539 + 3540 +(% class="col-xs-4 com_det" %) 3541 +((( 3542 +(% class="text-right" %) 3543 +[[(0)>>path:/Nieuws/Audit%20BMI%20en%20BORG||class="thumbnail"]] 3544 +))) 3545 +))) 3546 +))) 3547 +))) 3548 +))) 3549 +))) 3550 +))) 3551 +))) 3552 + 3553 +(% class="macro" data-macro="startmacro:blogPostLayoutCards|-|reference=~"Nieuws.Nieuwsbrief december 2023~" params=~"displayTitle=true|useSummary=true|list-limit=10|list-layout=cards|list-blog=Nieuws.WebHome|list-paginated=no|list-results-count=10|postIndex=0|previousPostReference=|postIndex=1|previousPostReference=Nieuws.Nieuwsbrief januari 2024|postIndex=2|previousPostReference=Nieuws.Audit BMI en BORG~"" %) 3554 +((( 3555 +(% class="macro-placeholder hidden" %) 3556 +((( 3557 +macro:blogPostLayoutCards 3558 +))) 3559 + 3560 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogCode~"" %) 3561 +((( 3562 +(% class="macro-placeholder hidden" %) 3563 +((( 3564 +macro:include 3565 +))) 3566 + 3567 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogParameters~"" %) 3568 +((( 3569 +(% class="macro-placeholder hidden" %) 3570 +((( 3571 +macro:include 3572 +))) 3573 + 3574 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|## Blog 3575 +#set($blogClassname = 'Blog.BlogClass') 3576 +#set($blogTemplate = 'Blog.BlogTemplate') 3577 +#set($blogSheet = 'Blog.BlogSheet') 3578 +## Blog entries 3579 +#set($blogPostClassname = 'Blog.BlogPostClass') 3580 +#set($blogPostTemplate = 'Blog.BlogPostTemplate') 3581 +#set($blogPostSheet = 'Blog.BlogPostSheet') 3582 +#set($blogPostObjectNumber = $xwiki.getDocument($blogPostTemplate).getObject($blogPostClassname).number) 3583 +#set($oldArticleClassname = 'XWiki.ArticleClass') 3584 +## Categories 3585 +#set($blogCategoryClassname = 'Blog.CategoryClass') 3586 +#set($blogCategoryTemplate = 'Blog.CategoryTemplate') 3587 +#set($blogCategorySheet = 'Blog.CategorySheet') 3588 +#set($blogCategoriesSheet = 'Blog.CategoriesSheet') 3589 +#set($oldBlogCategoryClassname = 'Blog.Categories') 3590 +#set($defaultCategoryParent = 'Blog.Categories') 3591 +## Style 3592 +#set($blogStyleDocumentName = 'Blog.BlogStyle') 3593 +#set($blogStyleDocument = $xwiki.getDocument($blogStyleDocumentName)) 3594 +## Clientside scripts 3595 +#set($blogScriptsDocumentName = 'Blog.BlogScripts') 3596 +#set($blogPublisher = $xwiki.getDocument('Blog.Publisher')) 3597 +## Misc 3598 +#set($thisURL = $doc.getURL($xcontext.action, $request.queryString)) 3599 +#set($isBlogPost = $doc.getObject($blogPostClassname)) 3600 +#set($defaultBlogSpace = 'Blog') 3601 +#set($isCategoryPage = $doc.getObject($blogCategoryClassname)) 3602 +#set($isCategoriesHomePage = $doc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 3603 +## 3604 +## 3605 +## 3606 +#** 3607 + * Displays an image, taken from the blog style document. 3608 + * 3609 + * @param $imgName The name of the icon from icons set to use. 3610 + *# 3611 +#macro(toolImage $imgName)(% class=~"icon-manage~"~%)$services.icon.render($imgName)(%~%)#end" %) 3612 +((( 3613 +(% class="macro-placeholder hidden" %) 3614 +((( 3615 +macro:velocity 3616 +))) 3617 +))) 3618 +))) 3619 + 3620 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|## 3621 +## 3622 +## 3623 +## Import the blog skin and javascripts. 3624 +$!xwiki.ssx.use($blogStyleDocumentName)## 3625 +$!xwiki.jsx.use($blogScriptsDocumentName)## 3626 +## 3627 +## import the hierarchy for the path the blog post footer (in displayEntryBlogLocation) 3628 +#template('hierarchy_macros.vm')## 3629 +## 3630 +## 3631 +#** 3632 + * Prints a blog. This is the main macro used in the BlogSheet. 3633 + * 3634 + * @param blogDoc the XDocument holding the blog definition object. 3635 + *### 3636 +#macro(printBlog $blogDoc) 3637 + \{\{include reference='Blog.CreatePost'/}} 3638 + 3639 + ## Use the blogPostList macro to display the blogposts 3640 + ## The blogPostList is used only in case of 'all' or 'paged' blog display type because the blogPostList macro 3641 + ## do not support FTM the monthly and weekly blog display types 3642 + #getBlogDisplayType($blogDoc $displayType) 3643 + #if ($displayType == 'weekly' || $displayType == 'monthly') 3644 + #getBlogEntries($blogDoc $entries) 3645 + #displayBlog($entries 'index' true true) 3646 + #displayNavigationLinks($blogDoc) 3647 + #else 3648 + #getBlogDisplayType($blogDoc $displayType) 3649 + #set ($paginated = 'no') 3650 + #if ($displayType == 'paginated') 3651 + #set ($paginated = 'yes') 3652 + #end 3653 + #getBlogPostsLayout($blogDoc $postsLayout) 3654 + (% class=~"hfeed index~" ~%)(((\{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$paginated~" layout=~"$!postsLayout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" /}}))) 3655 + #end 3656 +#end 3657 +## 3658 +## 3659 +## 3660 +#** 3661 + * Shows blog information. In view mode, the description is printed. In edit mode, allows changing blog settings: title, 3662 + * description, blog type (global or in-space), index display type (fixed size pagination, weekly index, monthly index, 3663 + * all entries). 3664 + * 3665 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 3666 + *### 3667 +#macro(showBlogInfo $blogDoc) 3668 + #if($blogDoc.getObject($blogClassname)) 3669 + ## Keep testing for inline action for backward compatibility with older blogs. 3670 + #if($xcontext.action == 'edit' || $xcontext.action == 'inline') 3671 + #macro(displayProperty $blogDoc $propname) 3672 + ; #displayPropName($xwiki.getClass($blogClassname).get($propname)): 3673 + : $blogDoc.display($propname) 3674 + #end 3675 + #displayProperty($blogDoc 'title') 3676 + #displayProperty($blogDoc 'description') 3677 + #displayProperty($blogDoc 'displayType') 3678 + #displayProperty($blogDoc 'itemsPerPage') 3679 + #displayProperty($blogDoc 'postsLayout') 3680 + #displayProperty($blogDoc 'postsLayoutParameters') 3681 + #else 3682 + $blogDoc.display('description') 3683 + #end 3684 + #elseif($doc.fullName == $blogSheet) 3685 += $services.localization.render('blog.code.blogsheet') = 3686 + \{\{translation key='blog.code.sheetexplanation'/}} 3687 + #else 3688 + \{\{warning}}\{\{translation key='blog.code.notblog'/}}\{\{/warning}} 3689 + #end 3690 +#end 3691 +## 3692 +## 3693 +## 3694 +#** 3695 + * Retrieve the blog document, which usually is either <tt><Space>.WebHome</tt> for whole-spaces blogs, or 3696 + * <tt><Space>.Blog</tt> for in-space blogs. If none of these documents contains a blog object, then the first 3697 + * (alphabetically) document in the target space that contains one is returned. Finally, if no document in the current 3698 + * space contains a blog object, then <tt>Blog.WebHome</tt> is returned as the default blog. 3699 + * 3700 + * @param space A <tt>String</tt>, the name of the space where to search. 3701 + * @param blogDoc The resulting XDocument. 3702 + *### 3703 +#macro(getBlogDocument $space $blogDoc) 3704 + #set ($result = $NULL) 3705 + ## Check if the current space has a WebPreferences page that contains a Blog.EnablePanelsConfigurationClass object. 3706 + ## It is possible to display a blog's panels info in a page that is not inside a blog and in that case we need to 3707 + ## identify the right blog based on a configuration object in a WebPreferences page. 3708 + #set ($spaceReference = $services.model.resolveSpace($space)) 3709 + #set ($preferencesDocRef = $services.model.createDocumentReference('WebPreferences', $spaceReference)) 3710 + #set ($preferencesDoc = $xwiki.getDocument($preferencesDocRef)) 3711 + #set ($preferencesObj = $preferencesDoc.getObject('Blog.EnablePanelsConfigurationClass')) 3712 + #if ($preferencesObj) 3713 + #set ($result = $xwiki.getDocument( $preferencesObj.getValue('blog'))) 3714 + #end 3715 + ## If no special case occurs (like the previous one), identify a blog by checking the current location from its hierarchy. 3716 + #if (~"$!result~" == '') 3717 + ## First, try the Space.WebHome, for a whole-space blog 3718 + #set($result = $xwiki.getDocument(~"$\{space}.WebHome~")) 3719 + #if(!$result.getObject($blogClassname)) 3720 + ## Second, try the Space.Blog document 3721 + #set($result = $xwiki.getDocument(~"$\{space}.Blog~")) 3722 + #if(!$result.getObject($blogClassname)) 3723 + ## Third, try searching for a blog document in the current space 3724 + ## Prevent the query fail when the space contains dots '.' 3725 + #set($blogDocs = $services.query.hql(~", BaseObject obj where doc.space = :space and obj.name = doc.fullName and obj.className = '$blogClassname' order by doc.name~").setLimit(1).setOffset(0).bindValue('space', $space).execute()) 3726 + #if($blogDocs.size() > 0) 3727 + #set($result = $xwiki.getDocument($blogDocs.get(0))) 3728 + #else 3729 + ## Fourth, try searching for a blog document that have the its 'postsLocation' set to the current space 3730 + #set($blogDocs = $services.query.hql(~", BaseObject obj, StringProperty as postsLocationProp where obj.name = doc.fullName and obj.className = '$blogClassname' and obj.id=postsLocationProp.id.id and postsLocationProp.id.name='postsLocation' and postsLocationProp.value=:postsLocation order by doc.name~").setLimit(1).setOffset(0).bindValue('postsLocation', $space).execute()) 3731 + #if($blogDocs.size() > 0) 3732 + #set($result = $xwiki.getDocument($blogDocs.get(0))) 3733 + #else 3734 + ## Last, fallback to Blog.WebHome, the default blog 3735 + #set($result = $xwiki.getDocument('Blog.WebHome')) 3736 + #end 3737 + #end 3738 + #end 3739 + #end 3740 + #end 3741 + #set ($blogDoc = $NULL) 3742 + #setVariable (~"$blogDoc~" $result) 3743 +#end 3744 +## 3745 +## 3746 +## 3747 +#** 3748 + * Retrieve the blog title. 3749 + * 3750 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass<tt> object with the <tt>title</tt> property set. 3751 + * @param title The resulting title. 3752 + *### 3753 +#macro(getBlogTitle $blogDoc $title) 3754 + ## Titles can contain velocity code (most commonly translations), so we should evaluate them. 3755 + #set ($title = $NULL) 3756 + #setVariable (~"$title~" $!blogDoc.displayTitle) 3757 +#end 3758 +## 3759 +## 3760 +## 3761 +#** 3762 + * Retrieve the blog description. 3763 + * 3764 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>description</tt> 3765 + * property set. 3766 + * @param description The resulting description. 3767 + *### 3768 +#macro(getBlogDescription $blogDoc $description) 3769 + #getBlogProperty($blogDoc 'description' '' $result) 3770 + #set ($description = $NULL) 3771 + #setVariable (~"$description~" $result) 3772 +#end 3773 +## 3774 +## 3775 +## 3776 +#** 3777 + * Retrieves a list of entries to be displayed. The entries are either part of the blog's space, or have the blog 3778 + * document set as a parent. The number and range of entries returned (from all those belonging to this blog) depends on 3779 + * the blog display type: paginated (fixed number of entries), weekly (all entries in a week), monthly (all entries in a 3780 + * month), or all. 3781 + * 3782 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 3783 + * @param entries The resulting list of entries to display, a list of XDocument names. 3784 + *### 3785 +#macro(getBlogEntries $blogDoc $entries) 3786 + #if (!$entries) 3787 + #setVariable (~"$entries~" []) 3788 + #end 3789 + #getAllBlogPostsQuery($query) 3790 + #isDefaultBlog($blogDoc $isDefault) 3791 + #set($queryParams = \{}) 3792 + #if ($isDefault) 3793 + #getCategoryAllBlogPostsQuery($query) 3794 + #set($query = ~"$\{query} and (doc.creator = :creator or (isPublished.value = 1 and hidden.value = 0)) and (doc.space = :space or catList like :catList escape '!')~") 3795 + #set($discard = $queryParams.put('creator', $xcontext.user)) 3796 + #set($discard = $queryParams.put('space', $blogDoc.space)) 3797 + #set($sanitizedBlogSpace = $blogDoc.space.replaceAll('([%_!])', '!$1').concat('.%')) 3798 + #set($discard = $queryParams.put('catList', $sanitizedBlogSpace)) 3799 + #else 3800 + #set($query = ~"$\{query} and (doc.space = :space or doc.parent = :parent)~") 3801 + #getBlogPostsLocation($blogDoc.space $blogPostsLocation) 3802 + #set($discard = $queryParams.put('space', $blogPostsLocation)) 3803 + #set($discard = $queryParams.put('parent', $blogDoc.space)) 3804 + #end 3805 + #getBlogDisplayType($blogDoc $displayType) 3806 + #if($displayType == 'weekly') 3807 + #getWeeklyBlogEntries($blogDoc $query $entries $queryParams) 3808 + #elseif($displayType == 'monthly') 3809 + #getMonthlyBlogEntries($blogDoc $query $entries $queryParams) 3810 + #elseif($displayType == 'all') 3811 + #getAllBlogEntries($blogDoc $query $entries $queryParams) 3812 + #else 3813 + #getPagedBlogEntries($blogDoc $query $entries $queryParams) 3814 + #end 3815 +#end 3816 +## 3817 +## 3818 +## 3819 +#** 3820 + * Retrieves a list of entries to be displayed. The entries are taken from a ~"page~" of the blog, a sequence of documents 3821 + * defined by the request parameters <tt>ipp</tt> (items per page) and <tt>page</tt> (the current page). Initially the 3822 + * first page is displayed, with the number of entries defined in the blog object in the <tt>itemsPerPage</tt> property 3823 + * (10 if not defined). 3824 + * 3825 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 3826 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 3827 + * refined to restrict to a given space, or to a given search criteria, etc. 3828 + * @param entries The resulting list of entries to display, a list of XDocument names. 3829 + * @param queryParams The parameters to bind with the query. 3830 + *### 3831 +#macro(getPagedBlogEntries $blogDoc $query $entries $queryParams) 3832 + #if (!$entries) 3833 + #setVariable (~"$entries~" []) 3834 + #end 3835 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 3836 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~")) 3837 + #bindQueryParameters($countQueryObj $queryParams) 3838 + #bindQueryParameters($queryObj $queryParams) 3839 + #set($totalEntries = $countQueryObj.count()) 3840 + #getBlogProperty($blogDoc 'itemsPerPage' '10' $defaultItemsPerPage) 3841 + #set($defaultItemsPerPage = $numbertool.toNumber($defaultItemsPerPage).intValue()) 3842 + ## This macro is defined in the default macros.vm library. It also sets $itemsPerPage and $startAt. 3843 + #preparePagedViewParams($totalEntries $defaultItemsPerPage) 3844 + #set($discard = $entries.addAll($queryObj.setLimit($itemsPerPage).setOffset($startAt).addFilter(~"unique~").execute())) 3845 +#end 3846 +## 3847 +## 3848 +## 3849 +#** 3850 + * Retrieves a list of entries to be displayed. The entries are taken from a week of the blog. The target week is 3851 + * defined by the request parameters <tt>week</tt> (the week number in the year, from 1 to 52) and <tt>year</tt> (4 3852 + * digit year). Initially the current week is displayed. 3853 + * 3854 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 3855 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 3856 + * refined to restrict to a given space, or to a given search criteria, etc. 3857 + * @param entries The resulting list of entries to display, a list of XDocument names. 3858 + * @param queryParams The parameters to bind with the query. 3859 + *### 3860 +#macro(getWeeklyBlogEntries $blogDoc $query $entries $queryParams) 3861 + #if (!$entries) 3862 + #setVariable (~"$entries~" []) 3863 + #end 3864 + #getRequestedWeek($weekDate) 3865 + #set($dateFormatter = $xwiki.jodatime.getDateTimeFormatterForPattern('yyyy-MM-dd')) 3866 + #set($minDay = $dateFormatter.print($weekDate.toMutableDateTime().weekOfWeekyear().roundFloor())) 3867 + #set($maxDay = $dateFormatter.print($weekDate.toMutableDateTime().weekOfWeekyear().roundCeiling())) 3868 + #set($query = ~"$\{query} and publishDate.value >= '$minDay' and publishDate.value < '$maxDay'~") 3869 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 3870 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 3871 + #bindQueryParameters($countQueryObj $queryParams) 3872 + #bindQueryParameters($queryObj $queryParams) 3873 + #set($totalEntries = $countQueryObj.count()) 3874 + #set($discard = $entries.addAll($queryObj.execute())) 3875 +#end 3876 +## 3877 +## 3878 +## 3879 +#** 3880 + * Retrieves a list of entries to be displayed. The entries are taken from a month of the blog. The target month is 3881 + * defined by the request parameters <tt>month</tt> (the month number, from 1 to 12) and <tt>year</tt> (4 3882 + * digit year). Initially the current month is displayed. 3883 + * 3884 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 3885 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 3886 + * refined to restrict to a given space, or to a given search criteria, etc. 3887 + * @param entries The resulting list of entries to display, a list of XDocument names. 3888 + * @param queryParams The parameters to bind with the query. 3889 + *### 3890 +#macro(getMonthlyBlogEntries $blogDoc $query $entries) 3891 + #if (!$entries) 3892 + #setVariable (~"$entries~" []) 3893 + #end 3894 + #getRequestedMonth($monthDate) 3895 + #set($dateFormatter = $xwiki.jodatime.getDateTimeFormatterForPattern('yyyy-MM-dd')) 3896 + #set($minDay = $dateFormatter.print($monthDate.toMutableDateTime().monthOfYear().roundFloor())) 3897 + #set($maxDay = $dateFormatter.print($monthDate.toMutableDateTime().monthOfYear().roundCeiling())) 3898 + #set($query = ~"$\{query} and publishDate.value >= '$minDay' and publishDate.value < '$maxDay'~") 3899 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 3900 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 3901 + #bindQueryParameters($countQueryObj $queryParams) 3902 + #bindQueryParameters($queryObj $queryParams) 3903 + #set($totalEntries = $countQueryObj.count()) 3904 + #set($discard = $entries.addAll($queryObj.execute())) 3905 +#end 3906 +## 3907 +## 3908 +## 3909 +#** 3910 + * Retrieves a list of entries to be displayed. All entries belonging to the current blog are returned. 3911 + * 3912 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 3913 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 3914 + * refined to restrict to a given space, or to a given search criteria, etc. 3915 + * @param entries The resulting list of entries to display, a list of XDocument names. 3916 + * @param queryParams The parameters to bind with the query. 3917 + *### 3918 +#macro(getAllBlogEntries $blogDoc $query $entries $queryParams) 3919 + #if (!$entries) 3920 + #setVariable (~"$entries~" []) 3921 + #end 3922 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 3923 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 3924 + #bindQueryParameters($countQueryObj $queryParams) 3925 + #bindQueryParameters($queryObj $queryParams) 3926 + #set($totalEntries = $countQueryObj.count()) 3927 + #set($discard = $entries.addAll($queryObj.execute())) 3928 +#end 3929 +## 3930 +## 3931 +## 3932 +#** 3933 + * Retrieves a list of entries to be displayed. Only (and all) unpublished entries are returned. 3934 + * 3935 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 3936 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 3937 + * refined to restrict to a given space, or to a given search criteria, etc. 3938 + * @param queryParams The parameters to bind with the query. 3939 + * @param entries The resulting list of entries to display, a list of XDocument names. 3940 + *### 3941 +#macro(getUnpublishedBlogEntries $blogDoc $query $queryParams $entries) 3942 + #if (!$entries) 3943 + #setVariable (~"$entries~" []) 3944 + #end 3945 + #set($query = ~"$\{query} and isPublished.value = 0~") 3946 + #set ($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 3947 + #set ($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 3948 + #bindQueryParameters($countQueryObj $queryParams) 3949 + #bindQueryParameters($queryObj $queryParams) 3950 + #set($totalEntries = $countQueryObj.count()) 3951 + #set($discard = $entries.addAll($queryObj.execute())) 3952 +#end 3953 +## 3954 +## 3955 +## 3956 +#** 3957 + * Retrieves a list of entries to be displayed. The entries are taken from all the wiki, and not from a specific blog. 3958 + * 3959 + * @param entries The resulting list of entries to display, a list of XDocument names. 3960 + *### 3961 +#macro(getGlobalBlogEntries $entries) 3962 + #if (!$entries) 3963 + #setVariable (~"$entries~" []) 3964 + #end 3965 + #getAllBlogPostsQuery($query) 3966 + #set($totalEntries = $services.query.hql($query).count()) 3967 + #set($defaultItemsPerPage = 20) 3968 + ## This macro is defined in the default macros.vm library. It also sets $itemsPerPage and $startAt. 3969 + #preparePagedViewParams($totalEntries $defaultItemsPerPage) 3970 + #set($discard = $entries.addAll($services.query.hql(~"$\{query} order by publishDate.value desc~").setLimit($itemsPerPage).setOffset($startAt).execute())) 3971 +#end 3972 +#** 3973 + * Return the base query for selecting blog entries. It filters only visible entries, but does not bind to a specific 3974 + * blog, nor specify a range or an ordering criteria. 3975 + * 3976 + * This macro is a duplicate of #getAllBlogPostsQuery($query). Keep it in case it's used outside, in custom code. 3977 + * 3978 + * @param query The basic query for selecting blog entries. 3979 + *# 3980 +#macro(getBlogEntriesBaseQuery $query) 3981 + #getAllBlogPostsQuery($query) 3982 +#end 3983 +#** 3984 + * Return the Query for selecting the all wiki blog posts without filtering 3985 + * 3986 + * @param query The basic query for selecting blog entries. 3987 + *# 3988 +#macro(getAllBlogPostsQuery $query) 3989 + #set ($query = $NULL) 3990 + #setVariable(~"$query~" ~", BaseObject as obj, IntegerProperty isPublished, 3991 + IntegerProperty hidden, DateProperty publishDate 3992 + where doc.fullName <> '$blogPostTemplate' and 3993 + obj.name=doc.fullName and obj.className='$blogPostClassname' and 3994 + isPublished.id.id = obj.id and isPublished.id.name = 'published' and 3995 + hidden.id.id = obj.id and hidden.id.name='hidden' and 3996 + publishDate.id.id = obj.id and publishDate.id.name='publishDate' and 3997 + (doc.creator = '$escapetool.sql($xcontext.user)' or (isPublished.value = 1 and hidden.value = 0))~") 3998 +#end 3999 +## 4000 +## 4001 +## 4002 +###** 4003 + * Return the Query for selecting the all wiki blog posts with categories filtering 4004 + * 4005 + * @param query The basic query for selecting blog entries. 4006 + *### 4007 +#macro(getCategoryAllBlogPostsQuery $query) 4008 + #set ($query = $NULL) 4009 + #getAllBlogPostsQuery($baseQuery) 4010 + #set ($baseQuery = $baseQuery.replace('DateProperty publishDate', 'DateProperty publishDate, DBStringListProperty as category left join category.list catList')) 4011 + #setVariable(~"$query~" ~"$\{baseQuery} and obj.id=category.id.id and category.id.name='category'~") 4012 +#end 4013 +## 4014 +## 4015 +## 4016 +#** 4017 + * Determines how is the blog index split into pages: paginated (fixed number of entries), weekly (all entries in a 4018 + * week), monthly (all entries in a month), or all. 4019 + * 4020 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>displayType</tt> 4021 + * property set. 4022 + * @param displayType The resulting string. If the blog object does not define anything, it is considered paginated. 4023 + *### 4024 +#macro(getBlogDisplayType $blogDoc $displayType) 4025 + #getBlogProperty($blogDoc 'displayType' 'paginated' $result) 4026 + #set ($displayType = $NULL) 4027 + #setVariable (~"$displayType~" $result) 4028 +#end 4029 +## 4030 +## 4031 +## 4032 +#** 4033 + * Displays a list of entries. 4034 + * 4035 + * @param entries The entries to display, a list of XDocument names. 4036 + * @param displaying What exactly is displayed: blog index, a single blog entry, a blog category, search results, 4037 + * unpublished entries, etc. This will be used as the classname(s) for the container div (hfeed). Currently 4038 + * used values: index, single, category, search, unpublished, hidden. 4039 + * @param onlyExtract If <tt>true</tt>, only display the extract of articles where available, otherwise display the full content. 4040 + * @param shouldDisplayTitles If <tt>true</tt>, display the blog title (blog posts shouldn't display the title when they're 4041 + * displayed alone on their page since it's the page title which is used in this case) 4042 + *### 4043 +#macro(displayBlog $entries $displaying $onlyExtract $shouldDisplayTitles) 4044 + #set($blogDay = '') 4045 + (% class=~"hfeed $!\{displaying}~" ~%)((( 4046 + (% class=~"blogDay~" ~%)((( 4047 + #foreach ($entryDoc in $xwiki.wrapDocs($entries)) 4048 + #getEntryObject($entryDoc $entryObj) 4049 + ## Although all entries should have one of the two objects, better check to be sure. 4050 + #if(~"$!\{entryObj}~" != '') 4051 + #getEntryDate($entryDoc $entryObj $entryDate) 4052 + ## Display a ~"calendar sheet~" for each day. All entries posted on the same day share one such sheet. 4053 + #set($entryDateStr = $xwiki.formatDate($entryDate, 'yyyyMMMMdd')) 4054 + #if($blogDay != $entryDateStr) 4055 + #if($blogDay != '') 4056 + ))) 4057 + (% class=~"blogDay~" ~%)((( 4058 + #end 4059 + #displayBlogDate($entryDate) 4060 + #set ($blogDay = $entryDateStr) 4061 + #end 4062 + ## Finally, display the entry. 4063 + #displayEntry($entryDoc $entryObj $onlyExtract $shouldDisplayTitles true) 4064 + #end 4065 + #end 4066 + )))## blogDay 4067 + )))## hfeed 4068 +#end 4069 +## 4070 +## 4071 +## 4072 +#** 4073 + * Get the entry object, either a new BlogPost or an old Article. 4074 + * 4075 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 4076 + * @param entryObj The resulting xobject of the blog post. 4077 + *### 4078 +#macro(getEntryObject $entryDoc $__entryObj) 4079 + #set($result = $entryDoc.getObject(~"$\{blogPostClassname}~")) 4080 + #if(!$result) 4081 + #set($result = $entryDoc.getObject(~"$\{oldArticleClassname}~")) 4082 + #end 4083 + ## NOTE: The reason we put an underscore in front of the variable name is to prevent the following line from 4084 + ## overwriting the $entryObj variable that may be defined before this macro is called. Of course, $__entryObj may be 4085 + ## overwritten in this case but it's less likely to have such a variable defined before. 4086 + #set ($__entryObj = $NULL) 4087 + #setVariable (~"$__entryObj~" $result) 4088 +#end 4089 +## 4090 +## 4091 +## 4092 +#** 4093 + * Gets the date associated with a blog entry. This is the publication date. For unpublished entries, initially this is 4094 + * the document creation date, but can be edited by the user. 4095 + * 4096 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 4097 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 4098 + * @param result The resulting date, an instance of <tt>java.util.Date</tt>. 4099 + *### 4100 +#macro(getEntryDate $entryDoc $entryObj $result) 4101 + #set ($result = $NULL) 4102 + #setVariable (~"$result~" $entryObj.getProperty('publishDate').value) 4103 +#end 4104 +## 4105 +## 4106 +## 4107 +#** 4108 + * Displays a date, nicely formatted as a calendar page. 4109 + * 4110 + * @param date The date to display, an instance of <tt>java.util.Date</tt>. 4111 + *### 4112 +#macro(displayBlogDate $date) 4113 + #set($year = $xwiki.formatDate($date, 'yyyy')) 4114 + ## 3 letter month name, like Jan, Dec. 4115 + #set($month = $xwiki.formatDate($date, 'MMM')) 4116 + ## Uncomment to get a full length month name, like January, December. 4117 + ## TODO: this could be defined somewhere in the blog style. 4118 + ## #set($month = $xwiki.formatDate($date, 'MMMM')) 4119 + #set($day = $xwiki.formatDate($date, 'dd')) 4120 + (% class=~"blogdate~" ~%) 4121 + == (% class=~"month~" ~%)$month(%~%) (% class=~"day~" ~%)$day(%~%) (% class=~"year~" ~%)$year(%~%) == 4122 +#end 4123 +## 4124 +## 4125 +## 4126 +#** 4127 + * Displays a blog article: management tools, header, content, footer. 4128 + * 4129 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 4130 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 4131 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 4132 + * @param shouldDisplayTitle If <tt>true</tt>, display the blog title (blog posts shouldn't display the title 4133 + * when they're displayed alone on their page since it's the page title which is used in this case) 4134 + * @param shouldDisplayActions If <tt>true</tt>, display the blog post actions buttons 4135 + *### 4136 +#macro(displayEntry $entryDoc $entryObj $onlyExtract $shouldDisplayTitle $shouldDisplayActions) 4137 + ## Only articles with an explicit hidden setting or an explicit unpublished setting are hidden 4138 + #isPublished($entryObj $isPublished) 4139 + #isHidden($entryObj $isHidden) 4140 + #if($doc.fullName == $entryDoc.fullName) 4141 + (% class=~"hentry single-article~" ~%)((( 4142 + #else 4143 + (% class=~"hentry#if(!$isPublished) unpublished-article#elseif($isHidden) hidden-article#end~" ~%)((( 4144 + #end 4145 + #if ($shouldDisplayActions) 4146 + #displayEntryTools($entryDoc $entryObj) 4147 + #end 4148 + #if($shouldDisplayTitle) 4149 + #displayEntryTitle($entryDoc $entryObj) 4150 + #end 4151 + #if($doc.fullName == $entryDoc.fullName) 4152 + #if(!$isPublished) 4153 + \{\{warning}}\{\{translation key='blog.code.published'/}}\{\{/warning}} 4154 + #elseif($isHidden) 4155 + \{\{warning}}\{\{translation key='blog.code.hidden'/}}\{\{/warning}} 4156 + #end 4157 + #end 4158 + #displayEntryContent($entryDoc $entryObj $onlyExtract) 4159 + #displayEntryFooter($entryDoc $entryObj) 4160 + )))## hentry 4161 +#end 4162 +## 4163 +## 4164 +## 4165 +#** 4166 + * Checks if the provided blog is published or not. 4167 + * 4168 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 4169 + * @param isPublished The resulting boolean, true if the entry is considered published. 4170 + *### 4171 +#macro(isPublished $entryObj $isPublished) 4172 + #set ($isPublished = $NULL) 4173 + ## This should work for both old articles, which don't have the 'published' property at all, and 4174 + ## are considered published by default, and new entries, that should have 1 if published. 4175 + #if (~"$!\{entryObj.getProperty('published').value}~" != '0') 4176 + #setVariable (~"$isPublished~" true) 4177 + #else 4178 + #setVariable (~"$isPublished~" false) 4179 + #end 4180 +#end 4181 +## 4182 +## 4183 +## 4184 +#** 4185 + * Checks if the provided blog is hidden or not. 4186 + * 4187 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass<tt> xclass. 4188 + * @param isHiddel The resulting boolean, true if the entry is considered hidden. 4189 + *### 4190 +#macro(isHidden $entryObj $isHidden) 4191 + #set ($isHidden = $NULL) 4192 + ## This should work for both old articles, which don't have the 'hidden' property at all, and 4193 + ## are considered visible by default, and new entries, that should have 1 if hidden. 4194 + #if (~"$!\{entryObj.getProperty('hidden').value}~" == '1') 4195 + #setVariable (~"$isHidden~" true) 4196 + #else 4197 + #setVariable (~"$isHidden~" false) 4198 + #end 4199 +#end 4200 +## 4201 +## 4202 +## 4203 +#** 4204 + * Displays several ~"tools~" for manipulating blog posts: hide/show, publish, edit. 4205 + * 4206 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 4207 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 4208 + *### 4209 +#macro(displayEntryTools $entryDoc $entryObj) 4210 + #if($xcontext.action == 'view') 4211 + (% class=~"blog-entry-toolbox~" ~%)((( 4212 + #displayPublishButton($entryDoc $entryObj) 4213 + #displayHideShowButton($entryDoc $entryObj) 4214 + #displayEditButton($entryDoc $entryObj) 4215 + #displayDeleteButton($entryDoc $entryObj) 4216 + ))) 4217 + #end 4218 +#end 4219 +## 4220 +## 4221 +## 4222 +#** 4223 + * Displays the publish button to the entry <strong>creator</strong>, if the article is not published yet. 4224 + * 4225 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 4226 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 4227 + * @todo AJAX calls. 4228 + *### 4229 +#macro(displayPublishButton $entryDoc $entryObj) 4230 + #isPublished($entryObj $isPublished) 4231 + #if(!$isPublished && $entryDoc.creator == $xcontext.user && $xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 4232 + [[#toolImage('world')>>path:$blogPublisher.getURL('view', ~"entryName=$\{escapetool.url($entryDoc.fullName)}&xredirect=$\{escapetool.url($thisURL)}&form_token=$!\{services.csrf.getToken()}~")||title=~"$services.localization.render('blog.code.notpublished')~"]]## 4233 + #end 4234 +#end 4235 +## 4236 +## 4237 +## 4238 +#** 4239 + * Displays the hide or show button to the entry <strong>creator</strong>, if the article is already published. 4240 + * 4241 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 4242 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 4243 + *### 4244 +#macro(displayHideShowButton $entryDoc $entryObj) 4245 + #isPublished($entryObj $isPublished) 4246 + #isHidden($entryObj $isHidden) 4247 + ## Only published articles can be hidden. Unpublished articles are considered already hidden. 4248 + #if($isPublished && $entryDoc.creator == $xcontext.user && $xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 4249 + #set ($queryString = \{ 4250 + 'xredirect' : $thisURL, 4251 + 'form_token' : $services.csrf.getToken() 4252 + }) 4253 + #if ($isHidden) 4254 + #set ($discard = $queryString.putAll(\{ 4255 + ~"$\{entryObj.getxWikiClass().getName()}_$\{entryObj.number}_hidden~" : 0, 4256 + 'comment' : $services.localization.render('blog.code.madevisible') 4257 + })) 4258 + #set ($lockURL = $entryDoc.getURL('save', $escapetool.url($queryString))) 4259 + [[#toolImage('unlock')>>path:$lockURL||class=~"blog-tool-show~" title=~"$services.localization.render('blog.code.makevisible')~"]]## 4260 + #else 4261 + #set ($discard = $queryString.putAll(\{ 4262 + ~"$\{entryObj.getxWikiClass().getName()}_$\{entryObj.number}_hidden~" : 1, 4263 + 'comment' : $services.localization.render('blog.code.hid') 4264 + })) 4265 + #set ($lockURL = $entryDoc.getURL('save', $escapetool.url($queryString))) 4266 + [[#toolImage('lock')>>path:$lockURL||class=~"blog-tool-hide~" title=~"$services.localization.render('blog.code.hide')~"]]## 4267 + #end 4268 + #end 4269 +#end 4270 +## 4271 +## 4272 +## 4273 +#** 4274 + * Displays the edit button to those that can edit the article. 4275 + * 4276 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 4277 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 4278 + *### 4279 +#macro(displayEditButton $entryDoc $entryObj) 4280 + #if($xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 4281 + ## Call getDefaultEditMode() for backward compatibility with older blog posts. 4282 + [[#toolImage('pencil')>>path:$entryDoc.getURL('edit')||title=~"$services.localization.render('blog.code.editpost')~"]]## 4283 + #end 4284 +#end 4285 +## 4286 +## 4287 +## 4288 +#** 4289 + * Displays the delete button to those that can edit the article. 4290 + * 4291 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 4292 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 4293 + * @todo AJAX calls. 4294 + *### 4295 +#macro(displayDeleteButton $entryDoc $entryObj) 4296 + #if($xwiki.hasAccessLevel('delete', $xcontext.user, $entryDoc.fullName)) 4297 + [[#toolImage('cross')>>path:$entryDoc.getURL('delete')||title=~"$services.localization.render('blog.code.deletepost')~"]]## 4298 + #end 4299 +#end 4300 +## 4301 +## 4302 +## 4303 +#** 4304 + * Displays the title of the entry. 4305 + * 4306 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 4307 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 4308 + *### 4309 +#macro(displayEntryTitle $entryDoc $entryObj) 4310 + #if($doc.fullName == $entryDoc.fullName) 4311 + (% class=~"entry-title~" ~%) 4312 + = $services.rendering.escape($entryDoc.display('title', 'view', $entryObj), $xwiki.getCurrentContentSyntaxId()) = 4313 + #else 4314 + (% class=~"entry-title~" ~%) 4315 + === [[$services.rendering.escape($entryDoc.display('title', 'view', $entryObj),$xwiki.getCurrentContentSyntaxId())>>doc:$services.rendering.escape($services.model.serialize($entryDoc.getDocumentReference(),'default'),$xwiki.getCurrentContentSyntaxId())]] === 4316 + #end 4317 +#end 4318 +## 4319 +## 4320 +## 4321 +#** 4322 + * Displays the body of the entry. 4323 + * 4324 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 4325 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 4326 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 4327 + *### 4328 +#macro(displayEntryContent $entryDoc $entryObj $onlyExtract) 4329 + (% class=~"#if($onlyExtract)entry-summary#\{else}entry-content#end~" ~%)((( 4330 + #getEntryContent($entryDoc $entryObj $onlyExtract $entryContent) 4331 + ## FIXME: This causes the blog's content to not be annotatable. See http://jira.xwiki.org/browse/XWIKI-6328 4332 + ## Should probably be replaced by a display macro call with a reference to the object property holding the post's content 4333 + \{\{html wiki=~"false~"}}$entryDoc.getRenderedContent($entryContent, $entryDoc.syntax.toIdString())\{\{/html}} 4334 + ))) ## entry-content 4335 + (% class=~"clearfloats~" ~%)((())) 4336 +#end 4337 +## 4338 +## 4339 +## 4340 +#** 4341 + * Extracts the body of the entry that should be displayed. If <tt>onlyExtract</tt> is <tt>true</tt>, display the content 4342 + * of the <tt>extract</tt> field (if not empty). 4343 + * 4344 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 4345 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 4346 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 4347 + * @param entryContent The resulting content. 4348 + * @param removeEllipsis If <tt>true</tt>, then don't display an ellipsis at the end of the content (only used when 4349 + * <tt>onlyExtract</tt> is <tt>true</tt>) 4350 + *### 4351 +#macro(getEntryContent $entryDoc $entryObj $onlyExtract $entryContent $removeEllipsis) 4352 + #if ($onlyExtract) 4353 + ## Note: We trim the summary so that if there's some white space it won't cause the summary to be used instead 4354 + ## of the content. 4355 + #set ($macro.result = $entryObj.getProperty('extract').value.trim()) 4356 + #end 4357 + #if(~"$!macro.result~" == '') 4358 + #set($macro.result = $entryObj.getProperty('content').value) 4359 +#* Disabled until the content can be cleanly cut. 4360 +* #if($onlyExtract && $result.length()>$maxchars) 4361 +* #set($i = $macro.result.lastIndexOf(~" ~", $maxchars)) 4362 +* #set($i = $i + 1) 4363 +* #set($macro.result = ~"$\{macro.result.substring(0,$i)} *[...>$\{entryDoc.fullName}]*~") 4364 +* #end 4365 +## *### 4366 + #elseif (!$removeEllipsis) 4367 + #if($entryDoc.syntax.toIdString() == 'xwiki/1.0') 4368 + #set($macro.result = ~"$\{macro.result} <a href='$\{entryDoc.getExternalURL()}' title='$services.localization.render('blog.code.readpost')'>...</a>~") 4369 + #elseif($entryDoc.syntax.toIdString() == 'xwiki/2.0' || $entryDoc.syntax.toIdString() == 'xwiki/2.1') 4370 + #set($macro.result = ~"$\{macro.result} [[...>>$\{services.rendering.escape($entryDoc.getFullName(), $entryDoc.getSyntax())}||title='$services.localization.render('blog.code.readpost')']]~") 4371 + #end 4372 + #end 4373 + #set ($entryContent = $NULL) 4374 + #setVariable (~"$entryContent~" $macro.result) 4375 +#end 4376 +## 4377 +## 4378 +## 4379 +#** 4380 + * Displays the footer of the entry. 4381 + * 4382 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 4383 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 4384 + *### 4385 +#macro(displayEntryFooter $entryDoc $entryObj) 4386 + (% class=~"entry-footer~" ~%)((( 4387 + #isPublished($entryObj $isPublished) 4388 + (% class='entry-author-label' ~%) 4389 + #if($isPublished) 4390 + \{\{translation key='blog.code.postedby'/}} ## 4391 + #else 4392 + \{\{translation key='blog.code.createdby'/}} ## 4393 + #end 4394 + \{\{html wiki=~"false~" clean=~"false~"}}<span class='author vcard'>#userfn($entryDoc.creator)</span>\{\{/html}} ## 4395 + #getEntryDate($entryDoc $entryObj $entryDate) 4396 + #listCategories($entryObj) #* 4397 + ## Since the publish date and update date are not set at the exact same time, there could be a small difference that 4398 + ## we assume cannot be more than 3 seconds. 4399 + *#(% class=~"separator~" ~%)·(%~%) [[\{\{translation key='blog.code.permalink'/}}>>$services.rendering.escape($entryDoc,'xwiki/2.0')||rel=~"bookmark~"]] ## 4400 + #if ($showcomments) 4401 + (% class=~"separator~" ~%)·(%~%) [[\{\{translation key='blog.code.comments'/}}>>$services.rendering.escape($entryDoc,'xwiki/2.0')||anchor=~"Comments~"]] (% class=~"itemCount~" ~%)($entryDoc.comments.size())(%~%) ## 4402 + #end ## 4403 + #if($entryDoc != $doc) ## 4404 + #displayEntryBlogLocation($entryDoc $entryObj) ## 4405 + #end 4406 + )))## entry-footer 4407 +#end 4408 +## 4409 +## 4410 +#** 4411 + * Display the blog for the entry (if it is not the currently displayed blog) 4412 + * 4413 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 4414 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 4415 + *### 4416 +#macro(displayEntryBlogLocation $entryDoc $entryObj) 4417 + #getBlogPostsLocation($entryDoc.getSpace() $blogPostsLocation) ## 4418 + #if(~"$!blogPostsLocation~" != ~"~") ## 4419 + #set($blogSpaceRef = $services.model.resolveSpace($blogPostsLocation)) ## 4420 + #set($blogDocRef = $services.model.createDocumentReference($services.model.getEntityReference('DOCUMENT', 'default').getName(), $blogSpaceRef)) ## 4421 + #if($doc.documentReference != $blogDocRef) ## 4422 + #set($blogDoc = $xwiki.getDocument($blogDocRef)) ## 4423 + #set($blogObj = $blogDoc.getObject('Blog.BlogClass')) ## 4424 + #if($blogObj) (% class=~"blogrefpath~" ~%)((( ## 4425 + $services.localization.render('blog.code.postedin') \{\{html clean=~"false~" wiki=~"false~"}} #hierarchy($blogDocRef, \{'selfIsActive' : true, 'local': true}) \{\{/html}} ## 4426 + )))(%~%)## 4427 + #end 4428 + #end 4429 + #end 4430 +#end 4431 +## 4432 +## 4433 +## 4434 +## 4435 +#** 4436 + * List the categories an entry belongs to. Used in the footer. The categories are instances of <tt>Blog.CategoryClass</tt>. 4437 + * 4438 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 4439 + *### 4440 +#macro(listCategories $entryObj) 4441 + #if($entryObj.getxWikiClass().getName() == $blogPostClassname) 4442 + #set($categories = $entryObj.getProperty('category').value) 4443 + #set($first = true) 4444 + #if($categories.size() > 0) 4445 + #foreach($category in $categories) 4446 + #set($categoryDoc = $!xwiki.getDocument($category)) 4447 + #if(!$categoryDoc.isNew() && $categoryDoc.getObject($\{blogCategoryClassname})) 4448 + #if($foreach.count == 1) 4449 + (% class='separator' ~%)·(%~%) $services.localization.render('blog.code.categories') ## 4450 + #else 4451 + , ## 4452 + #end## 4453 + [[$!\{categoryDoc.getObject($blogCategoryClassname).getValue('name')}>>$\{category}||rel='tag']]## 4454 + #end## 4455 + #end## 4456 + #end 4457 + #end 4458 +#end 4459 +## 4460 +## 4461 +## 4462 +#** 4463 + * Displays blog pagination links (older and newer entries). 4464 + * 4465 + * @param blogDoc the XDocument holding the blog definition object. 4466 + *### 4467 +#macro(displayNavigationLinks $blogDoc) 4468 + (% class=~"clearfloats~" ~%)((())) 4469 + #getBlogDisplayType($blogDoc $displayType) 4470 + #if($displayType == 'weekly') 4471 + (% class=~"pagingLinks~" ~%)((( 4472 + #getRequestedWeek($weekDate) 4473 + $weekDate.addWeeks(-1)## 4474 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.previousweek'/}}>>$doc.documentReference.name||queryString=~"year=$weekDate.weekyear&week=$weekDate.weekOfWeekyear~" class=~"button~"]]**(%~%) 4475 + #sep() 4476 + $weekDate.addWeeks(2)## 2 because we already subtracted 1 above 4477 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.nextweek'/}} »>>$doc.documentReference.name||queryString=~"year=$weekDate.weekyear&week=$weekDate.weekOfWeekyear~" class=~"button~"]]**(%~%) 4478 + ))) 4479 + #elseif($displayType == 'monthly') 4480 + (% class=~"pagingLinks~" ~%)((( 4481 + #getRequestedMonth($monthDate) 4482 + $monthDate.addMonths(-1)## 4483 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.previousmonth'/}}>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"year=$monthDate.year&month=$monthDate.monthOfYear~" class=~"button~"]]**(%~%) 4484 + #sep() 4485 + $monthDate.addMonths(2)## 2 because we already subtracted 1 above 4486 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.nextmonth'/}} »>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"year=$monthDate.year&month=$monthDate.monthOfYear~" class=~"button~"]]**(%~%) 4487 + ))) 4488 + #elseif($displayType == 'all') 4489 + #else 4490 + ## Paginated 4491 + #if(($totalPages > 1)) 4492 + #set($queryString = '') 4493 + #foreach($p in $request.getParameterNames()) 4494 + #if($p != 'page' && $p != 'ipp') 4495 + #foreach($v in $request.getParameterValues($p)) 4496 + #set($queryString = ~"$\{queryString}&$\{escapetool.url($p)}=$\{escapetool.url($v)}~") 4497 + #end 4498 + #end 4499 + #end 4500 + (% class=~"pagingLinks~" ~%)((( 4501 + #if ($currentPageNumber < $totalPages) 4502 + #set($currentPageNumber = $currentPageNumber + 1) 4503 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.olderposts'/}}>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"page=$\{currentPageNumber}&ipp=$\{itemsPerPage}$queryString~" class=~"button~"]]**(%~%) 4504 + #set($currentPageNumber = $currentPageNumber - 1) 4505 + #end 4506 + #if ($currentPageNumber > 1) 4507 + #if ($currentPageNumber < $totalPages) 4508 + #sep() 4509 + #end 4510 + #set($currentPageNumber = $currentPageNumber - 1) 4511 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.newerposts'/}} »>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"page=$\{currentPageNumber}&ipp=$\{itemsPerPage}$queryString~" class=~"button~"]]**(%~%) 4512 + #set($currentPageNumber = $currentPageNumber + 1) 4513 + #end 4514 + (% class=~"clear~" ~%)(%~%) 4515 + )))## pagingLinks 4516 + #end 4517 + #end 4518 +#end 4519 +## 4520 +## 4521 +## 4522 +#** 4523 + * Displays a message box with ~"publish~" icon. 4524 + * 4525 + * @param message A text message concerning blog article publishing 4526 + *### 4527 +#macro(publishMessageBox $message) 4528 +(% class=~"plainmessage publish-message~" ~%)((($services.icon.render('world') $message))) 4529 +#end 4530 +#** 4531 + * Displays a message box with ~"show/hide~" icon. 4532 + * 4533 + * @param message A text message concerning blog article hiding 4534 + *### 4535 +#macro(hideMessageBox $message) 4536 +(% class=~"plainmessage hide-message~" ~%)((($services.icon.render('unlock') $message))) 4537 +#end 4538 +## 4539 +## 4540 +## 4541 +#** 4542 + * Determine the requested week, for using in a weekly-indexed blog. The relevant request parameters are 4543 + * <tt>year</tt> and <tt>week</tt>. By default, the current week is used. 4544 + * 4545 + * @param monthDate The resulting week, a JODATime MutableDateTime. 4546 + *### 4547 +#macro(getRequestedWeek $weekDate) 4548 + #set ($weekDate = $NULL) 4549 + #setVariable (~"$weekDate~" $xwiki.jodatime.mutableDateTime) 4550 + #if(~"$!\{request.year}~" != '') 4551 + #set ($discard = $weekDate.setYear($numbertool.toNumber($request.year).intValue())) 4552 + #end 4553 + #if(~"$!\{request.week}~" != '') 4554 + #set ($discard = $weekDate.setWeekOfWeekyear($numbertool.toNumber($request.week).intValue())) 4555 + #end 4556 +#end 4557 +## 4558 +## 4559 +## 4560 +#** 4561 + * Determine the requested month, for using in a monthly-indexed blog. The relevant request parameters are 4562 + * <tt>year</tt> and <tt>month</tt>. By default, the current month is used. 4563 + * 4564 + * @param monthDate The resulting month, a JODATime MutableDateTime. 4565 + *### 4566 +#macro(getRequestedMonth $monthDate) 4567 + #set ($monthDate = $NULL) 4568 + #setVariable (~"$monthDate~" $xwiki.jodatime.mutableDateTime) 4569 + #if(~"$!\{request.year}~" != '') 4570 + #set ($discard = $monthDate.setYear($numbertool.toNumber($request.year).intValue())) 4571 + #end 4572 + #if(~"$!\{request.month}~" != '') 4573 + #set ($discard = $monthDate.setMonthOfYear($numbertool.toNumber($request.month).intValue())) 4574 + #end 4575 +#end 4576 +## 4577 +## 4578 +## 4579 +#** 4580 + * Retrieve a blog property (title, display type, etc). 4581 + * 4582 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 4583 + * @param propertyName The name of the property to be retrieved. One of the <tt>Blog.BlogClass</tt>'s properties. 4584 + * @param defaultValue The default value to use in case the blog object does not define one. 4585 + * @param propertyValue The resulting value. 4586 + *### 4587 +#macro(getBlogProperty $blogDoc $propertyName $defaultValue $propertyValue) 4588 + #set($result = ~"$!\{blogDoc.getObject($\{blogClassname}).getProperty($propertyName).value}~") 4589 + #if($result == '') 4590 + #set($result = $defaultValue) 4591 + #end 4592 + #set ($propertyValue = $NULL) 4593 + #setVariable (~"$propertyValue~" $result) 4594 +#end 4595 + 4596 +#** 4597 + * If an error occurs when executing an action, set a specific response status and display an error message. 4598 + * 4599 + * @param status The response status. 4600 + * @param text The user readable error to be displayed. Can be a translation key. 4601 + * @param parameters The parameters to use when decoding the translation key. 4602 + *### 4603 +#macro(blog__actionResponseError $status $text $parameters) 4604 + $response.setStatus($status) 4605 + #if($request.ajax) 4606 + $services.localization.render($text, $!parameters) 4607 + #else 4608 + \{\{error}}$services.localization.render($text, $!parameters)\{\{/error}} 4609 + #end 4610 +#end 4611 +## 4612 +## 4613 +## 4614 +#** 4615 + * Check if a blog is the Default blog (The one in the 'Blog' space). 4616 + * 4617 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 4618 + * @param isDefault The resulting boolean. 4619 + *### 4620 +#macro(isDefaultBlog $blogDoc $isDefault) 4621 + #set ($result = false) 4622 + #if ($blogDoc.space == 'Blog') 4623 + #set ($result = true) 4624 + #end 4625 + #setVariable(~"$isDefault~" $result) 4626 +#end 4627 +## 4628 +## 4629 +## 4630 +#** 4631 + * Retrieve the blog posts location (space). 4632 + * 4633 + * @param blogSpace The blog space. It should contain a document with a <tt>Blog.BlogClass</tt> object 4634 + * @param postsLocation The resulting location. 4635 + *### 4636 +#macro(getBlogPostsLocation $blogSpace $postsLocation) 4637 + #getBlogDocument($blogSpace $blogDoc) 4638 + #getBlogProperty($blogDoc 'postsLocation' $blogDoc.space $result) 4639 + #set ($postsLocation = $NULL) 4640 + #setVariable (~"$postsLocation~" $result) 4641 +#end 4642 +## 4643 +## 4644 +## 4645 +#** 4646 + * Retrieve the blog categories location (space). 4647 + * 4648 + * @param blogSpace The blog space. It should contain a document with a <tt>Blog.BlogClass</tt> object 4649 + * @param categoriesLocation The resulting location. 4650 + *### 4651 +#macro(getBlogCategoriesLocation $blogSpace $categoriesLocation) 4652 + #getBlogDocument($blogSpace $blogDoc) 4653 + #getBlogProperty($blogDoc 'categoriesLocation' 'Blog' $result) 4654 + #set ($postsLocation = $NULL) 4655 + #setVariable (~"$categoriesLocation~" $result) 4656 +#end 4657 +###** 4658 + * Return the Query for selecting the blog posts based on the context where the posts are displayed. 4659 + * for example there is 4 different panel contexts: 4660 + * aBlog.aPost or aBlog.WebHome 4661 + * aCategorySpace.aCategory 4662 + * aCategorySpace.WebHome 4663 + * Blog.aPost or Blog.WebHome 4664 + * 4665 + * @param query The query for selecting blog entries. 4666 + * @param queryParams The parameters to bind with the generated query. 4667 + * @param targetDoc The document in which the articles will be displayed. 4668 + *### 4669 +#macro(getAllBlogPostsQueryBasedOnDisplayContext $targetDoc $query $queryParams) 4670 + #set ($query = $NULL) 4671 + #set ($queryParams = $NULL) 4672 + #getCategoryAllBlogPostsQuery($resultQuery) 4673 + #set ($resultQueryParams = \{}) 4674 + #set($sanitizedSpace = $targetDoc.space.replaceAll('([%_!])', '!$1').concat('.%')) 4675 + #if ($targetDoc.space == $defaultBlogSpace && !$targetDoc.getObject($blogCategoryClassname)) 4676 + ## Get all posts that are in the default blog space('Blog') or in category 'Blog.%' 4677 + #set ($discard = $resultQueryParams.put('space', $targetDoc.space)) 4678 + #set ($discard = $resultQueryParams.put('catList', $sanitizedSpace)) 4679 + #set($resultQuery = ~"$\{resultQuery} and (doc.space = :space or catList like :catList escape '!')~") 4680 + #elseif($targetDoc.getObject($blogCategoryClassname)) 4681 + ## Get all posts that are in a category aCategorySpace.aCategory 4682 + #set ($discard = $resultQueryParams.put('catList', $targetDoc.fullName)) 4683 + #set($resultQuery = ~"$\{resultQuery} and (catList=:catList)~") 4684 + #elseif($targetDoc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 4685 + ## Get all posts that are in a category aCategorySpace.% 4686 + #set ($discard = $resultQueryParams.put('catList', $sanitizedSpace)) 4687 + ## Exclude incategorized posts 4688 + #set ($excludedCategory = ~"$\{targetDoc.space}.WebHome~") 4689 + #if ($targetDoc.space == $defaultBlogSpace) 4690 + #set ($excludedCategory = ~"Blog.Categories.WebHome~") 4691 + #end 4692 + #set ($discard = $resultQueryParams.put('excludedCategory', $excludedCategory)) 4693 + #set($resultQuery = ~"$\{resultQuery} and (catList<>:excludedCategory) and (catList like :catList escape '!')~") 4694 + #else 4695 + ## Get all posts in blog space aBlog 4696 + #getAllBlogPostsQuery($resultQuery) 4697 + #getBlogPostsLocation($targetDoc.space $postsLocation) 4698 + #set ($discard = $resultQueryParams.put('space', $postsLocation)) 4699 + #set($resultQuery = ~"$\{resultQuery} and doc.space = :space~") 4700 + #end 4701 + #setVariable(~"$query~" $resultQuery) 4702 + #setVariable(~"$queryParams~" $resultQueryParams) 4703 +#end 4704 +## 4705 +## 4706 +## 4707 +###** 4708 + * Display blog posts based on the context where the posts are displayed. 4709 + * for example there is 4 different panel contexts: 4710 + * aBlog.aPost or aBlog.WebHome 4711 + * aCategorySpace.aCategory 4712 + * aCategorySpace.WebHome 4713 + * Blog.aPost or Blog.WebHome 4714 + * 4715 + * @param targetDoc The document in which the articles will be displayed. 4716 + * @param postsVisiblity The visibility of the posts to display: recent, unpublished, ... 4717 + * @param layout Layout of the the posts to display 4718 + * @param layoutParams additional layout parameters, it is a string like 'param1=val1|...|paramN=valueN' 4719 + * @param limit the number of posts to display 4720 + *### 4721 +#macro(displayBlogPostsBasedOnContext $targetDoc $postsVisiblity $layout $layoutParams $limit) 4722 + #if ($postLayout == 'full') 4723 + #set ($macro.paginated = 'yes') 4724 + #end 4725 + #if ($targetDoc.space == $defaultBlogSpace && !$targetDoc.getObject($blogCategoryClassname)) 4726 + ## Display all posts that are in the default blog space('Blog') or in category 'Blog.%' 4727 + #getBlogPostsLayout($xwiki.getDocument(~"$\{defaultBlogSpace}.WebHome~") $postsLayout) 4728 + #if (~"$!layout~" == '') 4729 + #set ($layout = $postsLayout) 4730 + #end 4731 + #if ($postsVisiblity == 'recent') 4732 + \{\{blogpostlist blog=~"$\{defaultBlogSpace.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')}.WebHome~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 4733 + #elseif($postsVisiblity == 'unpublished') 4734 + \{\{blogpostlist blog=~"$\{defaultBlogSpace.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')}.WebHome~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 4735 + #end 4736 + #elseif($targetDoc.getObject($blogCategoryClassname)) 4737 + ## Display all posts that are in a category aCategorySpace.aCategory 4738 + #getBlogDocumentForCategoriesSpace($targetDoc.space $blogDoc) 4739 + #getBlogPostsLayout($blogDoc $postsLayout) 4740 + #if (~"$!layout~" == '') 4741 + #set ($layout = $postsLayout) 4742 + #end 4743 + #if ($postsVisiblity == 'recent') 4744 + \{\{blogpostlist category=~"$targetDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 4745 + #elseif($postsVisiblity == 'unpublished') 4746 + \{\{blogpostlist category=~"$targetDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 4747 + #end 4748 + #elseif($targetDoc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 4749 + ## Display all posts that are in a category aCategorySpace.% 4750 + #getBlogDocumentForCategoriesSpace($targetDoc.space $blogDoc) 4751 + #getBlogPostsLayout($blogDoc $postsLayout) 4752 + #if (~"$!layout~" == '') 4753 + #set ($layout = $postsLayout) 4754 + #end 4755 + #if ($postsVisiblity == 'recent') 4756 + \{\{blogpostlist category=~"$targetDoc.space.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 4757 + #elseif($postsVisiblity == 'unpublished') 4758 + \{\{blogpostlist category=~"$targetDoc.space.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 4759 + #end 4760 + #else 4761 + ## Display all posts in blog space aBlog 4762 + #getBlogDocument($targetDoc.space $blogDoc) 4763 + #getBlogPostsLayout($blogDoc $postsLayout) 4764 + #if (~"$!layout~" == '') 4765 + #set ($layout = $postsLayout) 4766 + #end 4767 + #if ($postsVisiblity == 'recent') 4768 + \{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 4769 + #elseif($postsVisiblity == 'unpublished') 4770 + \{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 4771 + #end 4772 + #end 4773 +#end 4774 +## 4775 +## 4776 +## 4777 +#** 4778 + * Bind parameters to a query object. 4779 + * 4780 + * @param queryObj The query object (org.xwiki.query.internal.ScriptQuery object) 4781 + * @param queryParams the query parameters. 4782 + *### 4783 +#macro(bindQueryParameters $queryObj $queryParams) 4784 + #set ($output = $queryObj) 4785 + #foreach( $key in $queryParams.keySet() ) 4786 + #set($output = $queryObj.bindValue($key, $queryParams.get($key))) 4787 + #end 4788 + #setVariable(~"$queryObj~" $output) 4789 +#end 4790 +## 4791 +## 4792 +## 4793 +#** 4794 + * Determines the blogposts layout: Default layout (Calendar), Image thumbnail 4795 + * 4796 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>postsLayout</tt> 4797 + * property set. 4798 + * @param postsLayout The resulting string. If the blog object does not define anything, it is considered full (default). 4799 + *### 4800 +#macro(getBlogPostsLayout $blogDoc $postsLayout) 4801 + #getBlogProperty($blogDoc 'postsLayout' 'full' $res) 4802 + #set ($postsLayout = $NULL) 4803 + #setVariable (~"$postsLayout~" $res) 4804 +#end 4805 +## 4806 +## 4807 +## 4808 +#** 4809 + * Retrieve the blog document of a categories space, which is the blog descriptor that have its categoriesLocation property set to the categories space 4810 + * 4811 + * @param categoriesSpace A <tt>String</tt>, the name of the space. 4812 + * @param blogDoc The resulting XDocument. 4813 + *### 4814 +#macro(getBlogDocumentForCategoriesSpace $categoriesSpace $blogDoc) 4815 + #set($macro.blogDocs = $services.query.hql(~", BaseObject obj, StringProperty as categoriesLocationProp where obj.name = doc.fullName and obj.className = '$blogClassname' and obj.id=categoriesLocationProp.id.id and categoriesLocationProp.id.name='categoriesLocation' and categoriesLocationProp.value=:categoriesLocation order by doc.creationDate~").setLimit(1).setOffset(0).bindValue('categoriesLocation', $categoriesSpace).execute()) 4816 + #if($macro.blogDocs.size() > 0 && $categoriesSpace != $defaultBlogSpace) 4817 + #set($macro.result = $xwiki.getDocument($macro.blogDocs.get(0))) 4818 + #else 4819 + ## Fallback to Blog.WebHome, the default blog 4820 + #set($macro.result = $xwiki.getDocument('Blog.WebHome')) 4821 + #end 4822 + #set ($blogDoc = $NULL) 4823 + #setVariable (~"$blogDoc~" $macro.result) 4824 +#end" %) 4825 +((( 4826 +(% class="macro-placeholder hidden" %) 4827 +((( 4828 +macro:velocity 4829 +))) 4830 +))) 4831 +))) 4832 + 4833 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogPostLayoutMacros~"" %) 4834 +((( 4835 +(% class="macro-placeholder hidden" %) 4836 +((( 4837 +macro:include 4838 +))) 4839 + 4840 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|#** 4841 + * Extract the layout parameters from a string. 4842 + * 4843 + * @param layoutParamsString The string representation of the layout parameters. 4844 + * It should contain a String following this format ~"paramName1=Value1|paramName2=Value2|...|paramNameK=ValueK~" 4845 + * @param layoutsParameters The resulting layout parameters Map. 4846 + *### 4847 +#macro(extractLayoutParametersFromString $layoutParamsString $layoutsParameters) 4848 + #set ($layoutsParameters = $NULL) 4849 + #set ($macro.layoutParams = \{}) 4850 + #if (~"$!layoutParamsString~" != '') 4851 + #set ($macro.paramsArr = $layoutParamsString.split('\\|')) 4852 + #foreach ($item in $macro.paramsArr) 4853 + #set ($itemSplit = $item.split('=')) 4854 + #if ($itemSplit.size() == 2) 4855 + #set ($discard = $macro.layoutParams.put($itemSplit[0].trim(), $itemSplit[1].trim())) 4856 + #end 4857 + #end 4858 + #end 4859 + #setVariable(~"$layoutsParameters~" $macro.layoutParams) 4860 +#end" %) 4861 +((( 4862 +(% class="macro-placeholder hidden" %) 4863 +((( 4864 +macro:velocity 4865 +))) 4866 +))) 4867 +))) 4868 + 4869 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-| #macro(displaySmallPost $pDoc $pObj) 4870 + #initLayoutVars($pDoc $pObj) 4871 + <div class=~"col-xs-12 col-sm-6 next_blog~"> 4872 + <a href=~"$pDoc.uRL~" class=~"thumbnail~"> 4873 + <div class=~"row~"> 4874 + <div class=~"col-xs-4~"> 4875 + #if ($imageAtt) 4876 + <img src=~"$escapetool.xml($pDoc.getAttachmentURL($pObj.getValue('image'),'download',$imgQs))~" /> 4877 + #end 4878 + </div> 4879 + <div class=~"col-xs-8 art_det~"> 4880 + <p class=~"text-left~"> 4881 + #if($displayTitle)$!postTitle #end 4882 + <br/><span class=~"date_info~"> $!dateStr </span> 4883 + </p> 4884 + #displayPostDetails($pDoc) 4885 + </div> 4886 + </div> 4887 + </a> 4888 + </div> 4889 + #end 4890 + ## 4891 + ## 4892 + ## 4893 + #macro(displayPinnedPost $pDoc $pObj) 4894 + #initLayoutVars($pDoc $pObj) 4895 + <div class=~"col-xs-12 col-sm-4 col-md-4 blog_pin~"> 4896 + <a href=~"$pDoc.uRL~" class=~"all-post-link~"> 4897 + <div class=~"thumbnail~"> 4898 + #if ($displayTitle && $displayTitleFirstOnPinnedPosts) 4899 + <h5 class=~"title-top~">$!postTitle <br/> <span class=~"date_info~"> $!dateStr </span> </h5> 4900 + #end 4901 + #if ($imageAtt) 4902 + <img src=~"$escapetool.xml($pDoc.getAttachmentURL($pObj.getValue('image'),'download',$imgQs))~" /> 4903 + #end 4904 + <div class=~"caption~"> 4905 + #if ($displayTitle && !$displayTitleFirstOnPinnedPosts) 4906 + <h5 class=~"text-left~">$!postTitle <br/> <span class=~"date_info~"> $!dateStr </span> </h5> 4907 + #end 4908 + #if ($displaySummaryOnPinnedPosts) 4909 + <div class=~"text-left post-summary~"> 4910 + #set ($postContent = $pObj.getProperty('extract').value) 4911 + $!pDoc.getRenderedContent($postContent, $pDoc.syntax.toIdString()) 4912 + </div> 4913 + #end 4914 + #displayPostDetails($pDoc) 4915 + </div> 4916 + </div> 4917 + </a> 4918 + </div> 4919 + #end 4920 + ## 4921 + ## 4922 + ## 4923 + #macro(formatPostDate $pDoc $pObj) 4924 + #set ($formattedDate = $NULL) 4925 + #set($dateStr = $datetool.format('medium', $postDate, $xcontext.locale)) 4926 + #if (~"$!dateStr~" != '') 4927 + #set ($dateArr = $dateStr.split(' ')) 4928 + #if ($dateArr.size() > 3) 4929 + #set ($dateStr = ~"~") 4930 + #foreach($s in $dateArr.subList(0, 3)) 4931 + #set ($dateStr = ~"$\{dateStr}$\{s} ~") 4932 + #end 4933 + #end 4934 + #end 4935 + #setVariable(~"$formattedDate~" $dateStr) 4936 + #end 4937 + ## 4938 + ## 4939 + ## 4940 + #macro(displayPostDetails $pDoc) 4941 + <div class=~"row post-details~"> 4942 + <div class=~"col-xs-8 detail~"> 4943 + #if ($services.like.displayButton($pDoc.documentReference)) 4944 + #set ($optLikeRecord = $services.like.getLikes($pDoc.documentReference)) 4945 + ## Retrieve the likes number in XWiki 12.9+ 4946 + #set ($likeNumber = $optLikeRecord.get()) 4947 + #if (!$stringtool.isNumeric($likeNumber.toString())) 4948 + ## Retrieve the likes number in XWiki versions greater than 12.7 and lower than 12.9. 4949 + #set ($likeNumber = $optLikeRecord.get().likeNumber) 4950 + #end 4951 + #if ($stringtool.isNumeric($likeNumber.toString())) 4952 + <div class=~"post-likes btn btn-default disabled badge~" 4953 + title=~"$escapetool.xml($services.localization.render('like.button.title', [$likeNumber]))~"> 4954 + $services.icon.renderHTML('heart') <span class=~"like-number~">$likeNumber</span> 4955 + </div> 4956 + #end 4957 + #elseif ($services.ratings) 4958 + #set ($ratingsConfigDoc = $services.ratings.getConfigurationDocument($pDoc.documentReference)) 4959 + #if ($ratingsConfigDoc.getValue('displayRatings') == 1) 4960 + <ul class=~"pull-left note list-inline~"> 4961 + #foreach ($x in [1..5]) 4962 + #set ($cls = ~"~") 4963 + #if ($x > $averageVote) 4964 + #set ($cls = ~"-o~") 4965 + #end 4966 + <li><span class=~"fa fa-star$\{cls}~"></span></li> 4967 + #end 4968 + </ul> 4969 + #end 4970 + #end 4971 + </div> 4972 + #if ($showPostComments) 4973 + <div class=~"col-xs-4 com_det~"> 4974 + <p class=~"text-right~">$services.icon.renderHTML('comments') ($!nbComments)</p> 4975 + </div> 4976 + #end 4977 + </div> 4978 + #end 4979 + ## 4980 + ## 4981 + ## 4982 + #macro(initLayoutVars $pDoc $pObj) 4983 + #set ($postTitle = $pDoc.display('title', 'view', $pObj)) 4984 + #set ($imageAtt = $pDoc.getAttachment($pObj.getValue('image'))) 4985 + #isPublished($pObj $isPublished) 4986 + #isHidden($pObj $isHidden) 4987 + #getEntryDate($pDoc $pObj $postDate) 4988 + #formatPostDate($postDate $dateStr) 4989 + #set ($averageVote = $mathtool.round($services.ratings.getAverageRating($pDoc.documentReference).getAverageVote())) 4990 + #set ($nbComments = $pDoc.getComments().size()) 4991 + #set ($showPostComments = true) 4992 + #if ($xwiki.getSpacePreferenceFor('showcomments', $pDoc.documentReference.lastSpaceReference, '1') == '0') 4993 + #set ($showPostComments = false) 4994 + #end 4995 + #end 4996 + ## 4997 + ## 4998 + ## 4999 + #macro(displayEditPinnedPostsButton) 5000 + #if (~"$!macroAdditionalParams.get('postIndex')~" == '0' && $services.security.authorization.hasAccess('edit', $pinnedPostsSourceDoc.fullName)) 5001 + #set ($discard = $xwiki.ssx.use('Blog.PinnedPostsSheet')) 5002 + #set ($discard = $xwiki.jsx.use('Blog.PinnedPostsSheet')) 5003 + #set ($editPinnedPostsURL = $xwiki.getURL($pinnedPostsSourceDoc.fullName, 'get', 'sheet=Blog.PinnedPostsSheet')) 5004 + #set ($newPinnedPostsObjectURL = $xwiki.getURL($pinnedPostsSourceDoc.fullName, 'objectadd', ~"classname=Blog.PinnedPostsClass&form_token=$!\{services.csrf.getToken()}~")) 5005 + <div class=~"edit-pinned-posts-container~"> 5006 + <div class=~"col-xs-12 col-sm-12 col-md-12 text-right edit-pinned-posts~"> 5007 + <a href=~"javascript:;~" class=~"editPinnedPosts~"> 5008 + <span class=~"glyphicon glyphicon-edit~" aria-hidden=~"true~"></span> $services.localization.render('blog.post.layout.cards.pinnedposts.edit') 5009 + </a> 5010 + <input type=~"hidden~" id=~"editPinnedPostsURL~" value=~"$escapetool.xml($editPinnedPostsURL)~"/> 5011 + <input type=~"hidden~" id=~"newPinnedPostsObjectURL~" value=~"$escapetool.xml($newPinnedPostsObjectURL)~"/> 5012 + </div> 5013 + </div> 5014 + #if (~"$!pinnedPostsObj~" == '') 5015 + <input type=~"hidden~" name=~"createPinnedPostsObject~" value=~"1~"/> 5016 + #end 5017 + #end 5018 + #end" %) 5019 +((( 5020 +(% class="macro-placeholder hidden" %) 5021 +((( 5022 +macro:velocity 5023 +))) 5024 +))) 5025 + 5026 +(% data-macro="startmacro:velocity|-||-| #if ($services.security.authorization.hasAccess('view', $xcontext.macro.params.reference)) 5027 + #set ($postDoc = $xwiki.getDocument($xcontext.macro.params.reference)) 5028 + #getEntryObject($postDoc $postObj) 5029 + #if (~"$!postObj~" != '') 5030 + #extractLayoutParametersFromString($xcontext.macro.params.params $macroAdditionalParams) 5031 + ## 5032 + #set ($displayTitle = true) 5033 + #if (~"$!macroAdditionalParams.get('displayTitle')~" == 'false') 5034 + #set ($displayTitle = false) 5035 + #end 5036 + ## 5037 + #set ($displayTitleFirstOnPinnedPosts = false) 5038 + #if (~"$!macroAdditionalParams.get('displayTitleFirstOnPinnedPosts')~" == 'true') 5039 + #set ($displayTitleFirstOnPinnedPosts = true) 5040 + #end 5041 + ## 5042 + #set ($displaySummaryOnPinnedPosts = true) 5043 + #if (~"$!macroAdditionalParams.get('displaySummaryOnPinnedPosts')~" == 'false') 5044 + #set ($displaySummaryOnPinnedPosts = false) 5045 + #end 5046 + ## 5047 + #set ($postIndex = $numbertool.toNumber($macroAdditionalParams.get('postIndex')).intValue()) 5048 + #if ($postIndex == 0) 5049 + #set ($stopBlogPostsDisplay = false) 5050 + #end 5051 + #set ($listResultsCount = $numbertool.toNumber($macroAdditionalParams.get('list-results-count')).intValue()) 5052 + #set($discard = $xwiki.ssx.use('Blog.BlogPostLayoutCards')) 5053 + #set($scaleWidth = 600) 5054 + #set($imgQs=~"width=$scaleWidth~") 5055 + ## Display pinned posts 5056 + ## Get the list of pinned posts 5057 + #set ($pinnedPosts = []) 5058 + #set ($pinnedPostsSourceDoc = $NULL) 5059 + #if (~"$!macroAdditionalParams.get('forceDisplayPinnedPosts')~" == 'true' && ~"$!macroAdditionalParams.get('list-blog')~" != '') 5060 + #set ($pinnedPostsSourceDoc = $xwiki.getDocument($macroAdditionalParams.get('list-blog'))) 5061 + ## Note that the 'list-blog' parameter contains the value of the 'blog' parameter of the blogpostlist macro passed to the current layout macro 5062 + #end 5063 + #if (~"$!pinnedPostsSourceDoc~" != '') 5064 + #set ($pinnedPostsObj = $pinnedPostsSourceDoc.getObject('Blog.PinnedPostsClass')) 5065 + #if (~"$!pinnedPostsObj~" != '') 5066 + #set ($orderedPinnedPostsJSON = $pinnedPostsObj.getProperty('orderedPosts').value) 5067 + #if (~"$!orderedPinnedPostsJSON~" != '') 5068 + #set ($pinnedPosts = $jsontool.parse($orderedPinnedPostsJSON)) 5069 + #else 5070 + #set ($pinnedPosts = $pinnedPostsObj.getProperty('posts').value) 5071 + #end 5072 + #end 5073 + #if ($postIndex == 0 && $pinnedPosts.size() > 0) 5074 + \{\{html clean=~"false~"}} 5075 + #set ($x = 0) 5076 + #set ($showPinnedPostsButton = true) 5077 + #foreach ($pinnedPost in $pinnedPosts) 5078 + #if ($x == 0) 5079 + <div class=~"row flex-container~"> 5080 + #if ($showPinnedPostsButton) 5081 + #displayEditPinnedPostsButton() 5082 + #set($showPinnedPostsButton = false) 5083 + #end 5084 + #end 5085 + #set ($pinnedPostDoc = $xwiki.getDocument($pinnedPost)) 5086 + #getEntryObject($pinnedPostDoc $pinnedPostObj) 5087 + #if (~"$!pinnedPostObj~" != '') 5088 + #displayPinnedPost($pinnedPostDoc $pinnedPostObj) 5089 + #end 5090 + #set ($x = $mathtool.add($x, 1)) 5091 + #if ($x == 3) 5092 + #set ($x = 0) 5093 + </div> 5094 + #end 5095 + #end 5096 + #if ($mathtool.mod($x, 3) != 0) 5097 + </div> 5098 + #end 5099 + \{\{/html}} 5100 + 5101 + ## If the first post is a pinned post : this means that all the posts are pinned 5102 + ## In this case, avoid displaying the posts again after the pinned posts section. 5103 + #if ($pinnedPosts.contains($xcontext.macro.params.reference)) 5104 + #set ($stopBlogPostsDisplay = true) 5105 + #end 5106 + #elseif($postIndex == 0 && $pinnedPosts.size() == 0) 5107 + 5108 + \{\{html}} 5109 + <div class=~"row no-pinnded-posts~"> 5110 + #displayEditPinnedPostsButton() 5111 + </div> 5112 + \{\{/html}} 5113 + 5114 + #end 5115 + #end 5116 + #if (!$stopBlogPostsDisplay) 5117 + \{\{html clean=~"false~"}} 5118 + #if (~"$!nbDisplayedPosts~" == '' || $postIndex == 0) 5119 + #set ($nbDisplayedPosts = 0) 5120 + #end 5121 + #if($mathtool.mod($nbDisplayedPosts, 2) == 0) 5122 + #if ($nbDisplayedPosts != 0) 5123 + </div> 5124 + #set ($lastHtmlTag = 'c') 5125 + #end 5126 + #if ($nbDisplayedPosts == $mathtool.sub($listResultsCount, 1)) 5127 + <div class=~"row~"> 5128 + #set ($lastHtmlTag = 'o') 5129 + #set ($lastRowClass = ~"~") 5130 + #else 5131 + <div class=~"row flex-container~"> 5132 + #set ($lastHtmlTag = 'o') 5133 + #set ($lastRowClass = ~"flex~") 5134 + #end 5135 + #end 5136 + #displaySmallPost($postDoc $postObj) 5137 + #if ($nbDisplayedPosts == $mathtool.sub($listResultsCount, 1)) 5138 + </div> 5139 + #set ($lastHtmlTag = 'c') 5140 + #end 5141 + #set ($nbDisplayedPosts = $mathtool.add($nbDisplayedPosts, 1)) 5142 + #if ($postIndex == $mathtool.sub($listResultsCount, 1)) 5143 + #if (~"$!lastHtmlTag~" == 'o') 5144 + #if ($mathtool.mod($nbDisplayedPosts, 2) == 1 && $lastRowClass == ~"flex~") 5145 + <div class=~"col-xs-12 col-sm-6~"></div> 5146 + #end 5147 + </div> 5148 + #end 5149 + #end 5150 + \{\{/html}} 5151 + #end 5152 + #else 5153 + \{\{error}}$services.localization.render('blog.blogpostlayout.notpost', [$xcontext.macro.params.reference])\{\{/error}} 5154 + #end 5155 + #else 5156 + \{\{error}}$services.localization.render('blog.blogpostlayout.post_view_not_allowed', [$xcontext.macro.params.reference])\{\{/error}} 5157 + #end" class="macro hidden macro-placeholder" %)macro:velocity(% data-macro="startmacro:html|-|clean=~"false~"|-|</div> 5158 +<div class=~"row flex-container~"> 5159 +<div class=~"col-xs-12 col-sm-6 next_blog~"> 5160 +<a href=~"/Nieuws/Nieuwsbrief%20december%202023~" class=~"thumbnail~"> 5161 +<div class=~"row~"> 5162 +<div class=~"col-xs-4~"> 5163 +<img src=~"/download/Nieuws/Nieuwsbrief%20december%202023/snow-cold-winter-white-sweet-weather-1363025-pxhere.com.jpg?width=600&rev=1.1~" /> 5164 +</div> 5165 +<div class=~"col-xs-8 art_det~"> 5166 +<p class=~"text-left~"> 5167 +Nieuwsbrief december 2023 <br/><span class=~"date_info~"> Jan 10, 2024, </span> 5168 +</p> 5169 +<div class=~"row post-details~"> 5170 +<div class=~"col-xs-8 detail~"> 5171 +<div class=~"post-likes btn btn-default disabled badge~" 5172 +title=~"Number of likes on this page: 1~"> 5173 +<span class=~"fa fa-heart~"></span> <span class=~"like-number~">1</span> 5174 +</div> 5175 +</div> 5176 +<div class=~"col-xs-4 com_det~"> 5177 +<p class=~"text-right~"><span class=~"fa fa-comments~"></span> (0)</p> 5178 +</div> 5179 +</div> 5180 +</div> 5181 +</div> 5182 +</a> 5183 +</div>" class="macro macro hidden macro-placeholder" %)macro:html 5184 +))) 5185 +))) 5186 + 5187 +(% class="row flex-container" %) 5188 +((( 5189 +(% class="col-xs-12 col-sm-6 next_blog" %) 5190 +((( 5191 +(% class="row" %) 5192 +((( 5193 +(% class="col-xs-4" %) 5194 +((( 5195 +[[~[~[image:/download/Nieuws/Nieuwsbrief%20december%202023/snow-cold-winter-white-sweet-weather-1363025-pxhere.com.jpg?width=600&rev=1.1~]~]>>path:/Nieuws/Nieuwsbrief%20december%202023||class="thumbnail"]] 5196 +))) 5197 + 5198 +(% class="col-xs-8 art_det" %) 5199 +((( 5200 +(% class="text-left" %) 5201 +[[Nieuwsbrief december 2023 5202 +(% class="date_info" %) Jan 10, 2024,>>path:/Nieuws/Nieuwsbrief%20december%202023||class="thumbnail"]] 5203 + 5204 +(% class="row post-details" %) 5205 +((( 5206 +(% class="col-xs-8 detail" %) 5207 +((( 5208 +(% class="post-likes btn btn-default disabled badge" title="Number of likes on this page: 1" %) 5209 +((( 5210 +[[(% class="like-number" %)1>>path:/Nieuws/Nieuwsbrief%20december%202023||class="thumbnail"]] 5211 +))) 5212 +))) 5213 + 5214 +(% class="col-xs-4 com_det" %) 5215 +((( 5216 +(% class="text-right" %) 5217 +[[(0)>>path:/Nieuws/Nieuwsbrief%20december%202023||class="thumbnail"]] 5218 +))) 5219 +))) 5220 +))) 5221 +))) 5222 +))) 5223 + 5224 +(% class="macro" data-macro="startmacro:blogPostLayoutCards|-|reference=~"Nieuws.Nieuwsbrief november 2023~" params=~"displayTitle=true|useSummary=true|list-limit=10|list-layout=cards|list-blog=Nieuws.WebHome|list-paginated=no|list-results-count=10|postIndex=0|previousPostReference=|postIndex=1|previousPostReference=Nieuws.Nieuwsbrief januari 2024|postIndex=2|previousPostReference=Nieuws.Audit BMI en BORG|postIndex=3|previousPostReference=Nieuws.Nieuwsbrief december 2023~"" %) 5225 +((( 5226 +(% class="macro-placeholder hidden" %) 5227 +((( 5228 +macro:blogPostLayoutCards 5229 +))) 5230 + 5231 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogCode~"" %) 5232 +((( 5233 +(% class="macro-placeholder hidden" %) 5234 +((( 5235 +macro:include 5236 +))) 5237 + 5238 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogParameters~"" %) 5239 +((( 5240 +(% class="macro-placeholder hidden" %) 5241 +((( 5242 +macro:include 5243 +))) 5244 + 5245 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|## Blog 5246 +#set($blogClassname = 'Blog.BlogClass') 5247 +#set($blogTemplate = 'Blog.BlogTemplate') 5248 +#set($blogSheet = 'Blog.BlogSheet') 5249 +## Blog entries 5250 +#set($blogPostClassname = 'Blog.BlogPostClass') 5251 +#set($blogPostTemplate = 'Blog.BlogPostTemplate') 5252 +#set($blogPostSheet = 'Blog.BlogPostSheet') 5253 +#set($blogPostObjectNumber = $xwiki.getDocument($blogPostTemplate).getObject($blogPostClassname).number) 5254 +#set($oldArticleClassname = 'XWiki.ArticleClass') 5255 +## Categories 5256 +#set($blogCategoryClassname = 'Blog.CategoryClass') 5257 +#set($blogCategoryTemplate = 'Blog.CategoryTemplate') 5258 +#set($blogCategorySheet = 'Blog.CategorySheet') 5259 +#set($blogCategoriesSheet = 'Blog.CategoriesSheet') 5260 +#set($oldBlogCategoryClassname = 'Blog.Categories') 5261 +#set($defaultCategoryParent = 'Blog.Categories') 5262 +## Style 5263 +#set($blogStyleDocumentName = 'Blog.BlogStyle') 5264 +#set($blogStyleDocument = $xwiki.getDocument($blogStyleDocumentName)) 5265 +## Clientside scripts 5266 +#set($blogScriptsDocumentName = 'Blog.BlogScripts') 5267 +#set($blogPublisher = $xwiki.getDocument('Blog.Publisher')) 5268 +## Misc 5269 +#set($thisURL = $doc.getURL($xcontext.action, $request.queryString)) 5270 +#set($isBlogPost = $doc.getObject($blogPostClassname)) 5271 +#set($defaultBlogSpace = 'Blog') 5272 +#set($isCategoryPage = $doc.getObject($blogCategoryClassname)) 5273 +#set($isCategoriesHomePage = $doc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 5274 +## 5275 +## 5276 +## 5277 +#** 5278 + * Displays an image, taken from the blog style document. 5279 + * 5280 + * @param $imgName The name of the icon from icons set to use. 5281 + *# 5282 +#macro(toolImage $imgName)(% class=~"icon-manage~"~%)$services.icon.render($imgName)(%~%)#end" %) 5283 +((( 5284 +(% class="macro-placeholder hidden" %) 5285 +((( 5286 +macro:velocity 5287 +))) 5288 +))) 5289 +))) 5290 + 5291 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|## 5292 +## 5293 +## 5294 +## Import the blog skin and javascripts. 5295 +$!xwiki.ssx.use($blogStyleDocumentName)## 5296 +$!xwiki.jsx.use($blogScriptsDocumentName)## 5297 +## 5298 +## import the hierarchy for the path the blog post footer (in displayEntryBlogLocation) 5299 +#template('hierarchy_macros.vm')## 5300 +## 5301 +## 5302 +#** 5303 + * Prints a blog. This is the main macro used in the BlogSheet. 5304 + * 5305 + * @param blogDoc the XDocument holding the blog definition object. 5306 + *### 5307 +#macro(printBlog $blogDoc) 5308 + \{\{include reference='Blog.CreatePost'/}} 5309 + 5310 + ## Use the blogPostList macro to display the blogposts 5311 + ## The blogPostList is used only in case of 'all' or 'paged' blog display type because the blogPostList macro 5312 + ## do not support FTM the monthly and weekly blog display types 5313 + #getBlogDisplayType($blogDoc $displayType) 5314 + #if ($displayType == 'weekly' || $displayType == 'monthly') 5315 + #getBlogEntries($blogDoc $entries) 5316 + #displayBlog($entries 'index' true true) 5317 + #displayNavigationLinks($blogDoc) 5318 + #else 5319 + #getBlogDisplayType($blogDoc $displayType) 5320 + #set ($paginated = 'no') 5321 + #if ($displayType == 'paginated') 5322 + #set ($paginated = 'yes') 5323 + #end 5324 + #getBlogPostsLayout($blogDoc $postsLayout) 5325 + (% class=~"hfeed index~" ~%)(((\{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$paginated~" layout=~"$!postsLayout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" /}}))) 5326 + #end 5327 +#end 5328 +## 5329 +## 5330 +## 5331 +#** 5332 + * Shows blog information. In view mode, the description is printed. In edit mode, allows changing blog settings: title, 5333 + * description, blog type (global or in-space), index display type (fixed size pagination, weekly index, monthly index, 5334 + * all entries). 5335 + * 5336 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 5337 + *### 5338 +#macro(showBlogInfo $blogDoc) 5339 + #if($blogDoc.getObject($blogClassname)) 5340 + ## Keep testing for inline action for backward compatibility with older blogs. 5341 + #if($xcontext.action == 'edit' || $xcontext.action == 'inline') 5342 + #macro(displayProperty $blogDoc $propname) 5343 + ; #displayPropName($xwiki.getClass($blogClassname).get($propname)): 5344 + : $blogDoc.display($propname) 5345 + #end 5346 + #displayProperty($blogDoc 'title') 5347 + #displayProperty($blogDoc 'description') 5348 + #displayProperty($blogDoc 'displayType') 5349 + #displayProperty($blogDoc 'itemsPerPage') 5350 + #displayProperty($blogDoc 'postsLayout') 5351 + #displayProperty($blogDoc 'postsLayoutParameters') 5352 + #else 5353 + $blogDoc.display('description') 5354 + #end 5355 + #elseif($doc.fullName == $blogSheet) 5356 += $services.localization.render('blog.code.blogsheet') = 5357 + \{\{translation key='blog.code.sheetexplanation'/}} 5358 + #else 5359 + \{\{warning}}\{\{translation key='blog.code.notblog'/}}\{\{/warning}} 5360 + #end 5361 +#end 5362 +## 5363 +## 5364 +## 5365 +#** 5366 + * Retrieve the blog document, which usually is either <tt><Space>.WebHome</tt> for whole-spaces blogs, or 5367 + * <tt><Space>.Blog</tt> for in-space blogs. If none of these documents contains a blog object, then the first 5368 + * (alphabetically) document in the target space that contains one is returned. Finally, if no document in the current 5369 + * space contains a blog object, then <tt>Blog.WebHome</tt> is returned as the default blog. 5370 + * 5371 + * @param space A <tt>String</tt>, the name of the space where to search. 5372 + * @param blogDoc The resulting XDocument. 5373 + *### 5374 +#macro(getBlogDocument $space $blogDoc) 5375 + #set ($result = $NULL) 5376 + ## Check if the current space has a WebPreferences page that contains a Blog.EnablePanelsConfigurationClass object. 5377 + ## It is possible to display a blog's panels info in a page that is not inside a blog and in that case we need to 5378 + ## identify the right blog based on a configuration object in a WebPreferences page. 5379 + #set ($spaceReference = $services.model.resolveSpace($space)) 5380 + #set ($preferencesDocRef = $services.model.createDocumentReference('WebPreferences', $spaceReference)) 5381 + #set ($preferencesDoc = $xwiki.getDocument($preferencesDocRef)) 5382 + #set ($preferencesObj = $preferencesDoc.getObject('Blog.EnablePanelsConfigurationClass')) 5383 + #if ($preferencesObj) 5384 + #set ($result = $xwiki.getDocument( $preferencesObj.getValue('blog'))) 5385 + #end 5386 + ## If no special case occurs (like the previous one), identify a blog by checking the current location from its hierarchy. 5387 + #if (~"$!result~" == '') 5388 + ## First, try the Space.WebHome, for a whole-space blog 5389 + #set($result = $xwiki.getDocument(~"$\{space}.WebHome~")) 5390 + #if(!$result.getObject($blogClassname)) 5391 + ## Second, try the Space.Blog document 5392 + #set($result = $xwiki.getDocument(~"$\{space}.Blog~")) 5393 + #if(!$result.getObject($blogClassname)) 5394 + ## Third, try searching for a blog document in the current space 5395 + ## Prevent the query fail when the space contains dots '.' 5396 + #set($blogDocs = $services.query.hql(~", BaseObject obj where doc.space = :space and obj.name = doc.fullName and obj.className = '$blogClassname' order by doc.name~").setLimit(1).setOffset(0).bindValue('space', $space).execute()) 5397 + #if($blogDocs.size() > 0) 5398 + #set($result = $xwiki.getDocument($blogDocs.get(0))) 5399 + #else 5400 + ## Fourth, try searching for a blog document that have the its 'postsLocation' set to the current space 5401 + #set($blogDocs = $services.query.hql(~", BaseObject obj, StringProperty as postsLocationProp where obj.name = doc.fullName and obj.className = '$blogClassname' and obj.id=postsLocationProp.id.id and postsLocationProp.id.name='postsLocation' and postsLocationProp.value=:postsLocation order by doc.name~").setLimit(1).setOffset(0).bindValue('postsLocation', $space).execute()) 5402 + #if($blogDocs.size() > 0) 5403 + #set($result = $xwiki.getDocument($blogDocs.get(0))) 5404 + #else 5405 + ## Last, fallback to Blog.WebHome, the default blog 5406 + #set($result = $xwiki.getDocument('Blog.WebHome')) 5407 + #end 5408 + #end 5409 + #end 5410 + #end 5411 + #end 5412 + #set ($blogDoc = $NULL) 5413 + #setVariable (~"$blogDoc~" $result) 5414 +#end 5415 +## 5416 +## 5417 +## 5418 +#** 5419 + * Retrieve the blog title. 5420 + * 5421 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass<tt> object with the <tt>title</tt> property set. 5422 + * @param title The resulting title. 5423 + *### 5424 +#macro(getBlogTitle $blogDoc $title) 5425 + ## Titles can contain velocity code (most commonly translations), so we should evaluate them. 5426 + #set ($title = $NULL) 5427 + #setVariable (~"$title~" $!blogDoc.displayTitle) 5428 +#end 5429 +## 5430 +## 5431 +## 5432 +#** 5433 + * Retrieve the blog description. 5434 + * 5435 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>description</tt> 5436 + * property set. 5437 + * @param description The resulting description. 5438 + *### 5439 +#macro(getBlogDescription $blogDoc $description) 5440 + #getBlogProperty($blogDoc 'description' '' $result) 5441 + #set ($description = $NULL) 5442 + #setVariable (~"$description~" $result) 5443 +#end 5444 +## 5445 +## 5446 +## 5447 +#** 5448 + * Retrieves a list of entries to be displayed. The entries are either part of the blog's space, or have the blog 5449 + * document set as a parent. The number and range of entries returned (from all those belonging to this blog) depends on 5450 + * the blog display type: paginated (fixed number of entries), weekly (all entries in a week), monthly (all entries in a 5451 + * month), or all. 5452 + * 5453 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 5454 + * @param entries The resulting list of entries to display, a list of XDocument names. 5455 + *### 5456 +#macro(getBlogEntries $blogDoc $entries) 5457 + #if (!$entries) 5458 + #setVariable (~"$entries~" []) 5459 + #end 5460 + #getAllBlogPostsQuery($query) 5461 + #isDefaultBlog($blogDoc $isDefault) 5462 + #set($queryParams = \{}) 5463 + #if ($isDefault) 5464 + #getCategoryAllBlogPostsQuery($query) 5465 + #set($query = ~"$\{query} and (doc.creator = :creator or (isPublished.value = 1 and hidden.value = 0)) and (doc.space = :space or catList like :catList escape '!')~") 5466 + #set($discard = $queryParams.put('creator', $xcontext.user)) 5467 + #set($discard = $queryParams.put('space', $blogDoc.space)) 5468 + #set($sanitizedBlogSpace = $blogDoc.space.replaceAll('([%_!])', '!$1').concat('.%')) 5469 + #set($discard = $queryParams.put('catList', $sanitizedBlogSpace)) 5470 + #else 5471 + #set($query = ~"$\{query} and (doc.space = :space or doc.parent = :parent)~") 5472 + #getBlogPostsLocation($blogDoc.space $blogPostsLocation) 5473 + #set($discard = $queryParams.put('space', $blogPostsLocation)) 5474 + #set($discard = $queryParams.put('parent', $blogDoc.space)) 5475 + #end 5476 + #getBlogDisplayType($blogDoc $displayType) 5477 + #if($displayType == 'weekly') 5478 + #getWeeklyBlogEntries($blogDoc $query $entries $queryParams) 5479 + #elseif($displayType == 'monthly') 5480 + #getMonthlyBlogEntries($blogDoc $query $entries $queryParams) 5481 + #elseif($displayType == 'all') 5482 + #getAllBlogEntries($blogDoc $query $entries $queryParams) 5483 + #else 5484 + #getPagedBlogEntries($blogDoc $query $entries $queryParams) 5485 + #end 5486 +#end 5487 +## 5488 +## 5489 +## 5490 +#** 5491 + * Retrieves a list of entries to be displayed. The entries are taken from a ~"page~" of the blog, a sequence of documents 5492 + * defined by the request parameters <tt>ipp</tt> (items per page) and <tt>page</tt> (the current page). Initially the 5493 + * first page is displayed, with the number of entries defined in the blog object in the <tt>itemsPerPage</tt> property 5494 + * (10 if not defined). 5495 + * 5496 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 5497 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 5498 + * refined to restrict to a given space, or to a given search criteria, etc. 5499 + * @param entries The resulting list of entries to display, a list of XDocument names. 5500 + * @param queryParams The parameters to bind with the query. 5501 + *### 5502 +#macro(getPagedBlogEntries $blogDoc $query $entries $queryParams) 5503 + #if (!$entries) 5504 + #setVariable (~"$entries~" []) 5505 + #end 5506 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 5507 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~")) 5508 + #bindQueryParameters($countQueryObj $queryParams) 5509 + #bindQueryParameters($queryObj $queryParams) 5510 + #set($totalEntries = $countQueryObj.count()) 5511 + #getBlogProperty($blogDoc 'itemsPerPage' '10' $defaultItemsPerPage) 5512 + #set($defaultItemsPerPage = $numbertool.toNumber($defaultItemsPerPage).intValue()) 5513 + ## This macro is defined in the default macros.vm library. It also sets $itemsPerPage and $startAt. 5514 + #preparePagedViewParams($totalEntries $defaultItemsPerPage) 5515 + #set($discard = $entries.addAll($queryObj.setLimit($itemsPerPage).setOffset($startAt).addFilter(~"unique~").execute())) 5516 +#end 5517 +## 5518 +## 5519 +## 5520 +#** 5521 + * Retrieves a list of entries to be displayed. The entries are taken from a week of the blog. The target week is 5522 + * defined by the request parameters <tt>week</tt> (the week number in the year, from 1 to 52) and <tt>year</tt> (4 5523 + * digit year). Initially the current week is displayed. 5524 + * 5525 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 5526 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 5527 + * refined to restrict to a given space, or to a given search criteria, etc. 5528 + * @param entries The resulting list of entries to display, a list of XDocument names. 5529 + * @param queryParams The parameters to bind with the query. 5530 + *### 5531 +#macro(getWeeklyBlogEntries $blogDoc $query $entries $queryParams) 5532 + #if (!$entries) 5533 + #setVariable (~"$entries~" []) 5534 + #end 5535 + #getRequestedWeek($weekDate) 5536 + #set($dateFormatter = $xwiki.jodatime.getDateTimeFormatterForPattern('yyyy-MM-dd')) 5537 + #set($minDay = $dateFormatter.print($weekDate.toMutableDateTime().weekOfWeekyear().roundFloor())) 5538 + #set($maxDay = $dateFormatter.print($weekDate.toMutableDateTime().weekOfWeekyear().roundCeiling())) 5539 + #set($query = ~"$\{query} and publishDate.value >= '$minDay' and publishDate.value < '$maxDay'~") 5540 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 5541 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 5542 + #bindQueryParameters($countQueryObj $queryParams) 5543 + #bindQueryParameters($queryObj $queryParams) 5544 + #set($totalEntries = $countQueryObj.count()) 5545 + #set($discard = $entries.addAll($queryObj.execute())) 5546 +#end 5547 +## 5548 +## 5549 +## 5550 +#** 5551 + * Retrieves a list of entries to be displayed. The entries are taken from a month of the blog. The target month is 5552 + * defined by the request parameters <tt>month</tt> (the month number, from 1 to 12) and <tt>year</tt> (4 5553 + * digit year). Initially the current month is displayed. 5554 + * 5555 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 5556 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 5557 + * refined to restrict to a given space, or to a given search criteria, etc. 5558 + * @param entries The resulting list of entries to display, a list of XDocument names. 5559 + * @param queryParams The parameters to bind with the query. 5560 + *### 5561 +#macro(getMonthlyBlogEntries $blogDoc $query $entries) 5562 + #if (!$entries) 5563 + #setVariable (~"$entries~" []) 5564 + #end 5565 + #getRequestedMonth($monthDate) 5566 + #set($dateFormatter = $xwiki.jodatime.getDateTimeFormatterForPattern('yyyy-MM-dd')) 5567 + #set($minDay = $dateFormatter.print($monthDate.toMutableDateTime().monthOfYear().roundFloor())) 5568 + #set($maxDay = $dateFormatter.print($monthDate.toMutableDateTime().monthOfYear().roundCeiling())) 5569 + #set($query = ~"$\{query} and publishDate.value >= '$minDay' and publishDate.value < '$maxDay'~") 5570 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 5571 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 5572 + #bindQueryParameters($countQueryObj $queryParams) 5573 + #bindQueryParameters($queryObj $queryParams) 5574 + #set($totalEntries = $countQueryObj.count()) 5575 + #set($discard = $entries.addAll($queryObj.execute())) 5576 +#end 5577 +## 5578 +## 5579 +## 5580 +#** 5581 + * Retrieves a list of entries to be displayed. All entries belonging to the current blog are returned. 5582 + * 5583 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 5584 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 5585 + * refined to restrict to a given space, or to a given search criteria, etc. 5586 + * @param entries The resulting list of entries to display, a list of XDocument names. 5587 + * @param queryParams The parameters to bind with the query. 5588 + *### 5589 +#macro(getAllBlogEntries $blogDoc $query $entries $queryParams) 5590 + #if (!$entries) 5591 + #setVariable (~"$entries~" []) 5592 + #end 5593 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 5594 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 5595 + #bindQueryParameters($countQueryObj $queryParams) 5596 + #bindQueryParameters($queryObj $queryParams) 5597 + #set($totalEntries = $countQueryObj.count()) 5598 + #set($discard = $entries.addAll($queryObj.execute())) 5599 +#end 5600 +## 5601 +## 5602 +## 5603 +#** 5604 + * Retrieves a list of entries to be displayed. Only (and all) unpublished entries are returned. 5605 + * 5606 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 5607 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 5608 + * refined to restrict to a given space, or to a given search criteria, etc. 5609 + * @param queryParams The parameters to bind with the query. 5610 + * @param entries The resulting list of entries to display, a list of XDocument names. 5611 + *### 5612 +#macro(getUnpublishedBlogEntries $blogDoc $query $queryParams $entries) 5613 + #if (!$entries) 5614 + #setVariable (~"$entries~" []) 5615 + #end 5616 + #set($query = ~"$\{query} and isPublished.value = 0~") 5617 + #set ($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 5618 + #set ($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 5619 + #bindQueryParameters($countQueryObj $queryParams) 5620 + #bindQueryParameters($queryObj $queryParams) 5621 + #set($totalEntries = $countQueryObj.count()) 5622 + #set($discard = $entries.addAll($queryObj.execute())) 5623 +#end 5624 +## 5625 +## 5626 +## 5627 +#** 5628 + * Retrieves a list of entries to be displayed. The entries are taken from all the wiki, and not from a specific blog. 5629 + * 5630 + * @param entries The resulting list of entries to display, a list of XDocument names. 5631 + *### 5632 +#macro(getGlobalBlogEntries $entries) 5633 + #if (!$entries) 5634 + #setVariable (~"$entries~" []) 5635 + #end 5636 + #getAllBlogPostsQuery($query) 5637 + #set($totalEntries = $services.query.hql($query).count()) 5638 + #set($defaultItemsPerPage = 20) 5639 + ## This macro is defined in the default macros.vm library. It also sets $itemsPerPage and $startAt. 5640 + #preparePagedViewParams($totalEntries $defaultItemsPerPage) 5641 + #set($discard = $entries.addAll($services.query.hql(~"$\{query} order by publishDate.value desc~").setLimit($itemsPerPage).setOffset($startAt).execute())) 5642 +#end 5643 +#** 5644 + * Return the base query for selecting blog entries. It filters only visible entries, but does not bind to a specific 5645 + * blog, nor specify a range or an ordering criteria. 5646 + * 5647 + * This macro is a duplicate of #getAllBlogPostsQuery($query). Keep it in case it's used outside, in custom code. 5648 + * 5649 + * @param query The basic query for selecting blog entries. 5650 + *# 5651 +#macro(getBlogEntriesBaseQuery $query) 5652 + #getAllBlogPostsQuery($query) 5653 +#end 5654 +#** 5655 + * Return the Query for selecting the all wiki blog posts without filtering 5656 + * 5657 + * @param query The basic query for selecting blog entries. 5658 + *# 5659 +#macro(getAllBlogPostsQuery $query) 5660 + #set ($query = $NULL) 5661 + #setVariable(~"$query~" ~", BaseObject as obj, IntegerProperty isPublished, 5662 + IntegerProperty hidden, DateProperty publishDate 5663 + where doc.fullName <> '$blogPostTemplate' and 5664 + obj.name=doc.fullName and obj.className='$blogPostClassname' and 5665 + isPublished.id.id = obj.id and isPublished.id.name = 'published' and 5666 + hidden.id.id = obj.id and hidden.id.name='hidden' and 5667 + publishDate.id.id = obj.id and publishDate.id.name='publishDate' and 5668 + (doc.creator = '$escapetool.sql($xcontext.user)' or (isPublished.value = 1 and hidden.value = 0))~") 5669 +#end 5670 +## 5671 +## 5672 +## 5673 +###** 5674 + * Return the Query for selecting the all wiki blog posts with categories filtering 5675 + * 5676 + * @param query The basic query for selecting blog entries. 5677 + *### 5678 +#macro(getCategoryAllBlogPostsQuery $query) 5679 + #set ($query = $NULL) 5680 + #getAllBlogPostsQuery($baseQuery) 5681 + #set ($baseQuery = $baseQuery.replace('DateProperty publishDate', 'DateProperty publishDate, DBStringListProperty as category left join category.list catList')) 5682 + #setVariable(~"$query~" ~"$\{baseQuery} and obj.id=category.id.id and category.id.name='category'~") 5683 +#end 5684 +## 5685 +## 5686 +## 5687 +#** 5688 + * Determines how is the blog index split into pages: paginated (fixed number of entries), weekly (all entries in a 5689 + * week), monthly (all entries in a month), or all. 5690 + * 5691 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>displayType</tt> 5692 + * property set. 5693 + * @param displayType The resulting string. If the blog object does not define anything, it is considered paginated. 5694 + *### 5695 +#macro(getBlogDisplayType $blogDoc $displayType) 5696 + #getBlogProperty($blogDoc 'displayType' 'paginated' $result) 5697 + #set ($displayType = $NULL) 5698 + #setVariable (~"$displayType~" $result) 5699 +#end 5700 +## 5701 +## 5702 +## 5703 +#** 5704 + * Displays a list of entries. 5705 + * 5706 + * @param entries The entries to display, a list of XDocument names. 5707 + * @param displaying What exactly is displayed: blog index, a single blog entry, a blog category, search results, 5708 + * unpublished entries, etc. This will be used as the classname(s) for the container div (hfeed). Currently 5709 + * used values: index, single, category, search, unpublished, hidden. 5710 + * @param onlyExtract If <tt>true</tt>, only display the extract of articles where available, otherwise display the full content. 5711 + * @param shouldDisplayTitles If <tt>true</tt>, display the blog title (blog posts shouldn't display the title when they're 5712 + * displayed alone on their page since it's the page title which is used in this case) 5713 + *### 5714 +#macro(displayBlog $entries $displaying $onlyExtract $shouldDisplayTitles) 5715 + #set($blogDay = '') 5716 + (% class=~"hfeed $!\{displaying}~" ~%)((( 5717 + (% class=~"blogDay~" ~%)((( 5718 + #foreach ($entryDoc in $xwiki.wrapDocs($entries)) 5719 + #getEntryObject($entryDoc $entryObj) 5720 + ## Although all entries should have one of the two objects, better check to be sure. 5721 + #if(~"$!\{entryObj}~" != '') 5722 + #getEntryDate($entryDoc $entryObj $entryDate) 5723 + ## Display a ~"calendar sheet~" for each day. All entries posted on the same day share one such sheet. 5724 + #set($entryDateStr = $xwiki.formatDate($entryDate, 'yyyyMMMMdd')) 5725 + #if($blogDay != $entryDateStr) 5726 + #if($blogDay != '') 5727 + ))) 5728 + (% class=~"blogDay~" ~%)((( 5729 + #end 5730 + #displayBlogDate($entryDate) 5731 + #set ($blogDay = $entryDateStr) 5732 + #end 5733 + ## Finally, display the entry. 5734 + #displayEntry($entryDoc $entryObj $onlyExtract $shouldDisplayTitles true) 5735 + #end 5736 + #end 5737 + )))## blogDay 5738 + )))## hfeed 5739 +#end 5740 +## 5741 +## 5742 +## 5743 +#** 5744 + * Get the entry object, either a new BlogPost or an old Article. 5745 + * 5746 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 5747 + * @param entryObj The resulting xobject of the blog post. 5748 + *### 5749 +#macro(getEntryObject $entryDoc $__entryObj) 5750 + #set($result = $entryDoc.getObject(~"$\{blogPostClassname}~")) 5751 + #if(!$result) 5752 + #set($result = $entryDoc.getObject(~"$\{oldArticleClassname}~")) 5753 + #end 5754 + ## NOTE: The reason we put an underscore in front of the variable name is to prevent the following line from 5755 + ## overwriting the $entryObj variable that may be defined before this macro is called. Of course, $__entryObj may be 5756 + ## overwritten in this case but it's less likely to have such a variable defined before. 5757 + #set ($__entryObj = $NULL) 5758 + #setVariable (~"$__entryObj~" $result) 5759 +#end 5760 +## 5761 +## 5762 +## 5763 +#** 5764 + * Gets the date associated with a blog entry. This is the publication date. For unpublished entries, initially this is 5765 + * the document creation date, but can be edited by the user. 5766 + * 5767 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 5768 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 5769 + * @param result The resulting date, an instance of <tt>java.util.Date</tt>. 5770 + *### 5771 +#macro(getEntryDate $entryDoc $entryObj $result) 5772 + #set ($result = $NULL) 5773 + #setVariable (~"$result~" $entryObj.getProperty('publishDate').value) 5774 +#end 5775 +## 5776 +## 5777 +## 5778 +#** 5779 + * Displays a date, nicely formatted as a calendar page. 5780 + * 5781 + * @param date The date to display, an instance of <tt>java.util.Date</tt>. 5782 + *### 5783 +#macro(displayBlogDate $date) 5784 + #set($year = $xwiki.formatDate($date, 'yyyy')) 5785 + ## 3 letter month name, like Jan, Dec. 5786 + #set($month = $xwiki.formatDate($date, 'MMM')) 5787 + ## Uncomment to get a full length month name, like January, December. 5788 + ## TODO: this could be defined somewhere in the blog style. 5789 + ## #set($month = $xwiki.formatDate($date, 'MMMM')) 5790 + #set($day = $xwiki.formatDate($date, 'dd')) 5791 + (% class=~"blogdate~" ~%) 5792 + == (% class=~"month~" ~%)$month(%~%) (% class=~"day~" ~%)$day(%~%) (% class=~"year~" ~%)$year(%~%) == 5793 +#end 5794 +## 5795 +## 5796 +## 5797 +#** 5798 + * Displays a blog article: management tools, header, content, footer. 5799 + * 5800 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 5801 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 5802 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 5803 + * @param shouldDisplayTitle If <tt>true</tt>, display the blog title (blog posts shouldn't display the title 5804 + * when they're displayed alone on their page since it's the page title which is used in this case) 5805 + * @param shouldDisplayActions If <tt>true</tt>, display the blog post actions buttons 5806 + *### 5807 +#macro(displayEntry $entryDoc $entryObj $onlyExtract $shouldDisplayTitle $shouldDisplayActions) 5808 + ## Only articles with an explicit hidden setting or an explicit unpublished setting are hidden 5809 + #isPublished($entryObj $isPublished) 5810 + #isHidden($entryObj $isHidden) 5811 + #if($doc.fullName == $entryDoc.fullName) 5812 + (% class=~"hentry single-article~" ~%)((( 5813 + #else 5814 + (% class=~"hentry#if(!$isPublished) unpublished-article#elseif($isHidden) hidden-article#end~" ~%)((( 5815 + #end 5816 + #if ($shouldDisplayActions) 5817 + #displayEntryTools($entryDoc $entryObj) 5818 + #end 5819 + #if($shouldDisplayTitle) 5820 + #displayEntryTitle($entryDoc $entryObj) 5821 + #end 5822 + #if($doc.fullName == $entryDoc.fullName) 5823 + #if(!$isPublished) 5824 + \{\{warning}}\{\{translation key='blog.code.published'/}}\{\{/warning}} 5825 + #elseif($isHidden) 5826 + \{\{warning}}\{\{translation key='blog.code.hidden'/}}\{\{/warning}} 5827 + #end 5828 + #end 5829 + #displayEntryContent($entryDoc $entryObj $onlyExtract) 5830 + #displayEntryFooter($entryDoc $entryObj) 5831 + )))## hentry 5832 +#end 5833 +## 5834 +## 5835 +## 5836 +#** 5837 + * Checks if the provided blog is published or not. 5838 + * 5839 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 5840 + * @param isPublished The resulting boolean, true if the entry is considered published. 5841 + *### 5842 +#macro(isPublished $entryObj $isPublished) 5843 + #set ($isPublished = $NULL) 5844 + ## This should work for both old articles, which don't have the 'published' property at all, and 5845 + ## are considered published by default, and new entries, that should have 1 if published. 5846 + #if (~"$!\{entryObj.getProperty('published').value}~" != '0') 5847 + #setVariable (~"$isPublished~" true) 5848 + #else 5849 + #setVariable (~"$isPublished~" false) 5850 + #end 5851 +#end 5852 +## 5853 +## 5854 +## 5855 +#** 5856 + * Checks if the provided blog is hidden or not. 5857 + * 5858 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass<tt> xclass. 5859 + * @param isHiddel The resulting boolean, true if the entry is considered hidden. 5860 + *### 5861 +#macro(isHidden $entryObj $isHidden) 5862 + #set ($isHidden = $NULL) 5863 + ## This should work for both old articles, which don't have the 'hidden' property at all, and 5864 + ## are considered visible by default, and new entries, that should have 1 if hidden. 5865 + #if (~"$!\{entryObj.getProperty('hidden').value}~" == '1') 5866 + #setVariable (~"$isHidden~" true) 5867 + #else 5868 + #setVariable (~"$isHidden~" false) 5869 + #end 5870 +#end 5871 +## 5872 +## 5873 +## 5874 +#** 5875 + * Displays several ~"tools~" for manipulating blog posts: hide/show, publish, edit. 5876 + * 5877 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 5878 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 5879 + *### 5880 +#macro(displayEntryTools $entryDoc $entryObj) 5881 + #if($xcontext.action == 'view') 5882 + (% class=~"blog-entry-toolbox~" ~%)((( 5883 + #displayPublishButton($entryDoc $entryObj) 5884 + #displayHideShowButton($entryDoc $entryObj) 5885 + #displayEditButton($entryDoc $entryObj) 5886 + #displayDeleteButton($entryDoc $entryObj) 5887 + ))) 5888 + #end 5889 +#end 5890 +## 5891 +## 5892 +## 5893 +#** 5894 + * Displays the publish button to the entry <strong>creator</strong>, if the article is not published yet. 5895 + * 5896 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 5897 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 5898 + * @todo AJAX calls. 5899 + *### 5900 +#macro(displayPublishButton $entryDoc $entryObj) 5901 + #isPublished($entryObj $isPublished) 5902 + #if(!$isPublished && $entryDoc.creator == $xcontext.user && $xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 5903 + [[#toolImage('world')>>path:$blogPublisher.getURL('view', ~"entryName=$\{escapetool.url($entryDoc.fullName)}&xredirect=$\{escapetool.url($thisURL)}&form_token=$!\{services.csrf.getToken()}~")||title=~"$services.localization.render('blog.code.notpublished')~"]]## 5904 + #end 5905 +#end 5906 +## 5907 +## 5908 +## 5909 +#** 5910 + * Displays the hide or show button to the entry <strong>creator</strong>, if the article is already published. 5911 + * 5912 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 5913 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 5914 + *### 5915 +#macro(displayHideShowButton $entryDoc $entryObj) 5916 + #isPublished($entryObj $isPublished) 5917 + #isHidden($entryObj $isHidden) 5918 + ## Only published articles can be hidden. Unpublished articles are considered already hidden. 5919 + #if($isPublished && $entryDoc.creator == $xcontext.user && $xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 5920 + #set ($queryString = \{ 5921 + 'xredirect' : $thisURL, 5922 + 'form_token' : $services.csrf.getToken() 5923 + }) 5924 + #if ($isHidden) 5925 + #set ($discard = $queryString.putAll(\{ 5926 + ~"$\{entryObj.getxWikiClass().getName()}_$\{entryObj.number}_hidden~" : 0, 5927 + 'comment' : $services.localization.render('blog.code.madevisible') 5928 + })) 5929 + #set ($lockURL = $entryDoc.getURL('save', $escapetool.url($queryString))) 5930 + [[#toolImage('unlock')>>path:$lockURL||class=~"blog-tool-show~" title=~"$services.localization.render('blog.code.makevisible')~"]]## 5931 + #else 5932 + #set ($discard = $queryString.putAll(\{ 5933 + ~"$\{entryObj.getxWikiClass().getName()}_$\{entryObj.number}_hidden~" : 1, 5934 + 'comment' : $services.localization.render('blog.code.hid') 5935 + })) 5936 + #set ($lockURL = $entryDoc.getURL('save', $escapetool.url($queryString))) 5937 + [[#toolImage('lock')>>path:$lockURL||class=~"blog-tool-hide~" title=~"$services.localization.render('blog.code.hide')~"]]## 5938 + #end 5939 + #end 5940 +#end 5941 +## 5942 +## 5943 +## 5944 +#** 5945 + * Displays the edit button to those that can edit the article. 5946 + * 5947 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 5948 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 5949 + *### 5950 +#macro(displayEditButton $entryDoc $entryObj) 5951 + #if($xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 5952 + ## Call getDefaultEditMode() for backward compatibility with older blog posts. 5953 + [[#toolImage('pencil')>>path:$entryDoc.getURL('edit')||title=~"$services.localization.render('blog.code.editpost')~"]]## 5954 + #end 5955 +#end 5956 +## 5957 +## 5958 +## 5959 +#** 5960 + * Displays the delete button to those that can edit the article. 5961 + * 5962 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 5963 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 5964 + * @todo AJAX calls. 5965 + *### 5966 +#macro(displayDeleteButton $entryDoc $entryObj) 5967 + #if($xwiki.hasAccessLevel('delete', $xcontext.user, $entryDoc.fullName)) 5968 + [[#toolImage('cross')>>path:$entryDoc.getURL('delete')||title=~"$services.localization.render('blog.code.deletepost')~"]]## 5969 + #end 5970 +#end 5971 +## 5972 +## 5973 +## 5974 +#** 5975 + * Displays the title of the entry. 5976 + * 5977 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 5978 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 5979 + *### 5980 +#macro(displayEntryTitle $entryDoc $entryObj) 5981 + #if($doc.fullName == $entryDoc.fullName) 5982 + (% class=~"entry-title~" ~%) 5983 + = $services.rendering.escape($entryDoc.display('title', 'view', $entryObj), $xwiki.getCurrentContentSyntaxId()) = 5984 + #else 5985 + (% class=~"entry-title~" ~%) 5986 + === [[$services.rendering.escape($entryDoc.display('title', 'view', $entryObj),$xwiki.getCurrentContentSyntaxId())>>doc:$services.rendering.escape($services.model.serialize($entryDoc.getDocumentReference(),'default'),$xwiki.getCurrentContentSyntaxId())]] === 5987 + #end 5988 +#end 5989 +## 5990 +## 5991 +## 5992 +#** 5993 + * Displays the body of the entry. 5994 + * 5995 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 5996 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 5997 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 5998 + *### 5999 +#macro(displayEntryContent $entryDoc $entryObj $onlyExtract) 6000 + (% class=~"#if($onlyExtract)entry-summary#\{else}entry-content#end~" ~%)((( 6001 + #getEntryContent($entryDoc $entryObj $onlyExtract $entryContent) 6002 + ## FIXME: This causes the blog's content to not be annotatable. See http://jira.xwiki.org/browse/XWIKI-6328 6003 + ## Should probably be replaced by a display macro call with a reference to the object property holding the post's content 6004 + \{\{html wiki=~"false~"}}$entryDoc.getRenderedContent($entryContent, $entryDoc.syntax.toIdString())\{\{/html}} 6005 + ))) ## entry-content 6006 + (% class=~"clearfloats~" ~%)((())) 6007 +#end 6008 +## 6009 +## 6010 +## 6011 +#** 6012 + * Extracts the body of the entry that should be displayed. If <tt>onlyExtract</tt> is <tt>true</tt>, display the content 6013 + * of the <tt>extract</tt> field (if not empty). 6014 + * 6015 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 6016 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 6017 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 6018 + * @param entryContent The resulting content. 6019 + * @param removeEllipsis If <tt>true</tt>, then don't display an ellipsis at the end of the content (only used when 6020 + * <tt>onlyExtract</tt> is <tt>true</tt>) 6021 + *### 6022 +#macro(getEntryContent $entryDoc $entryObj $onlyExtract $entryContent $removeEllipsis) 6023 + #if ($onlyExtract) 6024 + ## Note: We trim the summary so that if there's some white space it won't cause the summary to be used instead 6025 + ## of the content. 6026 + #set ($macro.result = $entryObj.getProperty('extract').value.trim()) 6027 + #end 6028 + #if(~"$!macro.result~" == '') 6029 + #set($macro.result = $entryObj.getProperty('content').value) 6030 +#* Disabled until the content can be cleanly cut. 6031 +* #if($onlyExtract && $result.length()>$maxchars) 6032 +* #set($i = $macro.result.lastIndexOf(~" ~", $maxchars)) 6033 +* #set($i = $i + 1) 6034 +* #set($macro.result = ~"$\{macro.result.substring(0,$i)} *[...>$\{entryDoc.fullName}]*~") 6035 +* #end 6036 +## *### 6037 + #elseif (!$removeEllipsis) 6038 + #if($entryDoc.syntax.toIdString() == 'xwiki/1.0') 6039 + #set($macro.result = ~"$\{macro.result} <a href='$\{entryDoc.getExternalURL()}' title='$services.localization.render('blog.code.readpost')'>...</a>~") 6040 + #elseif($entryDoc.syntax.toIdString() == 'xwiki/2.0' || $entryDoc.syntax.toIdString() == 'xwiki/2.1') 6041 + #set($macro.result = ~"$\{macro.result} [[...>>$\{services.rendering.escape($entryDoc.getFullName(), $entryDoc.getSyntax())}||title='$services.localization.render('blog.code.readpost')']]~") 6042 + #end 6043 + #end 6044 + #set ($entryContent = $NULL) 6045 + #setVariable (~"$entryContent~" $macro.result) 6046 +#end 6047 +## 6048 +## 6049 +## 6050 +#** 6051 + * Displays the footer of the entry. 6052 + * 6053 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 6054 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 6055 + *### 6056 +#macro(displayEntryFooter $entryDoc $entryObj) 6057 + (% class=~"entry-footer~" ~%)((( 6058 + #isPublished($entryObj $isPublished) 6059 + (% class='entry-author-label' ~%) 6060 + #if($isPublished) 6061 + \{\{translation key='blog.code.postedby'/}} ## 6062 + #else 6063 + \{\{translation key='blog.code.createdby'/}} ## 6064 + #end 6065 + \{\{html wiki=~"false~" clean=~"false~"}}<span class='author vcard'>#userfn($entryDoc.creator)</span>\{\{/html}} ## 6066 + #getEntryDate($entryDoc $entryObj $entryDate) 6067 + #listCategories($entryObj) #* 6068 + ## Since the publish date and update date are not set at the exact same time, there could be a small difference that 6069 + ## we assume cannot be more than 3 seconds. 6070 + *#(% class=~"separator~" ~%)·(%~%) [[\{\{translation key='blog.code.permalink'/}}>>$services.rendering.escape($entryDoc,'xwiki/2.0')||rel=~"bookmark~"]] ## 6071 + #if ($showcomments) 6072 + (% class=~"separator~" ~%)·(%~%) [[\{\{translation key='blog.code.comments'/}}>>$services.rendering.escape($entryDoc,'xwiki/2.0')||anchor=~"Comments~"]] (% class=~"itemCount~" ~%)($entryDoc.comments.size())(%~%) ## 6073 + #end ## 6074 + #if($entryDoc != $doc) ## 6075 + #displayEntryBlogLocation($entryDoc $entryObj) ## 6076 + #end 6077 + )))## entry-footer 6078 +#end 6079 +## 6080 +## 6081 +#** 6082 + * Display the blog for the entry (if it is not the currently displayed blog) 6083 + * 6084 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 6085 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 6086 + *### 6087 +#macro(displayEntryBlogLocation $entryDoc $entryObj) 6088 + #getBlogPostsLocation($entryDoc.getSpace() $blogPostsLocation) ## 6089 + #if(~"$!blogPostsLocation~" != ~"~") ## 6090 + #set($blogSpaceRef = $services.model.resolveSpace($blogPostsLocation)) ## 6091 + #set($blogDocRef = $services.model.createDocumentReference($services.model.getEntityReference('DOCUMENT', 'default').getName(), $blogSpaceRef)) ## 6092 + #if($doc.documentReference != $blogDocRef) ## 6093 + #set($blogDoc = $xwiki.getDocument($blogDocRef)) ## 6094 + #set($blogObj = $blogDoc.getObject('Blog.BlogClass')) ## 6095 + #if($blogObj) (% class=~"blogrefpath~" ~%)((( ## 6096 + $services.localization.render('blog.code.postedin') \{\{html clean=~"false~" wiki=~"false~"}} #hierarchy($blogDocRef, \{'selfIsActive' : true, 'local': true}) \{\{/html}} ## 6097 + )))(%~%)## 6098 + #end 6099 + #end 6100 + #end 6101 +#end 6102 +## 6103 +## 6104 +## 6105 +## 6106 +#** 6107 + * List the categories an entry belongs to. Used in the footer. The categories are instances of <tt>Blog.CategoryClass</tt>. 6108 + * 6109 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 6110 + *### 6111 +#macro(listCategories $entryObj) 6112 + #if($entryObj.getxWikiClass().getName() == $blogPostClassname) 6113 + #set($categories = $entryObj.getProperty('category').value) 6114 + #set($first = true) 6115 + #if($categories.size() > 0) 6116 + #foreach($category in $categories) 6117 + #set($categoryDoc = $!xwiki.getDocument($category)) 6118 + #if(!$categoryDoc.isNew() && $categoryDoc.getObject($\{blogCategoryClassname})) 6119 + #if($foreach.count == 1) 6120 + (% class='separator' ~%)·(%~%) $services.localization.render('blog.code.categories') ## 6121 + #else 6122 + , ## 6123 + #end## 6124 + [[$!\{categoryDoc.getObject($blogCategoryClassname).getValue('name')}>>$\{category}||rel='tag']]## 6125 + #end## 6126 + #end## 6127 + #end 6128 + #end 6129 +#end 6130 +## 6131 +## 6132 +## 6133 +#** 6134 + * Displays blog pagination links (older and newer entries). 6135 + * 6136 + * @param blogDoc the XDocument holding the blog definition object. 6137 + *### 6138 +#macro(displayNavigationLinks $blogDoc) 6139 + (% class=~"clearfloats~" ~%)((())) 6140 + #getBlogDisplayType($blogDoc $displayType) 6141 + #if($displayType == 'weekly') 6142 + (% class=~"pagingLinks~" ~%)((( 6143 + #getRequestedWeek($weekDate) 6144 + $weekDate.addWeeks(-1)## 6145 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.previousweek'/}}>>$doc.documentReference.name||queryString=~"year=$weekDate.weekyear&week=$weekDate.weekOfWeekyear~" class=~"button~"]]**(%~%) 6146 + #sep() 6147 + $weekDate.addWeeks(2)## 2 because we already subtracted 1 above 6148 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.nextweek'/}} »>>$doc.documentReference.name||queryString=~"year=$weekDate.weekyear&week=$weekDate.weekOfWeekyear~" class=~"button~"]]**(%~%) 6149 + ))) 6150 + #elseif($displayType == 'monthly') 6151 + (% class=~"pagingLinks~" ~%)((( 6152 + #getRequestedMonth($monthDate) 6153 + $monthDate.addMonths(-1)## 6154 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.previousmonth'/}}>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"year=$monthDate.year&month=$monthDate.monthOfYear~" class=~"button~"]]**(%~%) 6155 + #sep() 6156 + $monthDate.addMonths(2)## 2 because we already subtracted 1 above 6157 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.nextmonth'/}} »>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"year=$monthDate.year&month=$monthDate.monthOfYear~" class=~"button~"]]**(%~%) 6158 + ))) 6159 + #elseif($displayType == 'all') 6160 + #else 6161 + ## Paginated 6162 + #if(($totalPages > 1)) 6163 + #set($queryString = '') 6164 + #foreach($p in $request.getParameterNames()) 6165 + #if($p != 'page' && $p != 'ipp') 6166 + #foreach($v in $request.getParameterValues($p)) 6167 + #set($queryString = ~"$\{queryString}&$\{escapetool.url($p)}=$\{escapetool.url($v)}~") 6168 + #end 6169 + #end 6170 + #end 6171 + (% class=~"pagingLinks~" ~%)((( 6172 + #if ($currentPageNumber < $totalPages) 6173 + #set($currentPageNumber = $currentPageNumber + 1) 6174 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.olderposts'/}}>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"page=$\{currentPageNumber}&ipp=$\{itemsPerPage}$queryString~" class=~"button~"]]**(%~%) 6175 + #set($currentPageNumber = $currentPageNumber - 1) 6176 + #end 6177 + #if ($currentPageNumber > 1) 6178 + #if ($currentPageNumber < $totalPages) 6179 + #sep() 6180 + #end 6181 + #set($currentPageNumber = $currentPageNumber - 1) 6182 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.newerposts'/}} »>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"page=$\{currentPageNumber}&ipp=$\{itemsPerPage}$queryString~" class=~"button~"]]**(%~%) 6183 + #set($currentPageNumber = $currentPageNumber + 1) 6184 + #end 6185 + (% class=~"clear~" ~%)(%~%) 6186 + )))## pagingLinks 6187 + #end 6188 + #end 6189 +#end 6190 +## 6191 +## 6192 +## 6193 +#** 6194 + * Displays a message box with ~"publish~" icon. 6195 + * 6196 + * @param message A text message concerning blog article publishing 6197 + *### 6198 +#macro(publishMessageBox $message) 6199 +(% class=~"plainmessage publish-message~" ~%)((($services.icon.render('world') $message))) 6200 +#end 6201 +#** 6202 + * Displays a message box with ~"show/hide~" icon. 6203 + * 6204 + * @param message A text message concerning blog article hiding 6205 + *### 6206 +#macro(hideMessageBox $message) 6207 +(% class=~"plainmessage hide-message~" ~%)((($services.icon.render('unlock') $message))) 6208 +#end 6209 +## 6210 +## 6211 +## 6212 +#** 6213 + * Determine the requested week, for using in a weekly-indexed blog. The relevant request parameters are 6214 + * <tt>year</tt> and <tt>week</tt>. By default, the current week is used. 6215 + * 6216 + * @param monthDate The resulting week, a JODATime MutableDateTime. 6217 + *### 6218 +#macro(getRequestedWeek $weekDate) 6219 + #set ($weekDate = $NULL) 6220 + #setVariable (~"$weekDate~" $xwiki.jodatime.mutableDateTime) 6221 + #if(~"$!\{request.year}~" != '') 6222 + #set ($discard = $weekDate.setYear($numbertool.toNumber($request.year).intValue())) 6223 + #end 6224 + #if(~"$!\{request.week}~" != '') 6225 + #set ($discard = $weekDate.setWeekOfWeekyear($numbertool.toNumber($request.week).intValue())) 6226 + #end 6227 +#end 6228 +## 6229 +## 6230 +## 6231 +#** 6232 + * Determine the requested month, for using in a monthly-indexed blog. The relevant request parameters are 6233 + * <tt>year</tt> and <tt>month</tt>. By default, the current month is used. 6234 + * 6235 + * @param monthDate The resulting month, a JODATime MutableDateTime. 6236 + *### 6237 +#macro(getRequestedMonth $monthDate) 6238 + #set ($monthDate = $NULL) 6239 + #setVariable (~"$monthDate~" $xwiki.jodatime.mutableDateTime) 6240 + #if(~"$!\{request.year}~" != '') 6241 + #set ($discard = $monthDate.setYear($numbertool.toNumber($request.year).intValue())) 6242 + #end 6243 + #if(~"$!\{request.month}~" != '') 6244 + #set ($discard = $monthDate.setMonthOfYear($numbertool.toNumber($request.month).intValue())) 6245 + #end 6246 +#end 6247 +## 6248 +## 6249 +## 6250 +#** 6251 + * Retrieve a blog property (title, display type, etc). 6252 + * 6253 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 6254 + * @param propertyName The name of the property to be retrieved. One of the <tt>Blog.BlogClass</tt>'s properties. 6255 + * @param defaultValue The default value to use in case the blog object does not define one. 6256 + * @param propertyValue The resulting value. 6257 + *### 6258 +#macro(getBlogProperty $blogDoc $propertyName $defaultValue $propertyValue) 6259 + #set($result = ~"$!\{blogDoc.getObject($\{blogClassname}).getProperty($propertyName).value}~") 6260 + #if($result == '') 6261 + #set($result = $defaultValue) 6262 + #end 6263 + #set ($propertyValue = $NULL) 6264 + #setVariable (~"$propertyValue~" $result) 6265 +#end 6266 + 6267 +#** 6268 + * If an error occurs when executing an action, set a specific response status and display an error message. 6269 + * 6270 + * @param status The response status. 6271 + * @param text The user readable error to be displayed. Can be a translation key. 6272 + * @param parameters The parameters to use when decoding the translation key. 6273 + *### 6274 +#macro(blog__actionResponseError $status $text $parameters) 6275 + $response.setStatus($status) 6276 + #if($request.ajax) 6277 + $services.localization.render($text, $!parameters) 6278 + #else 6279 + \{\{error}}$services.localization.render($text, $!parameters)\{\{/error}} 6280 + #end 6281 +#end 6282 +## 6283 +## 6284 +## 6285 +#** 6286 + * Check if a blog is the Default blog (The one in the 'Blog' space). 6287 + * 6288 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 6289 + * @param isDefault The resulting boolean. 6290 + *### 6291 +#macro(isDefaultBlog $blogDoc $isDefault) 6292 + #set ($result = false) 6293 + #if ($blogDoc.space == 'Blog') 6294 + #set ($result = true) 6295 + #end 6296 + #setVariable(~"$isDefault~" $result) 6297 +#end 6298 +## 6299 +## 6300 +## 6301 +#** 6302 + * Retrieve the blog posts location (space). 6303 + * 6304 + * @param blogSpace The blog space. It should contain a document with a <tt>Blog.BlogClass</tt> object 6305 + * @param postsLocation The resulting location. 6306 + *### 6307 +#macro(getBlogPostsLocation $blogSpace $postsLocation) 6308 + #getBlogDocument($blogSpace $blogDoc) 6309 + #getBlogProperty($blogDoc 'postsLocation' $blogDoc.space $result) 6310 + #set ($postsLocation = $NULL) 6311 + #setVariable (~"$postsLocation~" $result) 6312 +#end 6313 +## 6314 +## 6315 +## 6316 +#** 6317 + * Retrieve the blog categories location (space). 6318 + * 6319 + * @param blogSpace The blog space. It should contain a document with a <tt>Blog.BlogClass</tt> object 6320 + * @param categoriesLocation The resulting location. 6321 + *### 6322 +#macro(getBlogCategoriesLocation $blogSpace $categoriesLocation) 6323 + #getBlogDocument($blogSpace $blogDoc) 6324 + #getBlogProperty($blogDoc 'categoriesLocation' 'Blog' $result) 6325 + #set ($postsLocation = $NULL) 6326 + #setVariable (~"$categoriesLocation~" $result) 6327 +#end 6328 +###** 6329 + * Return the Query for selecting the blog posts based on the context where the posts are displayed. 6330 + * for example there is 4 different panel contexts: 6331 + * aBlog.aPost or aBlog.WebHome 6332 + * aCategorySpace.aCategory 6333 + * aCategorySpace.WebHome 6334 + * Blog.aPost or Blog.WebHome 6335 + * 6336 + * @param query The query for selecting blog entries. 6337 + * @param queryParams The parameters to bind with the generated query. 6338 + * @param targetDoc The document in which the articles will be displayed. 6339 + *### 6340 +#macro(getAllBlogPostsQueryBasedOnDisplayContext $targetDoc $query $queryParams) 6341 + #set ($query = $NULL) 6342 + #set ($queryParams = $NULL) 6343 + #getCategoryAllBlogPostsQuery($resultQuery) 6344 + #set ($resultQueryParams = \{}) 6345 + #set($sanitizedSpace = $targetDoc.space.replaceAll('([%_!])', '!$1').concat('.%')) 6346 + #if ($targetDoc.space == $defaultBlogSpace && !$targetDoc.getObject($blogCategoryClassname)) 6347 + ## Get all posts that are in the default blog space('Blog') or in category 'Blog.%' 6348 + #set ($discard = $resultQueryParams.put('space', $targetDoc.space)) 6349 + #set ($discard = $resultQueryParams.put('catList', $sanitizedSpace)) 6350 + #set($resultQuery = ~"$\{resultQuery} and (doc.space = :space or catList like :catList escape '!')~") 6351 + #elseif($targetDoc.getObject($blogCategoryClassname)) 6352 + ## Get all posts that are in a category aCategorySpace.aCategory 6353 + #set ($discard = $resultQueryParams.put('catList', $targetDoc.fullName)) 6354 + #set($resultQuery = ~"$\{resultQuery} and (catList=:catList)~") 6355 + #elseif($targetDoc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 6356 + ## Get all posts that are in a category aCategorySpace.% 6357 + #set ($discard = $resultQueryParams.put('catList', $sanitizedSpace)) 6358 + ## Exclude incategorized posts 6359 + #set ($excludedCategory = ~"$\{targetDoc.space}.WebHome~") 6360 + #if ($targetDoc.space == $defaultBlogSpace) 6361 + #set ($excludedCategory = ~"Blog.Categories.WebHome~") 6362 + #end 6363 + #set ($discard = $resultQueryParams.put('excludedCategory', $excludedCategory)) 6364 + #set($resultQuery = ~"$\{resultQuery} and (catList<>:excludedCategory) and (catList like :catList escape '!')~") 6365 + #else 6366 + ## Get all posts in blog space aBlog 6367 + #getAllBlogPostsQuery($resultQuery) 6368 + #getBlogPostsLocation($targetDoc.space $postsLocation) 6369 + #set ($discard = $resultQueryParams.put('space', $postsLocation)) 6370 + #set($resultQuery = ~"$\{resultQuery} and doc.space = :space~") 6371 + #end 6372 + #setVariable(~"$query~" $resultQuery) 6373 + #setVariable(~"$queryParams~" $resultQueryParams) 6374 +#end 6375 +## 6376 +## 6377 +## 6378 +###** 6379 + * Display blog posts based on the context where the posts are displayed. 6380 + * for example there is 4 different panel contexts: 6381 + * aBlog.aPost or aBlog.WebHome 6382 + * aCategorySpace.aCategory 6383 + * aCategorySpace.WebHome 6384 + * Blog.aPost or Blog.WebHome 6385 + * 6386 + * @param targetDoc The document in which the articles will be displayed. 6387 + * @param postsVisiblity The visibility of the posts to display: recent, unpublished, ... 6388 + * @param layout Layout of the the posts to display 6389 + * @param layoutParams additional layout parameters, it is a string like 'param1=val1|...|paramN=valueN' 6390 + * @param limit the number of posts to display 6391 + *### 6392 +#macro(displayBlogPostsBasedOnContext $targetDoc $postsVisiblity $layout $layoutParams $limit) 6393 + #if ($postLayout == 'full') 6394 + #set ($macro.paginated = 'yes') 6395 + #end 6396 + #if ($targetDoc.space == $defaultBlogSpace && !$targetDoc.getObject($blogCategoryClassname)) 6397 + ## Display all posts that are in the default blog space('Blog') or in category 'Blog.%' 6398 + #getBlogPostsLayout($xwiki.getDocument(~"$\{defaultBlogSpace}.WebHome~") $postsLayout) 6399 + #if (~"$!layout~" == '') 6400 + #set ($layout = $postsLayout) 6401 + #end 6402 + #if ($postsVisiblity == 'recent') 6403 + \{\{blogpostlist blog=~"$\{defaultBlogSpace.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')}.WebHome~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 6404 + #elseif($postsVisiblity == 'unpublished') 6405 + \{\{blogpostlist blog=~"$\{defaultBlogSpace.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')}.WebHome~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 6406 + #end 6407 + #elseif($targetDoc.getObject($blogCategoryClassname)) 6408 + ## Display all posts that are in a category aCategorySpace.aCategory 6409 + #getBlogDocumentForCategoriesSpace($targetDoc.space $blogDoc) 6410 + #getBlogPostsLayout($blogDoc $postsLayout) 6411 + #if (~"$!layout~" == '') 6412 + #set ($layout = $postsLayout) 6413 + #end 6414 + #if ($postsVisiblity == 'recent') 6415 + \{\{blogpostlist category=~"$targetDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 6416 + #elseif($postsVisiblity == 'unpublished') 6417 + \{\{blogpostlist category=~"$targetDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 6418 + #end 6419 + #elseif($targetDoc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 6420 + ## Display all posts that are in a category aCategorySpace.% 6421 + #getBlogDocumentForCategoriesSpace($targetDoc.space $blogDoc) 6422 + #getBlogPostsLayout($blogDoc $postsLayout) 6423 + #if (~"$!layout~" == '') 6424 + #set ($layout = $postsLayout) 6425 + #end 6426 + #if ($postsVisiblity == 'recent') 6427 + \{\{blogpostlist category=~"$targetDoc.space.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 6428 + #elseif($postsVisiblity == 'unpublished') 6429 + \{\{blogpostlist category=~"$targetDoc.space.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 6430 + #end 6431 + #else 6432 + ## Display all posts in blog space aBlog 6433 + #getBlogDocument($targetDoc.space $blogDoc) 6434 + #getBlogPostsLayout($blogDoc $postsLayout) 6435 + #if (~"$!layout~" == '') 6436 + #set ($layout = $postsLayout) 6437 + #end 6438 + #if ($postsVisiblity == 'recent') 6439 + \{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 6440 + #elseif($postsVisiblity == 'unpublished') 6441 + \{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 6442 + #end 6443 + #end 6444 +#end 6445 +## 6446 +## 6447 +## 6448 +#** 6449 + * Bind parameters to a query object. 6450 + * 6451 + * @param queryObj The query object (org.xwiki.query.internal.ScriptQuery object) 6452 + * @param queryParams the query parameters. 6453 + *### 6454 +#macro(bindQueryParameters $queryObj $queryParams) 6455 + #set ($output = $queryObj) 6456 + #foreach( $key in $queryParams.keySet() ) 6457 + #set($output = $queryObj.bindValue($key, $queryParams.get($key))) 6458 + #end 6459 + #setVariable(~"$queryObj~" $output) 6460 +#end 6461 +## 6462 +## 6463 +## 6464 +#** 6465 + * Determines the blogposts layout: Default layout (Calendar), Image thumbnail 6466 + * 6467 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>postsLayout</tt> 6468 + * property set. 6469 + * @param postsLayout The resulting string. If the blog object does not define anything, it is considered full (default). 6470 + *### 6471 +#macro(getBlogPostsLayout $blogDoc $postsLayout) 6472 + #getBlogProperty($blogDoc 'postsLayout' 'full' $res) 6473 + #set ($postsLayout = $NULL) 6474 + #setVariable (~"$postsLayout~" $res) 6475 +#end 6476 +## 6477 +## 6478 +## 6479 +#** 6480 + * Retrieve the blog document of a categories space, which is the blog descriptor that have its categoriesLocation property set to the categories space 6481 + * 6482 + * @param categoriesSpace A <tt>String</tt>, the name of the space. 6483 + * @param blogDoc The resulting XDocument. 6484 + *### 6485 +#macro(getBlogDocumentForCategoriesSpace $categoriesSpace $blogDoc) 6486 + #set($macro.blogDocs = $services.query.hql(~", BaseObject obj, StringProperty as categoriesLocationProp where obj.name = doc.fullName and obj.className = '$blogClassname' and obj.id=categoriesLocationProp.id.id and categoriesLocationProp.id.name='categoriesLocation' and categoriesLocationProp.value=:categoriesLocation order by doc.creationDate~").setLimit(1).setOffset(0).bindValue('categoriesLocation', $categoriesSpace).execute()) 6487 + #if($macro.blogDocs.size() > 0 && $categoriesSpace != $defaultBlogSpace) 6488 + #set($macro.result = $xwiki.getDocument($macro.blogDocs.get(0))) 6489 + #else 6490 + ## Fallback to Blog.WebHome, the default blog 6491 + #set($macro.result = $xwiki.getDocument('Blog.WebHome')) 6492 + #end 6493 + #set ($blogDoc = $NULL) 6494 + #setVariable (~"$blogDoc~" $macro.result) 6495 +#end" %) 6496 +((( 6497 +(% class="macro-placeholder hidden" %) 6498 +((( 6499 +macro:velocity 6500 +))) 6501 +))) 6502 +))) 6503 + 6504 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogPostLayoutMacros~"" %) 6505 +((( 6506 +(% class="macro-placeholder hidden" %) 6507 +((( 6508 +macro:include 6509 +))) 6510 + 6511 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|#** 6512 + * Extract the layout parameters from a string. 6513 + * 6514 + * @param layoutParamsString The string representation of the layout parameters. 6515 + * It should contain a String following this format ~"paramName1=Value1|paramName2=Value2|...|paramNameK=ValueK~" 6516 + * @param layoutsParameters The resulting layout parameters Map. 6517 + *### 6518 +#macro(extractLayoutParametersFromString $layoutParamsString $layoutsParameters) 6519 + #set ($layoutsParameters = $NULL) 6520 + #set ($macro.layoutParams = \{}) 6521 + #if (~"$!layoutParamsString~" != '') 6522 + #set ($macro.paramsArr = $layoutParamsString.split('\\|')) 6523 + #foreach ($item in $macro.paramsArr) 6524 + #set ($itemSplit = $item.split('=')) 6525 + #if ($itemSplit.size() == 2) 6526 + #set ($discard = $macro.layoutParams.put($itemSplit[0].trim(), $itemSplit[1].trim())) 6527 + #end 6528 + #end 6529 + #end 6530 + #setVariable(~"$layoutsParameters~" $macro.layoutParams) 6531 +#end" %) 6532 +((( 6533 +(% class="macro-placeholder hidden" %) 6534 +((( 6535 +macro:velocity 6536 +))) 6537 +))) 6538 +))) 6539 + 6540 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-| #macro(displaySmallPost $pDoc $pObj) 6541 + #initLayoutVars($pDoc $pObj) 6542 + <div class=~"col-xs-12 col-sm-6 next_blog~"> 6543 + <a href=~"$pDoc.uRL~" class=~"thumbnail~"> 6544 + <div class=~"row~"> 6545 + <div class=~"col-xs-4~"> 6546 + #if ($imageAtt) 6547 + <img src=~"$escapetool.xml($pDoc.getAttachmentURL($pObj.getValue('image'),'download',$imgQs))~" /> 6548 + #end 6549 + </div> 6550 + <div class=~"col-xs-8 art_det~"> 6551 + <p class=~"text-left~"> 6552 + #if($displayTitle)$!postTitle #end 6553 + <br/><span class=~"date_info~"> $!dateStr </span> 6554 + </p> 6555 + #displayPostDetails($pDoc) 6556 + </div> 6557 + </div> 6558 + </a> 6559 + </div> 6560 + #end 6561 + ## 6562 + ## 6563 + ## 6564 + #macro(displayPinnedPost $pDoc $pObj) 6565 + #initLayoutVars($pDoc $pObj) 6566 + <div class=~"col-xs-12 col-sm-4 col-md-4 blog_pin~"> 6567 + <a href=~"$pDoc.uRL~" class=~"all-post-link~"> 6568 + <div class=~"thumbnail~"> 6569 + #if ($displayTitle && $displayTitleFirstOnPinnedPosts) 6570 + <h5 class=~"title-top~">$!postTitle <br/> <span class=~"date_info~"> $!dateStr </span> </h5> 6571 + #end 6572 + #if ($imageAtt) 6573 + <img src=~"$escapetool.xml($pDoc.getAttachmentURL($pObj.getValue('image'),'download',$imgQs))~" /> 6574 + #end 6575 + <div class=~"caption~"> 6576 + #if ($displayTitle && !$displayTitleFirstOnPinnedPosts) 6577 + <h5 class=~"text-left~">$!postTitle <br/> <span class=~"date_info~"> $!dateStr </span> </h5> 6578 + #end 6579 + #if ($displaySummaryOnPinnedPosts) 6580 + <div class=~"text-left post-summary~"> 6581 + #set ($postContent = $pObj.getProperty('extract').value) 6582 + $!pDoc.getRenderedContent($postContent, $pDoc.syntax.toIdString()) 6583 + </div> 6584 + #end 6585 + #displayPostDetails($pDoc) 6586 + </div> 6587 + </div> 6588 + </a> 6589 + </div> 6590 + #end 6591 + ## 6592 + ## 6593 + ## 6594 + #macro(formatPostDate $pDoc $pObj) 6595 + #set ($formattedDate = $NULL) 6596 + #set($dateStr = $datetool.format('medium', $postDate, $xcontext.locale)) 6597 + #if (~"$!dateStr~" != '') 6598 + #set ($dateArr = $dateStr.split(' ')) 6599 + #if ($dateArr.size() > 3) 6600 + #set ($dateStr = ~"~") 6601 + #foreach($s in $dateArr.subList(0, 3)) 6602 + #set ($dateStr = ~"$\{dateStr}$\{s} ~") 6603 + #end 6604 + #end 6605 + #end 6606 + #setVariable(~"$formattedDate~" $dateStr) 6607 + #end 6608 + ## 6609 + ## 6610 + ## 6611 + #macro(displayPostDetails $pDoc) 6612 + <div class=~"row post-details~"> 6613 + <div class=~"col-xs-8 detail~"> 6614 + #if ($services.like.displayButton($pDoc.documentReference)) 6615 + #set ($optLikeRecord = $services.like.getLikes($pDoc.documentReference)) 6616 + ## Retrieve the likes number in XWiki 12.9+ 6617 + #set ($likeNumber = $optLikeRecord.get()) 6618 + #if (!$stringtool.isNumeric($likeNumber.toString())) 6619 + ## Retrieve the likes number in XWiki versions greater than 12.7 and lower than 12.9. 6620 + #set ($likeNumber = $optLikeRecord.get().likeNumber) 6621 + #end 6622 + #if ($stringtool.isNumeric($likeNumber.toString())) 6623 + <div class=~"post-likes btn btn-default disabled badge~" 6624 + title=~"$escapetool.xml($services.localization.render('like.button.title', [$likeNumber]))~"> 6625 + $services.icon.renderHTML('heart') <span class=~"like-number~">$likeNumber</span> 6626 + </div> 6627 + #end 6628 + #elseif ($services.ratings) 6629 + #set ($ratingsConfigDoc = $services.ratings.getConfigurationDocument($pDoc.documentReference)) 6630 + #if ($ratingsConfigDoc.getValue('displayRatings') == 1) 6631 + <ul class=~"pull-left note list-inline~"> 6632 + #foreach ($x in [1..5]) 6633 + #set ($cls = ~"~") 6634 + #if ($x > $averageVote) 6635 + #set ($cls = ~"-o~") 6636 + #end 6637 + <li><span class=~"fa fa-star$\{cls}~"></span></li> 6638 + #end 6639 + </ul> 6640 + #end 6641 + #end 6642 + </div> 6643 + #if ($showPostComments) 6644 + <div class=~"col-xs-4 com_det~"> 6645 + <p class=~"text-right~">$services.icon.renderHTML('comments') ($!nbComments)</p> 6646 + </div> 6647 + #end 6648 + </div> 6649 + #end 6650 + ## 6651 + ## 6652 + ## 6653 + #macro(initLayoutVars $pDoc $pObj) 6654 + #set ($postTitle = $pDoc.display('title', 'view', $pObj)) 6655 + #set ($imageAtt = $pDoc.getAttachment($pObj.getValue('image'))) 6656 + #isPublished($pObj $isPublished) 6657 + #isHidden($pObj $isHidden) 6658 + #getEntryDate($pDoc $pObj $postDate) 6659 + #formatPostDate($postDate $dateStr) 6660 + #set ($averageVote = $mathtool.round($services.ratings.getAverageRating($pDoc.documentReference).getAverageVote())) 6661 + #set ($nbComments = $pDoc.getComments().size()) 6662 + #set ($showPostComments = true) 6663 + #if ($xwiki.getSpacePreferenceFor('showcomments', $pDoc.documentReference.lastSpaceReference, '1') == '0') 6664 + #set ($showPostComments = false) 6665 + #end 6666 + #end 6667 + ## 6668 + ## 6669 + ## 6670 + #macro(displayEditPinnedPostsButton) 6671 + #if (~"$!macroAdditionalParams.get('postIndex')~" == '0' && $services.security.authorization.hasAccess('edit', $pinnedPostsSourceDoc.fullName)) 6672 + #set ($discard = $xwiki.ssx.use('Blog.PinnedPostsSheet')) 6673 + #set ($discard = $xwiki.jsx.use('Blog.PinnedPostsSheet')) 6674 + #set ($editPinnedPostsURL = $xwiki.getURL($pinnedPostsSourceDoc.fullName, 'get', 'sheet=Blog.PinnedPostsSheet')) 6675 + #set ($newPinnedPostsObjectURL = $xwiki.getURL($pinnedPostsSourceDoc.fullName, 'objectadd', ~"classname=Blog.PinnedPostsClass&form_token=$!\{services.csrf.getToken()}~")) 6676 + <div class=~"edit-pinned-posts-container~"> 6677 + <div class=~"col-xs-12 col-sm-12 col-md-12 text-right edit-pinned-posts~"> 6678 + <a href=~"javascript:;~" class=~"editPinnedPosts~"> 6679 + <span class=~"glyphicon glyphicon-edit~" aria-hidden=~"true~"></span> $services.localization.render('blog.post.layout.cards.pinnedposts.edit') 6680 + </a> 6681 + <input type=~"hidden~" id=~"editPinnedPostsURL~" value=~"$escapetool.xml($editPinnedPostsURL)~"/> 6682 + <input type=~"hidden~" id=~"newPinnedPostsObjectURL~" value=~"$escapetool.xml($newPinnedPostsObjectURL)~"/> 6683 + </div> 6684 + </div> 6685 + #if (~"$!pinnedPostsObj~" == '') 6686 + <input type=~"hidden~" name=~"createPinnedPostsObject~" value=~"1~"/> 6687 + #end 6688 + #end 6689 + #end" %) 6690 +((( 6691 +(% class="macro-placeholder hidden" %) 6692 +((( 6693 +macro:velocity 6694 +))) 6695 +))) 6696 + 6697 +(% class="macro" data-macro="startmacro:velocity|-||-| #if ($services.security.authorization.hasAccess('view', $xcontext.macro.params.reference)) 6698 + #set ($postDoc = $xwiki.getDocument($xcontext.macro.params.reference)) 6699 + #getEntryObject($postDoc $postObj) 6700 + #if (~"$!postObj~" != '') 6701 + #extractLayoutParametersFromString($xcontext.macro.params.params $macroAdditionalParams) 6702 + ## 6703 + #set ($displayTitle = true) 6704 + #if (~"$!macroAdditionalParams.get('displayTitle')~" == 'false') 6705 + #set ($displayTitle = false) 6706 + #end 6707 + ## 6708 + #set ($displayTitleFirstOnPinnedPosts = false) 6709 + #if (~"$!macroAdditionalParams.get('displayTitleFirstOnPinnedPosts')~" == 'true') 6710 + #set ($displayTitleFirstOnPinnedPosts = true) 6711 + #end 6712 + ## 6713 + #set ($displaySummaryOnPinnedPosts = true) 6714 + #if (~"$!macroAdditionalParams.get('displaySummaryOnPinnedPosts')~" == 'false') 6715 + #set ($displaySummaryOnPinnedPosts = false) 6716 + #end 6717 + ## 6718 + #set ($postIndex = $numbertool.toNumber($macroAdditionalParams.get('postIndex')).intValue()) 6719 + #if ($postIndex == 0) 6720 + #set ($stopBlogPostsDisplay = false) 6721 + #end 6722 + #set ($listResultsCount = $numbertool.toNumber($macroAdditionalParams.get('list-results-count')).intValue()) 6723 + #set($discard = $xwiki.ssx.use('Blog.BlogPostLayoutCards')) 6724 + #set($scaleWidth = 600) 6725 + #set($imgQs=~"width=$scaleWidth~") 6726 + ## Display pinned posts 6727 + ## Get the list of pinned posts 6728 + #set ($pinnedPosts = []) 6729 + #set ($pinnedPostsSourceDoc = $NULL) 6730 + #if (~"$!macroAdditionalParams.get('forceDisplayPinnedPosts')~" == 'true' && ~"$!macroAdditionalParams.get('list-blog')~" != '') 6731 + #set ($pinnedPostsSourceDoc = $xwiki.getDocument($macroAdditionalParams.get('list-blog'))) 6732 + ## Note that the 'list-blog' parameter contains the value of the 'blog' parameter of the blogpostlist macro passed to the current layout macro 6733 + #end 6734 + #if (~"$!pinnedPostsSourceDoc~" != '') 6735 + #set ($pinnedPostsObj = $pinnedPostsSourceDoc.getObject('Blog.PinnedPostsClass')) 6736 + #if (~"$!pinnedPostsObj~" != '') 6737 + #set ($orderedPinnedPostsJSON = $pinnedPostsObj.getProperty('orderedPosts').value) 6738 + #if (~"$!orderedPinnedPostsJSON~" != '') 6739 + #set ($pinnedPosts = $jsontool.parse($orderedPinnedPostsJSON)) 6740 + #else 6741 + #set ($pinnedPosts = $pinnedPostsObj.getProperty('posts').value) 6742 + #end 6743 + #end 6744 + #if ($postIndex == 0 && $pinnedPosts.size() > 0) 6745 + \{\{html clean=~"false~"}} 6746 + #set ($x = 0) 6747 + #set ($showPinnedPostsButton = true) 6748 + #foreach ($pinnedPost in $pinnedPosts) 6749 + #if ($x == 0) 6750 + <div class=~"row flex-container~"> 6751 + #if ($showPinnedPostsButton) 6752 + #displayEditPinnedPostsButton() 6753 + #set($showPinnedPostsButton = false) 6754 + #end 6755 + #end 6756 + #set ($pinnedPostDoc = $xwiki.getDocument($pinnedPost)) 6757 + #getEntryObject($pinnedPostDoc $pinnedPostObj) 6758 + #if (~"$!pinnedPostObj~" != '') 6759 + #displayPinnedPost($pinnedPostDoc $pinnedPostObj) 6760 + #end 6761 + #set ($x = $mathtool.add($x, 1)) 6762 + #if ($x == 3) 6763 + #set ($x = 0) 6764 + </div> 6765 + #end 6766 + #end 6767 + #if ($mathtool.mod($x, 3) != 0) 6768 + </div> 6769 + #end 6770 + \{\{/html}} 6771 + 6772 + ## If the first post is a pinned post : this means that all the posts are pinned 6773 + ## In this case, avoid displaying the posts again after the pinned posts section. 6774 + #if ($pinnedPosts.contains($xcontext.macro.params.reference)) 6775 + #set ($stopBlogPostsDisplay = true) 6776 + #end 6777 + #elseif($postIndex == 0 && $pinnedPosts.size() == 0) 6778 + 6779 + \{\{html}} 6780 + <div class=~"row no-pinnded-posts~"> 6781 + #displayEditPinnedPostsButton() 6782 + </div> 6783 + \{\{/html}} 6784 + 6785 + #end 6786 + #end 6787 + #if (!$stopBlogPostsDisplay) 6788 + \{\{html clean=~"false~"}} 6789 + #if (~"$!nbDisplayedPosts~" == '' || $postIndex == 0) 6790 + #set ($nbDisplayedPosts = 0) 6791 + #end 6792 + #if($mathtool.mod($nbDisplayedPosts, 2) == 0) 6793 + #if ($nbDisplayedPosts != 0) 6794 + </div> 6795 + #set ($lastHtmlTag = 'c') 6796 + #end 6797 + #if ($nbDisplayedPosts == $mathtool.sub($listResultsCount, 1)) 6798 + <div class=~"row~"> 6799 + #set ($lastHtmlTag = 'o') 6800 + #set ($lastRowClass = ~"~") 6801 + #else 6802 + <div class=~"row flex-container~"> 6803 + #set ($lastHtmlTag = 'o') 6804 + #set ($lastRowClass = ~"flex~") 6805 + #end 6806 + #end 6807 + #displaySmallPost($postDoc $postObj) 6808 + #if ($nbDisplayedPosts == $mathtool.sub($listResultsCount, 1)) 6809 + </div> 6810 + #set ($lastHtmlTag = 'c') 6811 + #end 6812 + #set ($nbDisplayedPosts = $mathtool.add($nbDisplayedPosts, 1)) 6813 + #if ($postIndex == $mathtool.sub($listResultsCount, 1)) 6814 + #if (~"$!lastHtmlTag~" == 'o') 6815 + #if ($mathtool.mod($nbDisplayedPosts, 2) == 1 && $lastRowClass == ~"flex~") 6816 + <div class=~"col-xs-12 col-sm-6~"></div> 6817 + #end 6818 + </div> 6819 + #end 6820 + #end 6821 + \{\{/html}} 6822 + #end 6823 + #else 6824 + \{\{error}}$services.localization.render('blog.blogpostlayout.notpost', [$xcontext.macro.params.reference])\{\{/error}} 6825 + #end 6826 + #else 6827 + \{\{error}}$services.localization.render('blog.blogpostlayout.post_view_not_allowed', [$xcontext.macro.params.reference])\{\{/error}} 6828 + #end" %) 6829 +((( 6830 +(% class="macro-placeholder hidden" %) 6831 +((( 6832 +macro:velocity 6833 +))) 6834 + 6835 +(% class="macro" data-macro="startmacro:html|-|clean=~"false~"|-|<div class=~"col-xs-12 col-sm-6 next_blog~"> 6836 +<a href=~"/Nieuws/Nieuwsbrief%20november%202023~" class=~"thumbnail~"> 6837 +<div class=~"row~"> 6838 +<div class=~"col-xs-4~"> 6839 +<img src=~"/download/Nieuws/Nieuwsbrief%20november%202023/man-person-light-bokeh-blur-road-1268120-pxhere.com.jpg?width=600&rev=1.1~" /> 6840 +</div> 6841 +<div class=~"col-xs-8 art_det~"> 6842 +<p class=~"text-left~"> 6843 +Nieuwsbrief november 2023 <br/><span class=~"date_info~"> Dec 18, 2023, </span> 6844 +</p> 6845 +<div class=~"row post-details~"> 6846 +<div class=~"col-xs-8 detail~"> 6847 +<div class=~"post-likes btn btn-default disabled badge~" 6848 +title=~"Number of likes on this page: 0~"> 6849 +<span class=~"fa fa-heart~"></span> <span class=~"like-number~">0</span> 6850 +</div> 6851 +</div> 6852 +<div class=~"col-xs-4 com_det~"> 6853 +<p class=~"text-right~"><span class=~"fa fa-comments~"></span> (0)</p> 6854 +</div> 6855 +</div> 6856 +</div> 6857 +</div> 6858 +</a> 6859 +</div>" %) 6860 +((( 6861 +(% class="macro-placeholder hidden" %) 6862 +((( 6863 +macro:html 6864 +))) 6865 + 6866 +(% class="col-xs-12 col-sm-6 next_blog" %) 6867 +((( 6868 +(% class="row" %) 6869 +((( 6870 +(% class="col-xs-4" %) 6871 +((( 6872 +[[~[~[image:/download/Nieuws/Nieuwsbrief%20november%202023/man-person-light-bokeh-blur-road-1268120-pxhere.com.jpg?width=600&rev=1.1~]~]>>path:/Nieuws/Nieuwsbrief%20november%202023||class="thumbnail"]] 6873 +))) 6874 + 6875 +(% class="col-xs-8 art_det" %) 6876 +((( 6877 +(% class="text-left" %) 6878 +[[Nieuwsbrief november 2023 6879 +(% class="date_info" %) Dec 18, 2023,>>path:/Nieuws/Nieuwsbrief%20november%202023||class="thumbnail"]] 6880 + 6881 +(% class="row post-details" %) 6882 +((( 6883 +(% class="col-xs-8 detail" %) 6884 +((( 6885 +(% class="post-likes btn btn-default disabled badge" title="Number of likes on this page: 0" %) 6886 +((( 6887 +[[(% class="like-number" %)0>>path:/Nieuws/Nieuwsbrief%20november%202023||class="thumbnail"]] 6888 +))) 6889 +))) 6890 + 6891 +(% class="col-xs-4 com_det" %) 6892 +((( 6893 +(% class="text-right" %) 6894 +[[(0)>>path:/Nieuws/Nieuwsbrief%20november%202023||class="thumbnail"]] 6895 +))) 6896 +))) 6897 +))) 6898 +))) 6899 +))) 6900 +))) 6901 +))) 6902 +))) 6903 + 6904 +(% class="macro" data-macro="startmacro:blogPostLayoutCards|-|reference=~"Nieuws.Feestelijke afsluiting, keuring materieel en tellen voorraden~" params=~"displayTitle=true|useSummary=true|list-limit=10|list-layout=cards|list-blog=Nieuws.WebHome|list-paginated=no|list-results-count=10|postIndex=0|previousPostReference=|postIndex=1|previousPostReference=Nieuws.Nieuwsbrief januari 2024|postIndex=2|previousPostReference=Nieuws.Audit BMI en BORG|postIndex=3|previousPostReference=Nieuws.Nieuwsbrief december 2023|postIndex=4|previousPostReference=Nieuws.Nieuwsbrief november 2023~"" %) 6905 +((( 6906 +(% class="macro-placeholder hidden" %) 6907 +((( 6908 +macro:blogPostLayoutCards 6909 +))) 6910 + 6911 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogCode~"" %) 6912 +((( 6913 +(% class="macro-placeholder hidden" %) 6914 +((( 6915 +macro:include 6916 +))) 6917 + 6918 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogParameters~"" %) 6919 +((( 6920 +(% class="macro-placeholder hidden" %) 6921 +((( 6922 +macro:include 6923 +))) 6924 + 6925 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|## Blog 6926 +#set($blogClassname = 'Blog.BlogClass') 6927 +#set($blogTemplate = 'Blog.BlogTemplate') 6928 +#set($blogSheet = 'Blog.BlogSheet') 6929 +## Blog entries 6930 +#set($blogPostClassname = 'Blog.BlogPostClass') 6931 +#set($blogPostTemplate = 'Blog.BlogPostTemplate') 6932 +#set($blogPostSheet = 'Blog.BlogPostSheet') 6933 +#set($blogPostObjectNumber = $xwiki.getDocument($blogPostTemplate).getObject($blogPostClassname).number) 6934 +#set($oldArticleClassname = 'XWiki.ArticleClass') 6935 +## Categories 6936 +#set($blogCategoryClassname = 'Blog.CategoryClass') 6937 +#set($blogCategoryTemplate = 'Blog.CategoryTemplate') 6938 +#set($blogCategorySheet = 'Blog.CategorySheet') 6939 +#set($blogCategoriesSheet = 'Blog.CategoriesSheet') 6940 +#set($oldBlogCategoryClassname = 'Blog.Categories') 6941 +#set($defaultCategoryParent = 'Blog.Categories') 6942 +## Style 6943 +#set($blogStyleDocumentName = 'Blog.BlogStyle') 6944 +#set($blogStyleDocument = $xwiki.getDocument($blogStyleDocumentName)) 6945 +## Clientside scripts 6946 +#set($blogScriptsDocumentName = 'Blog.BlogScripts') 6947 +#set($blogPublisher = $xwiki.getDocument('Blog.Publisher')) 6948 +## Misc 6949 +#set($thisURL = $doc.getURL($xcontext.action, $request.queryString)) 6950 +#set($isBlogPost = $doc.getObject($blogPostClassname)) 6951 +#set($defaultBlogSpace = 'Blog') 6952 +#set($isCategoryPage = $doc.getObject($blogCategoryClassname)) 6953 +#set($isCategoriesHomePage = $doc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 6954 +## 6955 +## 6956 +## 6957 +#** 6958 + * Displays an image, taken from the blog style document. 6959 + * 6960 + * @param $imgName The name of the icon from icons set to use. 6961 + *# 6962 +#macro(toolImage $imgName)(% class=~"icon-manage~"~%)$services.icon.render($imgName)(%~%)#end" %) 6963 +((( 6964 +(% class="macro-placeholder hidden" %) 6965 +((( 6966 +macro:velocity 6967 +))) 6968 +))) 6969 +))) 6970 + 6971 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|## 6972 +## 6973 +## 6974 +## Import the blog skin and javascripts. 6975 +$!xwiki.ssx.use($blogStyleDocumentName)## 6976 +$!xwiki.jsx.use($blogScriptsDocumentName)## 6977 +## 6978 +## import the hierarchy for the path the blog post footer (in displayEntryBlogLocation) 6979 +#template('hierarchy_macros.vm')## 6980 +## 6981 +## 6982 +#** 6983 + * Prints a blog. This is the main macro used in the BlogSheet. 6984 + * 6985 + * @param blogDoc the XDocument holding the blog definition object. 6986 + *### 6987 +#macro(printBlog $blogDoc) 6988 + \{\{include reference='Blog.CreatePost'/}} 6989 + 6990 + ## Use the blogPostList macro to display the blogposts 6991 + ## The blogPostList is used only in case of 'all' or 'paged' blog display type because the blogPostList macro 6992 + ## do not support FTM the monthly and weekly blog display types 6993 + #getBlogDisplayType($blogDoc $displayType) 6994 + #if ($displayType == 'weekly' || $displayType == 'monthly') 6995 + #getBlogEntries($blogDoc $entries) 6996 + #displayBlog($entries 'index' true true) 6997 + #displayNavigationLinks($blogDoc) 6998 + #else 6999 + #getBlogDisplayType($blogDoc $displayType) 7000 + #set ($paginated = 'no') 7001 + #if ($displayType == 'paginated') 7002 + #set ($paginated = 'yes') 7003 + #end 7004 + #getBlogPostsLayout($blogDoc $postsLayout) 7005 + (% class=~"hfeed index~" ~%)(((\{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$paginated~" layout=~"$!postsLayout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" /}}))) 7006 + #end 7007 +#end 7008 +## 7009 +## 7010 +## 7011 +#** 7012 + * Shows blog information. In view mode, the description is printed. In edit mode, allows changing blog settings: title, 7013 + * description, blog type (global or in-space), index display type (fixed size pagination, weekly index, monthly index, 7014 + * all entries). 7015 + * 7016 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 7017 + *### 7018 +#macro(showBlogInfo $blogDoc) 7019 + #if($blogDoc.getObject($blogClassname)) 7020 + ## Keep testing for inline action for backward compatibility with older blogs. 7021 + #if($xcontext.action == 'edit' || $xcontext.action == 'inline') 7022 + #macro(displayProperty $blogDoc $propname) 7023 + ; #displayPropName($xwiki.getClass($blogClassname).get($propname)): 7024 + : $blogDoc.display($propname) 7025 + #end 7026 + #displayProperty($blogDoc 'title') 7027 + #displayProperty($blogDoc 'description') 7028 + #displayProperty($blogDoc 'displayType') 7029 + #displayProperty($blogDoc 'itemsPerPage') 7030 + #displayProperty($blogDoc 'postsLayout') 7031 + #displayProperty($blogDoc 'postsLayoutParameters') 7032 + #else 7033 + $blogDoc.display('description') 7034 + #end 7035 + #elseif($doc.fullName == $blogSheet) 7036 += $services.localization.render('blog.code.blogsheet') = 7037 + \{\{translation key='blog.code.sheetexplanation'/}} 7038 + #else 7039 + \{\{warning}}\{\{translation key='blog.code.notblog'/}}\{\{/warning}} 7040 + #end 7041 +#end 7042 +## 7043 +## 7044 +## 7045 +#** 7046 + * Retrieve the blog document, which usually is either <tt><Space>.WebHome</tt> for whole-spaces blogs, or 7047 + * <tt><Space>.Blog</tt> for in-space blogs. If none of these documents contains a blog object, then the first 7048 + * (alphabetically) document in the target space that contains one is returned. Finally, if no document in the current 7049 + * space contains a blog object, then <tt>Blog.WebHome</tt> is returned as the default blog. 7050 + * 7051 + * @param space A <tt>String</tt>, the name of the space where to search. 7052 + * @param blogDoc The resulting XDocument. 7053 + *### 7054 +#macro(getBlogDocument $space $blogDoc) 7055 + #set ($result = $NULL) 7056 + ## Check if the current space has a WebPreferences page that contains a Blog.EnablePanelsConfigurationClass object. 7057 + ## It is possible to display a blog's panels info in a page that is not inside a blog and in that case we need to 7058 + ## identify the right blog based on a configuration object in a WebPreferences page. 7059 + #set ($spaceReference = $services.model.resolveSpace($space)) 7060 + #set ($preferencesDocRef = $services.model.createDocumentReference('WebPreferences', $spaceReference)) 7061 + #set ($preferencesDoc = $xwiki.getDocument($preferencesDocRef)) 7062 + #set ($preferencesObj = $preferencesDoc.getObject('Blog.EnablePanelsConfigurationClass')) 7063 + #if ($preferencesObj) 7064 + #set ($result = $xwiki.getDocument( $preferencesObj.getValue('blog'))) 7065 + #end 7066 + ## If no special case occurs (like the previous one), identify a blog by checking the current location from its hierarchy. 7067 + #if (~"$!result~" == '') 7068 + ## First, try the Space.WebHome, for a whole-space blog 7069 + #set($result = $xwiki.getDocument(~"$\{space}.WebHome~")) 7070 + #if(!$result.getObject($blogClassname)) 7071 + ## Second, try the Space.Blog document 7072 + #set($result = $xwiki.getDocument(~"$\{space}.Blog~")) 7073 + #if(!$result.getObject($blogClassname)) 7074 + ## Third, try searching for a blog document in the current space 7075 + ## Prevent the query fail when the space contains dots '.' 7076 + #set($blogDocs = $services.query.hql(~", BaseObject obj where doc.space = :space and obj.name = doc.fullName and obj.className = '$blogClassname' order by doc.name~").setLimit(1).setOffset(0).bindValue('space', $space).execute()) 7077 + #if($blogDocs.size() > 0) 7078 + #set($result = $xwiki.getDocument($blogDocs.get(0))) 7079 + #else 7080 + ## Fourth, try searching for a blog document that have the its 'postsLocation' set to the current space 7081 + #set($blogDocs = $services.query.hql(~", BaseObject obj, StringProperty as postsLocationProp where obj.name = doc.fullName and obj.className = '$blogClassname' and obj.id=postsLocationProp.id.id and postsLocationProp.id.name='postsLocation' and postsLocationProp.value=:postsLocation order by doc.name~").setLimit(1).setOffset(0).bindValue('postsLocation', $space).execute()) 7082 + #if($blogDocs.size() > 0) 7083 + #set($result = $xwiki.getDocument($blogDocs.get(0))) 7084 + #else 7085 + ## Last, fallback to Blog.WebHome, the default blog 7086 + #set($result = $xwiki.getDocument('Blog.WebHome')) 7087 + #end 7088 + #end 7089 + #end 7090 + #end 7091 + #end 7092 + #set ($blogDoc = $NULL) 7093 + #setVariable (~"$blogDoc~" $result) 7094 +#end 7095 +## 7096 +## 7097 +## 7098 +#** 7099 + * Retrieve the blog title. 7100 + * 7101 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass<tt> object with the <tt>title</tt> property set. 7102 + * @param title The resulting title. 7103 + *### 7104 +#macro(getBlogTitle $blogDoc $title) 7105 + ## Titles can contain velocity code (most commonly translations), so we should evaluate them. 7106 + #set ($title = $NULL) 7107 + #setVariable (~"$title~" $!blogDoc.displayTitle) 7108 +#end 7109 +## 7110 +## 7111 +## 7112 +#** 7113 + * Retrieve the blog description. 7114 + * 7115 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>description</tt> 7116 + * property set. 7117 + * @param description The resulting description. 7118 + *### 7119 +#macro(getBlogDescription $blogDoc $description) 7120 + #getBlogProperty($blogDoc 'description' '' $result) 7121 + #set ($description = $NULL) 7122 + #setVariable (~"$description~" $result) 7123 +#end 7124 +## 7125 +## 7126 +## 7127 +#** 7128 + * Retrieves a list of entries to be displayed. The entries are either part of the blog's space, or have the blog 7129 + * document set as a parent. The number and range of entries returned (from all those belonging to this blog) depends on 7130 + * the blog display type: paginated (fixed number of entries), weekly (all entries in a week), monthly (all entries in a 7131 + * month), or all. 7132 + * 7133 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 7134 + * @param entries The resulting list of entries to display, a list of XDocument names. 7135 + *### 7136 +#macro(getBlogEntries $blogDoc $entries) 7137 + #if (!$entries) 7138 + #setVariable (~"$entries~" []) 7139 + #end 7140 + #getAllBlogPostsQuery($query) 7141 + #isDefaultBlog($blogDoc $isDefault) 7142 + #set($queryParams = \{}) 7143 + #if ($isDefault) 7144 + #getCategoryAllBlogPostsQuery($query) 7145 + #set($query = ~"$\{query} and (doc.creator = :creator or (isPublished.value = 1 and hidden.value = 0)) and (doc.space = :space or catList like :catList escape '!')~") 7146 + #set($discard = $queryParams.put('creator', $xcontext.user)) 7147 + #set($discard = $queryParams.put('space', $blogDoc.space)) 7148 + #set($sanitizedBlogSpace = $blogDoc.space.replaceAll('([%_!])', '!$1').concat('.%')) 7149 + #set($discard = $queryParams.put('catList', $sanitizedBlogSpace)) 7150 + #else 7151 + #set($query = ~"$\{query} and (doc.space = :space or doc.parent = :parent)~") 7152 + #getBlogPostsLocation($blogDoc.space $blogPostsLocation) 7153 + #set($discard = $queryParams.put('space', $blogPostsLocation)) 7154 + #set($discard = $queryParams.put('parent', $blogDoc.space)) 7155 + #end 7156 + #getBlogDisplayType($blogDoc $displayType) 7157 + #if($displayType == 'weekly') 7158 + #getWeeklyBlogEntries($blogDoc $query $entries $queryParams) 7159 + #elseif($displayType == 'monthly') 7160 + #getMonthlyBlogEntries($blogDoc $query $entries $queryParams) 7161 + #elseif($displayType == 'all') 7162 + #getAllBlogEntries($blogDoc $query $entries $queryParams) 7163 + #else 7164 + #getPagedBlogEntries($blogDoc $query $entries $queryParams) 7165 + #end 7166 +#end 7167 +## 7168 +## 7169 +## 7170 +#** 7171 + * Retrieves a list of entries to be displayed. The entries are taken from a ~"page~" of the blog, a sequence of documents 7172 + * defined by the request parameters <tt>ipp</tt> (items per page) and <tt>page</tt> (the current page). Initially the 7173 + * first page is displayed, with the number of entries defined in the blog object in the <tt>itemsPerPage</tt> property 7174 + * (10 if not defined). 7175 + * 7176 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 7177 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 7178 + * refined to restrict to a given space, or to a given search criteria, etc. 7179 + * @param entries The resulting list of entries to display, a list of XDocument names. 7180 + * @param queryParams The parameters to bind with the query. 7181 + *### 7182 +#macro(getPagedBlogEntries $blogDoc $query $entries $queryParams) 7183 + #if (!$entries) 7184 + #setVariable (~"$entries~" []) 7185 + #end 7186 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 7187 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~")) 7188 + #bindQueryParameters($countQueryObj $queryParams) 7189 + #bindQueryParameters($queryObj $queryParams) 7190 + #set($totalEntries = $countQueryObj.count()) 7191 + #getBlogProperty($blogDoc 'itemsPerPage' '10' $defaultItemsPerPage) 7192 + #set($defaultItemsPerPage = $numbertool.toNumber($defaultItemsPerPage).intValue()) 7193 + ## This macro is defined in the default macros.vm library. It also sets $itemsPerPage and $startAt. 7194 + #preparePagedViewParams($totalEntries $defaultItemsPerPage) 7195 + #set($discard = $entries.addAll($queryObj.setLimit($itemsPerPage).setOffset($startAt).addFilter(~"unique~").execute())) 7196 +#end 7197 +## 7198 +## 7199 +## 7200 +#** 7201 + * Retrieves a list of entries to be displayed. The entries are taken from a week of the blog. The target week is 7202 + * defined by the request parameters <tt>week</tt> (the week number in the year, from 1 to 52) and <tt>year</tt> (4 7203 + * digit year). Initially the current week is displayed. 7204 + * 7205 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 7206 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 7207 + * refined to restrict to a given space, or to a given search criteria, etc. 7208 + * @param entries The resulting list of entries to display, a list of XDocument names. 7209 + * @param queryParams The parameters to bind with the query. 7210 + *### 7211 +#macro(getWeeklyBlogEntries $blogDoc $query $entries $queryParams) 7212 + #if (!$entries) 7213 + #setVariable (~"$entries~" []) 7214 + #end 7215 + #getRequestedWeek($weekDate) 7216 + #set($dateFormatter = $xwiki.jodatime.getDateTimeFormatterForPattern('yyyy-MM-dd')) 7217 + #set($minDay = $dateFormatter.print($weekDate.toMutableDateTime().weekOfWeekyear().roundFloor())) 7218 + #set($maxDay = $dateFormatter.print($weekDate.toMutableDateTime().weekOfWeekyear().roundCeiling())) 7219 + #set($query = ~"$\{query} and publishDate.value >= '$minDay' and publishDate.value < '$maxDay'~") 7220 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 7221 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 7222 + #bindQueryParameters($countQueryObj $queryParams) 7223 + #bindQueryParameters($queryObj $queryParams) 7224 + #set($totalEntries = $countQueryObj.count()) 7225 + #set($discard = $entries.addAll($queryObj.execute())) 7226 +#end 7227 +## 7228 +## 7229 +## 7230 +#** 7231 + * Retrieves a list of entries to be displayed. The entries are taken from a month of the blog. The target month is 7232 + * defined by the request parameters <tt>month</tt> (the month number, from 1 to 12) and <tt>year</tt> (4 7233 + * digit year). Initially the current month is displayed. 7234 + * 7235 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 7236 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 7237 + * refined to restrict to a given space, or to a given search criteria, etc. 7238 + * @param entries The resulting list of entries to display, a list of XDocument names. 7239 + * @param queryParams The parameters to bind with the query. 7240 + *### 7241 +#macro(getMonthlyBlogEntries $blogDoc $query $entries) 7242 + #if (!$entries) 7243 + #setVariable (~"$entries~" []) 7244 + #end 7245 + #getRequestedMonth($monthDate) 7246 + #set($dateFormatter = $xwiki.jodatime.getDateTimeFormatterForPattern('yyyy-MM-dd')) 7247 + #set($minDay = $dateFormatter.print($monthDate.toMutableDateTime().monthOfYear().roundFloor())) 7248 + #set($maxDay = $dateFormatter.print($monthDate.toMutableDateTime().monthOfYear().roundCeiling())) 7249 + #set($query = ~"$\{query} and publishDate.value >= '$minDay' and publishDate.value < '$maxDay'~") 7250 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 7251 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 7252 + #bindQueryParameters($countQueryObj $queryParams) 7253 + #bindQueryParameters($queryObj $queryParams) 7254 + #set($totalEntries = $countQueryObj.count()) 7255 + #set($discard = $entries.addAll($queryObj.execute())) 7256 +#end 7257 +## 7258 +## 7259 +## 7260 +#** 7261 + * Retrieves a list of entries to be displayed. All entries belonging to the current blog are returned. 7262 + * 7263 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 7264 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 7265 + * refined to restrict to a given space, or to a given search criteria, etc. 7266 + * @param entries The resulting list of entries to display, a list of XDocument names. 7267 + * @param queryParams The parameters to bind with the query. 7268 + *### 7269 +#macro(getAllBlogEntries $blogDoc $query $entries $queryParams) 7270 + #if (!$entries) 7271 + #setVariable (~"$entries~" []) 7272 + #end 7273 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 7274 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 7275 + #bindQueryParameters($countQueryObj $queryParams) 7276 + #bindQueryParameters($queryObj $queryParams) 7277 + #set($totalEntries = $countQueryObj.count()) 7278 + #set($discard = $entries.addAll($queryObj.execute())) 7279 +#end 7280 +## 7281 +## 7282 +## 7283 +#** 7284 + * Retrieves a list of entries to be displayed. Only (and all) unpublished entries are returned. 7285 + * 7286 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 7287 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 7288 + * refined to restrict to a given space, or to a given search criteria, etc. 7289 + * @param queryParams The parameters to bind with the query. 7290 + * @param entries The resulting list of entries to display, a list of XDocument names. 7291 + *### 7292 +#macro(getUnpublishedBlogEntries $blogDoc $query $queryParams $entries) 7293 + #if (!$entries) 7294 + #setVariable (~"$entries~" []) 7295 + #end 7296 + #set($query = ~"$\{query} and isPublished.value = 0~") 7297 + #set ($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 7298 + #set ($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 7299 + #bindQueryParameters($countQueryObj $queryParams) 7300 + #bindQueryParameters($queryObj $queryParams) 7301 + #set($totalEntries = $countQueryObj.count()) 7302 + #set($discard = $entries.addAll($queryObj.execute())) 7303 +#end 7304 +## 7305 +## 7306 +## 7307 +#** 7308 + * Retrieves a list of entries to be displayed. The entries are taken from all the wiki, and not from a specific blog. 7309 + * 7310 + * @param entries The resulting list of entries to display, a list of XDocument names. 7311 + *### 7312 +#macro(getGlobalBlogEntries $entries) 7313 + #if (!$entries) 7314 + #setVariable (~"$entries~" []) 7315 + #end 7316 + #getAllBlogPostsQuery($query) 7317 + #set($totalEntries = $services.query.hql($query).count()) 7318 + #set($defaultItemsPerPage = 20) 7319 + ## This macro is defined in the default macros.vm library. It also sets $itemsPerPage and $startAt. 7320 + #preparePagedViewParams($totalEntries $defaultItemsPerPage) 7321 + #set($discard = $entries.addAll($services.query.hql(~"$\{query} order by publishDate.value desc~").setLimit($itemsPerPage).setOffset($startAt).execute())) 7322 +#end 7323 +#** 7324 + * Return the base query for selecting blog entries. It filters only visible entries, but does not bind to a specific 7325 + * blog, nor specify a range or an ordering criteria. 7326 + * 7327 + * This macro is a duplicate of #getAllBlogPostsQuery($query). Keep it in case it's used outside, in custom code. 7328 + * 7329 + * @param query The basic query for selecting blog entries. 7330 + *# 7331 +#macro(getBlogEntriesBaseQuery $query) 7332 + #getAllBlogPostsQuery($query) 7333 +#end 7334 +#** 7335 + * Return the Query for selecting the all wiki blog posts without filtering 7336 + * 7337 + * @param query The basic query for selecting blog entries. 7338 + *# 7339 +#macro(getAllBlogPostsQuery $query) 7340 + #set ($query = $NULL) 7341 + #setVariable(~"$query~" ~", BaseObject as obj, IntegerProperty isPublished, 7342 + IntegerProperty hidden, DateProperty publishDate 7343 + where doc.fullName <> '$blogPostTemplate' and 7344 + obj.name=doc.fullName and obj.className='$blogPostClassname' and 7345 + isPublished.id.id = obj.id and isPublished.id.name = 'published' and 7346 + hidden.id.id = obj.id and hidden.id.name='hidden' and 7347 + publishDate.id.id = obj.id and publishDate.id.name='publishDate' and 7348 + (doc.creator = '$escapetool.sql($xcontext.user)' or (isPublished.value = 1 and hidden.value = 0))~") 7349 +#end 7350 +## 7351 +## 7352 +## 7353 +###** 7354 + * Return the Query for selecting the all wiki blog posts with categories filtering 7355 + * 7356 + * @param query The basic query for selecting blog entries. 7357 + *### 7358 +#macro(getCategoryAllBlogPostsQuery $query) 7359 + #set ($query = $NULL) 7360 + #getAllBlogPostsQuery($baseQuery) 7361 + #set ($baseQuery = $baseQuery.replace('DateProperty publishDate', 'DateProperty publishDate, DBStringListProperty as category left join category.list catList')) 7362 + #setVariable(~"$query~" ~"$\{baseQuery} and obj.id=category.id.id and category.id.name='category'~") 7363 +#end 7364 +## 7365 +## 7366 +## 7367 +#** 7368 + * Determines how is the blog index split into pages: paginated (fixed number of entries), weekly (all entries in a 7369 + * week), monthly (all entries in a month), or all. 7370 + * 7371 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>displayType</tt> 7372 + * property set. 7373 + * @param displayType The resulting string. If the blog object does not define anything, it is considered paginated. 7374 + *### 7375 +#macro(getBlogDisplayType $blogDoc $displayType) 7376 + #getBlogProperty($blogDoc 'displayType' 'paginated' $result) 7377 + #set ($displayType = $NULL) 7378 + #setVariable (~"$displayType~" $result) 7379 +#end 7380 +## 7381 +## 7382 +## 7383 +#** 7384 + * Displays a list of entries. 7385 + * 7386 + * @param entries The entries to display, a list of XDocument names. 7387 + * @param displaying What exactly is displayed: blog index, a single blog entry, a blog category, search results, 7388 + * unpublished entries, etc. This will be used as the classname(s) for the container div (hfeed). Currently 7389 + * used values: index, single, category, search, unpublished, hidden. 7390 + * @param onlyExtract If <tt>true</tt>, only display the extract of articles where available, otherwise display the full content. 7391 + * @param shouldDisplayTitles If <tt>true</tt>, display the blog title (blog posts shouldn't display the title when they're 7392 + * displayed alone on their page since it's the page title which is used in this case) 7393 + *### 7394 +#macro(displayBlog $entries $displaying $onlyExtract $shouldDisplayTitles) 7395 + #set($blogDay = '') 7396 + (% class=~"hfeed $!\{displaying}~" ~%)((( 7397 + (% class=~"blogDay~" ~%)((( 7398 + #foreach ($entryDoc in $xwiki.wrapDocs($entries)) 7399 + #getEntryObject($entryDoc $entryObj) 7400 + ## Although all entries should have one of the two objects, better check to be sure. 7401 + #if(~"$!\{entryObj}~" != '') 7402 + #getEntryDate($entryDoc $entryObj $entryDate) 7403 + ## Display a ~"calendar sheet~" for each day. All entries posted on the same day share one such sheet. 7404 + #set($entryDateStr = $xwiki.formatDate($entryDate, 'yyyyMMMMdd')) 7405 + #if($blogDay != $entryDateStr) 7406 + #if($blogDay != '') 7407 + ))) 7408 + (% class=~"blogDay~" ~%)((( 7409 + #end 7410 + #displayBlogDate($entryDate) 7411 + #set ($blogDay = $entryDateStr) 7412 + #end 7413 + ## Finally, display the entry. 7414 + #displayEntry($entryDoc $entryObj $onlyExtract $shouldDisplayTitles true) 7415 + #end 7416 + #end 7417 + )))## blogDay 7418 + )))## hfeed 7419 +#end 7420 +## 7421 +## 7422 +## 7423 +#** 7424 + * Get the entry object, either a new BlogPost or an old Article. 7425 + * 7426 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 7427 + * @param entryObj The resulting xobject of the blog post. 7428 + *### 7429 +#macro(getEntryObject $entryDoc $__entryObj) 7430 + #set($result = $entryDoc.getObject(~"$\{blogPostClassname}~")) 7431 + #if(!$result) 7432 + #set($result = $entryDoc.getObject(~"$\{oldArticleClassname}~")) 7433 + #end 7434 + ## NOTE: The reason we put an underscore in front of the variable name is to prevent the following line from 7435 + ## overwriting the $entryObj variable that may be defined before this macro is called. Of course, $__entryObj may be 7436 + ## overwritten in this case but it's less likely to have such a variable defined before. 7437 + #set ($__entryObj = $NULL) 7438 + #setVariable (~"$__entryObj~" $result) 7439 +#end 7440 +## 7441 +## 7442 +## 7443 +#** 7444 + * Gets the date associated with a blog entry. This is the publication date. For unpublished entries, initially this is 7445 + * the document creation date, but can be edited by the user. 7446 + * 7447 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 7448 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 7449 + * @param result The resulting date, an instance of <tt>java.util.Date</tt>. 7450 + *### 7451 +#macro(getEntryDate $entryDoc $entryObj $result) 7452 + #set ($result = $NULL) 7453 + #setVariable (~"$result~" $entryObj.getProperty('publishDate').value) 7454 +#end 7455 +## 7456 +## 7457 +## 7458 +#** 7459 + * Displays a date, nicely formatted as a calendar page. 7460 + * 7461 + * @param date The date to display, an instance of <tt>java.util.Date</tt>. 7462 + *### 7463 +#macro(displayBlogDate $date) 7464 + #set($year = $xwiki.formatDate($date, 'yyyy')) 7465 + ## 3 letter month name, like Jan, Dec. 7466 + #set($month = $xwiki.formatDate($date, 'MMM')) 7467 + ## Uncomment to get a full length month name, like January, December. 7468 + ## TODO: this could be defined somewhere in the blog style. 7469 + ## #set($month = $xwiki.formatDate($date, 'MMMM')) 7470 + #set($day = $xwiki.formatDate($date, 'dd')) 7471 + (% class=~"blogdate~" ~%) 7472 + == (% class=~"month~" ~%)$month(%~%) (% class=~"day~" ~%)$day(%~%) (% class=~"year~" ~%)$year(%~%) == 7473 +#end 7474 +## 7475 +## 7476 +## 7477 +#** 7478 + * Displays a blog article: management tools, header, content, footer. 7479 + * 7480 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 7481 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 7482 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 7483 + * @param shouldDisplayTitle If <tt>true</tt>, display the blog title (blog posts shouldn't display the title 7484 + * when they're displayed alone on their page since it's the page title which is used in this case) 7485 + * @param shouldDisplayActions If <tt>true</tt>, display the blog post actions buttons 7486 + *### 7487 +#macro(displayEntry $entryDoc $entryObj $onlyExtract $shouldDisplayTitle $shouldDisplayActions) 7488 + ## Only articles with an explicit hidden setting or an explicit unpublished setting are hidden 7489 + #isPublished($entryObj $isPublished) 7490 + #isHidden($entryObj $isHidden) 7491 + #if($doc.fullName == $entryDoc.fullName) 7492 + (% class=~"hentry single-article~" ~%)((( 7493 + #else 7494 + (% class=~"hentry#if(!$isPublished) unpublished-article#elseif($isHidden) hidden-article#end~" ~%)((( 7495 + #end 7496 + #if ($shouldDisplayActions) 7497 + #displayEntryTools($entryDoc $entryObj) 7498 + #end 7499 + #if($shouldDisplayTitle) 7500 + #displayEntryTitle($entryDoc $entryObj) 7501 + #end 7502 + #if($doc.fullName == $entryDoc.fullName) 7503 + #if(!$isPublished) 7504 + \{\{warning}}\{\{translation key='blog.code.published'/}}\{\{/warning}} 7505 + #elseif($isHidden) 7506 + \{\{warning}}\{\{translation key='blog.code.hidden'/}}\{\{/warning}} 7507 + #end 7508 + #end 7509 + #displayEntryContent($entryDoc $entryObj $onlyExtract) 7510 + #displayEntryFooter($entryDoc $entryObj) 7511 + )))## hentry 7512 +#end 7513 +## 7514 +## 7515 +## 7516 +#** 7517 + * Checks if the provided blog is published or not. 7518 + * 7519 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 7520 + * @param isPublished The resulting boolean, true if the entry is considered published. 7521 + *### 7522 +#macro(isPublished $entryObj $isPublished) 7523 + #set ($isPublished = $NULL) 7524 + ## This should work for both old articles, which don't have the 'published' property at all, and 7525 + ## are considered published by default, and new entries, that should have 1 if published. 7526 + #if (~"$!\{entryObj.getProperty('published').value}~" != '0') 7527 + #setVariable (~"$isPublished~" true) 7528 + #else 7529 + #setVariable (~"$isPublished~" false) 7530 + #end 7531 +#end 7532 +## 7533 +## 7534 +## 7535 +#** 7536 + * Checks if the provided blog is hidden or not. 7537 + * 7538 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass<tt> xclass. 7539 + * @param isHiddel The resulting boolean, true if the entry is considered hidden. 7540 + *### 7541 +#macro(isHidden $entryObj $isHidden) 7542 + #set ($isHidden = $NULL) 7543 + ## This should work for both old articles, which don't have the 'hidden' property at all, and 7544 + ## are considered visible by default, and new entries, that should have 1 if hidden. 7545 + #if (~"$!\{entryObj.getProperty('hidden').value}~" == '1') 7546 + #setVariable (~"$isHidden~" true) 7547 + #else 7548 + #setVariable (~"$isHidden~" false) 7549 + #end 7550 +#end 7551 +## 7552 +## 7553 +## 7554 +#** 7555 + * Displays several ~"tools~" for manipulating blog posts: hide/show, publish, edit. 7556 + * 7557 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 7558 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 7559 + *### 7560 +#macro(displayEntryTools $entryDoc $entryObj) 7561 + #if($xcontext.action == 'view') 7562 + (% class=~"blog-entry-toolbox~" ~%)((( 7563 + #displayPublishButton($entryDoc $entryObj) 7564 + #displayHideShowButton($entryDoc $entryObj) 7565 + #displayEditButton($entryDoc $entryObj) 7566 + #displayDeleteButton($entryDoc $entryObj) 7567 + ))) 7568 + #end 7569 +#end 7570 +## 7571 +## 7572 +## 7573 +#** 7574 + * Displays the publish button to the entry <strong>creator</strong>, if the article is not published yet. 7575 + * 7576 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 7577 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 7578 + * @todo AJAX calls. 7579 + *### 7580 +#macro(displayPublishButton $entryDoc $entryObj) 7581 + #isPublished($entryObj $isPublished) 7582 + #if(!$isPublished && $entryDoc.creator == $xcontext.user && $xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 7583 + [[#toolImage('world')>>path:$blogPublisher.getURL('view', ~"entryName=$\{escapetool.url($entryDoc.fullName)}&xredirect=$\{escapetool.url($thisURL)}&form_token=$!\{services.csrf.getToken()}~")||title=~"$services.localization.render('blog.code.notpublished')~"]]## 7584 + #end 7585 +#end 7586 +## 7587 +## 7588 +## 7589 +#** 7590 + * Displays the hide or show button to the entry <strong>creator</strong>, if the article is already published. 7591 + * 7592 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 7593 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 7594 + *### 7595 +#macro(displayHideShowButton $entryDoc $entryObj) 7596 + #isPublished($entryObj $isPublished) 7597 + #isHidden($entryObj $isHidden) 7598 + ## Only published articles can be hidden. Unpublished articles are considered already hidden. 7599 + #if($isPublished && $entryDoc.creator == $xcontext.user && $xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 7600 + #set ($queryString = \{ 7601 + 'xredirect' : $thisURL, 7602 + 'form_token' : $services.csrf.getToken() 7603 + }) 7604 + #if ($isHidden) 7605 + #set ($discard = $queryString.putAll(\{ 7606 + ~"$\{entryObj.getxWikiClass().getName()}_$\{entryObj.number}_hidden~" : 0, 7607 + 'comment' : $services.localization.render('blog.code.madevisible') 7608 + })) 7609 + #set ($lockURL = $entryDoc.getURL('save', $escapetool.url($queryString))) 7610 + [[#toolImage('unlock')>>path:$lockURL||class=~"blog-tool-show~" title=~"$services.localization.render('blog.code.makevisible')~"]]## 7611 + #else 7612 + #set ($discard = $queryString.putAll(\{ 7613 + ~"$\{entryObj.getxWikiClass().getName()}_$\{entryObj.number}_hidden~" : 1, 7614 + 'comment' : $services.localization.render('blog.code.hid') 7615 + })) 7616 + #set ($lockURL = $entryDoc.getURL('save', $escapetool.url($queryString))) 7617 + [[#toolImage('lock')>>path:$lockURL||class=~"blog-tool-hide~" title=~"$services.localization.render('blog.code.hide')~"]]## 7618 + #end 7619 + #end 7620 +#end 7621 +## 7622 +## 7623 +## 7624 +#** 7625 + * Displays the edit button to those that can edit the article. 7626 + * 7627 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 7628 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 7629 + *### 7630 +#macro(displayEditButton $entryDoc $entryObj) 7631 + #if($xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 7632 + ## Call getDefaultEditMode() for backward compatibility with older blog posts. 7633 + [[#toolImage('pencil')>>path:$entryDoc.getURL('edit')||title=~"$services.localization.render('blog.code.editpost')~"]]## 7634 + #end 7635 +#end 7636 +## 7637 +## 7638 +## 7639 +#** 7640 + * Displays the delete button to those that can edit the article. 7641 + * 7642 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 7643 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 7644 + * @todo AJAX calls. 7645 + *### 7646 +#macro(displayDeleteButton $entryDoc $entryObj) 7647 + #if($xwiki.hasAccessLevel('delete', $xcontext.user, $entryDoc.fullName)) 7648 + [[#toolImage('cross')>>path:$entryDoc.getURL('delete')||title=~"$services.localization.render('blog.code.deletepost')~"]]## 7649 + #end 7650 +#end 7651 +## 7652 +## 7653 +## 7654 +#** 7655 + * Displays the title of the entry. 7656 + * 7657 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 7658 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 7659 + *### 7660 +#macro(displayEntryTitle $entryDoc $entryObj) 7661 + #if($doc.fullName == $entryDoc.fullName) 7662 + (% class=~"entry-title~" ~%) 7663 + = $services.rendering.escape($entryDoc.display('title', 'view', $entryObj), $xwiki.getCurrentContentSyntaxId()) = 7664 + #else 7665 + (% class=~"entry-title~" ~%) 7666 + === [[$services.rendering.escape($entryDoc.display('title', 'view', $entryObj),$xwiki.getCurrentContentSyntaxId())>>doc:$services.rendering.escape($services.model.serialize($entryDoc.getDocumentReference(),'default'),$xwiki.getCurrentContentSyntaxId())]] === 7667 + #end 7668 +#end 7669 +## 7670 +## 7671 +## 7672 +#** 7673 + * Displays the body of the entry. 7674 + * 7675 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 7676 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 7677 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 7678 + *### 7679 +#macro(displayEntryContent $entryDoc $entryObj $onlyExtract) 7680 + (% class=~"#if($onlyExtract)entry-summary#\{else}entry-content#end~" ~%)((( 7681 + #getEntryContent($entryDoc $entryObj $onlyExtract $entryContent) 7682 + ## FIXME: This causes the blog's content to not be annotatable. See http://jira.xwiki.org/browse/XWIKI-6328 7683 + ## Should probably be replaced by a display macro call with a reference to the object property holding the post's content 7684 + \{\{html wiki=~"false~"}}$entryDoc.getRenderedContent($entryContent, $entryDoc.syntax.toIdString())\{\{/html}} 7685 + ))) ## entry-content 7686 + (% class=~"clearfloats~" ~%)((())) 7687 +#end 7688 +## 7689 +## 7690 +## 7691 +#** 7692 + * Extracts the body of the entry that should be displayed. If <tt>onlyExtract</tt> is <tt>true</tt>, display the content 7693 + * of the <tt>extract</tt> field (if not empty). 7694 + * 7695 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 7696 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 7697 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 7698 + * @param entryContent The resulting content. 7699 + * @param removeEllipsis If <tt>true</tt>, then don't display an ellipsis at the end of the content (only used when 7700 + * <tt>onlyExtract</tt> is <tt>true</tt>) 7701 + *### 7702 +#macro(getEntryContent $entryDoc $entryObj $onlyExtract $entryContent $removeEllipsis) 7703 + #if ($onlyExtract) 7704 + ## Note: We trim the summary so that if there's some white space it won't cause the summary to be used instead 7705 + ## of the content. 7706 + #set ($macro.result = $entryObj.getProperty('extract').value.trim()) 7707 + #end 7708 + #if(~"$!macro.result~" == '') 7709 + #set($macro.result = $entryObj.getProperty('content').value) 7710 +#* Disabled until the content can be cleanly cut. 7711 +* #if($onlyExtract && $result.length()>$maxchars) 7712 +* #set($i = $macro.result.lastIndexOf(~" ~", $maxchars)) 7713 +* #set($i = $i + 1) 7714 +* #set($macro.result = ~"$\{macro.result.substring(0,$i)} *[...>$\{entryDoc.fullName}]*~") 7715 +* #end 7716 +## *### 7717 + #elseif (!$removeEllipsis) 7718 + #if($entryDoc.syntax.toIdString() == 'xwiki/1.0') 7719 + #set($macro.result = ~"$\{macro.result} <a href='$\{entryDoc.getExternalURL()}' title='$services.localization.render('blog.code.readpost')'>...</a>~") 7720 + #elseif($entryDoc.syntax.toIdString() == 'xwiki/2.0' || $entryDoc.syntax.toIdString() == 'xwiki/2.1') 7721 + #set($macro.result = ~"$\{macro.result} [[...>>$\{services.rendering.escape($entryDoc.getFullName(), $entryDoc.getSyntax())}||title='$services.localization.render('blog.code.readpost')']]~") 7722 + #end 7723 + #end 7724 + #set ($entryContent = $NULL) 7725 + #setVariable (~"$entryContent~" $macro.result) 7726 +#end 7727 +## 7728 +## 7729 +## 7730 +#** 7731 + * Displays the footer of the entry. 7732 + * 7733 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 7734 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 7735 + *### 7736 +#macro(displayEntryFooter $entryDoc $entryObj) 7737 + (% class=~"entry-footer~" ~%)((( 7738 + #isPublished($entryObj $isPublished) 7739 + (% class='entry-author-label' ~%) 7740 + #if($isPublished) 7741 + \{\{translation key='blog.code.postedby'/}} ## 7742 + #else 7743 + \{\{translation key='blog.code.createdby'/}} ## 7744 + #end 7745 + \{\{html wiki=~"false~" clean=~"false~"}}<span class='author vcard'>#userfn($entryDoc.creator)</span>\{\{/html}} ## 7746 + #getEntryDate($entryDoc $entryObj $entryDate) 7747 + #listCategories($entryObj) #* 7748 + ## Since the publish date and update date are not set at the exact same time, there could be a small difference that 7749 + ## we assume cannot be more than 3 seconds. 7750 + *#(% class=~"separator~" ~%)·(%~%) [[\{\{translation key='blog.code.permalink'/}}>>$services.rendering.escape($entryDoc,'xwiki/2.0')||rel=~"bookmark~"]] ## 7751 + #if ($showcomments) 7752 + (% class=~"separator~" ~%)·(%~%) [[\{\{translation key='blog.code.comments'/}}>>$services.rendering.escape($entryDoc,'xwiki/2.0')||anchor=~"Comments~"]] (% class=~"itemCount~" ~%)($entryDoc.comments.size())(%~%) ## 7753 + #end ## 7754 + #if($entryDoc != $doc) ## 7755 + #displayEntryBlogLocation($entryDoc $entryObj) ## 7756 + #end 7757 + )))## entry-footer 7758 +#end 7759 +## 7760 +## 7761 +#** 7762 + * Display the blog for the entry (if it is not the currently displayed blog) 7763 + * 7764 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 7765 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 7766 + *### 7767 +#macro(displayEntryBlogLocation $entryDoc $entryObj) 7768 + #getBlogPostsLocation($entryDoc.getSpace() $blogPostsLocation) ## 7769 + #if(~"$!blogPostsLocation~" != ~"~") ## 7770 + #set($blogSpaceRef = $services.model.resolveSpace($blogPostsLocation)) ## 7771 + #set($blogDocRef = $services.model.createDocumentReference($services.model.getEntityReference('DOCUMENT', 'default').getName(), $blogSpaceRef)) ## 7772 + #if($doc.documentReference != $blogDocRef) ## 7773 + #set($blogDoc = $xwiki.getDocument($blogDocRef)) ## 7774 + #set($blogObj = $blogDoc.getObject('Blog.BlogClass')) ## 7775 + #if($blogObj) (% class=~"blogrefpath~" ~%)((( ## 7776 + $services.localization.render('blog.code.postedin') \{\{html clean=~"false~" wiki=~"false~"}} #hierarchy($blogDocRef, \{'selfIsActive' : true, 'local': true}) \{\{/html}} ## 7777 + )))(%~%)## 7778 + #end 7779 + #end 7780 + #end 7781 +#end 7782 +## 7783 +## 7784 +## 7785 +## 7786 +#** 7787 + * List the categories an entry belongs to. Used in the footer. The categories are instances of <tt>Blog.CategoryClass</tt>. 7788 + * 7789 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 7790 + *### 7791 +#macro(listCategories $entryObj) 7792 + #if($entryObj.getxWikiClass().getName() == $blogPostClassname) 7793 + #set($categories = $entryObj.getProperty('category').value) 7794 + #set($first = true) 7795 + #if($categories.size() > 0) 7796 + #foreach($category in $categories) 7797 + #set($categoryDoc = $!xwiki.getDocument($category)) 7798 + #if(!$categoryDoc.isNew() && $categoryDoc.getObject($\{blogCategoryClassname})) 7799 + #if($foreach.count == 1) 7800 + (% class='separator' ~%)·(%~%) $services.localization.render('blog.code.categories') ## 7801 + #else 7802 + , ## 7803 + #end## 7804 + [[$!\{categoryDoc.getObject($blogCategoryClassname).getValue('name')}>>$\{category}||rel='tag']]## 7805 + #end## 7806 + #end## 7807 + #end 7808 + #end 7809 +#end 7810 +## 7811 +## 7812 +## 7813 +#** 7814 + * Displays blog pagination links (older and newer entries). 7815 + * 7816 + * @param blogDoc the XDocument holding the blog definition object. 7817 + *### 7818 +#macro(displayNavigationLinks $blogDoc) 7819 + (% class=~"clearfloats~" ~%)((())) 7820 + #getBlogDisplayType($blogDoc $displayType) 7821 + #if($displayType == 'weekly') 7822 + (% class=~"pagingLinks~" ~%)((( 7823 + #getRequestedWeek($weekDate) 7824 + $weekDate.addWeeks(-1)## 7825 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.previousweek'/}}>>$doc.documentReference.name||queryString=~"year=$weekDate.weekyear&week=$weekDate.weekOfWeekyear~" class=~"button~"]]**(%~%) 7826 + #sep() 7827 + $weekDate.addWeeks(2)## 2 because we already subtracted 1 above 7828 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.nextweek'/}} »>>$doc.documentReference.name||queryString=~"year=$weekDate.weekyear&week=$weekDate.weekOfWeekyear~" class=~"button~"]]**(%~%) 7829 + ))) 7830 + #elseif($displayType == 'monthly') 7831 + (% class=~"pagingLinks~" ~%)((( 7832 + #getRequestedMonth($monthDate) 7833 + $monthDate.addMonths(-1)## 7834 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.previousmonth'/}}>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"year=$monthDate.year&month=$monthDate.monthOfYear~" class=~"button~"]]**(%~%) 7835 + #sep() 7836 + $monthDate.addMonths(2)## 2 because we already subtracted 1 above 7837 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.nextmonth'/}} »>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"year=$monthDate.year&month=$monthDate.monthOfYear~" class=~"button~"]]**(%~%) 7838 + ))) 7839 + #elseif($displayType == 'all') 7840 + #else 7841 + ## Paginated 7842 + #if(($totalPages > 1)) 7843 + #set($queryString = '') 7844 + #foreach($p in $request.getParameterNames()) 7845 + #if($p != 'page' && $p != 'ipp') 7846 + #foreach($v in $request.getParameterValues($p)) 7847 + #set($queryString = ~"$\{queryString}&$\{escapetool.url($p)}=$\{escapetool.url($v)}~") 7848 + #end 7849 + #end 7850 + #end 7851 + (% class=~"pagingLinks~" ~%)((( 7852 + #if ($currentPageNumber < $totalPages) 7853 + #set($currentPageNumber = $currentPageNumber + 1) 7854 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.olderposts'/}}>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"page=$\{currentPageNumber}&ipp=$\{itemsPerPage}$queryString~" class=~"button~"]]**(%~%) 7855 + #set($currentPageNumber = $currentPageNumber - 1) 7856 + #end 7857 + #if ($currentPageNumber > 1) 7858 + #if ($currentPageNumber < $totalPages) 7859 + #sep() 7860 + #end 7861 + #set($currentPageNumber = $currentPageNumber - 1) 7862 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.newerposts'/}} »>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"page=$\{currentPageNumber}&ipp=$\{itemsPerPage}$queryString~" class=~"button~"]]**(%~%) 7863 + #set($currentPageNumber = $currentPageNumber + 1) 7864 + #end 7865 + (% class=~"clear~" ~%)(%~%) 7866 + )))## pagingLinks 7867 + #end 7868 + #end 7869 +#end 7870 +## 7871 +## 7872 +## 7873 +#** 7874 + * Displays a message box with ~"publish~" icon. 7875 + * 7876 + * @param message A text message concerning blog article publishing 7877 + *### 7878 +#macro(publishMessageBox $message) 7879 +(% class=~"plainmessage publish-message~" ~%)((($services.icon.render('world') $message))) 7880 +#end 7881 +#** 7882 + * Displays a message box with ~"show/hide~" icon. 7883 + * 7884 + * @param message A text message concerning blog article hiding 7885 + *### 7886 +#macro(hideMessageBox $message) 7887 +(% class=~"plainmessage hide-message~" ~%)((($services.icon.render('unlock') $message))) 7888 +#end 7889 +## 7890 +## 7891 +## 7892 +#** 7893 + * Determine the requested week, for using in a weekly-indexed blog. The relevant request parameters are 7894 + * <tt>year</tt> and <tt>week</tt>. By default, the current week is used. 7895 + * 7896 + * @param monthDate The resulting week, a JODATime MutableDateTime. 7897 + *### 7898 +#macro(getRequestedWeek $weekDate) 7899 + #set ($weekDate = $NULL) 7900 + #setVariable (~"$weekDate~" $xwiki.jodatime.mutableDateTime) 7901 + #if(~"$!\{request.year}~" != '') 7902 + #set ($discard = $weekDate.setYear($numbertool.toNumber($request.year).intValue())) 7903 + #end 7904 + #if(~"$!\{request.week}~" != '') 7905 + #set ($discard = $weekDate.setWeekOfWeekyear($numbertool.toNumber($request.week).intValue())) 7906 + #end 7907 +#end 7908 +## 7909 +## 7910 +## 7911 +#** 7912 + * Determine the requested month, for using in a monthly-indexed blog. The relevant request parameters are 7913 + * <tt>year</tt> and <tt>month</tt>. By default, the current month is used. 7914 + * 7915 + * @param monthDate The resulting month, a JODATime MutableDateTime. 7916 + *### 7917 +#macro(getRequestedMonth $monthDate) 7918 + #set ($monthDate = $NULL) 7919 + #setVariable (~"$monthDate~" $xwiki.jodatime.mutableDateTime) 7920 + #if(~"$!\{request.year}~" != '') 7921 + #set ($discard = $monthDate.setYear($numbertool.toNumber($request.year).intValue())) 7922 + #end 7923 + #if(~"$!\{request.month}~" != '') 7924 + #set ($discard = $monthDate.setMonthOfYear($numbertool.toNumber($request.month).intValue())) 7925 + #end 7926 +#end 7927 +## 7928 +## 7929 +## 7930 +#** 7931 + * Retrieve a blog property (title, display type, etc). 7932 + * 7933 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 7934 + * @param propertyName The name of the property to be retrieved. One of the <tt>Blog.BlogClass</tt>'s properties. 7935 + * @param defaultValue The default value to use in case the blog object does not define one. 7936 + * @param propertyValue The resulting value. 7937 + *### 7938 +#macro(getBlogProperty $blogDoc $propertyName $defaultValue $propertyValue) 7939 + #set($result = ~"$!\{blogDoc.getObject($\{blogClassname}).getProperty($propertyName).value}~") 7940 + #if($result == '') 7941 + #set($result = $defaultValue) 7942 + #end 7943 + #set ($propertyValue = $NULL) 7944 + #setVariable (~"$propertyValue~" $result) 7945 +#end 7946 + 7947 +#** 7948 + * If an error occurs when executing an action, set a specific response status and display an error message. 7949 + * 7950 + * @param status The response status. 7951 + * @param text The user readable error to be displayed. Can be a translation key. 7952 + * @param parameters The parameters to use when decoding the translation key. 7953 + *### 7954 +#macro(blog__actionResponseError $status $text $parameters) 7955 + $response.setStatus($status) 7956 + #if($request.ajax) 7957 + $services.localization.render($text, $!parameters) 7958 + #else 7959 + \{\{error}}$services.localization.render($text, $!parameters)\{\{/error}} 7960 + #end 7961 +#end 7962 +## 7963 +## 7964 +## 7965 +#** 7966 + * Check if a blog is the Default blog (The one in the 'Blog' space). 7967 + * 7968 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 7969 + * @param isDefault The resulting boolean. 7970 + *### 7971 +#macro(isDefaultBlog $blogDoc $isDefault) 7972 + #set ($result = false) 7973 + #if ($blogDoc.space == 'Blog') 7974 + #set ($result = true) 7975 + #end 7976 + #setVariable(~"$isDefault~" $result) 7977 +#end 7978 +## 7979 +## 7980 +## 7981 +#** 7982 + * Retrieve the blog posts location (space). 7983 + * 7984 + * @param blogSpace The blog space. It should contain a document with a <tt>Blog.BlogClass</tt> object 7985 + * @param postsLocation The resulting location. 7986 + *### 7987 +#macro(getBlogPostsLocation $blogSpace $postsLocation) 7988 + #getBlogDocument($blogSpace $blogDoc) 7989 + #getBlogProperty($blogDoc 'postsLocation' $blogDoc.space $result) 7990 + #set ($postsLocation = $NULL) 7991 + #setVariable (~"$postsLocation~" $result) 7992 +#end 7993 +## 7994 +## 7995 +## 7996 +#** 7997 + * Retrieve the blog categories location (space). 7998 + * 7999 + * @param blogSpace The blog space. It should contain a document with a <tt>Blog.BlogClass</tt> object 8000 + * @param categoriesLocation The resulting location. 8001 + *### 8002 +#macro(getBlogCategoriesLocation $blogSpace $categoriesLocation) 8003 + #getBlogDocument($blogSpace $blogDoc) 8004 + #getBlogProperty($blogDoc 'categoriesLocation' 'Blog' $result) 8005 + #set ($postsLocation = $NULL) 8006 + #setVariable (~"$categoriesLocation~" $result) 8007 +#end 8008 +###** 8009 + * Return the Query for selecting the blog posts based on the context where the posts are displayed. 8010 + * for example there is 4 different panel contexts: 8011 + * aBlog.aPost or aBlog.WebHome 8012 + * aCategorySpace.aCategory 8013 + * aCategorySpace.WebHome 8014 + * Blog.aPost or Blog.WebHome 8015 + * 8016 + * @param query The query for selecting blog entries. 8017 + * @param queryParams The parameters to bind with the generated query. 8018 + * @param targetDoc The document in which the articles will be displayed. 8019 + *### 8020 +#macro(getAllBlogPostsQueryBasedOnDisplayContext $targetDoc $query $queryParams) 8021 + #set ($query = $NULL) 8022 + #set ($queryParams = $NULL) 8023 + #getCategoryAllBlogPostsQuery($resultQuery) 8024 + #set ($resultQueryParams = \{}) 8025 + #set($sanitizedSpace = $targetDoc.space.replaceAll('([%_!])', '!$1').concat('.%')) 8026 + #if ($targetDoc.space == $defaultBlogSpace && !$targetDoc.getObject($blogCategoryClassname)) 8027 + ## Get all posts that are in the default blog space('Blog') or in category 'Blog.%' 8028 + #set ($discard = $resultQueryParams.put('space', $targetDoc.space)) 8029 + #set ($discard = $resultQueryParams.put('catList', $sanitizedSpace)) 8030 + #set($resultQuery = ~"$\{resultQuery} and (doc.space = :space or catList like :catList escape '!')~") 8031 + #elseif($targetDoc.getObject($blogCategoryClassname)) 8032 + ## Get all posts that are in a category aCategorySpace.aCategory 8033 + #set ($discard = $resultQueryParams.put('catList', $targetDoc.fullName)) 8034 + #set($resultQuery = ~"$\{resultQuery} and (catList=:catList)~") 8035 + #elseif($targetDoc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 8036 + ## Get all posts that are in a category aCategorySpace.% 8037 + #set ($discard = $resultQueryParams.put('catList', $sanitizedSpace)) 8038 + ## Exclude incategorized posts 8039 + #set ($excludedCategory = ~"$\{targetDoc.space}.WebHome~") 8040 + #if ($targetDoc.space == $defaultBlogSpace) 8041 + #set ($excludedCategory = ~"Blog.Categories.WebHome~") 8042 + #end 8043 + #set ($discard = $resultQueryParams.put('excludedCategory', $excludedCategory)) 8044 + #set($resultQuery = ~"$\{resultQuery} and (catList<>:excludedCategory) and (catList like :catList escape '!')~") 8045 + #else 8046 + ## Get all posts in blog space aBlog 8047 + #getAllBlogPostsQuery($resultQuery) 8048 + #getBlogPostsLocation($targetDoc.space $postsLocation) 8049 + #set ($discard = $resultQueryParams.put('space', $postsLocation)) 8050 + #set($resultQuery = ~"$\{resultQuery} and doc.space = :space~") 8051 + #end 8052 + #setVariable(~"$query~" $resultQuery) 8053 + #setVariable(~"$queryParams~" $resultQueryParams) 8054 +#end 8055 +## 8056 +## 8057 +## 8058 +###** 8059 + * Display blog posts based on the context where the posts are displayed. 8060 + * for example there is 4 different panel contexts: 8061 + * aBlog.aPost or aBlog.WebHome 8062 + * aCategorySpace.aCategory 8063 + * aCategorySpace.WebHome 8064 + * Blog.aPost or Blog.WebHome 8065 + * 8066 + * @param targetDoc The document in which the articles will be displayed. 8067 + * @param postsVisiblity The visibility of the posts to display: recent, unpublished, ... 8068 + * @param layout Layout of the the posts to display 8069 + * @param layoutParams additional layout parameters, it is a string like 'param1=val1|...|paramN=valueN' 8070 + * @param limit the number of posts to display 8071 + *### 8072 +#macro(displayBlogPostsBasedOnContext $targetDoc $postsVisiblity $layout $layoutParams $limit) 8073 + #if ($postLayout == 'full') 8074 + #set ($macro.paginated = 'yes') 8075 + #end 8076 + #if ($targetDoc.space == $defaultBlogSpace && !$targetDoc.getObject($blogCategoryClassname)) 8077 + ## Display all posts that are in the default blog space('Blog') or in category 'Blog.%' 8078 + #getBlogPostsLayout($xwiki.getDocument(~"$\{defaultBlogSpace}.WebHome~") $postsLayout) 8079 + #if (~"$!layout~" == '') 8080 + #set ($layout = $postsLayout) 8081 + #end 8082 + #if ($postsVisiblity == 'recent') 8083 + \{\{blogpostlist blog=~"$\{defaultBlogSpace.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')}.WebHome~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 8084 + #elseif($postsVisiblity == 'unpublished') 8085 + \{\{blogpostlist blog=~"$\{defaultBlogSpace.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')}.WebHome~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 8086 + #end 8087 + #elseif($targetDoc.getObject($blogCategoryClassname)) 8088 + ## Display all posts that are in a category aCategorySpace.aCategory 8089 + #getBlogDocumentForCategoriesSpace($targetDoc.space $blogDoc) 8090 + #getBlogPostsLayout($blogDoc $postsLayout) 8091 + #if (~"$!layout~" == '') 8092 + #set ($layout = $postsLayout) 8093 + #end 8094 + #if ($postsVisiblity == 'recent') 8095 + \{\{blogpostlist category=~"$targetDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 8096 + #elseif($postsVisiblity == 'unpublished') 8097 + \{\{blogpostlist category=~"$targetDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 8098 + #end 8099 + #elseif($targetDoc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 8100 + ## Display all posts that are in a category aCategorySpace.% 8101 + #getBlogDocumentForCategoriesSpace($targetDoc.space $blogDoc) 8102 + #getBlogPostsLayout($blogDoc $postsLayout) 8103 + #if (~"$!layout~" == '') 8104 + #set ($layout = $postsLayout) 8105 + #end 8106 + #if ($postsVisiblity == 'recent') 8107 + \{\{blogpostlist category=~"$targetDoc.space.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 8108 + #elseif($postsVisiblity == 'unpublished') 8109 + \{\{blogpostlist category=~"$targetDoc.space.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 8110 + #end 8111 + #else 8112 + ## Display all posts in blog space aBlog 8113 + #getBlogDocument($targetDoc.space $blogDoc) 8114 + #getBlogPostsLayout($blogDoc $postsLayout) 8115 + #if (~"$!layout~" == '') 8116 + #set ($layout = $postsLayout) 8117 + #end 8118 + #if ($postsVisiblity == 'recent') 8119 + \{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 8120 + #elseif($postsVisiblity == 'unpublished') 8121 + \{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 8122 + #end 8123 + #end 8124 +#end 8125 +## 8126 +## 8127 +## 8128 +#** 8129 + * Bind parameters to a query object. 8130 + * 8131 + * @param queryObj The query object (org.xwiki.query.internal.ScriptQuery object) 8132 + * @param queryParams the query parameters. 8133 + *### 8134 +#macro(bindQueryParameters $queryObj $queryParams) 8135 + #set ($output = $queryObj) 8136 + #foreach( $key in $queryParams.keySet() ) 8137 + #set($output = $queryObj.bindValue($key, $queryParams.get($key))) 8138 + #end 8139 + #setVariable(~"$queryObj~" $output) 8140 +#end 8141 +## 8142 +## 8143 +## 8144 +#** 8145 + * Determines the blogposts layout: Default layout (Calendar), Image thumbnail 8146 + * 8147 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>postsLayout</tt> 8148 + * property set. 8149 + * @param postsLayout The resulting string. If the blog object does not define anything, it is considered full (default). 8150 + *### 8151 +#macro(getBlogPostsLayout $blogDoc $postsLayout) 8152 + #getBlogProperty($blogDoc 'postsLayout' 'full' $res) 8153 + #set ($postsLayout = $NULL) 8154 + #setVariable (~"$postsLayout~" $res) 8155 +#end 8156 +## 8157 +## 8158 +## 8159 +#** 8160 + * Retrieve the blog document of a categories space, which is the blog descriptor that have its categoriesLocation property set to the categories space 8161 + * 8162 + * @param categoriesSpace A <tt>String</tt>, the name of the space. 8163 + * @param blogDoc The resulting XDocument. 8164 + *### 8165 +#macro(getBlogDocumentForCategoriesSpace $categoriesSpace $blogDoc) 8166 + #set($macro.blogDocs = $services.query.hql(~", BaseObject obj, StringProperty as categoriesLocationProp where obj.name = doc.fullName and obj.className = '$blogClassname' and obj.id=categoriesLocationProp.id.id and categoriesLocationProp.id.name='categoriesLocation' and categoriesLocationProp.value=:categoriesLocation order by doc.creationDate~").setLimit(1).setOffset(0).bindValue('categoriesLocation', $categoriesSpace).execute()) 8167 + #if($macro.blogDocs.size() > 0 && $categoriesSpace != $defaultBlogSpace) 8168 + #set($macro.result = $xwiki.getDocument($macro.blogDocs.get(0))) 8169 + #else 8170 + ## Fallback to Blog.WebHome, the default blog 8171 + #set($macro.result = $xwiki.getDocument('Blog.WebHome')) 8172 + #end 8173 + #set ($blogDoc = $NULL) 8174 + #setVariable (~"$blogDoc~" $macro.result) 8175 +#end" %) 8176 +((( 8177 +(% class="macro-placeholder hidden" %) 8178 +((( 8179 +macro:velocity 8180 +))) 8181 +))) 8182 +))) 8183 + 8184 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogPostLayoutMacros~"" %) 8185 +((( 8186 +(% class="macro-placeholder hidden" %) 8187 +((( 8188 +macro:include 8189 +))) 8190 + 8191 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|#** 8192 + * Extract the layout parameters from a string. 8193 + * 8194 + * @param layoutParamsString The string representation of the layout parameters. 8195 + * It should contain a String following this format ~"paramName1=Value1|paramName2=Value2|...|paramNameK=ValueK~" 8196 + * @param layoutsParameters The resulting layout parameters Map. 8197 + *### 8198 +#macro(extractLayoutParametersFromString $layoutParamsString $layoutsParameters) 8199 + #set ($layoutsParameters = $NULL) 8200 + #set ($macro.layoutParams = \{}) 8201 + #if (~"$!layoutParamsString~" != '') 8202 + #set ($macro.paramsArr = $layoutParamsString.split('\\|')) 8203 + #foreach ($item in $macro.paramsArr) 8204 + #set ($itemSplit = $item.split('=')) 8205 + #if ($itemSplit.size() == 2) 8206 + #set ($discard = $macro.layoutParams.put($itemSplit[0].trim(), $itemSplit[1].trim())) 8207 + #end 8208 + #end 8209 + #end 8210 + #setVariable(~"$layoutsParameters~" $macro.layoutParams) 8211 +#end" %) 8212 +((( 8213 +(% class="macro-placeholder hidden" %) 8214 +((( 8215 +macro:velocity 8216 +))) 8217 +))) 8218 +))) 8219 + 8220 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-| #macro(displaySmallPost $pDoc $pObj) 8221 + #initLayoutVars($pDoc $pObj) 8222 + <div class=~"col-xs-12 col-sm-6 next_blog~"> 8223 + <a href=~"$pDoc.uRL~" class=~"thumbnail~"> 8224 + <div class=~"row~"> 8225 + <div class=~"col-xs-4~"> 8226 + #if ($imageAtt) 8227 + <img src=~"$escapetool.xml($pDoc.getAttachmentURL($pObj.getValue('image'),'download',$imgQs))~" /> 8228 + #end 8229 + </div> 8230 + <div class=~"col-xs-8 art_det~"> 8231 + <p class=~"text-left~"> 8232 + #if($displayTitle)$!postTitle #end 8233 + <br/><span class=~"date_info~"> $!dateStr </span> 8234 + </p> 8235 + #displayPostDetails($pDoc) 8236 + </div> 8237 + </div> 8238 + </a> 8239 + </div> 8240 + #end 8241 + ## 8242 + ## 8243 + ## 8244 + #macro(displayPinnedPost $pDoc $pObj) 8245 + #initLayoutVars($pDoc $pObj) 8246 + <div class=~"col-xs-12 col-sm-4 col-md-4 blog_pin~"> 8247 + <a href=~"$pDoc.uRL~" class=~"all-post-link~"> 8248 + <div class=~"thumbnail~"> 8249 + #if ($displayTitle && $displayTitleFirstOnPinnedPosts) 8250 + <h5 class=~"title-top~">$!postTitle <br/> <span class=~"date_info~"> $!dateStr </span> </h5> 8251 + #end 8252 + #if ($imageAtt) 8253 + <img src=~"$escapetool.xml($pDoc.getAttachmentURL($pObj.getValue('image'),'download',$imgQs))~" /> 8254 + #end 8255 + <div class=~"caption~"> 8256 + #if ($displayTitle && !$displayTitleFirstOnPinnedPosts) 8257 + <h5 class=~"text-left~">$!postTitle <br/> <span class=~"date_info~"> $!dateStr </span> </h5> 8258 + #end 8259 + #if ($displaySummaryOnPinnedPosts) 8260 + <div class=~"text-left post-summary~"> 8261 + #set ($postContent = $pObj.getProperty('extract').value) 8262 + $!pDoc.getRenderedContent($postContent, $pDoc.syntax.toIdString()) 8263 + </div> 8264 + #end 8265 + #displayPostDetails($pDoc) 8266 + </div> 8267 + </div> 8268 + </a> 8269 + </div> 8270 + #end 8271 + ## 8272 + ## 8273 + ## 8274 + #macro(formatPostDate $pDoc $pObj) 8275 + #set ($formattedDate = $NULL) 8276 + #set($dateStr = $datetool.format('medium', $postDate, $xcontext.locale)) 8277 + #if (~"$!dateStr~" != '') 8278 + #set ($dateArr = $dateStr.split(' ')) 8279 + #if ($dateArr.size() > 3) 8280 + #set ($dateStr = ~"~") 8281 + #foreach($s in $dateArr.subList(0, 3)) 8282 + #set ($dateStr = ~"$\{dateStr}$\{s} ~") 8283 + #end 8284 + #end 8285 + #end 8286 + #setVariable(~"$formattedDate~" $dateStr) 8287 + #end 8288 + ## 8289 + ## 8290 + ## 8291 + #macro(displayPostDetails $pDoc) 8292 + <div class=~"row post-details~"> 8293 + <div class=~"col-xs-8 detail~"> 8294 + #if ($services.like.displayButton($pDoc.documentReference)) 8295 + #set ($optLikeRecord = $services.like.getLikes($pDoc.documentReference)) 8296 + ## Retrieve the likes number in XWiki 12.9+ 8297 + #set ($likeNumber = $optLikeRecord.get()) 8298 + #if (!$stringtool.isNumeric($likeNumber.toString())) 8299 + ## Retrieve the likes number in XWiki versions greater than 12.7 and lower than 12.9. 8300 + #set ($likeNumber = $optLikeRecord.get().likeNumber) 8301 + #end 8302 + #if ($stringtool.isNumeric($likeNumber.toString())) 8303 + <div class=~"post-likes btn btn-default disabled badge~" 8304 + title=~"$escapetool.xml($services.localization.render('like.button.title', [$likeNumber]))~"> 8305 + $services.icon.renderHTML('heart') <span class=~"like-number~">$likeNumber</span> 8306 + </div> 8307 + #end 8308 + #elseif ($services.ratings) 8309 + #set ($ratingsConfigDoc = $services.ratings.getConfigurationDocument($pDoc.documentReference)) 8310 + #if ($ratingsConfigDoc.getValue('displayRatings') == 1) 8311 + <ul class=~"pull-left note list-inline~"> 8312 + #foreach ($x in [1..5]) 8313 + #set ($cls = ~"~") 8314 + #if ($x > $averageVote) 8315 + #set ($cls = ~"-o~") 8316 + #end 8317 + <li><span class=~"fa fa-star$\{cls}~"></span></li> 8318 + #end 8319 + </ul> 8320 + #end 8321 + #end 8322 + </div> 8323 + #if ($showPostComments) 8324 + <div class=~"col-xs-4 com_det~"> 8325 + <p class=~"text-right~">$services.icon.renderHTML('comments') ($!nbComments)</p> 8326 + </div> 8327 + #end 8328 + </div> 8329 + #end 8330 + ## 8331 + ## 8332 + ## 8333 + #macro(initLayoutVars $pDoc $pObj) 8334 + #set ($postTitle = $pDoc.display('title', 'view', $pObj)) 8335 + #set ($imageAtt = $pDoc.getAttachment($pObj.getValue('image'))) 8336 + #isPublished($pObj $isPublished) 8337 + #isHidden($pObj $isHidden) 8338 + #getEntryDate($pDoc $pObj $postDate) 8339 + #formatPostDate($postDate $dateStr) 8340 + #set ($averageVote = $mathtool.round($services.ratings.getAverageRating($pDoc.documentReference).getAverageVote())) 8341 + #set ($nbComments = $pDoc.getComments().size()) 8342 + #set ($showPostComments = true) 8343 + #if ($xwiki.getSpacePreferenceFor('showcomments', $pDoc.documentReference.lastSpaceReference, '1') == '0') 8344 + #set ($showPostComments = false) 8345 + #end 8346 + #end 8347 + ## 8348 + ## 8349 + ## 8350 + #macro(displayEditPinnedPostsButton) 8351 + #if (~"$!macroAdditionalParams.get('postIndex')~" == '0' && $services.security.authorization.hasAccess('edit', $pinnedPostsSourceDoc.fullName)) 8352 + #set ($discard = $xwiki.ssx.use('Blog.PinnedPostsSheet')) 8353 + #set ($discard = $xwiki.jsx.use('Blog.PinnedPostsSheet')) 8354 + #set ($editPinnedPostsURL = $xwiki.getURL($pinnedPostsSourceDoc.fullName, 'get', 'sheet=Blog.PinnedPostsSheet')) 8355 + #set ($newPinnedPostsObjectURL = $xwiki.getURL($pinnedPostsSourceDoc.fullName, 'objectadd', ~"classname=Blog.PinnedPostsClass&form_token=$!\{services.csrf.getToken()}~")) 8356 + <div class=~"edit-pinned-posts-container~"> 8357 + <div class=~"col-xs-12 col-sm-12 col-md-12 text-right edit-pinned-posts~"> 8358 + <a href=~"javascript:;~" class=~"editPinnedPosts~"> 8359 + <span class=~"glyphicon glyphicon-edit~" aria-hidden=~"true~"></span> $services.localization.render('blog.post.layout.cards.pinnedposts.edit') 8360 + </a> 8361 + <input type=~"hidden~" id=~"editPinnedPostsURL~" value=~"$escapetool.xml($editPinnedPostsURL)~"/> 8362 + <input type=~"hidden~" id=~"newPinnedPostsObjectURL~" value=~"$escapetool.xml($newPinnedPostsObjectURL)~"/> 8363 + </div> 8364 + </div> 8365 + #if (~"$!pinnedPostsObj~" == '') 8366 + <input type=~"hidden~" name=~"createPinnedPostsObject~" value=~"1~"/> 8367 + #end 8368 + #end 8369 + #end" %) 8370 +((( 8371 +(% class="macro-placeholder hidden" %) 8372 +((( 8373 +macro:velocity 8374 +))) 8375 +))) 8376 + 8377 +(% data-macro="startmacro:velocity|-||-| #if ($services.security.authorization.hasAccess('view', $xcontext.macro.params.reference)) 8378 + #set ($postDoc = $xwiki.getDocument($xcontext.macro.params.reference)) 8379 + #getEntryObject($postDoc $postObj) 8380 + #if (~"$!postObj~" != '') 8381 + #extractLayoutParametersFromString($xcontext.macro.params.params $macroAdditionalParams) 8382 + ## 8383 + #set ($displayTitle = true) 8384 + #if (~"$!macroAdditionalParams.get('displayTitle')~" == 'false') 8385 + #set ($displayTitle = false) 8386 + #end 8387 + ## 8388 + #set ($displayTitleFirstOnPinnedPosts = false) 8389 + #if (~"$!macroAdditionalParams.get('displayTitleFirstOnPinnedPosts')~" == 'true') 8390 + #set ($displayTitleFirstOnPinnedPosts = true) 8391 + #end 8392 + ## 8393 + #set ($displaySummaryOnPinnedPosts = true) 8394 + #if (~"$!macroAdditionalParams.get('displaySummaryOnPinnedPosts')~" == 'false') 8395 + #set ($displaySummaryOnPinnedPosts = false) 8396 + #end 8397 + ## 8398 + #set ($postIndex = $numbertool.toNumber($macroAdditionalParams.get('postIndex')).intValue()) 8399 + #if ($postIndex == 0) 8400 + #set ($stopBlogPostsDisplay = false) 8401 + #end 8402 + #set ($listResultsCount = $numbertool.toNumber($macroAdditionalParams.get('list-results-count')).intValue()) 8403 + #set($discard = $xwiki.ssx.use('Blog.BlogPostLayoutCards')) 8404 + #set($scaleWidth = 600) 8405 + #set($imgQs=~"width=$scaleWidth~") 8406 + ## Display pinned posts 8407 + ## Get the list of pinned posts 8408 + #set ($pinnedPosts = []) 8409 + #set ($pinnedPostsSourceDoc = $NULL) 8410 + #if (~"$!macroAdditionalParams.get('forceDisplayPinnedPosts')~" == 'true' && ~"$!macroAdditionalParams.get('list-blog')~" != '') 8411 + #set ($pinnedPostsSourceDoc = $xwiki.getDocument($macroAdditionalParams.get('list-blog'))) 8412 + ## Note that the 'list-blog' parameter contains the value of the 'blog' parameter of the blogpostlist macro passed to the current layout macro 8413 + #end 8414 + #if (~"$!pinnedPostsSourceDoc~" != '') 8415 + #set ($pinnedPostsObj = $pinnedPostsSourceDoc.getObject('Blog.PinnedPostsClass')) 8416 + #if (~"$!pinnedPostsObj~" != '') 8417 + #set ($orderedPinnedPostsJSON = $pinnedPostsObj.getProperty('orderedPosts').value) 8418 + #if (~"$!orderedPinnedPostsJSON~" != '') 8419 + #set ($pinnedPosts = $jsontool.parse($orderedPinnedPostsJSON)) 8420 + #else 8421 + #set ($pinnedPosts = $pinnedPostsObj.getProperty('posts').value) 8422 + #end 8423 + #end 8424 + #if ($postIndex == 0 && $pinnedPosts.size() > 0) 8425 + \{\{html clean=~"false~"}} 8426 + #set ($x = 0) 8427 + #set ($showPinnedPostsButton = true) 8428 + #foreach ($pinnedPost in $pinnedPosts) 8429 + #if ($x == 0) 8430 + <div class=~"row flex-container~"> 8431 + #if ($showPinnedPostsButton) 8432 + #displayEditPinnedPostsButton() 8433 + #set($showPinnedPostsButton = false) 8434 + #end 8435 + #end 8436 + #set ($pinnedPostDoc = $xwiki.getDocument($pinnedPost)) 8437 + #getEntryObject($pinnedPostDoc $pinnedPostObj) 8438 + #if (~"$!pinnedPostObj~" != '') 8439 + #displayPinnedPost($pinnedPostDoc $pinnedPostObj) 8440 + #end 8441 + #set ($x = $mathtool.add($x, 1)) 8442 + #if ($x == 3) 8443 + #set ($x = 0) 8444 + </div> 8445 + #end 8446 + #end 8447 + #if ($mathtool.mod($x, 3) != 0) 8448 + </div> 8449 + #end 8450 + \{\{/html}} 8451 + 8452 + ## If the first post is a pinned post : this means that all the posts are pinned 8453 + ## In this case, avoid displaying the posts again after the pinned posts section. 8454 + #if ($pinnedPosts.contains($xcontext.macro.params.reference)) 8455 + #set ($stopBlogPostsDisplay = true) 8456 + #end 8457 + #elseif($postIndex == 0 && $pinnedPosts.size() == 0) 8458 + 8459 + \{\{html}} 8460 + <div class=~"row no-pinnded-posts~"> 8461 + #displayEditPinnedPostsButton() 8462 + </div> 8463 + \{\{/html}} 8464 + 8465 + #end 8466 + #end 8467 + #if (!$stopBlogPostsDisplay) 8468 + \{\{html clean=~"false~"}} 8469 + #if (~"$!nbDisplayedPosts~" == '' || $postIndex == 0) 8470 + #set ($nbDisplayedPosts = 0) 8471 + #end 8472 + #if($mathtool.mod($nbDisplayedPosts, 2) == 0) 8473 + #if ($nbDisplayedPosts != 0) 8474 + </div> 8475 + #set ($lastHtmlTag = 'c') 8476 + #end 8477 + #if ($nbDisplayedPosts == $mathtool.sub($listResultsCount, 1)) 8478 + <div class=~"row~"> 8479 + #set ($lastHtmlTag = 'o') 8480 + #set ($lastRowClass = ~"~") 8481 + #else 8482 + <div class=~"row flex-container~"> 8483 + #set ($lastHtmlTag = 'o') 8484 + #set ($lastRowClass = ~"flex~") 8485 + #end 8486 + #end 8487 + #displaySmallPost($postDoc $postObj) 8488 + #if ($nbDisplayedPosts == $mathtool.sub($listResultsCount, 1)) 8489 + </div> 8490 + #set ($lastHtmlTag = 'c') 8491 + #end 8492 + #set ($nbDisplayedPosts = $mathtool.add($nbDisplayedPosts, 1)) 8493 + #if ($postIndex == $mathtool.sub($listResultsCount, 1)) 8494 + #if (~"$!lastHtmlTag~" == 'o') 8495 + #if ($mathtool.mod($nbDisplayedPosts, 2) == 1 && $lastRowClass == ~"flex~") 8496 + <div class=~"col-xs-12 col-sm-6~"></div> 8497 + #end 8498 + </div> 8499 + #end 8500 + #end 8501 + \{\{/html}} 8502 + #end 8503 + #else 8504 + \{\{error}}$services.localization.render('blog.blogpostlayout.notpost', [$xcontext.macro.params.reference])\{\{/error}} 8505 + #end 8506 + #else 8507 + \{\{error}}$services.localization.render('blog.blogpostlayout.post_view_not_allowed', [$xcontext.macro.params.reference])\{\{/error}} 8508 + #end" class="macro hidden macro-placeholder" %)macro:velocity(% data-macro="startmacro:html|-|clean=~"false~"|-|</div> 8509 +<div class=~"row flex-container~"> 8510 +<div class=~"col-xs-12 col-sm-6 next_blog~"> 8511 +<a href=~"/Nieuws/Feestelijke%20afsluiting%2C%20keuring%20materieel%20en%20tellen%20voorraden~" class=~"thumbnail~"> 8512 +<div class=~"row~"> 8513 +<div class=~"col-xs-4~"> 8514 +<img src=~"/download/Nieuws/Feestelijke%20afsluiting%2C%20keuring%20materieel%20en%20tellen%20voorraden/OIP.jpg?width=600&rev=1.1~" /> 8515 +</div> 8516 +<div class=~"col-xs-8 art_det~"> 8517 +<p class=~"text-left~"> 8518 +Feestelijke afsluiting, keuring materieel en tellen voorraden <br/><span class=~"date_info~"> Dec 14, 2023, </span> 8519 +</p> 8520 +<div class=~"row post-details~"> 8521 +<div class=~"col-xs-8 detail~"> 8522 +<div class=~"post-likes btn btn-default disabled badge~" 8523 +title=~"Number of likes on this page: 1~"> 8524 +<span class=~"fa fa-heart~"></span> <span class=~"like-number~">1</span> 8525 +</div> 8526 +</div> 8527 +<div class=~"col-xs-4 com_det~"> 8528 +<p class=~"text-right~"><span class=~"fa fa-comments~"></span> (0)</p> 8529 +</div> 8530 +</div> 8531 +</div> 8532 +</div> 8533 +</a> 8534 +</div>" class="macro macro hidden macro-placeholder" %)macro:html 8535 +))) 8536 +))) 8537 + 8538 +(% class="row flex-container" %) 8539 +((( 8540 +(% class="col-xs-12 col-sm-6 next_blog" %) 8541 +((( 8542 +(% class="row" %) 8543 +((( 8544 +(% class="col-xs-4" %) 8545 +((( 8546 +[[~[~[image:/download/Nieuws/Feestelijke%20afsluiting%2C%20keuring%20materieel%20en%20tellen%20voorraden/OIP.jpg?width=600&rev=1.1~]~]>>path:/Nieuws/Feestelijke%20afsluiting%2C%20keuring%20materieel%20en%20tellen%20voorraden||class="thumbnail"]] 8547 +))) 8548 + 8549 +(% class="col-xs-8 art_det" %) 8550 +((( 8551 +(% class="text-left" %) 8552 +[[Feestelijke afsluiting, keuring materieel en tellen voorraden 8553 +(% class="date_info" %) Dec 14, 2023,>>path:/Nieuws/Feestelijke%20afsluiting%2C%20keuring%20materieel%20en%20tellen%20voorraden||class="thumbnail"]] 8554 + 8555 +(% class="row post-details" %) 8556 +((( 8557 +(% class="col-xs-8 detail" %) 8558 +((( 8559 +(% class="post-likes btn btn-default disabled badge" title="Number of likes on this page: 1" %) 8560 +((( 8561 +[[(% class="like-number" %)1>>path:/Nieuws/Feestelijke%20afsluiting%2C%20keuring%20materieel%20en%20tellen%20voorraden||class="thumbnail"]] 8562 +))) 8563 +))) 8564 + 8565 +(% class="col-xs-4 com_det" %) 8566 +((( 8567 +(% class="text-right" %) 8568 +[[(0)>>path:/Nieuws/Feestelijke%20afsluiting%2C%20keuring%20materieel%20en%20tellen%20voorraden||class="thumbnail"]] 8569 +))) 8570 +))) 8571 +))) 8572 +))) 8573 +))) 8574 + 8575 +(% class="macro" data-macro="startmacro:blogPostLayoutCards|-|reference=~"Nieuws.RI&E 2023~" params=~"displayTitle=true|useSummary=true|list-limit=10|list-layout=cards|list-blog=Nieuws.WebHome|list-paginated=no|list-results-count=10|postIndex=0|previousPostReference=|postIndex=1|previousPostReference=Nieuws.Nieuwsbrief januari 2024|postIndex=2|previousPostReference=Nieuws.Audit BMI en BORG|postIndex=3|previousPostReference=Nieuws.Nieuwsbrief december 2023|postIndex=4|previousPostReference=Nieuws.Nieuwsbrief november 2023|postIndex=5|previousPostReference=Nieuws.Feestelijke afsluiting, keuring materieel en tellen voorraden~"" %) 8576 +((( 8577 +(% class="macro-placeholder hidden" %) 8578 +((( 8579 +macro:blogPostLayoutCards 8580 +))) 8581 + 8582 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogCode~"" %) 8583 +((( 8584 +(% class="macro-placeholder hidden" %) 8585 +((( 8586 +macro:include 8587 +))) 8588 + 8589 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogParameters~"" %) 8590 +((( 8591 +(% class="macro-placeholder hidden" %) 8592 +((( 8593 +macro:include 8594 +))) 8595 + 8596 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|## Blog 8597 +#set($blogClassname = 'Blog.BlogClass') 8598 +#set($blogTemplate = 'Blog.BlogTemplate') 8599 +#set($blogSheet = 'Blog.BlogSheet') 8600 +## Blog entries 8601 +#set($blogPostClassname = 'Blog.BlogPostClass') 8602 +#set($blogPostTemplate = 'Blog.BlogPostTemplate') 8603 +#set($blogPostSheet = 'Blog.BlogPostSheet') 8604 +#set($blogPostObjectNumber = $xwiki.getDocument($blogPostTemplate).getObject($blogPostClassname).number) 8605 +#set($oldArticleClassname = 'XWiki.ArticleClass') 8606 +## Categories 8607 +#set($blogCategoryClassname = 'Blog.CategoryClass') 8608 +#set($blogCategoryTemplate = 'Blog.CategoryTemplate') 8609 +#set($blogCategorySheet = 'Blog.CategorySheet') 8610 +#set($blogCategoriesSheet = 'Blog.CategoriesSheet') 8611 +#set($oldBlogCategoryClassname = 'Blog.Categories') 8612 +#set($defaultCategoryParent = 'Blog.Categories') 8613 +## Style 8614 +#set($blogStyleDocumentName = 'Blog.BlogStyle') 8615 +#set($blogStyleDocument = $xwiki.getDocument($blogStyleDocumentName)) 8616 +## Clientside scripts 8617 +#set($blogScriptsDocumentName = 'Blog.BlogScripts') 8618 +#set($blogPublisher = $xwiki.getDocument('Blog.Publisher')) 8619 +## Misc 8620 +#set($thisURL = $doc.getURL($xcontext.action, $request.queryString)) 8621 +#set($isBlogPost = $doc.getObject($blogPostClassname)) 8622 +#set($defaultBlogSpace = 'Blog') 8623 +#set($isCategoryPage = $doc.getObject($blogCategoryClassname)) 8624 +#set($isCategoriesHomePage = $doc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 8625 +## 8626 +## 8627 +## 8628 +#** 8629 + * Displays an image, taken from the blog style document. 8630 + * 8631 + * @param $imgName The name of the icon from icons set to use. 8632 + *# 8633 +#macro(toolImage $imgName)(% class=~"icon-manage~"~%)$services.icon.render($imgName)(%~%)#end" %) 8634 +((( 8635 +(% class="macro-placeholder hidden" %) 8636 +((( 8637 +macro:velocity 8638 +))) 8639 +))) 8640 +))) 8641 + 8642 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|## 8643 +## 8644 +## 8645 +## Import the blog skin and javascripts. 8646 +$!xwiki.ssx.use($blogStyleDocumentName)## 8647 +$!xwiki.jsx.use($blogScriptsDocumentName)## 8648 +## 8649 +## import the hierarchy for the path the blog post footer (in displayEntryBlogLocation) 8650 +#template('hierarchy_macros.vm')## 8651 +## 8652 +## 8653 +#** 8654 + * Prints a blog. This is the main macro used in the BlogSheet. 8655 + * 8656 + * @param blogDoc the XDocument holding the blog definition object. 8657 + *### 8658 +#macro(printBlog $blogDoc) 8659 + \{\{include reference='Blog.CreatePost'/}} 8660 + 8661 + ## Use the blogPostList macro to display the blogposts 8662 + ## The blogPostList is used only in case of 'all' or 'paged' blog display type because the blogPostList macro 8663 + ## do not support FTM the monthly and weekly blog display types 8664 + #getBlogDisplayType($blogDoc $displayType) 8665 + #if ($displayType == 'weekly' || $displayType == 'monthly') 8666 + #getBlogEntries($blogDoc $entries) 8667 + #displayBlog($entries 'index' true true) 8668 + #displayNavigationLinks($blogDoc) 8669 + #else 8670 + #getBlogDisplayType($blogDoc $displayType) 8671 + #set ($paginated = 'no') 8672 + #if ($displayType == 'paginated') 8673 + #set ($paginated = 'yes') 8674 + #end 8675 + #getBlogPostsLayout($blogDoc $postsLayout) 8676 + (% class=~"hfeed index~" ~%)(((\{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$paginated~" layout=~"$!postsLayout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" /}}))) 8677 + #end 8678 +#end 8679 +## 8680 +## 8681 +## 8682 +#** 8683 + * Shows blog information. In view mode, the description is printed. In edit mode, allows changing blog settings: title, 8684 + * description, blog type (global or in-space), index display type (fixed size pagination, weekly index, monthly index, 8685 + * all entries). 8686 + * 8687 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 8688 + *### 8689 +#macro(showBlogInfo $blogDoc) 8690 + #if($blogDoc.getObject($blogClassname)) 8691 + ## Keep testing for inline action for backward compatibility with older blogs. 8692 + #if($xcontext.action == 'edit' || $xcontext.action == 'inline') 8693 + #macro(displayProperty $blogDoc $propname) 8694 + ; #displayPropName($xwiki.getClass($blogClassname).get($propname)): 8695 + : $blogDoc.display($propname) 8696 + #end 8697 + #displayProperty($blogDoc 'title') 8698 + #displayProperty($blogDoc 'description') 8699 + #displayProperty($blogDoc 'displayType') 8700 + #displayProperty($blogDoc 'itemsPerPage') 8701 + #displayProperty($blogDoc 'postsLayout') 8702 + #displayProperty($blogDoc 'postsLayoutParameters') 8703 + #else 8704 + $blogDoc.display('description') 8705 + #end 8706 + #elseif($doc.fullName == $blogSheet) 8707 += $services.localization.render('blog.code.blogsheet') = 8708 + \{\{translation key='blog.code.sheetexplanation'/}} 8709 + #else 8710 + \{\{warning}}\{\{translation key='blog.code.notblog'/}}\{\{/warning}} 8711 + #end 8712 +#end 8713 +## 8714 +## 8715 +## 8716 +#** 8717 + * Retrieve the blog document, which usually is either <tt><Space>.WebHome</tt> for whole-spaces blogs, or 8718 + * <tt><Space>.Blog</tt> for in-space blogs. If none of these documents contains a blog object, then the first 8719 + * (alphabetically) document in the target space that contains one is returned. Finally, if no document in the current 8720 + * space contains a blog object, then <tt>Blog.WebHome</tt> is returned as the default blog. 8721 + * 8722 + * @param space A <tt>String</tt>, the name of the space where to search. 8723 + * @param blogDoc The resulting XDocument. 8724 + *### 8725 +#macro(getBlogDocument $space $blogDoc) 8726 + #set ($result = $NULL) 8727 + ## Check if the current space has a WebPreferences page that contains a Blog.EnablePanelsConfigurationClass object. 8728 + ## It is possible to display a blog's panels info in a page that is not inside a blog and in that case we need to 8729 + ## identify the right blog based on a configuration object in a WebPreferences page. 8730 + #set ($spaceReference = $services.model.resolveSpace($space)) 8731 + #set ($preferencesDocRef = $services.model.createDocumentReference('WebPreferences', $spaceReference)) 8732 + #set ($preferencesDoc = $xwiki.getDocument($preferencesDocRef)) 8733 + #set ($preferencesObj = $preferencesDoc.getObject('Blog.EnablePanelsConfigurationClass')) 8734 + #if ($preferencesObj) 8735 + #set ($result = $xwiki.getDocument( $preferencesObj.getValue('blog'))) 8736 + #end 8737 + ## If no special case occurs (like the previous one), identify a blog by checking the current location from its hierarchy. 8738 + #if (~"$!result~" == '') 8739 + ## First, try the Space.WebHome, for a whole-space blog 8740 + #set($result = $xwiki.getDocument(~"$\{space}.WebHome~")) 8741 + #if(!$result.getObject($blogClassname)) 8742 + ## Second, try the Space.Blog document 8743 + #set($result = $xwiki.getDocument(~"$\{space}.Blog~")) 8744 + #if(!$result.getObject($blogClassname)) 8745 + ## Third, try searching for a blog document in the current space 8746 + ## Prevent the query fail when the space contains dots '.' 8747 + #set($blogDocs = $services.query.hql(~", BaseObject obj where doc.space = :space and obj.name = doc.fullName and obj.className = '$blogClassname' order by doc.name~").setLimit(1).setOffset(0).bindValue('space', $space).execute()) 8748 + #if($blogDocs.size() > 0) 8749 + #set($result = $xwiki.getDocument($blogDocs.get(0))) 8750 + #else 8751 + ## Fourth, try searching for a blog document that have the its 'postsLocation' set to the current space 8752 + #set($blogDocs = $services.query.hql(~", BaseObject obj, StringProperty as postsLocationProp where obj.name = doc.fullName and obj.className = '$blogClassname' and obj.id=postsLocationProp.id.id and postsLocationProp.id.name='postsLocation' and postsLocationProp.value=:postsLocation order by doc.name~").setLimit(1).setOffset(0).bindValue('postsLocation', $space).execute()) 8753 + #if($blogDocs.size() > 0) 8754 + #set($result = $xwiki.getDocument($blogDocs.get(0))) 8755 + #else 8756 + ## Last, fallback to Blog.WebHome, the default blog 8757 + #set($result = $xwiki.getDocument('Blog.WebHome')) 8758 + #end 8759 + #end 8760 + #end 8761 + #end 8762 + #end 8763 + #set ($blogDoc = $NULL) 8764 + #setVariable (~"$blogDoc~" $result) 8765 +#end 8766 +## 8767 +## 8768 +## 8769 +#** 8770 + * Retrieve the blog title. 8771 + * 8772 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass<tt> object with the <tt>title</tt> property set. 8773 + * @param title The resulting title. 8774 + *### 8775 +#macro(getBlogTitle $blogDoc $title) 8776 + ## Titles can contain velocity code (most commonly translations), so we should evaluate them. 8777 + #set ($title = $NULL) 8778 + #setVariable (~"$title~" $!blogDoc.displayTitle) 8779 +#end 8780 +## 8781 +## 8782 +## 8783 +#** 8784 + * Retrieve the blog description. 8785 + * 8786 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>description</tt> 8787 + * property set. 8788 + * @param description The resulting description. 8789 + *### 8790 +#macro(getBlogDescription $blogDoc $description) 8791 + #getBlogProperty($blogDoc 'description' '' $result) 8792 + #set ($description = $NULL) 8793 + #setVariable (~"$description~" $result) 8794 +#end 8795 +## 8796 +## 8797 +## 8798 +#** 8799 + * Retrieves a list of entries to be displayed. The entries are either part of the blog's space, or have the blog 8800 + * document set as a parent. The number and range of entries returned (from all those belonging to this blog) depends on 8801 + * the blog display type: paginated (fixed number of entries), weekly (all entries in a week), monthly (all entries in a 8802 + * month), or all. 8803 + * 8804 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 8805 + * @param entries The resulting list of entries to display, a list of XDocument names. 8806 + *### 8807 +#macro(getBlogEntries $blogDoc $entries) 8808 + #if (!$entries) 8809 + #setVariable (~"$entries~" []) 8810 + #end 8811 + #getAllBlogPostsQuery($query) 8812 + #isDefaultBlog($blogDoc $isDefault) 8813 + #set($queryParams = \{}) 8814 + #if ($isDefault) 8815 + #getCategoryAllBlogPostsQuery($query) 8816 + #set($query = ~"$\{query} and (doc.creator = :creator or (isPublished.value = 1 and hidden.value = 0)) and (doc.space = :space or catList like :catList escape '!')~") 8817 + #set($discard = $queryParams.put('creator', $xcontext.user)) 8818 + #set($discard = $queryParams.put('space', $blogDoc.space)) 8819 + #set($sanitizedBlogSpace = $blogDoc.space.replaceAll('([%_!])', '!$1').concat('.%')) 8820 + #set($discard = $queryParams.put('catList', $sanitizedBlogSpace)) 8821 + #else 8822 + #set($query = ~"$\{query} and (doc.space = :space or doc.parent = :parent)~") 8823 + #getBlogPostsLocation($blogDoc.space $blogPostsLocation) 8824 + #set($discard = $queryParams.put('space', $blogPostsLocation)) 8825 + #set($discard = $queryParams.put('parent', $blogDoc.space)) 8826 + #end 8827 + #getBlogDisplayType($blogDoc $displayType) 8828 + #if($displayType == 'weekly') 8829 + #getWeeklyBlogEntries($blogDoc $query $entries $queryParams) 8830 + #elseif($displayType == 'monthly') 8831 + #getMonthlyBlogEntries($blogDoc $query $entries $queryParams) 8832 + #elseif($displayType == 'all') 8833 + #getAllBlogEntries($blogDoc $query $entries $queryParams) 8834 + #else 8835 + #getPagedBlogEntries($blogDoc $query $entries $queryParams) 8836 + #end 8837 +#end 8838 +## 8839 +## 8840 +## 8841 +#** 8842 + * Retrieves a list of entries to be displayed. The entries are taken from a ~"page~" of the blog, a sequence of documents 8843 + * defined by the request parameters <tt>ipp</tt> (items per page) and <tt>page</tt> (the current page). Initially the 8844 + * first page is displayed, with the number of entries defined in the blog object in the <tt>itemsPerPage</tt> property 8845 + * (10 if not defined). 8846 + * 8847 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 8848 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 8849 + * refined to restrict to a given space, or to a given search criteria, etc. 8850 + * @param entries The resulting list of entries to display, a list of XDocument names. 8851 + * @param queryParams The parameters to bind with the query. 8852 + *### 8853 +#macro(getPagedBlogEntries $blogDoc $query $entries $queryParams) 8854 + #if (!$entries) 8855 + #setVariable (~"$entries~" []) 8856 + #end 8857 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 8858 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~")) 8859 + #bindQueryParameters($countQueryObj $queryParams) 8860 + #bindQueryParameters($queryObj $queryParams) 8861 + #set($totalEntries = $countQueryObj.count()) 8862 + #getBlogProperty($blogDoc 'itemsPerPage' '10' $defaultItemsPerPage) 8863 + #set($defaultItemsPerPage = $numbertool.toNumber($defaultItemsPerPage).intValue()) 8864 + ## This macro is defined in the default macros.vm library. It also sets $itemsPerPage and $startAt. 8865 + #preparePagedViewParams($totalEntries $defaultItemsPerPage) 8866 + #set($discard = $entries.addAll($queryObj.setLimit($itemsPerPage).setOffset($startAt).addFilter(~"unique~").execute())) 8867 +#end 8868 +## 8869 +## 8870 +## 8871 +#** 8872 + * Retrieves a list of entries to be displayed. The entries are taken from a week of the blog. The target week is 8873 + * defined by the request parameters <tt>week</tt> (the week number in the year, from 1 to 52) and <tt>year</tt> (4 8874 + * digit year). Initially the current week is displayed. 8875 + * 8876 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 8877 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 8878 + * refined to restrict to a given space, or to a given search criteria, etc. 8879 + * @param entries The resulting list of entries to display, a list of XDocument names. 8880 + * @param queryParams The parameters to bind with the query. 8881 + *### 8882 +#macro(getWeeklyBlogEntries $blogDoc $query $entries $queryParams) 8883 + #if (!$entries) 8884 + #setVariable (~"$entries~" []) 8885 + #end 8886 + #getRequestedWeek($weekDate) 8887 + #set($dateFormatter = $xwiki.jodatime.getDateTimeFormatterForPattern('yyyy-MM-dd')) 8888 + #set($minDay = $dateFormatter.print($weekDate.toMutableDateTime().weekOfWeekyear().roundFloor())) 8889 + #set($maxDay = $dateFormatter.print($weekDate.toMutableDateTime().weekOfWeekyear().roundCeiling())) 8890 + #set($query = ~"$\{query} and publishDate.value >= '$minDay' and publishDate.value < '$maxDay'~") 8891 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 8892 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 8893 + #bindQueryParameters($countQueryObj $queryParams) 8894 + #bindQueryParameters($queryObj $queryParams) 8895 + #set($totalEntries = $countQueryObj.count()) 8896 + #set($discard = $entries.addAll($queryObj.execute())) 8897 +#end 8898 +## 8899 +## 8900 +## 8901 +#** 8902 + * Retrieves a list of entries to be displayed. The entries are taken from a month of the blog. The target month is 8903 + * defined by the request parameters <tt>month</tt> (the month number, from 1 to 12) and <tt>year</tt> (4 8904 + * digit year). Initially the current month is displayed. 8905 + * 8906 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 8907 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 8908 + * refined to restrict to a given space, or to a given search criteria, etc. 8909 + * @param entries The resulting list of entries to display, a list of XDocument names. 8910 + * @param queryParams The parameters to bind with the query. 8911 + *### 8912 +#macro(getMonthlyBlogEntries $blogDoc $query $entries) 8913 + #if (!$entries) 8914 + #setVariable (~"$entries~" []) 8915 + #end 8916 + #getRequestedMonth($monthDate) 8917 + #set($dateFormatter = $xwiki.jodatime.getDateTimeFormatterForPattern('yyyy-MM-dd')) 8918 + #set($minDay = $dateFormatter.print($monthDate.toMutableDateTime().monthOfYear().roundFloor())) 8919 + #set($maxDay = $dateFormatter.print($monthDate.toMutableDateTime().monthOfYear().roundCeiling())) 8920 + #set($query = ~"$\{query} and publishDate.value >= '$minDay' and publishDate.value < '$maxDay'~") 8921 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 8922 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 8923 + #bindQueryParameters($countQueryObj $queryParams) 8924 + #bindQueryParameters($queryObj $queryParams) 8925 + #set($totalEntries = $countQueryObj.count()) 8926 + #set($discard = $entries.addAll($queryObj.execute())) 8927 +#end 8928 +## 8929 +## 8930 +## 8931 +#** 8932 + * Retrieves a list of entries to be displayed. All entries belonging to the current blog are returned. 8933 + * 8934 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 8935 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 8936 + * refined to restrict to a given space, or to a given search criteria, etc. 8937 + * @param entries The resulting list of entries to display, a list of XDocument names. 8938 + * @param queryParams The parameters to bind with the query. 8939 + *### 8940 +#macro(getAllBlogEntries $blogDoc $query $entries $queryParams) 8941 + #if (!$entries) 8942 + #setVariable (~"$entries~" []) 8943 + #end 8944 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 8945 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 8946 + #bindQueryParameters($countQueryObj $queryParams) 8947 + #bindQueryParameters($queryObj $queryParams) 8948 + #set($totalEntries = $countQueryObj.count()) 8949 + #set($discard = $entries.addAll($queryObj.execute())) 8950 +#end 8951 +## 8952 +## 8953 +## 8954 +#** 8955 + * Retrieves a list of entries to be displayed. Only (and all) unpublished entries are returned. 8956 + * 8957 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 8958 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 8959 + * refined to restrict to a given space, or to a given search criteria, etc. 8960 + * @param queryParams The parameters to bind with the query. 8961 + * @param entries The resulting list of entries to display, a list of XDocument names. 8962 + *### 8963 +#macro(getUnpublishedBlogEntries $blogDoc $query $queryParams $entries) 8964 + #if (!$entries) 8965 + #setVariable (~"$entries~" []) 8966 + #end 8967 + #set($query = ~"$\{query} and isPublished.value = 0~") 8968 + #set ($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 8969 + #set ($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 8970 + #bindQueryParameters($countQueryObj $queryParams) 8971 + #bindQueryParameters($queryObj $queryParams) 8972 + #set($totalEntries = $countQueryObj.count()) 8973 + #set($discard = $entries.addAll($queryObj.execute())) 8974 +#end 8975 +## 8976 +## 8977 +## 8978 +#** 8979 + * Retrieves a list of entries to be displayed. The entries are taken from all the wiki, and not from a specific blog. 8980 + * 8981 + * @param entries The resulting list of entries to display, a list of XDocument names. 8982 + *### 8983 +#macro(getGlobalBlogEntries $entries) 8984 + #if (!$entries) 8985 + #setVariable (~"$entries~" []) 8986 + #end 8987 + #getAllBlogPostsQuery($query) 8988 + #set($totalEntries = $services.query.hql($query).count()) 8989 + #set($defaultItemsPerPage = 20) 8990 + ## This macro is defined in the default macros.vm library. It also sets $itemsPerPage and $startAt. 8991 + #preparePagedViewParams($totalEntries $defaultItemsPerPage) 8992 + #set($discard = $entries.addAll($services.query.hql(~"$\{query} order by publishDate.value desc~").setLimit($itemsPerPage).setOffset($startAt).execute())) 8993 +#end 8994 +#** 8995 + * Return the base query for selecting blog entries. It filters only visible entries, but does not bind to a specific 8996 + * blog, nor specify a range or an ordering criteria. 8997 + * 8998 + * This macro is a duplicate of #getAllBlogPostsQuery($query). Keep it in case it's used outside, in custom code. 8999 + * 9000 + * @param query The basic query for selecting blog entries. 9001 + *# 9002 +#macro(getBlogEntriesBaseQuery $query) 9003 + #getAllBlogPostsQuery($query) 9004 +#end 9005 +#** 9006 + * Return the Query for selecting the all wiki blog posts without filtering 9007 + * 9008 + * @param query The basic query for selecting blog entries. 9009 + *# 9010 +#macro(getAllBlogPostsQuery $query) 9011 + #set ($query = $NULL) 9012 + #setVariable(~"$query~" ~", BaseObject as obj, IntegerProperty isPublished, 9013 + IntegerProperty hidden, DateProperty publishDate 9014 + where doc.fullName <> '$blogPostTemplate' and 9015 + obj.name=doc.fullName and obj.className='$blogPostClassname' and 9016 + isPublished.id.id = obj.id and isPublished.id.name = 'published' and 9017 + hidden.id.id = obj.id and hidden.id.name='hidden' and 9018 + publishDate.id.id = obj.id and publishDate.id.name='publishDate' and 9019 + (doc.creator = '$escapetool.sql($xcontext.user)' or (isPublished.value = 1 and hidden.value = 0))~") 9020 +#end 9021 +## 9022 +## 9023 +## 9024 +###** 9025 + * Return the Query for selecting the all wiki blog posts with categories filtering 9026 + * 9027 + * @param query The basic query for selecting blog entries. 9028 + *### 9029 +#macro(getCategoryAllBlogPostsQuery $query) 9030 + #set ($query = $NULL) 9031 + #getAllBlogPostsQuery($baseQuery) 9032 + #set ($baseQuery = $baseQuery.replace('DateProperty publishDate', 'DateProperty publishDate, DBStringListProperty as category left join category.list catList')) 9033 + #setVariable(~"$query~" ~"$\{baseQuery} and obj.id=category.id.id and category.id.name='category'~") 9034 +#end 9035 +## 9036 +## 9037 +## 9038 +#** 9039 + * Determines how is the blog index split into pages: paginated (fixed number of entries), weekly (all entries in a 9040 + * week), monthly (all entries in a month), or all. 9041 + * 9042 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>displayType</tt> 9043 + * property set. 9044 + * @param displayType The resulting string. If the blog object does not define anything, it is considered paginated. 9045 + *### 9046 +#macro(getBlogDisplayType $blogDoc $displayType) 9047 + #getBlogProperty($blogDoc 'displayType' 'paginated' $result) 9048 + #set ($displayType = $NULL) 9049 + #setVariable (~"$displayType~" $result) 9050 +#end 9051 +## 9052 +## 9053 +## 9054 +#** 9055 + * Displays a list of entries. 9056 + * 9057 + * @param entries The entries to display, a list of XDocument names. 9058 + * @param displaying What exactly is displayed: blog index, a single blog entry, a blog category, search results, 9059 + * unpublished entries, etc. This will be used as the classname(s) for the container div (hfeed). Currently 9060 + * used values: index, single, category, search, unpublished, hidden. 9061 + * @param onlyExtract If <tt>true</tt>, only display the extract of articles where available, otherwise display the full content. 9062 + * @param shouldDisplayTitles If <tt>true</tt>, display the blog title (blog posts shouldn't display the title when they're 9063 + * displayed alone on their page since it's the page title which is used in this case) 9064 + *### 9065 +#macro(displayBlog $entries $displaying $onlyExtract $shouldDisplayTitles) 9066 + #set($blogDay = '') 9067 + (% class=~"hfeed $!\{displaying}~" ~%)((( 9068 + (% class=~"blogDay~" ~%)((( 9069 + #foreach ($entryDoc in $xwiki.wrapDocs($entries)) 9070 + #getEntryObject($entryDoc $entryObj) 9071 + ## Although all entries should have one of the two objects, better check to be sure. 9072 + #if(~"$!\{entryObj}~" != '') 9073 + #getEntryDate($entryDoc $entryObj $entryDate) 9074 + ## Display a ~"calendar sheet~" for each day. All entries posted on the same day share one such sheet. 9075 + #set($entryDateStr = $xwiki.formatDate($entryDate, 'yyyyMMMMdd')) 9076 + #if($blogDay != $entryDateStr) 9077 + #if($blogDay != '') 9078 + ))) 9079 + (% class=~"blogDay~" ~%)((( 9080 + #end 9081 + #displayBlogDate($entryDate) 9082 + #set ($blogDay = $entryDateStr) 9083 + #end 9084 + ## Finally, display the entry. 9085 + #displayEntry($entryDoc $entryObj $onlyExtract $shouldDisplayTitles true) 9086 + #end 9087 + #end 9088 + )))## blogDay 9089 + )))## hfeed 9090 +#end 9091 +## 9092 +## 9093 +## 9094 +#** 9095 + * Get the entry object, either a new BlogPost or an old Article. 9096 + * 9097 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 9098 + * @param entryObj The resulting xobject of the blog post. 9099 + *### 9100 +#macro(getEntryObject $entryDoc $__entryObj) 9101 + #set($result = $entryDoc.getObject(~"$\{blogPostClassname}~")) 9102 + #if(!$result) 9103 + #set($result = $entryDoc.getObject(~"$\{oldArticleClassname}~")) 9104 + #end 9105 + ## NOTE: The reason we put an underscore in front of the variable name is to prevent the following line from 9106 + ## overwriting the $entryObj variable that may be defined before this macro is called. Of course, $__entryObj may be 9107 + ## overwritten in this case but it's less likely to have such a variable defined before. 9108 + #set ($__entryObj = $NULL) 9109 + #setVariable (~"$__entryObj~" $result) 9110 +#end 9111 +## 9112 +## 9113 +## 9114 +#** 9115 + * Gets the date associated with a blog entry. This is the publication date. For unpublished entries, initially this is 9116 + * the document creation date, but can be edited by the user. 9117 + * 9118 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 9119 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 9120 + * @param result The resulting date, an instance of <tt>java.util.Date</tt>. 9121 + *### 9122 +#macro(getEntryDate $entryDoc $entryObj $result) 9123 + #set ($result = $NULL) 9124 + #setVariable (~"$result~" $entryObj.getProperty('publishDate').value) 9125 +#end 9126 +## 9127 +## 9128 +## 9129 +#** 9130 + * Displays a date, nicely formatted as a calendar page. 9131 + * 9132 + * @param date The date to display, an instance of <tt>java.util.Date</tt>. 9133 + *### 9134 +#macro(displayBlogDate $date) 9135 + #set($year = $xwiki.formatDate($date, 'yyyy')) 9136 + ## 3 letter month name, like Jan, Dec. 9137 + #set($month = $xwiki.formatDate($date, 'MMM')) 9138 + ## Uncomment to get a full length month name, like January, December. 9139 + ## TODO: this could be defined somewhere in the blog style. 9140 + ## #set($month = $xwiki.formatDate($date, 'MMMM')) 9141 + #set($day = $xwiki.formatDate($date, 'dd')) 9142 + (% class=~"blogdate~" ~%) 9143 + == (% class=~"month~" ~%)$month(%~%) (% class=~"day~" ~%)$day(%~%) (% class=~"year~" ~%)$year(%~%) == 9144 +#end 9145 +## 9146 +## 9147 +## 9148 +#** 9149 + * Displays a blog article: management tools, header, content, footer. 9150 + * 9151 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 9152 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 9153 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 9154 + * @param shouldDisplayTitle If <tt>true</tt>, display the blog title (blog posts shouldn't display the title 9155 + * when they're displayed alone on their page since it's the page title which is used in this case) 9156 + * @param shouldDisplayActions If <tt>true</tt>, display the blog post actions buttons 9157 + *### 9158 +#macro(displayEntry $entryDoc $entryObj $onlyExtract $shouldDisplayTitle $shouldDisplayActions) 9159 + ## Only articles with an explicit hidden setting or an explicit unpublished setting are hidden 9160 + #isPublished($entryObj $isPublished) 9161 + #isHidden($entryObj $isHidden) 9162 + #if($doc.fullName == $entryDoc.fullName) 9163 + (% class=~"hentry single-article~" ~%)((( 9164 + #else 9165 + (% class=~"hentry#if(!$isPublished) unpublished-article#elseif($isHidden) hidden-article#end~" ~%)((( 9166 + #end 9167 + #if ($shouldDisplayActions) 9168 + #displayEntryTools($entryDoc $entryObj) 9169 + #end 9170 + #if($shouldDisplayTitle) 9171 + #displayEntryTitle($entryDoc $entryObj) 9172 + #end 9173 + #if($doc.fullName == $entryDoc.fullName) 9174 + #if(!$isPublished) 9175 + \{\{warning}}\{\{translation key='blog.code.published'/}}\{\{/warning}} 9176 + #elseif($isHidden) 9177 + \{\{warning}}\{\{translation key='blog.code.hidden'/}}\{\{/warning}} 9178 + #end 9179 + #end 9180 + #displayEntryContent($entryDoc $entryObj $onlyExtract) 9181 + #displayEntryFooter($entryDoc $entryObj) 9182 + )))## hentry 9183 +#end 9184 +## 9185 +## 9186 +## 9187 +#** 9188 + * Checks if the provided blog is published or not. 9189 + * 9190 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 9191 + * @param isPublished The resulting boolean, true if the entry is considered published. 9192 + *### 9193 +#macro(isPublished $entryObj $isPublished) 9194 + #set ($isPublished = $NULL) 9195 + ## This should work for both old articles, which don't have the 'published' property at all, and 9196 + ## are considered published by default, and new entries, that should have 1 if published. 9197 + #if (~"$!\{entryObj.getProperty('published').value}~" != '0') 9198 + #setVariable (~"$isPublished~" true) 9199 + #else 9200 + #setVariable (~"$isPublished~" false) 9201 + #end 9202 +#end 9203 +## 9204 +## 9205 +## 9206 +#** 9207 + * Checks if the provided blog is hidden or not. 9208 + * 9209 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass<tt> xclass. 9210 + * @param isHiddel The resulting boolean, true if the entry is considered hidden. 9211 + *### 9212 +#macro(isHidden $entryObj $isHidden) 9213 + #set ($isHidden = $NULL) 9214 + ## This should work for both old articles, which don't have the 'hidden' property at all, and 9215 + ## are considered visible by default, and new entries, that should have 1 if hidden. 9216 + #if (~"$!\{entryObj.getProperty('hidden').value}~" == '1') 9217 + #setVariable (~"$isHidden~" true) 9218 + #else 9219 + #setVariable (~"$isHidden~" false) 9220 + #end 9221 +#end 9222 +## 9223 +## 9224 +## 9225 +#** 9226 + * Displays several ~"tools~" for manipulating blog posts: hide/show, publish, edit. 9227 + * 9228 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 9229 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 9230 + *### 9231 +#macro(displayEntryTools $entryDoc $entryObj) 9232 + #if($xcontext.action == 'view') 9233 + (% class=~"blog-entry-toolbox~" ~%)((( 9234 + #displayPublishButton($entryDoc $entryObj) 9235 + #displayHideShowButton($entryDoc $entryObj) 9236 + #displayEditButton($entryDoc $entryObj) 9237 + #displayDeleteButton($entryDoc $entryObj) 9238 + ))) 9239 + #end 9240 +#end 9241 +## 9242 +## 9243 +## 9244 +#** 9245 + * Displays the publish button to the entry <strong>creator</strong>, if the article is not published yet. 9246 + * 9247 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 9248 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 9249 + * @todo AJAX calls. 9250 + *### 9251 +#macro(displayPublishButton $entryDoc $entryObj) 9252 + #isPublished($entryObj $isPublished) 9253 + #if(!$isPublished && $entryDoc.creator == $xcontext.user && $xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 9254 + [[#toolImage('world')>>path:$blogPublisher.getURL('view', ~"entryName=$\{escapetool.url($entryDoc.fullName)}&xredirect=$\{escapetool.url($thisURL)}&form_token=$!\{services.csrf.getToken()}~")||title=~"$services.localization.render('blog.code.notpublished')~"]]## 9255 + #end 9256 +#end 9257 +## 9258 +## 9259 +## 9260 +#** 9261 + * Displays the hide or show button to the entry <strong>creator</strong>, if the article is already published. 9262 + * 9263 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 9264 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 9265 + *### 9266 +#macro(displayHideShowButton $entryDoc $entryObj) 9267 + #isPublished($entryObj $isPublished) 9268 + #isHidden($entryObj $isHidden) 9269 + ## Only published articles can be hidden. Unpublished articles are considered already hidden. 9270 + #if($isPublished && $entryDoc.creator == $xcontext.user && $xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 9271 + #set ($queryString = \{ 9272 + 'xredirect' : $thisURL, 9273 + 'form_token' : $services.csrf.getToken() 9274 + }) 9275 + #if ($isHidden) 9276 + #set ($discard = $queryString.putAll(\{ 9277 + ~"$\{entryObj.getxWikiClass().getName()}_$\{entryObj.number}_hidden~" : 0, 9278 + 'comment' : $services.localization.render('blog.code.madevisible') 9279 + })) 9280 + #set ($lockURL = $entryDoc.getURL('save', $escapetool.url($queryString))) 9281 + [[#toolImage('unlock')>>path:$lockURL||class=~"blog-tool-show~" title=~"$services.localization.render('blog.code.makevisible')~"]]## 9282 + #else 9283 + #set ($discard = $queryString.putAll(\{ 9284 + ~"$\{entryObj.getxWikiClass().getName()}_$\{entryObj.number}_hidden~" : 1, 9285 + 'comment' : $services.localization.render('blog.code.hid') 9286 + })) 9287 + #set ($lockURL = $entryDoc.getURL('save', $escapetool.url($queryString))) 9288 + [[#toolImage('lock')>>path:$lockURL||class=~"blog-tool-hide~" title=~"$services.localization.render('blog.code.hide')~"]]## 9289 + #end 9290 + #end 9291 +#end 9292 +## 9293 +## 9294 +## 9295 +#** 9296 + * Displays the edit button to those that can edit the article. 9297 + * 9298 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 9299 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 9300 + *### 9301 +#macro(displayEditButton $entryDoc $entryObj) 9302 + #if($xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 9303 + ## Call getDefaultEditMode() for backward compatibility with older blog posts. 9304 + [[#toolImage('pencil')>>path:$entryDoc.getURL('edit')||title=~"$services.localization.render('blog.code.editpost')~"]]## 9305 + #end 9306 +#end 9307 +## 9308 +## 9309 +## 9310 +#** 9311 + * Displays the delete button to those that can edit the article. 9312 + * 9313 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 9314 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 9315 + * @todo AJAX calls. 9316 + *### 9317 +#macro(displayDeleteButton $entryDoc $entryObj) 9318 + #if($xwiki.hasAccessLevel('delete', $xcontext.user, $entryDoc.fullName)) 9319 + [[#toolImage('cross')>>path:$entryDoc.getURL('delete')||title=~"$services.localization.render('blog.code.deletepost')~"]]## 9320 + #end 9321 +#end 9322 +## 9323 +## 9324 +## 9325 +#** 9326 + * Displays the title of the entry. 9327 + * 9328 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 9329 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 9330 + *### 9331 +#macro(displayEntryTitle $entryDoc $entryObj) 9332 + #if($doc.fullName == $entryDoc.fullName) 9333 + (% class=~"entry-title~" ~%) 9334 + = $services.rendering.escape($entryDoc.display('title', 'view', $entryObj), $xwiki.getCurrentContentSyntaxId()) = 9335 + #else 9336 + (% class=~"entry-title~" ~%) 9337 + === [[$services.rendering.escape($entryDoc.display('title', 'view', $entryObj),$xwiki.getCurrentContentSyntaxId())>>doc:$services.rendering.escape($services.model.serialize($entryDoc.getDocumentReference(),'default'),$xwiki.getCurrentContentSyntaxId())]] === 9338 + #end 9339 +#end 9340 +## 9341 +## 9342 +## 9343 +#** 9344 + * Displays the body of the entry. 9345 + * 9346 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 9347 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 9348 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 9349 + *### 9350 +#macro(displayEntryContent $entryDoc $entryObj $onlyExtract) 9351 + (% class=~"#if($onlyExtract)entry-summary#\{else}entry-content#end~" ~%)((( 9352 + #getEntryContent($entryDoc $entryObj $onlyExtract $entryContent) 9353 + ## FIXME: This causes the blog's content to not be annotatable. See http://jira.xwiki.org/browse/XWIKI-6328 9354 + ## Should probably be replaced by a display macro call with a reference to the object property holding the post's content 9355 + \{\{html wiki=~"false~"}}$entryDoc.getRenderedContent($entryContent, $entryDoc.syntax.toIdString())\{\{/html}} 9356 + ))) ## entry-content 9357 + (% class=~"clearfloats~" ~%)((())) 9358 +#end 9359 +## 9360 +## 9361 +## 9362 +#** 9363 + * Extracts the body of the entry that should be displayed. If <tt>onlyExtract</tt> is <tt>true</tt>, display the content 9364 + * of the <tt>extract</tt> field (if not empty). 9365 + * 9366 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 9367 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 9368 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 9369 + * @param entryContent The resulting content. 9370 + * @param removeEllipsis If <tt>true</tt>, then don't display an ellipsis at the end of the content (only used when 9371 + * <tt>onlyExtract</tt> is <tt>true</tt>) 9372 + *### 9373 +#macro(getEntryContent $entryDoc $entryObj $onlyExtract $entryContent $removeEllipsis) 9374 + #if ($onlyExtract) 9375 + ## Note: We trim the summary so that if there's some white space it won't cause the summary to be used instead 9376 + ## of the content. 9377 + #set ($macro.result = $entryObj.getProperty('extract').value.trim()) 9378 + #end 9379 + #if(~"$!macro.result~" == '') 9380 + #set($macro.result = $entryObj.getProperty('content').value) 9381 +#* Disabled until the content can be cleanly cut. 9382 +* #if($onlyExtract && $result.length()>$maxchars) 9383 +* #set($i = $macro.result.lastIndexOf(~" ~", $maxchars)) 9384 +* #set($i = $i + 1) 9385 +* #set($macro.result = ~"$\{macro.result.substring(0,$i)} *[...>$\{entryDoc.fullName}]*~") 9386 +* #end 9387 +## *### 9388 + #elseif (!$removeEllipsis) 9389 + #if($entryDoc.syntax.toIdString() == 'xwiki/1.0') 9390 + #set($macro.result = ~"$\{macro.result} <a href='$\{entryDoc.getExternalURL()}' title='$services.localization.render('blog.code.readpost')'>...</a>~") 9391 + #elseif($entryDoc.syntax.toIdString() == 'xwiki/2.0' || $entryDoc.syntax.toIdString() == 'xwiki/2.1') 9392 + #set($macro.result = ~"$\{macro.result} [[...>>$\{services.rendering.escape($entryDoc.getFullName(), $entryDoc.getSyntax())}||title='$services.localization.render('blog.code.readpost')']]~") 9393 + #end 9394 + #end 9395 + #set ($entryContent = $NULL) 9396 + #setVariable (~"$entryContent~" $macro.result) 9397 +#end 9398 +## 9399 +## 9400 +## 9401 +#** 9402 + * Displays the footer of the entry. 9403 + * 9404 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 9405 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 9406 + *### 9407 +#macro(displayEntryFooter $entryDoc $entryObj) 9408 + (% class=~"entry-footer~" ~%)((( 9409 + #isPublished($entryObj $isPublished) 9410 + (% class='entry-author-label' ~%) 9411 + #if($isPublished) 9412 + \{\{translation key='blog.code.postedby'/}} ## 9413 + #else 9414 + \{\{translation key='blog.code.createdby'/}} ## 9415 + #end 9416 + \{\{html wiki=~"false~" clean=~"false~"}}<span class='author vcard'>#userfn($entryDoc.creator)</span>\{\{/html}} ## 9417 + #getEntryDate($entryDoc $entryObj $entryDate) 9418 + #listCategories($entryObj) #* 9419 + ## Since the publish date and update date are not set at the exact same time, there could be a small difference that 9420 + ## we assume cannot be more than 3 seconds. 9421 + *#(% class=~"separator~" ~%)·(%~%) [[\{\{translation key='blog.code.permalink'/}}>>$services.rendering.escape($entryDoc,'xwiki/2.0')||rel=~"bookmark~"]] ## 9422 + #if ($showcomments) 9423 + (% class=~"separator~" ~%)·(%~%) [[\{\{translation key='blog.code.comments'/}}>>$services.rendering.escape($entryDoc,'xwiki/2.0')||anchor=~"Comments~"]] (% class=~"itemCount~" ~%)($entryDoc.comments.size())(%~%) ## 9424 + #end ## 9425 + #if($entryDoc != $doc) ## 9426 + #displayEntryBlogLocation($entryDoc $entryObj) ## 9427 + #end 9428 + )))## entry-footer 9429 +#end 9430 +## 9431 +## 9432 +#** 9433 + * Display the blog for the entry (if it is not the currently displayed blog) 9434 + * 9435 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 9436 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 9437 + *### 9438 +#macro(displayEntryBlogLocation $entryDoc $entryObj) 9439 + #getBlogPostsLocation($entryDoc.getSpace() $blogPostsLocation) ## 9440 + #if(~"$!blogPostsLocation~" != ~"~") ## 9441 + #set($blogSpaceRef = $services.model.resolveSpace($blogPostsLocation)) ## 9442 + #set($blogDocRef = $services.model.createDocumentReference($services.model.getEntityReference('DOCUMENT', 'default').getName(), $blogSpaceRef)) ## 9443 + #if($doc.documentReference != $blogDocRef) ## 9444 + #set($blogDoc = $xwiki.getDocument($blogDocRef)) ## 9445 + #set($blogObj = $blogDoc.getObject('Blog.BlogClass')) ## 9446 + #if($blogObj) (% class=~"blogrefpath~" ~%)((( ## 9447 + $services.localization.render('blog.code.postedin') \{\{html clean=~"false~" wiki=~"false~"}} #hierarchy($blogDocRef, \{'selfIsActive' : true, 'local': true}) \{\{/html}} ## 9448 + )))(%~%)## 9449 + #end 9450 + #end 9451 + #end 9452 +#end 9453 +## 9454 +## 9455 +## 9456 +## 9457 +#** 9458 + * List the categories an entry belongs to. Used in the footer. The categories are instances of <tt>Blog.CategoryClass</tt>. 9459 + * 9460 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 9461 + *### 9462 +#macro(listCategories $entryObj) 9463 + #if($entryObj.getxWikiClass().getName() == $blogPostClassname) 9464 + #set($categories = $entryObj.getProperty('category').value) 9465 + #set($first = true) 9466 + #if($categories.size() > 0) 9467 + #foreach($category in $categories) 9468 + #set($categoryDoc = $!xwiki.getDocument($category)) 9469 + #if(!$categoryDoc.isNew() && $categoryDoc.getObject($\{blogCategoryClassname})) 9470 + #if($foreach.count == 1) 9471 + (% class='separator' ~%)·(%~%) $services.localization.render('blog.code.categories') ## 9472 + #else 9473 + , ## 9474 + #end## 9475 + [[$!\{categoryDoc.getObject($blogCategoryClassname).getValue('name')}>>$\{category}||rel='tag']]## 9476 + #end## 9477 + #end## 9478 + #end 9479 + #end 9480 +#end 9481 +## 9482 +## 9483 +## 9484 +#** 9485 + * Displays blog pagination links (older and newer entries). 9486 + * 9487 + * @param blogDoc the XDocument holding the blog definition object. 9488 + *### 9489 +#macro(displayNavigationLinks $blogDoc) 9490 + (% class=~"clearfloats~" ~%)((())) 9491 + #getBlogDisplayType($blogDoc $displayType) 9492 + #if($displayType == 'weekly') 9493 + (% class=~"pagingLinks~" ~%)((( 9494 + #getRequestedWeek($weekDate) 9495 + $weekDate.addWeeks(-1)## 9496 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.previousweek'/}}>>$doc.documentReference.name||queryString=~"year=$weekDate.weekyear&week=$weekDate.weekOfWeekyear~" class=~"button~"]]**(%~%) 9497 + #sep() 9498 + $weekDate.addWeeks(2)## 2 because we already subtracted 1 above 9499 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.nextweek'/}} »>>$doc.documentReference.name||queryString=~"year=$weekDate.weekyear&week=$weekDate.weekOfWeekyear~" class=~"button~"]]**(%~%) 9500 + ))) 9501 + #elseif($displayType == 'monthly') 9502 + (% class=~"pagingLinks~" ~%)((( 9503 + #getRequestedMonth($monthDate) 9504 + $monthDate.addMonths(-1)## 9505 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.previousmonth'/}}>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"year=$monthDate.year&month=$monthDate.monthOfYear~" class=~"button~"]]**(%~%) 9506 + #sep() 9507 + $monthDate.addMonths(2)## 2 because we already subtracted 1 above 9508 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.nextmonth'/}} »>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"year=$monthDate.year&month=$monthDate.monthOfYear~" class=~"button~"]]**(%~%) 9509 + ))) 9510 + #elseif($displayType == 'all') 9511 + #else 9512 + ## Paginated 9513 + #if(($totalPages > 1)) 9514 + #set($queryString = '') 9515 + #foreach($p in $request.getParameterNames()) 9516 + #if($p != 'page' && $p != 'ipp') 9517 + #foreach($v in $request.getParameterValues($p)) 9518 + #set($queryString = ~"$\{queryString}&$\{escapetool.url($p)}=$\{escapetool.url($v)}~") 9519 + #end 9520 + #end 9521 + #end 9522 + (% class=~"pagingLinks~" ~%)((( 9523 + #if ($currentPageNumber < $totalPages) 9524 + #set($currentPageNumber = $currentPageNumber + 1) 9525 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.olderposts'/}}>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"page=$\{currentPageNumber}&ipp=$\{itemsPerPage}$queryString~" class=~"button~"]]**(%~%) 9526 + #set($currentPageNumber = $currentPageNumber - 1) 9527 + #end 9528 + #if ($currentPageNumber > 1) 9529 + #if ($currentPageNumber < $totalPages) 9530 + #sep() 9531 + #end 9532 + #set($currentPageNumber = $currentPageNumber - 1) 9533 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.newerposts'/}} »>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"page=$\{currentPageNumber}&ipp=$\{itemsPerPage}$queryString~" class=~"button~"]]**(%~%) 9534 + #set($currentPageNumber = $currentPageNumber + 1) 9535 + #end 9536 + (% class=~"clear~" ~%)(%~%) 9537 + )))## pagingLinks 9538 + #end 9539 + #end 9540 +#end 9541 +## 9542 +## 9543 +## 9544 +#** 9545 + * Displays a message box with ~"publish~" icon. 9546 + * 9547 + * @param message A text message concerning blog article publishing 9548 + *### 9549 +#macro(publishMessageBox $message) 9550 +(% class=~"plainmessage publish-message~" ~%)((($services.icon.render('world') $message))) 9551 +#end 9552 +#** 9553 + * Displays a message box with ~"show/hide~" icon. 9554 + * 9555 + * @param message A text message concerning blog article hiding 9556 + *### 9557 +#macro(hideMessageBox $message) 9558 +(% class=~"plainmessage hide-message~" ~%)((($services.icon.render('unlock') $message))) 9559 +#end 9560 +## 9561 +## 9562 +## 9563 +#** 9564 + * Determine the requested week, for using in a weekly-indexed blog. The relevant request parameters are 9565 + * <tt>year</tt> and <tt>week</tt>. By default, the current week is used. 9566 + * 9567 + * @param monthDate The resulting week, a JODATime MutableDateTime. 9568 + *### 9569 +#macro(getRequestedWeek $weekDate) 9570 + #set ($weekDate = $NULL) 9571 + #setVariable (~"$weekDate~" $xwiki.jodatime.mutableDateTime) 9572 + #if(~"$!\{request.year}~" != '') 9573 + #set ($discard = $weekDate.setYear($numbertool.toNumber($request.year).intValue())) 9574 + #end 9575 + #if(~"$!\{request.week}~" != '') 9576 + #set ($discard = $weekDate.setWeekOfWeekyear($numbertool.toNumber($request.week).intValue())) 9577 + #end 9578 +#end 9579 +## 9580 +## 9581 +## 9582 +#** 9583 + * Determine the requested month, for using in a monthly-indexed blog. The relevant request parameters are 9584 + * <tt>year</tt> and <tt>month</tt>. By default, the current month is used. 9585 + * 9586 + * @param monthDate The resulting month, a JODATime MutableDateTime. 9587 + *### 9588 +#macro(getRequestedMonth $monthDate) 9589 + #set ($monthDate = $NULL) 9590 + #setVariable (~"$monthDate~" $xwiki.jodatime.mutableDateTime) 9591 + #if(~"$!\{request.year}~" != '') 9592 + #set ($discard = $monthDate.setYear($numbertool.toNumber($request.year).intValue())) 9593 + #end 9594 + #if(~"$!\{request.month}~" != '') 9595 + #set ($discard = $monthDate.setMonthOfYear($numbertool.toNumber($request.month).intValue())) 9596 + #end 9597 +#end 9598 +## 9599 +## 9600 +## 9601 +#** 9602 + * Retrieve a blog property (title, display type, etc). 9603 + * 9604 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 9605 + * @param propertyName The name of the property to be retrieved. One of the <tt>Blog.BlogClass</tt>'s properties. 9606 + * @param defaultValue The default value to use in case the blog object does not define one. 9607 + * @param propertyValue The resulting value. 9608 + *### 9609 +#macro(getBlogProperty $blogDoc $propertyName $defaultValue $propertyValue) 9610 + #set($result = ~"$!\{blogDoc.getObject($\{blogClassname}).getProperty($propertyName).value}~") 9611 + #if($result == '') 9612 + #set($result = $defaultValue) 9613 + #end 9614 + #set ($propertyValue = $NULL) 9615 + #setVariable (~"$propertyValue~" $result) 9616 +#end 9617 + 9618 +#** 9619 + * If an error occurs when executing an action, set a specific response status and display an error message. 9620 + * 9621 + * @param status The response status. 9622 + * @param text The user readable error to be displayed. Can be a translation key. 9623 + * @param parameters The parameters to use when decoding the translation key. 9624 + *### 9625 +#macro(blog__actionResponseError $status $text $parameters) 9626 + $response.setStatus($status) 9627 + #if($request.ajax) 9628 + $services.localization.render($text, $!parameters) 9629 + #else 9630 + \{\{error}}$services.localization.render($text, $!parameters)\{\{/error}} 9631 + #end 9632 +#end 9633 +## 9634 +## 9635 +## 9636 +#** 9637 + * Check if a blog is the Default blog (The one in the 'Blog' space). 9638 + * 9639 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 9640 + * @param isDefault The resulting boolean. 9641 + *### 9642 +#macro(isDefaultBlog $blogDoc $isDefault) 9643 + #set ($result = false) 9644 + #if ($blogDoc.space == 'Blog') 9645 + #set ($result = true) 9646 + #end 9647 + #setVariable(~"$isDefault~" $result) 9648 +#end 9649 +## 9650 +## 9651 +## 9652 +#** 9653 + * Retrieve the blog posts location (space). 9654 + * 9655 + * @param blogSpace The blog space. It should contain a document with a <tt>Blog.BlogClass</tt> object 9656 + * @param postsLocation The resulting location. 9657 + *### 9658 +#macro(getBlogPostsLocation $blogSpace $postsLocation) 9659 + #getBlogDocument($blogSpace $blogDoc) 9660 + #getBlogProperty($blogDoc 'postsLocation' $blogDoc.space $result) 9661 + #set ($postsLocation = $NULL) 9662 + #setVariable (~"$postsLocation~" $result) 9663 +#end 9664 +## 9665 +## 9666 +## 9667 +#** 9668 + * Retrieve the blog categories location (space). 9669 + * 9670 + * @param blogSpace The blog space. It should contain a document with a <tt>Blog.BlogClass</tt> object 9671 + * @param categoriesLocation The resulting location. 9672 + *### 9673 +#macro(getBlogCategoriesLocation $blogSpace $categoriesLocation) 9674 + #getBlogDocument($blogSpace $blogDoc) 9675 + #getBlogProperty($blogDoc 'categoriesLocation' 'Blog' $result) 9676 + #set ($postsLocation = $NULL) 9677 + #setVariable (~"$categoriesLocation~" $result) 9678 +#end 9679 +###** 9680 + * Return the Query for selecting the blog posts based on the context where the posts are displayed. 9681 + * for example there is 4 different panel contexts: 9682 + * aBlog.aPost or aBlog.WebHome 9683 + * aCategorySpace.aCategory 9684 + * aCategorySpace.WebHome 9685 + * Blog.aPost or Blog.WebHome 9686 + * 9687 + * @param query The query for selecting blog entries. 9688 + * @param queryParams The parameters to bind with the generated query. 9689 + * @param targetDoc The document in which the articles will be displayed. 9690 + *### 9691 +#macro(getAllBlogPostsQueryBasedOnDisplayContext $targetDoc $query $queryParams) 9692 + #set ($query = $NULL) 9693 + #set ($queryParams = $NULL) 9694 + #getCategoryAllBlogPostsQuery($resultQuery) 9695 + #set ($resultQueryParams = \{}) 9696 + #set($sanitizedSpace = $targetDoc.space.replaceAll('([%_!])', '!$1').concat('.%')) 9697 + #if ($targetDoc.space == $defaultBlogSpace && !$targetDoc.getObject($blogCategoryClassname)) 9698 + ## Get all posts that are in the default blog space('Blog') or in category 'Blog.%' 9699 + #set ($discard = $resultQueryParams.put('space', $targetDoc.space)) 9700 + #set ($discard = $resultQueryParams.put('catList', $sanitizedSpace)) 9701 + #set($resultQuery = ~"$\{resultQuery} and (doc.space = :space or catList like :catList escape '!')~") 9702 + #elseif($targetDoc.getObject($blogCategoryClassname)) 9703 + ## Get all posts that are in a category aCategorySpace.aCategory 9704 + #set ($discard = $resultQueryParams.put('catList', $targetDoc.fullName)) 9705 + #set($resultQuery = ~"$\{resultQuery} and (catList=:catList)~") 9706 + #elseif($targetDoc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 9707 + ## Get all posts that are in a category aCategorySpace.% 9708 + #set ($discard = $resultQueryParams.put('catList', $sanitizedSpace)) 9709 + ## Exclude incategorized posts 9710 + #set ($excludedCategory = ~"$\{targetDoc.space}.WebHome~") 9711 + #if ($targetDoc.space == $defaultBlogSpace) 9712 + #set ($excludedCategory = ~"Blog.Categories.WebHome~") 9713 + #end 9714 + #set ($discard = $resultQueryParams.put('excludedCategory', $excludedCategory)) 9715 + #set($resultQuery = ~"$\{resultQuery} and (catList<>:excludedCategory) and (catList like :catList escape '!')~") 9716 + #else 9717 + ## Get all posts in blog space aBlog 9718 + #getAllBlogPostsQuery($resultQuery) 9719 + #getBlogPostsLocation($targetDoc.space $postsLocation) 9720 + #set ($discard = $resultQueryParams.put('space', $postsLocation)) 9721 + #set($resultQuery = ~"$\{resultQuery} and doc.space = :space~") 9722 + #end 9723 + #setVariable(~"$query~" $resultQuery) 9724 + #setVariable(~"$queryParams~" $resultQueryParams) 9725 +#end 9726 +## 9727 +## 9728 +## 9729 +###** 9730 + * Display blog posts based on the context where the posts are displayed. 9731 + * for example there is 4 different panel contexts: 9732 + * aBlog.aPost or aBlog.WebHome 9733 + * aCategorySpace.aCategory 9734 + * aCategorySpace.WebHome 9735 + * Blog.aPost or Blog.WebHome 9736 + * 9737 + * @param targetDoc The document in which the articles will be displayed. 9738 + * @param postsVisiblity The visibility of the posts to display: recent, unpublished, ... 9739 + * @param layout Layout of the the posts to display 9740 + * @param layoutParams additional layout parameters, it is a string like 'param1=val1|...|paramN=valueN' 9741 + * @param limit the number of posts to display 9742 + *### 9743 +#macro(displayBlogPostsBasedOnContext $targetDoc $postsVisiblity $layout $layoutParams $limit) 9744 + #if ($postLayout == 'full') 9745 + #set ($macro.paginated = 'yes') 9746 + #end 9747 + #if ($targetDoc.space == $defaultBlogSpace && !$targetDoc.getObject($blogCategoryClassname)) 9748 + ## Display all posts that are in the default blog space('Blog') or in category 'Blog.%' 9749 + #getBlogPostsLayout($xwiki.getDocument(~"$\{defaultBlogSpace}.WebHome~") $postsLayout) 9750 + #if (~"$!layout~" == '') 9751 + #set ($layout = $postsLayout) 9752 + #end 9753 + #if ($postsVisiblity == 'recent') 9754 + \{\{blogpostlist blog=~"$\{defaultBlogSpace.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')}.WebHome~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 9755 + #elseif($postsVisiblity == 'unpublished') 9756 + \{\{blogpostlist blog=~"$\{defaultBlogSpace.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')}.WebHome~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 9757 + #end 9758 + #elseif($targetDoc.getObject($blogCategoryClassname)) 9759 + ## Display all posts that are in a category aCategorySpace.aCategory 9760 + #getBlogDocumentForCategoriesSpace($targetDoc.space $blogDoc) 9761 + #getBlogPostsLayout($blogDoc $postsLayout) 9762 + #if (~"$!layout~" == '') 9763 + #set ($layout = $postsLayout) 9764 + #end 9765 + #if ($postsVisiblity == 'recent') 9766 + \{\{blogpostlist category=~"$targetDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 9767 + #elseif($postsVisiblity == 'unpublished') 9768 + \{\{blogpostlist category=~"$targetDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 9769 + #end 9770 + #elseif($targetDoc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 9771 + ## Display all posts that are in a category aCategorySpace.% 9772 + #getBlogDocumentForCategoriesSpace($targetDoc.space $blogDoc) 9773 + #getBlogPostsLayout($blogDoc $postsLayout) 9774 + #if (~"$!layout~" == '') 9775 + #set ($layout = $postsLayout) 9776 + #end 9777 + #if ($postsVisiblity == 'recent') 9778 + \{\{blogpostlist category=~"$targetDoc.space.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 9779 + #elseif($postsVisiblity == 'unpublished') 9780 + \{\{blogpostlist category=~"$targetDoc.space.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 9781 + #end 9782 + #else 9783 + ## Display all posts in blog space aBlog 9784 + #getBlogDocument($targetDoc.space $blogDoc) 9785 + #getBlogPostsLayout($blogDoc $postsLayout) 9786 + #if (~"$!layout~" == '') 9787 + #set ($layout = $postsLayout) 9788 + #end 9789 + #if ($postsVisiblity == 'recent') 9790 + \{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 9791 + #elseif($postsVisiblity == 'unpublished') 9792 + \{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 9793 + #end 9794 + #end 9795 +#end 9796 +## 9797 +## 9798 +## 9799 +#** 9800 + * Bind parameters to a query object. 9801 + * 9802 + * @param queryObj The query object (org.xwiki.query.internal.ScriptQuery object) 9803 + * @param queryParams the query parameters. 9804 + *### 9805 +#macro(bindQueryParameters $queryObj $queryParams) 9806 + #set ($output = $queryObj) 9807 + #foreach( $key in $queryParams.keySet() ) 9808 + #set($output = $queryObj.bindValue($key, $queryParams.get($key))) 9809 + #end 9810 + #setVariable(~"$queryObj~" $output) 9811 +#end 9812 +## 9813 +## 9814 +## 9815 +#** 9816 + * Determines the blogposts layout: Default layout (Calendar), Image thumbnail 9817 + * 9818 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>postsLayout</tt> 9819 + * property set. 9820 + * @param postsLayout The resulting string. If the blog object does not define anything, it is considered full (default). 9821 + *### 9822 +#macro(getBlogPostsLayout $blogDoc $postsLayout) 9823 + #getBlogProperty($blogDoc 'postsLayout' 'full' $res) 9824 + #set ($postsLayout = $NULL) 9825 + #setVariable (~"$postsLayout~" $res) 9826 +#end 9827 +## 9828 +## 9829 +## 9830 +#** 9831 + * Retrieve the blog document of a categories space, which is the blog descriptor that have its categoriesLocation property set to the categories space 9832 + * 9833 + * @param categoriesSpace A <tt>String</tt>, the name of the space. 9834 + * @param blogDoc The resulting XDocument. 9835 + *### 9836 +#macro(getBlogDocumentForCategoriesSpace $categoriesSpace $blogDoc) 9837 + #set($macro.blogDocs = $services.query.hql(~", BaseObject obj, StringProperty as categoriesLocationProp where obj.name = doc.fullName and obj.className = '$blogClassname' and obj.id=categoriesLocationProp.id.id and categoriesLocationProp.id.name='categoriesLocation' and categoriesLocationProp.value=:categoriesLocation order by doc.creationDate~").setLimit(1).setOffset(0).bindValue('categoriesLocation', $categoriesSpace).execute()) 9838 + #if($macro.blogDocs.size() > 0 && $categoriesSpace != $defaultBlogSpace) 9839 + #set($macro.result = $xwiki.getDocument($macro.blogDocs.get(0))) 9840 + #else 9841 + ## Fallback to Blog.WebHome, the default blog 9842 + #set($macro.result = $xwiki.getDocument('Blog.WebHome')) 9843 + #end 9844 + #set ($blogDoc = $NULL) 9845 + #setVariable (~"$blogDoc~" $macro.result) 9846 +#end" %) 9847 +((( 9848 +(% class="macro-placeholder hidden" %) 9849 +((( 9850 +macro:velocity 9851 +))) 9852 +))) 9853 +))) 9854 + 9855 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogPostLayoutMacros~"" %) 9856 +((( 9857 +(% class="macro-placeholder hidden" %) 9858 +((( 9859 +macro:include 9860 +))) 9861 + 9862 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|#** 9863 + * Extract the layout parameters from a string. 9864 + * 9865 + * @param layoutParamsString The string representation of the layout parameters. 9866 + * It should contain a String following this format ~"paramName1=Value1|paramName2=Value2|...|paramNameK=ValueK~" 9867 + * @param layoutsParameters The resulting layout parameters Map. 9868 + *### 9869 +#macro(extractLayoutParametersFromString $layoutParamsString $layoutsParameters) 9870 + #set ($layoutsParameters = $NULL) 9871 + #set ($macro.layoutParams = \{}) 9872 + #if (~"$!layoutParamsString~" != '') 9873 + #set ($macro.paramsArr = $layoutParamsString.split('\\|')) 9874 + #foreach ($item in $macro.paramsArr) 9875 + #set ($itemSplit = $item.split('=')) 9876 + #if ($itemSplit.size() == 2) 9877 + #set ($discard = $macro.layoutParams.put($itemSplit[0].trim(), $itemSplit[1].trim())) 9878 + #end 9879 + #end 9880 + #end 9881 + #setVariable(~"$layoutsParameters~" $macro.layoutParams) 9882 +#end" %) 9883 +((( 9884 +(% class="macro-placeholder hidden" %) 9885 +((( 9886 +macro:velocity 9887 +))) 9888 +))) 9889 +))) 9890 + 9891 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-| #macro(displaySmallPost $pDoc $pObj) 9892 + #initLayoutVars($pDoc $pObj) 9893 + <div class=~"col-xs-12 col-sm-6 next_blog~"> 9894 + <a href=~"$pDoc.uRL~" class=~"thumbnail~"> 9895 + <div class=~"row~"> 9896 + <div class=~"col-xs-4~"> 9897 + #if ($imageAtt) 9898 + <img src=~"$escapetool.xml($pDoc.getAttachmentURL($pObj.getValue('image'),'download',$imgQs))~" /> 9899 + #end 9900 + </div> 9901 + <div class=~"col-xs-8 art_det~"> 9902 + <p class=~"text-left~"> 9903 + #if($displayTitle)$!postTitle #end 9904 + <br/><span class=~"date_info~"> $!dateStr </span> 9905 + </p> 9906 + #displayPostDetails($pDoc) 9907 + </div> 9908 + </div> 9909 + </a> 9910 + </div> 9911 + #end 9912 + ## 9913 + ## 9914 + ## 9915 + #macro(displayPinnedPost $pDoc $pObj) 9916 + #initLayoutVars($pDoc $pObj) 9917 + <div class=~"col-xs-12 col-sm-4 col-md-4 blog_pin~"> 9918 + <a href=~"$pDoc.uRL~" class=~"all-post-link~"> 9919 + <div class=~"thumbnail~"> 9920 + #if ($displayTitle && $displayTitleFirstOnPinnedPosts) 9921 + <h5 class=~"title-top~">$!postTitle <br/> <span class=~"date_info~"> $!dateStr </span> </h5> 9922 + #end 9923 + #if ($imageAtt) 9924 + <img src=~"$escapetool.xml($pDoc.getAttachmentURL($pObj.getValue('image'),'download',$imgQs))~" /> 9925 + #end 9926 + <div class=~"caption~"> 9927 + #if ($displayTitle && !$displayTitleFirstOnPinnedPosts) 9928 + <h5 class=~"text-left~">$!postTitle <br/> <span class=~"date_info~"> $!dateStr </span> </h5> 9929 + #end 9930 + #if ($displaySummaryOnPinnedPosts) 9931 + <div class=~"text-left post-summary~"> 9932 + #set ($postContent = $pObj.getProperty('extract').value) 9933 + $!pDoc.getRenderedContent($postContent, $pDoc.syntax.toIdString()) 9934 + </div> 9935 + #end 9936 + #displayPostDetails($pDoc) 9937 + </div> 9938 + </div> 9939 + </a> 9940 + </div> 9941 + #end 9942 + ## 9943 + ## 9944 + ## 9945 + #macro(formatPostDate $pDoc $pObj) 9946 + #set ($formattedDate = $NULL) 9947 + #set($dateStr = $datetool.format('medium', $postDate, $xcontext.locale)) 9948 + #if (~"$!dateStr~" != '') 9949 + #set ($dateArr = $dateStr.split(' ')) 9950 + #if ($dateArr.size() > 3) 9951 + #set ($dateStr = ~"~") 9952 + #foreach($s in $dateArr.subList(0, 3)) 9953 + #set ($dateStr = ~"$\{dateStr}$\{s} ~") 9954 + #end 9955 + #end 9956 + #end 9957 + #setVariable(~"$formattedDate~" $dateStr) 9958 + #end 9959 + ## 9960 + ## 9961 + ## 9962 + #macro(displayPostDetails $pDoc) 9963 + <div class=~"row post-details~"> 9964 + <div class=~"col-xs-8 detail~"> 9965 + #if ($services.like.displayButton($pDoc.documentReference)) 9966 + #set ($optLikeRecord = $services.like.getLikes($pDoc.documentReference)) 9967 + ## Retrieve the likes number in XWiki 12.9+ 9968 + #set ($likeNumber = $optLikeRecord.get()) 9969 + #if (!$stringtool.isNumeric($likeNumber.toString())) 9970 + ## Retrieve the likes number in XWiki versions greater than 12.7 and lower than 12.9. 9971 + #set ($likeNumber = $optLikeRecord.get().likeNumber) 9972 + #end 9973 + #if ($stringtool.isNumeric($likeNumber.toString())) 9974 + <div class=~"post-likes btn btn-default disabled badge~" 9975 + title=~"$escapetool.xml($services.localization.render('like.button.title', [$likeNumber]))~"> 9976 + $services.icon.renderHTML('heart') <span class=~"like-number~">$likeNumber</span> 9977 + </div> 9978 + #end 9979 + #elseif ($services.ratings) 9980 + #set ($ratingsConfigDoc = $services.ratings.getConfigurationDocument($pDoc.documentReference)) 9981 + #if ($ratingsConfigDoc.getValue('displayRatings') == 1) 9982 + <ul class=~"pull-left note list-inline~"> 9983 + #foreach ($x in [1..5]) 9984 + #set ($cls = ~"~") 9985 + #if ($x > $averageVote) 9986 + #set ($cls = ~"-o~") 9987 + #end 9988 + <li><span class=~"fa fa-star$\{cls}~"></span></li> 9989 + #end 9990 + </ul> 9991 + #end 9992 + #end 9993 + </div> 9994 + #if ($showPostComments) 9995 + <div class=~"col-xs-4 com_det~"> 9996 + <p class=~"text-right~">$services.icon.renderHTML('comments') ($!nbComments)</p> 9997 + </div> 9998 + #end 9999 + </div> 10000 + #end 10001 + ## 10002 + ## 10003 + ## 10004 + #macro(initLayoutVars $pDoc $pObj) 10005 + #set ($postTitle = $pDoc.display('title', 'view', $pObj)) 10006 + #set ($imageAtt = $pDoc.getAttachment($pObj.getValue('image'))) 10007 + #isPublished($pObj $isPublished) 10008 + #isHidden($pObj $isHidden) 10009 + #getEntryDate($pDoc $pObj $postDate) 10010 + #formatPostDate($postDate $dateStr) 10011 + #set ($averageVote = $mathtool.round($services.ratings.getAverageRating($pDoc.documentReference).getAverageVote())) 10012 + #set ($nbComments = $pDoc.getComments().size()) 10013 + #set ($showPostComments = true) 10014 + #if ($xwiki.getSpacePreferenceFor('showcomments', $pDoc.documentReference.lastSpaceReference, '1') == '0') 10015 + #set ($showPostComments = false) 10016 + #end 10017 + #end 10018 + ## 10019 + ## 10020 + ## 10021 + #macro(displayEditPinnedPostsButton) 10022 + #if (~"$!macroAdditionalParams.get('postIndex')~" == '0' && $services.security.authorization.hasAccess('edit', $pinnedPostsSourceDoc.fullName)) 10023 + #set ($discard = $xwiki.ssx.use('Blog.PinnedPostsSheet')) 10024 + #set ($discard = $xwiki.jsx.use('Blog.PinnedPostsSheet')) 10025 + #set ($editPinnedPostsURL = $xwiki.getURL($pinnedPostsSourceDoc.fullName, 'get', 'sheet=Blog.PinnedPostsSheet')) 10026 + #set ($newPinnedPostsObjectURL = $xwiki.getURL($pinnedPostsSourceDoc.fullName, 'objectadd', ~"classname=Blog.PinnedPostsClass&form_token=$!\{services.csrf.getToken()}~")) 10027 + <div class=~"edit-pinned-posts-container~"> 10028 + <div class=~"col-xs-12 col-sm-12 col-md-12 text-right edit-pinned-posts~"> 10029 + <a href=~"javascript:;~" class=~"editPinnedPosts~"> 10030 + <span class=~"glyphicon glyphicon-edit~" aria-hidden=~"true~"></span> $services.localization.render('blog.post.layout.cards.pinnedposts.edit') 10031 + </a> 10032 + <input type=~"hidden~" id=~"editPinnedPostsURL~" value=~"$escapetool.xml($editPinnedPostsURL)~"/> 10033 + <input type=~"hidden~" id=~"newPinnedPostsObjectURL~" value=~"$escapetool.xml($newPinnedPostsObjectURL)~"/> 10034 + </div> 10035 + </div> 10036 + #if (~"$!pinnedPostsObj~" == '') 10037 + <input type=~"hidden~" name=~"createPinnedPostsObject~" value=~"1~"/> 10038 + #end 10039 + #end 10040 + #end" %) 10041 +((( 10042 +(% class="macro-placeholder hidden" %) 10043 +((( 10044 +macro:velocity 10045 +))) 10046 +))) 10047 + 10048 +(% class="macro" data-macro="startmacro:velocity|-||-| #if ($services.security.authorization.hasAccess('view', $xcontext.macro.params.reference)) 10049 + #set ($postDoc = $xwiki.getDocument($xcontext.macro.params.reference)) 10050 + #getEntryObject($postDoc $postObj) 10051 + #if (~"$!postObj~" != '') 10052 + #extractLayoutParametersFromString($xcontext.macro.params.params $macroAdditionalParams) 10053 + ## 10054 + #set ($displayTitle = true) 10055 + #if (~"$!macroAdditionalParams.get('displayTitle')~" == 'false') 10056 + #set ($displayTitle = false) 10057 + #end 10058 + ## 10059 + #set ($displayTitleFirstOnPinnedPosts = false) 10060 + #if (~"$!macroAdditionalParams.get('displayTitleFirstOnPinnedPosts')~" == 'true') 10061 + #set ($displayTitleFirstOnPinnedPosts = true) 10062 + #end 10063 + ## 10064 + #set ($displaySummaryOnPinnedPosts = true) 10065 + #if (~"$!macroAdditionalParams.get('displaySummaryOnPinnedPosts')~" == 'false') 10066 + #set ($displaySummaryOnPinnedPosts = false) 10067 + #end 10068 + ## 10069 + #set ($postIndex = $numbertool.toNumber($macroAdditionalParams.get('postIndex')).intValue()) 10070 + #if ($postIndex == 0) 10071 + #set ($stopBlogPostsDisplay = false) 10072 + #end 10073 + #set ($listResultsCount = $numbertool.toNumber($macroAdditionalParams.get('list-results-count')).intValue()) 10074 + #set($discard = $xwiki.ssx.use('Blog.BlogPostLayoutCards')) 10075 + #set($scaleWidth = 600) 10076 + #set($imgQs=~"width=$scaleWidth~") 10077 + ## Display pinned posts 10078 + ## Get the list of pinned posts 10079 + #set ($pinnedPosts = []) 10080 + #set ($pinnedPostsSourceDoc = $NULL) 10081 + #if (~"$!macroAdditionalParams.get('forceDisplayPinnedPosts')~" == 'true' && ~"$!macroAdditionalParams.get('list-blog')~" != '') 10082 + #set ($pinnedPostsSourceDoc = $xwiki.getDocument($macroAdditionalParams.get('list-blog'))) 10083 + ## Note that the 'list-blog' parameter contains the value of the 'blog' parameter of the blogpostlist macro passed to the current layout macro 10084 + #end 10085 + #if (~"$!pinnedPostsSourceDoc~" != '') 10086 + #set ($pinnedPostsObj = $pinnedPostsSourceDoc.getObject('Blog.PinnedPostsClass')) 10087 + #if (~"$!pinnedPostsObj~" != '') 10088 + #set ($orderedPinnedPostsJSON = $pinnedPostsObj.getProperty('orderedPosts').value) 10089 + #if (~"$!orderedPinnedPostsJSON~" != '') 10090 + #set ($pinnedPosts = $jsontool.parse($orderedPinnedPostsJSON)) 10091 + #else 10092 + #set ($pinnedPosts = $pinnedPostsObj.getProperty('posts').value) 10093 + #end 10094 + #end 10095 + #if ($postIndex == 0 && $pinnedPosts.size() > 0) 10096 + \{\{html clean=~"false~"}} 10097 + #set ($x = 0) 10098 + #set ($showPinnedPostsButton = true) 10099 + #foreach ($pinnedPost in $pinnedPosts) 10100 + #if ($x == 0) 10101 + <div class=~"row flex-container~"> 10102 + #if ($showPinnedPostsButton) 10103 + #displayEditPinnedPostsButton() 10104 + #set($showPinnedPostsButton = false) 10105 + #end 10106 + #end 10107 + #set ($pinnedPostDoc = $xwiki.getDocument($pinnedPost)) 10108 + #getEntryObject($pinnedPostDoc $pinnedPostObj) 10109 + #if (~"$!pinnedPostObj~" != '') 10110 + #displayPinnedPost($pinnedPostDoc $pinnedPostObj) 10111 + #end 10112 + #set ($x = $mathtool.add($x, 1)) 10113 + #if ($x == 3) 10114 + #set ($x = 0) 10115 + </div> 10116 + #end 10117 + #end 10118 + #if ($mathtool.mod($x, 3) != 0) 10119 + </div> 10120 + #end 10121 + \{\{/html}} 10122 + 10123 + ## If the first post is a pinned post : this means that all the posts are pinned 10124 + ## In this case, avoid displaying the posts again after the pinned posts section. 10125 + #if ($pinnedPosts.contains($xcontext.macro.params.reference)) 10126 + #set ($stopBlogPostsDisplay = true) 10127 + #end 10128 + #elseif($postIndex == 0 && $pinnedPosts.size() == 0) 10129 + 10130 + \{\{html}} 10131 + <div class=~"row no-pinnded-posts~"> 10132 + #displayEditPinnedPostsButton() 10133 + </div> 10134 + \{\{/html}} 10135 + 10136 + #end 10137 + #end 10138 + #if (!$stopBlogPostsDisplay) 10139 + \{\{html clean=~"false~"}} 10140 + #if (~"$!nbDisplayedPosts~" == '' || $postIndex == 0) 10141 + #set ($nbDisplayedPosts = 0) 10142 + #end 10143 + #if($mathtool.mod($nbDisplayedPosts, 2) == 0) 10144 + #if ($nbDisplayedPosts != 0) 10145 + </div> 10146 + #set ($lastHtmlTag = 'c') 10147 + #end 10148 + #if ($nbDisplayedPosts == $mathtool.sub($listResultsCount, 1)) 10149 + <div class=~"row~"> 10150 + #set ($lastHtmlTag = 'o') 10151 + #set ($lastRowClass = ~"~") 10152 + #else 10153 + <div class=~"row flex-container~"> 10154 + #set ($lastHtmlTag = 'o') 10155 + #set ($lastRowClass = ~"flex~") 10156 + #end 10157 + #end 10158 + #displaySmallPost($postDoc $postObj) 10159 + #if ($nbDisplayedPosts == $mathtool.sub($listResultsCount, 1)) 10160 + </div> 10161 + #set ($lastHtmlTag = 'c') 10162 + #end 10163 + #set ($nbDisplayedPosts = $mathtool.add($nbDisplayedPosts, 1)) 10164 + #if ($postIndex == $mathtool.sub($listResultsCount, 1)) 10165 + #if (~"$!lastHtmlTag~" == 'o') 10166 + #if ($mathtool.mod($nbDisplayedPosts, 2) == 1 && $lastRowClass == ~"flex~") 10167 + <div class=~"col-xs-12 col-sm-6~"></div> 10168 + #end 10169 + </div> 10170 + #end 10171 + #end 10172 + \{\{/html}} 10173 + #end 10174 + #else 10175 + \{\{error}}$services.localization.render('blog.blogpostlayout.notpost', [$xcontext.macro.params.reference])\{\{/error}} 10176 + #end 10177 + #else 10178 + \{\{error}}$services.localization.render('blog.blogpostlayout.post_view_not_allowed', [$xcontext.macro.params.reference])\{\{/error}} 10179 + #end" %) 10180 +((( 10181 +(% class="macro-placeholder hidden" %) 10182 +((( 10183 +macro:velocity 10184 +))) 10185 + 10186 +(% class="macro" data-macro="startmacro:html|-|clean=~"false~"|-|<div class=~"col-xs-12 col-sm-6 next_blog~"> 10187 +<a href=~"/Nieuws/RI%26E%202023~" class=~"thumbnail~"> 10188 +<div class=~"row~"> 10189 +<div class=~"col-xs-4~"> 10190 +<img src=~"/download/Nieuws/RI%26E%202023/Knipsel.JPG?width=600&rev=1.1~" /> 10191 +</div> 10192 +<div class=~"col-xs-8 art_det~"> 10193 +<p class=~"text-left~"> 10194 +RI en E 2023 <br/><span class=~"date_info~"> Dec 13, 2023, </span> 10195 +</p> 10196 +<div class=~"row post-details~"> 10197 +<div class=~"col-xs-8 detail~"> 10198 +<div class=~"post-likes btn btn-default disabled badge~" 10199 +title=~"Number of likes on this page: 1~"> 10200 +<span class=~"fa fa-heart~"></span> <span class=~"like-number~">1</span> 10201 +</div> 10202 +</div> 10203 +<div class=~"col-xs-4 com_det~"> 10204 +<p class=~"text-right~"><span class=~"fa fa-comments~"></span> (0)</p> 10205 +</div> 10206 +</div> 10207 +</div> 10208 +</div> 10209 +</a> 10210 +</div>" %) 10211 +((( 10212 +(% class="macro-placeholder hidden" %) 10213 +((( 10214 +macro:html 10215 +))) 10216 + 10217 +(% class="col-xs-12 col-sm-6 next_blog" %) 10218 +((( 10219 +(% class="row" %) 10220 +((( 10221 +(% class="col-xs-4" %) 10222 +((( 10223 +[[~[~[image:/download/Nieuws/RI%26E%202023/Knipsel.JPG?width=600&rev=1.1~]~]>>path:/Nieuws/RI%26E%202023||class="thumbnail"]] 10224 +))) 10225 + 10226 +(% class="col-xs-8 art_det" %) 10227 +((( 10228 +(% class="text-left" %) 10229 +[[RI en E 2023 10230 +(% class="date_info" %) Dec 13, 2023,>>path:/Nieuws/RI%26E%202023||class="thumbnail"]] 10231 + 10232 +(% class="row post-details" %) 10233 +((( 10234 +(% class="col-xs-8 detail" %) 10235 +((( 10236 +(% class="post-likes btn btn-default disabled badge" title="Number of likes on this page: 1" %) 10237 +((( 10238 +[[(% class="like-number" %)1>>path:/Nieuws/RI%26E%202023||class="thumbnail"]] 10239 +))) 10240 +))) 10241 + 10242 +(% class="col-xs-4 com_det" %) 10243 +((( 10244 +(% class="text-right" %) 10245 +[[(0)>>path:/Nieuws/RI%26E%202023||class="thumbnail"]] 10246 +))) 10247 +))) 10248 +))) 10249 +))) 10250 +))) 10251 +))) 10252 +))) 10253 +))) 10254 + 10255 +(% class="macro" data-macro="startmacro:blogPostLayoutCards|-|reference=~"Nieuws.Aankondiging update Syntess 6\\\\.9\\\\.0181~" params=~"displayTitle=true|useSummary=true|list-limit=10|list-layout=cards|list-blog=Nieuws.WebHome|list-paginated=no|list-results-count=10|postIndex=0|previousPostReference=|postIndex=1|previousPostReference=Nieuws.Nieuwsbrief januari 2024|postIndex=2|previousPostReference=Nieuws.Audit BMI en BORG|postIndex=3|previousPostReference=Nieuws.Nieuwsbrief december 2023|postIndex=4|previousPostReference=Nieuws.Nieuwsbrief november 2023|postIndex=5|previousPostReference=Nieuws.Feestelijke afsluiting, keuring materieel en tellen voorraden|postIndex=6|previousPostReference=Nieuws.RI&E 2023~"" %) 10256 +((( 10257 +(% class="macro-placeholder hidden" %) 10258 +((( 10259 +macro:blogPostLayoutCards 10260 +))) 10261 + 10262 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogCode~"" %) 10263 +((( 10264 +(% class="macro-placeholder hidden" %) 10265 +((( 10266 +macro:include 10267 +))) 10268 + 10269 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogParameters~"" %) 10270 +((( 10271 +(% class="macro-placeholder hidden" %) 10272 +((( 10273 +macro:include 10274 +))) 10275 + 10276 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|## Blog 10277 +#set($blogClassname = 'Blog.BlogClass') 10278 +#set($blogTemplate = 'Blog.BlogTemplate') 10279 +#set($blogSheet = 'Blog.BlogSheet') 10280 +## Blog entries 10281 +#set($blogPostClassname = 'Blog.BlogPostClass') 10282 +#set($blogPostTemplate = 'Blog.BlogPostTemplate') 10283 +#set($blogPostSheet = 'Blog.BlogPostSheet') 10284 +#set($blogPostObjectNumber = $xwiki.getDocument($blogPostTemplate).getObject($blogPostClassname).number) 10285 +#set($oldArticleClassname = 'XWiki.ArticleClass') 10286 +## Categories 10287 +#set($blogCategoryClassname = 'Blog.CategoryClass') 10288 +#set($blogCategoryTemplate = 'Blog.CategoryTemplate') 10289 +#set($blogCategorySheet = 'Blog.CategorySheet') 10290 +#set($blogCategoriesSheet = 'Blog.CategoriesSheet') 10291 +#set($oldBlogCategoryClassname = 'Blog.Categories') 10292 +#set($defaultCategoryParent = 'Blog.Categories') 10293 +## Style 10294 +#set($blogStyleDocumentName = 'Blog.BlogStyle') 10295 +#set($blogStyleDocument = $xwiki.getDocument($blogStyleDocumentName)) 10296 +## Clientside scripts 10297 +#set($blogScriptsDocumentName = 'Blog.BlogScripts') 10298 +#set($blogPublisher = $xwiki.getDocument('Blog.Publisher')) 10299 +## Misc 10300 +#set($thisURL = $doc.getURL($xcontext.action, $request.queryString)) 10301 +#set($isBlogPost = $doc.getObject($blogPostClassname)) 10302 +#set($defaultBlogSpace = 'Blog') 10303 +#set($isCategoryPage = $doc.getObject($blogCategoryClassname)) 10304 +#set($isCategoriesHomePage = $doc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 10305 +## 10306 +## 10307 +## 10308 +#** 10309 + * Displays an image, taken from the blog style document. 10310 + * 10311 + * @param $imgName The name of the icon from icons set to use. 10312 + *# 10313 +#macro(toolImage $imgName)(% class=~"icon-manage~"~%)$services.icon.render($imgName)(%~%)#end" %) 10314 +((( 10315 +(% class="macro-placeholder hidden" %) 10316 +((( 10317 +macro:velocity 10318 +))) 10319 +))) 10320 +))) 10321 + 10322 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|## 10323 +## 10324 +## 10325 +## Import the blog skin and javascripts. 10326 +$!xwiki.ssx.use($blogStyleDocumentName)## 10327 +$!xwiki.jsx.use($blogScriptsDocumentName)## 10328 +## 10329 +## import the hierarchy for the path the blog post footer (in displayEntryBlogLocation) 10330 +#template('hierarchy_macros.vm')## 10331 +## 10332 +## 10333 +#** 10334 + * Prints a blog. This is the main macro used in the BlogSheet. 10335 + * 10336 + * @param blogDoc the XDocument holding the blog definition object. 10337 + *### 10338 +#macro(printBlog $blogDoc) 10339 + \{\{include reference='Blog.CreatePost'/}} 10340 + 10341 + ## Use the blogPostList macro to display the blogposts 10342 + ## The blogPostList is used only in case of 'all' or 'paged' blog display type because the blogPostList macro 10343 + ## do not support FTM the monthly and weekly blog display types 10344 + #getBlogDisplayType($blogDoc $displayType) 10345 + #if ($displayType == 'weekly' || $displayType == 'monthly') 10346 + #getBlogEntries($blogDoc $entries) 10347 + #displayBlog($entries 'index' true true) 10348 + #displayNavigationLinks($blogDoc) 10349 + #else 10350 + #getBlogDisplayType($blogDoc $displayType) 10351 + #set ($paginated = 'no') 10352 + #if ($displayType == 'paginated') 10353 + #set ($paginated = 'yes') 10354 + #end 10355 + #getBlogPostsLayout($blogDoc $postsLayout) 10356 + (% class=~"hfeed index~" ~%)(((\{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$paginated~" layout=~"$!postsLayout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" /}}))) 10357 + #end 10358 +#end 10359 +## 10360 +## 10361 +## 10362 +#** 10363 + * Shows blog information. In view mode, the description is printed. In edit mode, allows changing blog settings: title, 10364 + * description, blog type (global or in-space), index display type (fixed size pagination, weekly index, monthly index, 10365 + * all entries). 10366 + * 10367 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 10368 + *### 10369 +#macro(showBlogInfo $blogDoc) 10370 + #if($blogDoc.getObject($blogClassname)) 10371 + ## Keep testing for inline action for backward compatibility with older blogs. 10372 + #if($xcontext.action == 'edit' || $xcontext.action == 'inline') 10373 + #macro(displayProperty $blogDoc $propname) 10374 + ; #displayPropName($xwiki.getClass($blogClassname).get($propname)): 10375 + : $blogDoc.display($propname) 10376 + #end 10377 + #displayProperty($blogDoc 'title') 10378 + #displayProperty($blogDoc 'description') 10379 + #displayProperty($blogDoc 'displayType') 10380 + #displayProperty($blogDoc 'itemsPerPage') 10381 + #displayProperty($blogDoc 'postsLayout') 10382 + #displayProperty($blogDoc 'postsLayoutParameters') 10383 + #else 10384 + $blogDoc.display('description') 10385 + #end 10386 + #elseif($doc.fullName == $blogSheet) 10387 += $services.localization.render('blog.code.blogsheet') = 10388 + \{\{translation key='blog.code.sheetexplanation'/}} 10389 + #else 10390 + \{\{warning}}\{\{translation key='blog.code.notblog'/}}\{\{/warning}} 10391 + #end 10392 +#end 10393 +## 10394 +## 10395 +## 10396 +#** 10397 + * Retrieve the blog document, which usually is either <tt><Space>.WebHome</tt> for whole-spaces blogs, or 10398 + * <tt><Space>.Blog</tt> for in-space blogs. If none of these documents contains a blog object, then the first 10399 + * (alphabetically) document in the target space that contains one is returned. Finally, if no document in the current 10400 + * space contains a blog object, then <tt>Blog.WebHome</tt> is returned as the default blog. 10401 + * 10402 + * @param space A <tt>String</tt>, the name of the space where to search. 10403 + * @param blogDoc The resulting XDocument. 10404 + *### 10405 +#macro(getBlogDocument $space $blogDoc) 10406 + #set ($result = $NULL) 10407 + ## Check if the current space has a WebPreferences page that contains a Blog.EnablePanelsConfigurationClass object. 10408 + ## It is possible to display a blog's panels info in a page that is not inside a blog and in that case we need to 10409 + ## identify the right blog based on a configuration object in a WebPreferences page. 10410 + #set ($spaceReference = $services.model.resolveSpace($space)) 10411 + #set ($preferencesDocRef = $services.model.createDocumentReference('WebPreferences', $spaceReference)) 10412 + #set ($preferencesDoc = $xwiki.getDocument($preferencesDocRef)) 10413 + #set ($preferencesObj = $preferencesDoc.getObject('Blog.EnablePanelsConfigurationClass')) 10414 + #if ($preferencesObj) 10415 + #set ($result = $xwiki.getDocument( $preferencesObj.getValue('blog'))) 10416 + #end 10417 + ## If no special case occurs (like the previous one), identify a blog by checking the current location from its hierarchy. 10418 + #if (~"$!result~" == '') 10419 + ## First, try the Space.WebHome, for a whole-space blog 10420 + #set($result = $xwiki.getDocument(~"$\{space}.WebHome~")) 10421 + #if(!$result.getObject($blogClassname)) 10422 + ## Second, try the Space.Blog document 10423 + #set($result = $xwiki.getDocument(~"$\{space}.Blog~")) 10424 + #if(!$result.getObject($blogClassname)) 10425 + ## Third, try searching for a blog document in the current space 10426 + ## Prevent the query fail when the space contains dots '.' 10427 + #set($blogDocs = $services.query.hql(~", BaseObject obj where doc.space = :space and obj.name = doc.fullName and obj.className = '$blogClassname' order by doc.name~").setLimit(1).setOffset(0).bindValue('space', $space).execute()) 10428 + #if($blogDocs.size() > 0) 10429 + #set($result = $xwiki.getDocument($blogDocs.get(0))) 10430 + #else 10431 + ## Fourth, try searching for a blog document that have the its 'postsLocation' set to the current space 10432 + #set($blogDocs = $services.query.hql(~", BaseObject obj, StringProperty as postsLocationProp where obj.name = doc.fullName and obj.className = '$blogClassname' and obj.id=postsLocationProp.id.id and postsLocationProp.id.name='postsLocation' and postsLocationProp.value=:postsLocation order by doc.name~").setLimit(1).setOffset(0).bindValue('postsLocation', $space).execute()) 10433 + #if($blogDocs.size() > 0) 10434 + #set($result = $xwiki.getDocument($blogDocs.get(0))) 10435 + #else 10436 + ## Last, fallback to Blog.WebHome, the default blog 10437 + #set($result = $xwiki.getDocument('Blog.WebHome')) 10438 + #end 10439 + #end 10440 + #end 10441 + #end 10442 + #end 10443 + #set ($blogDoc = $NULL) 10444 + #setVariable (~"$blogDoc~" $result) 10445 +#end 10446 +## 10447 +## 10448 +## 10449 +#** 10450 + * Retrieve the blog title. 10451 + * 10452 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass<tt> object with the <tt>title</tt> property set. 10453 + * @param title The resulting title. 10454 + *### 10455 +#macro(getBlogTitle $blogDoc $title) 10456 + ## Titles can contain velocity code (most commonly translations), so we should evaluate them. 10457 + #set ($title = $NULL) 10458 + #setVariable (~"$title~" $!blogDoc.displayTitle) 10459 +#end 10460 +## 10461 +## 10462 +## 10463 +#** 10464 + * Retrieve the blog description. 10465 + * 10466 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>description</tt> 10467 + * property set. 10468 + * @param description The resulting description. 10469 + *### 10470 +#macro(getBlogDescription $blogDoc $description) 10471 + #getBlogProperty($blogDoc 'description' '' $result) 10472 + #set ($description = $NULL) 10473 + #setVariable (~"$description~" $result) 10474 +#end 10475 +## 10476 +## 10477 +## 10478 +#** 10479 + * Retrieves a list of entries to be displayed. The entries are either part of the blog's space, or have the blog 10480 + * document set as a parent. The number and range of entries returned (from all those belonging to this blog) depends on 10481 + * the blog display type: paginated (fixed number of entries), weekly (all entries in a week), monthly (all entries in a 10482 + * month), or all. 10483 + * 10484 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 10485 + * @param entries The resulting list of entries to display, a list of XDocument names. 10486 + *### 10487 +#macro(getBlogEntries $blogDoc $entries) 10488 + #if (!$entries) 10489 + #setVariable (~"$entries~" []) 10490 + #end 10491 + #getAllBlogPostsQuery($query) 10492 + #isDefaultBlog($blogDoc $isDefault) 10493 + #set($queryParams = \{}) 10494 + #if ($isDefault) 10495 + #getCategoryAllBlogPostsQuery($query) 10496 + #set($query = ~"$\{query} and (doc.creator = :creator or (isPublished.value = 1 and hidden.value = 0)) and (doc.space = :space or catList like :catList escape '!')~") 10497 + #set($discard = $queryParams.put('creator', $xcontext.user)) 10498 + #set($discard = $queryParams.put('space', $blogDoc.space)) 10499 + #set($sanitizedBlogSpace = $blogDoc.space.replaceAll('([%_!])', '!$1').concat('.%')) 10500 + #set($discard = $queryParams.put('catList', $sanitizedBlogSpace)) 10501 + #else 10502 + #set($query = ~"$\{query} and (doc.space = :space or doc.parent = :parent)~") 10503 + #getBlogPostsLocation($blogDoc.space $blogPostsLocation) 10504 + #set($discard = $queryParams.put('space', $blogPostsLocation)) 10505 + #set($discard = $queryParams.put('parent', $blogDoc.space)) 10506 + #end 10507 + #getBlogDisplayType($blogDoc $displayType) 10508 + #if($displayType == 'weekly') 10509 + #getWeeklyBlogEntries($blogDoc $query $entries $queryParams) 10510 + #elseif($displayType == 'monthly') 10511 + #getMonthlyBlogEntries($blogDoc $query $entries $queryParams) 10512 + #elseif($displayType == 'all') 10513 + #getAllBlogEntries($blogDoc $query $entries $queryParams) 10514 + #else 10515 + #getPagedBlogEntries($blogDoc $query $entries $queryParams) 10516 + #end 10517 +#end 10518 +## 10519 +## 10520 +## 10521 +#** 10522 + * Retrieves a list of entries to be displayed. The entries are taken from a ~"page~" of the blog, a sequence of documents 10523 + * defined by the request parameters <tt>ipp</tt> (items per page) and <tt>page</tt> (the current page). Initially the 10524 + * first page is displayed, with the number of entries defined in the blog object in the <tt>itemsPerPage</tt> property 10525 + * (10 if not defined). 10526 + * 10527 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 10528 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 10529 + * refined to restrict to a given space, or to a given search criteria, etc. 10530 + * @param entries The resulting list of entries to display, a list of XDocument names. 10531 + * @param queryParams The parameters to bind with the query. 10532 + *### 10533 +#macro(getPagedBlogEntries $blogDoc $query $entries $queryParams) 10534 + #if (!$entries) 10535 + #setVariable (~"$entries~" []) 10536 + #end 10537 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 10538 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~")) 10539 + #bindQueryParameters($countQueryObj $queryParams) 10540 + #bindQueryParameters($queryObj $queryParams) 10541 + #set($totalEntries = $countQueryObj.count()) 10542 + #getBlogProperty($blogDoc 'itemsPerPage' '10' $defaultItemsPerPage) 10543 + #set($defaultItemsPerPage = $numbertool.toNumber($defaultItemsPerPage).intValue()) 10544 + ## This macro is defined in the default macros.vm library. It also sets $itemsPerPage and $startAt. 10545 + #preparePagedViewParams($totalEntries $defaultItemsPerPage) 10546 + #set($discard = $entries.addAll($queryObj.setLimit($itemsPerPage).setOffset($startAt).addFilter(~"unique~").execute())) 10547 +#end 10548 +## 10549 +## 10550 +## 10551 +#** 10552 + * Retrieves a list of entries to be displayed. The entries are taken from a week of the blog. The target week is 10553 + * defined by the request parameters <tt>week</tt> (the week number in the year, from 1 to 52) and <tt>year</tt> (4 10554 + * digit year). Initially the current week is displayed. 10555 + * 10556 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 10557 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 10558 + * refined to restrict to a given space, or to a given search criteria, etc. 10559 + * @param entries The resulting list of entries to display, a list of XDocument names. 10560 + * @param queryParams The parameters to bind with the query. 10561 + *### 10562 +#macro(getWeeklyBlogEntries $blogDoc $query $entries $queryParams) 10563 + #if (!$entries) 10564 + #setVariable (~"$entries~" []) 10565 + #end 10566 + #getRequestedWeek($weekDate) 10567 + #set($dateFormatter = $xwiki.jodatime.getDateTimeFormatterForPattern('yyyy-MM-dd')) 10568 + #set($minDay = $dateFormatter.print($weekDate.toMutableDateTime().weekOfWeekyear().roundFloor())) 10569 + #set($maxDay = $dateFormatter.print($weekDate.toMutableDateTime().weekOfWeekyear().roundCeiling())) 10570 + #set($query = ~"$\{query} and publishDate.value >= '$minDay' and publishDate.value < '$maxDay'~") 10571 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 10572 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 10573 + #bindQueryParameters($countQueryObj $queryParams) 10574 + #bindQueryParameters($queryObj $queryParams) 10575 + #set($totalEntries = $countQueryObj.count()) 10576 + #set($discard = $entries.addAll($queryObj.execute())) 10577 +#end 10578 +## 10579 +## 10580 +## 10581 +#** 10582 + * Retrieves a list of entries to be displayed. The entries are taken from a month of the blog. The target month is 10583 + * defined by the request parameters <tt>month</tt> (the month number, from 1 to 12) and <tt>year</tt> (4 10584 + * digit year). Initially the current month is displayed. 10585 + * 10586 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 10587 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 10588 + * refined to restrict to a given space, or to a given search criteria, etc. 10589 + * @param entries The resulting list of entries to display, a list of XDocument names. 10590 + * @param queryParams The parameters to bind with the query. 10591 + *### 10592 +#macro(getMonthlyBlogEntries $blogDoc $query $entries) 10593 + #if (!$entries) 10594 + #setVariable (~"$entries~" []) 10595 + #end 10596 + #getRequestedMonth($monthDate) 10597 + #set($dateFormatter = $xwiki.jodatime.getDateTimeFormatterForPattern('yyyy-MM-dd')) 10598 + #set($minDay = $dateFormatter.print($monthDate.toMutableDateTime().monthOfYear().roundFloor())) 10599 + #set($maxDay = $dateFormatter.print($monthDate.toMutableDateTime().monthOfYear().roundCeiling())) 10600 + #set($query = ~"$\{query} and publishDate.value >= '$minDay' and publishDate.value < '$maxDay'~") 10601 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 10602 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 10603 + #bindQueryParameters($countQueryObj $queryParams) 10604 + #bindQueryParameters($queryObj $queryParams) 10605 + #set($totalEntries = $countQueryObj.count()) 10606 + #set($discard = $entries.addAll($queryObj.execute())) 10607 +#end 10608 +## 10609 +## 10610 +## 10611 +#** 10612 + * Retrieves a list of entries to be displayed. All entries belonging to the current blog are returned. 10613 + * 10614 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 10615 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 10616 + * refined to restrict to a given space, or to a given search criteria, etc. 10617 + * @param entries The resulting list of entries to display, a list of XDocument names. 10618 + * @param queryParams The parameters to bind with the query. 10619 + *### 10620 +#macro(getAllBlogEntries $blogDoc $query $entries $queryParams) 10621 + #if (!$entries) 10622 + #setVariable (~"$entries~" []) 10623 + #end 10624 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 10625 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 10626 + #bindQueryParameters($countQueryObj $queryParams) 10627 + #bindQueryParameters($queryObj $queryParams) 10628 + #set($totalEntries = $countQueryObj.count()) 10629 + #set($discard = $entries.addAll($queryObj.execute())) 10630 +#end 10631 +## 10632 +## 10633 +## 10634 +#** 10635 + * Retrieves a list of entries to be displayed. Only (and all) unpublished entries are returned. 10636 + * 10637 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 10638 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 10639 + * refined to restrict to a given space, or to a given search criteria, etc. 10640 + * @param queryParams The parameters to bind with the query. 10641 + * @param entries The resulting list of entries to display, a list of XDocument names. 10642 + *### 10643 +#macro(getUnpublishedBlogEntries $blogDoc $query $queryParams $entries) 10644 + #if (!$entries) 10645 + #setVariable (~"$entries~" []) 10646 + #end 10647 + #set($query = ~"$\{query} and isPublished.value = 0~") 10648 + #set ($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 10649 + #set ($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 10650 + #bindQueryParameters($countQueryObj $queryParams) 10651 + #bindQueryParameters($queryObj $queryParams) 10652 + #set($totalEntries = $countQueryObj.count()) 10653 + #set($discard = $entries.addAll($queryObj.execute())) 10654 +#end 10655 +## 10656 +## 10657 +## 10658 +#** 10659 + * Retrieves a list of entries to be displayed. The entries are taken from all the wiki, and not from a specific blog. 10660 + * 10661 + * @param entries The resulting list of entries to display, a list of XDocument names. 10662 + *### 10663 +#macro(getGlobalBlogEntries $entries) 10664 + #if (!$entries) 10665 + #setVariable (~"$entries~" []) 10666 + #end 10667 + #getAllBlogPostsQuery($query) 10668 + #set($totalEntries = $services.query.hql($query).count()) 10669 + #set($defaultItemsPerPage = 20) 10670 + ## This macro is defined in the default macros.vm library. It also sets $itemsPerPage and $startAt. 10671 + #preparePagedViewParams($totalEntries $defaultItemsPerPage) 10672 + #set($discard = $entries.addAll($services.query.hql(~"$\{query} order by publishDate.value desc~").setLimit($itemsPerPage).setOffset($startAt).execute())) 10673 +#end 10674 +#** 10675 + * Return the base query for selecting blog entries. It filters only visible entries, but does not bind to a specific 10676 + * blog, nor specify a range or an ordering criteria. 10677 + * 10678 + * This macro is a duplicate of #getAllBlogPostsQuery($query). Keep it in case it's used outside, in custom code. 10679 + * 10680 + * @param query The basic query for selecting blog entries. 10681 + *# 10682 +#macro(getBlogEntriesBaseQuery $query) 10683 + #getAllBlogPostsQuery($query) 10684 +#end 10685 +#** 10686 + * Return the Query for selecting the all wiki blog posts without filtering 10687 + * 10688 + * @param query The basic query for selecting blog entries. 10689 + *# 10690 +#macro(getAllBlogPostsQuery $query) 10691 + #set ($query = $NULL) 10692 + #setVariable(~"$query~" ~", BaseObject as obj, IntegerProperty isPublished, 10693 + IntegerProperty hidden, DateProperty publishDate 10694 + where doc.fullName <> '$blogPostTemplate' and 10695 + obj.name=doc.fullName and obj.className='$blogPostClassname' and 10696 + isPublished.id.id = obj.id and isPublished.id.name = 'published' and 10697 + hidden.id.id = obj.id and hidden.id.name='hidden' and 10698 + publishDate.id.id = obj.id and publishDate.id.name='publishDate' and 10699 + (doc.creator = '$escapetool.sql($xcontext.user)' or (isPublished.value = 1 and hidden.value = 0))~") 10700 +#end 10701 +## 10702 +## 10703 +## 10704 +###** 10705 + * Return the Query for selecting the all wiki blog posts with categories filtering 10706 + * 10707 + * @param query The basic query for selecting blog entries. 10708 + *### 10709 +#macro(getCategoryAllBlogPostsQuery $query) 10710 + #set ($query = $NULL) 10711 + #getAllBlogPostsQuery($baseQuery) 10712 + #set ($baseQuery = $baseQuery.replace('DateProperty publishDate', 'DateProperty publishDate, DBStringListProperty as category left join category.list catList')) 10713 + #setVariable(~"$query~" ~"$\{baseQuery} and obj.id=category.id.id and category.id.name='category'~") 10714 +#end 10715 +## 10716 +## 10717 +## 10718 +#** 10719 + * Determines how is the blog index split into pages: paginated (fixed number of entries), weekly (all entries in a 10720 + * week), monthly (all entries in a month), or all. 10721 + * 10722 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>displayType</tt> 10723 + * property set. 10724 + * @param displayType The resulting string. If the blog object does not define anything, it is considered paginated. 10725 + *### 10726 +#macro(getBlogDisplayType $blogDoc $displayType) 10727 + #getBlogProperty($blogDoc 'displayType' 'paginated' $result) 10728 + #set ($displayType = $NULL) 10729 + #setVariable (~"$displayType~" $result) 10730 +#end 10731 +## 10732 +## 10733 +## 10734 +#** 10735 + * Displays a list of entries. 10736 + * 10737 + * @param entries The entries to display, a list of XDocument names. 10738 + * @param displaying What exactly is displayed: blog index, a single blog entry, a blog category, search results, 10739 + * unpublished entries, etc. This will be used as the classname(s) for the container div (hfeed). Currently 10740 + * used values: index, single, category, search, unpublished, hidden. 10741 + * @param onlyExtract If <tt>true</tt>, only display the extract of articles where available, otherwise display the full content. 10742 + * @param shouldDisplayTitles If <tt>true</tt>, display the blog title (blog posts shouldn't display the title when they're 10743 + * displayed alone on their page since it's the page title which is used in this case) 10744 + *### 10745 +#macro(displayBlog $entries $displaying $onlyExtract $shouldDisplayTitles) 10746 + #set($blogDay = '') 10747 + (% class=~"hfeed $!\{displaying}~" ~%)((( 10748 + (% class=~"blogDay~" ~%)((( 10749 + #foreach ($entryDoc in $xwiki.wrapDocs($entries)) 10750 + #getEntryObject($entryDoc $entryObj) 10751 + ## Although all entries should have one of the two objects, better check to be sure. 10752 + #if(~"$!\{entryObj}~" != '') 10753 + #getEntryDate($entryDoc $entryObj $entryDate) 10754 + ## Display a ~"calendar sheet~" for each day. All entries posted on the same day share one such sheet. 10755 + #set($entryDateStr = $xwiki.formatDate($entryDate, 'yyyyMMMMdd')) 10756 + #if($blogDay != $entryDateStr) 10757 + #if($blogDay != '') 10758 + ))) 10759 + (% class=~"blogDay~" ~%)((( 10760 + #end 10761 + #displayBlogDate($entryDate) 10762 + #set ($blogDay = $entryDateStr) 10763 + #end 10764 + ## Finally, display the entry. 10765 + #displayEntry($entryDoc $entryObj $onlyExtract $shouldDisplayTitles true) 10766 + #end 10767 + #end 10768 + )))## blogDay 10769 + )))## hfeed 10770 +#end 10771 +## 10772 +## 10773 +## 10774 +#** 10775 + * Get the entry object, either a new BlogPost or an old Article. 10776 + * 10777 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 10778 + * @param entryObj The resulting xobject of the blog post. 10779 + *### 10780 +#macro(getEntryObject $entryDoc $__entryObj) 10781 + #set($result = $entryDoc.getObject(~"$\{blogPostClassname}~")) 10782 + #if(!$result) 10783 + #set($result = $entryDoc.getObject(~"$\{oldArticleClassname}~")) 10784 + #end 10785 + ## NOTE: The reason we put an underscore in front of the variable name is to prevent the following line from 10786 + ## overwriting the $entryObj variable that may be defined before this macro is called. Of course, $__entryObj may be 10787 + ## overwritten in this case but it's less likely to have such a variable defined before. 10788 + #set ($__entryObj = $NULL) 10789 + #setVariable (~"$__entryObj~" $result) 10790 +#end 10791 +## 10792 +## 10793 +## 10794 +#** 10795 + * Gets the date associated with a blog entry. This is the publication date. For unpublished entries, initially this is 10796 + * the document creation date, but can be edited by the user. 10797 + * 10798 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 10799 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 10800 + * @param result The resulting date, an instance of <tt>java.util.Date</tt>. 10801 + *### 10802 +#macro(getEntryDate $entryDoc $entryObj $result) 10803 + #set ($result = $NULL) 10804 + #setVariable (~"$result~" $entryObj.getProperty('publishDate').value) 10805 +#end 10806 +## 10807 +## 10808 +## 10809 +#** 10810 + * Displays a date, nicely formatted as a calendar page. 10811 + * 10812 + * @param date The date to display, an instance of <tt>java.util.Date</tt>. 10813 + *### 10814 +#macro(displayBlogDate $date) 10815 + #set($year = $xwiki.formatDate($date, 'yyyy')) 10816 + ## 3 letter month name, like Jan, Dec. 10817 + #set($month = $xwiki.formatDate($date, 'MMM')) 10818 + ## Uncomment to get a full length month name, like January, December. 10819 + ## TODO: this could be defined somewhere in the blog style. 10820 + ## #set($month = $xwiki.formatDate($date, 'MMMM')) 10821 + #set($day = $xwiki.formatDate($date, 'dd')) 10822 + (% class=~"blogdate~" ~%) 10823 + == (% class=~"month~" ~%)$month(%~%) (% class=~"day~" ~%)$day(%~%) (% class=~"year~" ~%)$year(%~%) == 10824 +#end 10825 +## 10826 +## 10827 +## 10828 +#** 10829 + * Displays a blog article: management tools, header, content, footer. 10830 + * 10831 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 10832 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 10833 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 10834 + * @param shouldDisplayTitle If <tt>true</tt>, display the blog title (blog posts shouldn't display the title 10835 + * when they're displayed alone on their page since it's the page title which is used in this case) 10836 + * @param shouldDisplayActions If <tt>true</tt>, display the blog post actions buttons 10837 + *### 10838 +#macro(displayEntry $entryDoc $entryObj $onlyExtract $shouldDisplayTitle $shouldDisplayActions) 10839 + ## Only articles with an explicit hidden setting or an explicit unpublished setting are hidden 10840 + #isPublished($entryObj $isPublished) 10841 + #isHidden($entryObj $isHidden) 10842 + #if($doc.fullName == $entryDoc.fullName) 10843 + (% class=~"hentry single-article~" ~%)((( 10844 + #else 10845 + (% class=~"hentry#if(!$isPublished) unpublished-article#elseif($isHidden) hidden-article#end~" ~%)((( 10846 + #end 10847 + #if ($shouldDisplayActions) 10848 + #displayEntryTools($entryDoc $entryObj) 10849 + #end 10850 + #if($shouldDisplayTitle) 10851 + #displayEntryTitle($entryDoc $entryObj) 10852 + #end 10853 + #if($doc.fullName == $entryDoc.fullName) 10854 + #if(!$isPublished) 10855 + \{\{warning}}\{\{translation key='blog.code.published'/}}\{\{/warning}} 10856 + #elseif($isHidden) 10857 + \{\{warning}}\{\{translation key='blog.code.hidden'/}}\{\{/warning}} 10858 + #end 10859 + #end 10860 + #displayEntryContent($entryDoc $entryObj $onlyExtract) 10861 + #displayEntryFooter($entryDoc $entryObj) 10862 + )))## hentry 10863 +#end 10864 +## 10865 +## 10866 +## 10867 +#** 10868 + * Checks if the provided blog is published or not. 10869 + * 10870 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 10871 + * @param isPublished The resulting boolean, true if the entry is considered published. 10872 + *### 10873 +#macro(isPublished $entryObj $isPublished) 10874 + #set ($isPublished = $NULL) 10875 + ## This should work for both old articles, which don't have the 'published' property at all, and 10876 + ## are considered published by default, and new entries, that should have 1 if published. 10877 + #if (~"$!\{entryObj.getProperty('published').value}~" != '0') 10878 + #setVariable (~"$isPublished~" true) 10879 + #else 10880 + #setVariable (~"$isPublished~" false) 10881 + #end 10882 +#end 10883 +## 10884 +## 10885 +## 10886 +#** 10887 + * Checks if the provided blog is hidden or not. 10888 + * 10889 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass<tt> xclass. 10890 + * @param isHiddel The resulting boolean, true if the entry is considered hidden. 10891 + *### 10892 +#macro(isHidden $entryObj $isHidden) 10893 + #set ($isHidden = $NULL) 10894 + ## This should work for both old articles, which don't have the 'hidden' property at all, and 10895 + ## are considered visible by default, and new entries, that should have 1 if hidden. 10896 + #if (~"$!\{entryObj.getProperty('hidden').value}~" == '1') 10897 + #setVariable (~"$isHidden~" true) 10898 + #else 10899 + #setVariable (~"$isHidden~" false) 10900 + #end 10901 +#end 10902 +## 10903 +## 10904 +## 10905 +#** 10906 + * Displays several ~"tools~" for manipulating blog posts: hide/show, publish, edit. 10907 + * 10908 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 10909 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 10910 + *### 10911 +#macro(displayEntryTools $entryDoc $entryObj) 10912 + #if($xcontext.action == 'view') 10913 + (% class=~"blog-entry-toolbox~" ~%)((( 10914 + #displayPublishButton($entryDoc $entryObj) 10915 + #displayHideShowButton($entryDoc $entryObj) 10916 + #displayEditButton($entryDoc $entryObj) 10917 + #displayDeleteButton($entryDoc $entryObj) 10918 + ))) 10919 + #end 10920 +#end 10921 +## 10922 +## 10923 +## 10924 +#** 10925 + * Displays the publish button to the entry <strong>creator</strong>, if the article is not published yet. 10926 + * 10927 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 10928 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 10929 + * @todo AJAX calls. 10930 + *### 10931 +#macro(displayPublishButton $entryDoc $entryObj) 10932 + #isPublished($entryObj $isPublished) 10933 + #if(!$isPublished && $entryDoc.creator == $xcontext.user && $xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 10934 + [[#toolImage('world')>>path:$blogPublisher.getURL('view', ~"entryName=$\{escapetool.url($entryDoc.fullName)}&xredirect=$\{escapetool.url($thisURL)}&form_token=$!\{services.csrf.getToken()}~")||title=~"$services.localization.render('blog.code.notpublished')~"]]## 10935 + #end 10936 +#end 10937 +## 10938 +## 10939 +## 10940 +#** 10941 + * Displays the hide or show button to the entry <strong>creator</strong>, if the article is already published. 10942 + * 10943 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 10944 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 10945 + *### 10946 +#macro(displayHideShowButton $entryDoc $entryObj) 10947 + #isPublished($entryObj $isPublished) 10948 + #isHidden($entryObj $isHidden) 10949 + ## Only published articles can be hidden. Unpublished articles are considered already hidden. 10950 + #if($isPublished && $entryDoc.creator == $xcontext.user && $xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 10951 + #set ($queryString = \{ 10952 + 'xredirect' : $thisURL, 10953 + 'form_token' : $services.csrf.getToken() 10954 + }) 10955 + #if ($isHidden) 10956 + #set ($discard = $queryString.putAll(\{ 10957 + ~"$\{entryObj.getxWikiClass().getName()}_$\{entryObj.number}_hidden~" : 0, 10958 + 'comment' : $services.localization.render('blog.code.madevisible') 10959 + })) 10960 + #set ($lockURL = $entryDoc.getURL('save', $escapetool.url($queryString))) 10961 + [[#toolImage('unlock')>>path:$lockURL||class=~"blog-tool-show~" title=~"$services.localization.render('blog.code.makevisible')~"]]## 10962 + #else 10963 + #set ($discard = $queryString.putAll(\{ 10964 + ~"$\{entryObj.getxWikiClass().getName()}_$\{entryObj.number}_hidden~" : 1, 10965 + 'comment' : $services.localization.render('blog.code.hid') 10966 + })) 10967 + #set ($lockURL = $entryDoc.getURL('save', $escapetool.url($queryString))) 10968 + [[#toolImage('lock')>>path:$lockURL||class=~"blog-tool-hide~" title=~"$services.localization.render('blog.code.hide')~"]]## 10969 + #end 10970 + #end 10971 +#end 10972 +## 10973 +## 10974 +## 10975 +#** 10976 + * Displays the edit button to those that can edit the article. 10977 + * 10978 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 10979 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 10980 + *### 10981 +#macro(displayEditButton $entryDoc $entryObj) 10982 + #if($xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 10983 + ## Call getDefaultEditMode() for backward compatibility with older blog posts. 10984 + [[#toolImage('pencil')>>path:$entryDoc.getURL('edit')||title=~"$services.localization.render('blog.code.editpost')~"]]## 10985 + #end 10986 +#end 10987 +## 10988 +## 10989 +## 10990 +#** 10991 + * Displays the delete button to those that can edit the article. 10992 + * 10993 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 10994 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 10995 + * @todo AJAX calls. 10996 + *### 10997 +#macro(displayDeleteButton $entryDoc $entryObj) 10998 + #if($xwiki.hasAccessLevel('delete', $xcontext.user, $entryDoc.fullName)) 10999 + [[#toolImage('cross')>>path:$entryDoc.getURL('delete')||title=~"$services.localization.render('blog.code.deletepost')~"]]## 11000 + #end 11001 +#end 11002 +## 11003 +## 11004 +## 11005 +#** 11006 + * Displays the title of the entry. 11007 + * 11008 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 11009 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 11010 + *### 11011 +#macro(displayEntryTitle $entryDoc $entryObj) 11012 + #if($doc.fullName == $entryDoc.fullName) 11013 + (% class=~"entry-title~" ~%) 11014 + = $services.rendering.escape($entryDoc.display('title', 'view', $entryObj), $xwiki.getCurrentContentSyntaxId()) = 11015 + #else 11016 + (% class=~"entry-title~" ~%) 11017 + === [[$services.rendering.escape($entryDoc.display('title', 'view', $entryObj),$xwiki.getCurrentContentSyntaxId())>>doc:$services.rendering.escape($services.model.serialize($entryDoc.getDocumentReference(),'default'),$xwiki.getCurrentContentSyntaxId())]] === 11018 + #end 11019 +#end 11020 +## 11021 +## 11022 +## 11023 +#** 11024 + * Displays the body of the entry. 11025 + * 11026 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 11027 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 11028 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 11029 + *### 11030 +#macro(displayEntryContent $entryDoc $entryObj $onlyExtract) 11031 + (% class=~"#if($onlyExtract)entry-summary#\{else}entry-content#end~" ~%)((( 11032 + #getEntryContent($entryDoc $entryObj $onlyExtract $entryContent) 11033 + ## FIXME: This causes the blog's content to not be annotatable. See http://jira.xwiki.org/browse/XWIKI-6328 11034 + ## Should probably be replaced by a display macro call with a reference to the object property holding the post's content 11035 + \{\{html wiki=~"false~"}}$entryDoc.getRenderedContent($entryContent, $entryDoc.syntax.toIdString())\{\{/html}} 11036 + ))) ## entry-content 11037 + (% class=~"clearfloats~" ~%)((())) 11038 +#end 11039 +## 11040 +## 11041 +## 11042 +#** 11043 + * Extracts the body of the entry that should be displayed. If <tt>onlyExtract</tt> is <tt>true</tt>, display the content 11044 + * of the <tt>extract</tt> field (if not empty). 11045 + * 11046 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 11047 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 11048 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 11049 + * @param entryContent The resulting content. 11050 + * @param removeEllipsis If <tt>true</tt>, then don't display an ellipsis at the end of the content (only used when 11051 + * <tt>onlyExtract</tt> is <tt>true</tt>) 11052 + *### 11053 +#macro(getEntryContent $entryDoc $entryObj $onlyExtract $entryContent $removeEllipsis) 11054 + #if ($onlyExtract) 11055 + ## Note: We trim the summary so that if there's some white space it won't cause the summary to be used instead 11056 + ## of the content. 11057 + #set ($macro.result = $entryObj.getProperty('extract').value.trim()) 11058 + #end 11059 + #if(~"$!macro.result~" == '') 11060 + #set($macro.result = $entryObj.getProperty('content').value) 11061 +#* Disabled until the content can be cleanly cut. 11062 +* #if($onlyExtract && $result.length()>$maxchars) 11063 +* #set($i = $macro.result.lastIndexOf(~" ~", $maxchars)) 11064 +* #set($i = $i + 1) 11065 +* #set($macro.result = ~"$\{macro.result.substring(0,$i)} *[...>$\{entryDoc.fullName}]*~") 11066 +* #end 11067 +## *### 11068 + #elseif (!$removeEllipsis) 11069 + #if($entryDoc.syntax.toIdString() == 'xwiki/1.0') 11070 + #set($macro.result = ~"$\{macro.result} <a href='$\{entryDoc.getExternalURL()}' title='$services.localization.render('blog.code.readpost')'>...</a>~") 11071 + #elseif($entryDoc.syntax.toIdString() == 'xwiki/2.0' || $entryDoc.syntax.toIdString() == 'xwiki/2.1') 11072 + #set($macro.result = ~"$\{macro.result} [[...>>$\{services.rendering.escape($entryDoc.getFullName(), $entryDoc.getSyntax())}||title='$services.localization.render('blog.code.readpost')']]~") 11073 + #end 11074 + #end 11075 + #set ($entryContent = $NULL) 11076 + #setVariable (~"$entryContent~" $macro.result) 11077 +#end 11078 +## 11079 +## 11080 +## 11081 +#** 11082 + * Displays the footer of the entry. 11083 + * 11084 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 11085 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 11086 + *### 11087 +#macro(displayEntryFooter $entryDoc $entryObj) 11088 + (% class=~"entry-footer~" ~%)((( 11089 + #isPublished($entryObj $isPublished) 11090 + (% class='entry-author-label' ~%) 11091 + #if($isPublished) 11092 + \{\{translation key='blog.code.postedby'/}} ## 11093 + #else 11094 + \{\{translation key='blog.code.createdby'/}} ## 11095 + #end 11096 + \{\{html wiki=~"false~" clean=~"false~"}}<span class='author vcard'>#userfn($entryDoc.creator)</span>\{\{/html}} ## 11097 + #getEntryDate($entryDoc $entryObj $entryDate) 11098 + #listCategories($entryObj) #* 11099 + ## Since the publish date and update date are not set at the exact same time, there could be a small difference that 11100 + ## we assume cannot be more than 3 seconds. 11101 + *#(% class=~"separator~" ~%)·(%~%) [[\{\{translation key='blog.code.permalink'/}}>>$services.rendering.escape($entryDoc,'xwiki/2.0')||rel=~"bookmark~"]] ## 11102 + #if ($showcomments) 11103 + (% class=~"separator~" ~%)·(%~%) [[\{\{translation key='blog.code.comments'/}}>>$services.rendering.escape($entryDoc,'xwiki/2.0')||anchor=~"Comments~"]] (% class=~"itemCount~" ~%)($entryDoc.comments.size())(%~%) ## 11104 + #end ## 11105 + #if($entryDoc != $doc) ## 11106 + #displayEntryBlogLocation($entryDoc $entryObj) ## 11107 + #end 11108 + )))## entry-footer 11109 +#end 11110 +## 11111 +## 11112 +#** 11113 + * Display the blog for the entry (if it is not the currently displayed blog) 11114 + * 11115 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 11116 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 11117 + *### 11118 +#macro(displayEntryBlogLocation $entryDoc $entryObj) 11119 + #getBlogPostsLocation($entryDoc.getSpace() $blogPostsLocation) ## 11120 + #if(~"$!blogPostsLocation~" != ~"~") ## 11121 + #set($blogSpaceRef = $services.model.resolveSpace($blogPostsLocation)) ## 11122 + #set($blogDocRef = $services.model.createDocumentReference($services.model.getEntityReference('DOCUMENT', 'default').getName(), $blogSpaceRef)) ## 11123 + #if($doc.documentReference != $blogDocRef) ## 11124 + #set($blogDoc = $xwiki.getDocument($blogDocRef)) ## 11125 + #set($blogObj = $blogDoc.getObject('Blog.BlogClass')) ## 11126 + #if($blogObj) (% class=~"blogrefpath~" ~%)((( ## 11127 + $services.localization.render('blog.code.postedin') \{\{html clean=~"false~" wiki=~"false~"}} #hierarchy($blogDocRef, \{'selfIsActive' : true, 'local': true}) \{\{/html}} ## 11128 + )))(%~%)## 11129 + #end 11130 + #end 11131 + #end 11132 +#end 11133 +## 11134 +## 11135 +## 11136 +## 11137 +#** 11138 + * List the categories an entry belongs to. Used in the footer. The categories are instances of <tt>Blog.CategoryClass</tt>. 11139 + * 11140 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 11141 + *### 11142 +#macro(listCategories $entryObj) 11143 + #if($entryObj.getxWikiClass().getName() == $blogPostClassname) 11144 + #set($categories = $entryObj.getProperty('category').value) 11145 + #set($first = true) 11146 + #if($categories.size() > 0) 11147 + #foreach($category in $categories) 11148 + #set($categoryDoc = $!xwiki.getDocument($category)) 11149 + #if(!$categoryDoc.isNew() && $categoryDoc.getObject($\{blogCategoryClassname})) 11150 + #if($foreach.count == 1) 11151 + (% class='separator' ~%)·(%~%) $services.localization.render('blog.code.categories') ## 11152 + #else 11153 + , ## 11154 + #end## 11155 + [[$!\{categoryDoc.getObject($blogCategoryClassname).getValue('name')}>>$\{category}||rel='tag']]## 11156 + #end## 11157 + #end## 11158 + #end 11159 + #end 11160 +#end 11161 +## 11162 +## 11163 +## 11164 +#** 11165 + * Displays blog pagination links (older and newer entries). 11166 + * 11167 + * @param blogDoc the XDocument holding the blog definition object. 11168 + *### 11169 +#macro(displayNavigationLinks $blogDoc) 11170 + (% class=~"clearfloats~" ~%)((())) 11171 + #getBlogDisplayType($blogDoc $displayType) 11172 + #if($displayType == 'weekly') 11173 + (% class=~"pagingLinks~" ~%)((( 11174 + #getRequestedWeek($weekDate) 11175 + $weekDate.addWeeks(-1)## 11176 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.previousweek'/}}>>$doc.documentReference.name||queryString=~"year=$weekDate.weekyear&week=$weekDate.weekOfWeekyear~" class=~"button~"]]**(%~%) 11177 + #sep() 11178 + $weekDate.addWeeks(2)## 2 because we already subtracted 1 above 11179 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.nextweek'/}} »>>$doc.documentReference.name||queryString=~"year=$weekDate.weekyear&week=$weekDate.weekOfWeekyear~" class=~"button~"]]**(%~%) 11180 + ))) 11181 + #elseif($displayType == 'monthly') 11182 + (% class=~"pagingLinks~" ~%)((( 11183 + #getRequestedMonth($monthDate) 11184 + $monthDate.addMonths(-1)## 11185 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.previousmonth'/}}>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"year=$monthDate.year&month=$monthDate.monthOfYear~" class=~"button~"]]**(%~%) 11186 + #sep() 11187 + $monthDate.addMonths(2)## 2 because we already subtracted 1 above 11188 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.nextmonth'/}} »>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"year=$monthDate.year&month=$monthDate.monthOfYear~" class=~"button~"]]**(%~%) 11189 + ))) 11190 + #elseif($displayType == 'all') 11191 + #else 11192 + ## Paginated 11193 + #if(($totalPages > 1)) 11194 + #set($queryString = '') 11195 + #foreach($p in $request.getParameterNames()) 11196 + #if($p != 'page' && $p != 'ipp') 11197 + #foreach($v in $request.getParameterValues($p)) 11198 + #set($queryString = ~"$\{queryString}&$\{escapetool.url($p)}=$\{escapetool.url($v)}~") 11199 + #end 11200 + #end 11201 + #end 11202 + (% class=~"pagingLinks~" ~%)((( 11203 + #if ($currentPageNumber < $totalPages) 11204 + #set($currentPageNumber = $currentPageNumber + 1) 11205 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.olderposts'/}}>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"page=$\{currentPageNumber}&ipp=$\{itemsPerPage}$queryString~" class=~"button~"]]**(%~%) 11206 + #set($currentPageNumber = $currentPageNumber - 1) 11207 + #end 11208 + #if ($currentPageNumber > 1) 11209 + #if ($currentPageNumber < $totalPages) 11210 + #sep() 11211 + #end 11212 + #set($currentPageNumber = $currentPageNumber - 1) 11213 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.newerposts'/}} »>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"page=$\{currentPageNumber}&ipp=$\{itemsPerPage}$queryString~" class=~"button~"]]**(%~%) 11214 + #set($currentPageNumber = $currentPageNumber + 1) 11215 + #end 11216 + (% class=~"clear~" ~%)(%~%) 11217 + )))## pagingLinks 11218 + #end 11219 + #end 11220 +#end 11221 +## 11222 +## 11223 +## 11224 +#** 11225 + * Displays a message box with ~"publish~" icon. 11226 + * 11227 + * @param message A text message concerning blog article publishing 11228 + *### 11229 +#macro(publishMessageBox $message) 11230 +(% class=~"plainmessage publish-message~" ~%)((($services.icon.render('world') $message))) 11231 +#end 11232 +#** 11233 + * Displays a message box with ~"show/hide~" icon. 11234 + * 11235 + * @param message A text message concerning blog article hiding 11236 + *### 11237 +#macro(hideMessageBox $message) 11238 +(% class=~"plainmessage hide-message~" ~%)((($services.icon.render('unlock') $message))) 11239 +#end 11240 +## 11241 +## 11242 +## 11243 +#** 11244 + * Determine the requested week, for using in a weekly-indexed blog. The relevant request parameters are 11245 + * <tt>year</tt> and <tt>week</tt>. By default, the current week is used. 11246 + * 11247 + * @param monthDate The resulting week, a JODATime MutableDateTime. 11248 + *### 11249 +#macro(getRequestedWeek $weekDate) 11250 + #set ($weekDate = $NULL) 11251 + #setVariable (~"$weekDate~" $xwiki.jodatime.mutableDateTime) 11252 + #if(~"$!\{request.year}~" != '') 11253 + #set ($discard = $weekDate.setYear($numbertool.toNumber($request.year).intValue())) 11254 + #end 11255 + #if(~"$!\{request.week}~" != '') 11256 + #set ($discard = $weekDate.setWeekOfWeekyear($numbertool.toNumber($request.week).intValue())) 11257 + #end 11258 +#end 11259 +## 11260 +## 11261 +## 11262 +#** 11263 + * Determine the requested month, for using in a monthly-indexed blog. The relevant request parameters are 11264 + * <tt>year</tt> and <tt>month</tt>. By default, the current month is used. 11265 + * 11266 + * @param monthDate The resulting month, a JODATime MutableDateTime. 11267 + *### 11268 +#macro(getRequestedMonth $monthDate) 11269 + #set ($monthDate = $NULL) 11270 + #setVariable (~"$monthDate~" $xwiki.jodatime.mutableDateTime) 11271 + #if(~"$!\{request.year}~" != '') 11272 + #set ($discard = $monthDate.setYear($numbertool.toNumber($request.year).intValue())) 11273 + #end 11274 + #if(~"$!\{request.month}~" != '') 11275 + #set ($discard = $monthDate.setMonthOfYear($numbertool.toNumber($request.month).intValue())) 11276 + #end 11277 +#end 11278 +## 11279 +## 11280 +## 11281 +#** 11282 + * Retrieve a blog property (title, display type, etc). 11283 + * 11284 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 11285 + * @param propertyName The name of the property to be retrieved. One of the <tt>Blog.BlogClass</tt>'s properties. 11286 + * @param defaultValue The default value to use in case the blog object does not define one. 11287 + * @param propertyValue The resulting value. 11288 + *### 11289 +#macro(getBlogProperty $blogDoc $propertyName $defaultValue $propertyValue) 11290 + #set($result = ~"$!\{blogDoc.getObject($\{blogClassname}).getProperty($propertyName).value}~") 11291 + #if($result == '') 11292 + #set($result = $defaultValue) 11293 + #end 11294 + #set ($propertyValue = $NULL) 11295 + #setVariable (~"$propertyValue~" $result) 11296 +#end 11297 + 11298 +#** 11299 + * If an error occurs when executing an action, set a specific response status and display an error message. 11300 + * 11301 + * @param status The response status. 11302 + * @param text The user readable error to be displayed. Can be a translation key. 11303 + * @param parameters The parameters to use when decoding the translation key. 11304 + *### 11305 +#macro(blog__actionResponseError $status $text $parameters) 11306 + $response.setStatus($status) 11307 + #if($request.ajax) 11308 + $services.localization.render($text, $!parameters) 11309 + #else 11310 + \{\{error}}$services.localization.render($text, $!parameters)\{\{/error}} 11311 + #end 11312 +#end 11313 +## 11314 +## 11315 +## 11316 +#** 11317 + * Check if a blog is the Default blog (The one in the 'Blog' space). 11318 + * 11319 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 11320 + * @param isDefault The resulting boolean. 11321 + *### 11322 +#macro(isDefaultBlog $blogDoc $isDefault) 11323 + #set ($result = false) 11324 + #if ($blogDoc.space == 'Blog') 11325 + #set ($result = true) 11326 + #end 11327 + #setVariable(~"$isDefault~" $result) 11328 +#end 11329 +## 11330 +## 11331 +## 11332 +#** 11333 + * Retrieve the blog posts location (space). 11334 + * 11335 + * @param blogSpace The blog space. It should contain a document with a <tt>Blog.BlogClass</tt> object 11336 + * @param postsLocation The resulting location. 11337 + *### 11338 +#macro(getBlogPostsLocation $blogSpace $postsLocation) 11339 + #getBlogDocument($blogSpace $blogDoc) 11340 + #getBlogProperty($blogDoc 'postsLocation' $blogDoc.space $result) 11341 + #set ($postsLocation = $NULL) 11342 + #setVariable (~"$postsLocation~" $result) 11343 +#end 11344 +## 11345 +## 11346 +## 11347 +#** 11348 + * Retrieve the blog categories location (space). 11349 + * 11350 + * @param blogSpace The blog space. It should contain a document with a <tt>Blog.BlogClass</tt> object 11351 + * @param categoriesLocation The resulting location. 11352 + *### 11353 +#macro(getBlogCategoriesLocation $blogSpace $categoriesLocation) 11354 + #getBlogDocument($blogSpace $blogDoc) 11355 + #getBlogProperty($blogDoc 'categoriesLocation' 'Blog' $result) 11356 + #set ($postsLocation = $NULL) 11357 + #setVariable (~"$categoriesLocation~" $result) 11358 +#end 11359 +###** 11360 + * Return the Query for selecting the blog posts based on the context where the posts are displayed. 11361 + * for example there is 4 different panel contexts: 11362 + * aBlog.aPost or aBlog.WebHome 11363 + * aCategorySpace.aCategory 11364 + * aCategorySpace.WebHome 11365 + * Blog.aPost or Blog.WebHome 11366 + * 11367 + * @param query The query for selecting blog entries. 11368 + * @param queryParams The parameters to bind with the generated query. 11369 + * @param targetDoc The document in which the articles will be displayed. 11370 + *### 11371 +#macro(getAllBlogPostsQueryBasedOnDisplayContext $targetDoc $query $queryParams) 11372 + #set ($query = $NULL) 11373 + #set ($queryParams = $NULL) 11374 + #getCategoryAllBlogPostsQuery($resultQuery) 11375 + #set ($resultQueryParams = \{}) 11376 + #set($sanitizedSpace = $targetDoc.space.replaceAll('([%_!])', '!$1').concat('.%')) 11377 + #if ($targetDoc.space == $defaultBlogSpace && !$targetDoc.getObject($blogCategoryClassname)) 11378 + ## Get all posts that are in the default blog space('Blog') or in category 'Blog.%' 11379 + #set ($discard = $resultQueryParams.put('space', $targetDoc.space)) 11380 + #set ($discard = $resultQueryParams.put('catList', $sanitizedSpace)) 11381 + #set($resultQuery = ~"$\{resultQuery} and (doc.space = :space or catList like :catList escape '!')~") 11382 + #elseif($targetDoc.getObject($blogCategoryClassname)) 11383 + ## Get all posts that are in a category aCategorySpace.aCategory 11384 + #set ($discard = $resultQueryParams.put('catList', $targetDoc.fullName)) 11385 + #set($resultQuery = ~"$\{resultQuery} and (catList=:catList)~") 11386 + #elseif($targetDoc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 11387 + ## Get all posts that are in a category aCategorySpace.% 11388 + #set ($discard = $resultQueryParams.put('catList', $sanitizedSpace)) 11389 + ## Exclude incategorized posts 11390 + #set ($excludedCategory = ~"$\{targetDoc.space}.WebHome~") 11391 + #if ($targetDoc.space == $defaultBlogSpace) 11392 + #set ($excludedCategory = ~"Blog.Categories.WebHome~") 11393 + #end 11394 + #set ($discard = $resultQueryParams.put('excludedCategory', $excludedCategory)) 11395 + #set($resultQuery = ~"$\{resultQuery} and (catList<>:excludedCategory) and (catList like :catList escape '!')~") 11396 + #else 11397 + ## Get all posts in blog space aBlog 11398 + #getAllBlogPostsQuery($resultQuery) 11399 + #getBlogPostsLocation($targetDoc.space $postsLocation) 11400 + #set ($discard = $resultQueryParams.put('space', $postsLocation)) 11401 + #set($resultQuery = ~"$\{resultQuery} and doc.space = :space~") 11402 + #end 11403 + #setVariable(~"$query~" $resultQuery) 11404 + #setVariable(~"$queryParams~" $resultQueryParams) 11405 +#end 11406 +## 11407 +## 11408 +## 11409 +###** 11410 + * Display blog posts based on the context where the posts are displayed. 11411 + * for example there is 4 different panel contexts: 11412 + * aBlog.aPost or aBlog.WebHome 11413 + * aCategorySpace.aCategory 11414 + * aCategorySpace.WebHome 11415 + * Blog.aPost or Blog.WebHome 11416 + * 11417 + * @param targetDoc The document in which the articles will be displayed. 11418 + * @param postsVisiblity The visibility of the posts to display: recent, unpublished, ... 11419 + * @param layout Layout of the the posts to display 11420 + * @param layoutParams additional layout parameters, it is a string like 'param1=val1|...|paramN=valueN' 11421 + * @param limit the number of posts to display 11422 + *### 11423 +#macro(displayBlogPostsBasedOnContext $targetDoc $postsVisiblity $layout $layoutParams $limit) 11424 + #if ($postLayout == 'full') 11425 + #set ($macro.paginated = 'yes') 11426 + #end 11427 + #if ($targetDoc.space == $defaultBlogSpace && !$targetDoc.getObject($blogCategoryClassname)) 11428 + ## Display all posts that are in the default blog space('Blog') or in category 'Blog.%' 11429 + #getBlogPostsLayout($xwiki.getDocument(~"$\{defaultBlogSpace}.WebHome~") $postsLayout) 11430 + #if (~"$!layout~" == '') 11431 + #set ($layout = $postsLayout) 11432 + #end 11433 + #if ($postsVisiblity == 'recent') 11434 + \{\{blogpostlist blog=~"$\{defaultBlogSpace.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')}.WebHome~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 11435 + #elseif($postsVisiblity == 'unpublished') 11436 + \{\{blogpostlist blog=~"$\{defaultBlogSpace.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')}.WebHome~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 11437 + #end 11438 + #elseif($targetDoc.getObject($blogCategoryClassname)) 11439 + ## Display all posts that are in a category aCategorySpace.aCategory 11440 + #getBlogDocumentForCategoriesSpace($targetDoc.space $blogDoc) 11441 + #getBlogPostsLayout($blogDoc $postsLayout) 11442 + #if (~"$!layout~" == '') 11443 + #set ($layout = $postsLayout) 11444 + #end 11445 + #if ($postsVisiblity == 'recent') 11446 + \{\{blogpostlist category=~"$targetDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 11447 + #elseif($postsVisiblity == 'unpublished') 11448 + \{\{blogpostlist category=~"$targetDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 11449 + #end 11450 + #elseif($targetDoc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 11451 + ## Display all posts that are in a category aCategorySpace.% 11452 + #getBlogDocumentForCategoriesSpace($targetDoc.space $blogDoc) 11453 + #getBlogPostsLayout($blogDoc $postsLayout) 11454 + #if (~"$!layout~" == '') 11455 + #set ($layout = $postsLayout) 11456 + #end 11457 + #if ($postsVisiblity == 'recent') 11458 + \{\{blogpostlist category=~"$targetDoc.space.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 11459 + #elseif($postsVisiblity == 'unpublished') 11460 + \{\{blogpostlist category=~"$targetDoc.space.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 11461 + #end 11462 + #else 11463 + ## Display all posts in blog space aBlog 11464 + #getBlogDocument($targetDoc.space $blogDoc) 11465 + #getBlogPostsLayout($blogDoc $postsLayout) 11466 + #if (~"$!layout~" == '') 11467 + #set ($layout = $postsLayout) 11468 + #end 11469 + #if ($postsVisiblity == 'recent') 11470 + \{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 11471 + #elseif($postsVisiblity == 'unpublished') 11472 + \{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 11473 + #end 11474 + #end 11475 +#end 11476 +## 11477 +## 11478 +## 11479 +#** 11480 + * Bind parameters to a query object. 11481 + * 11482 + * @param queryObj The query object (org.xwiki.query.internal.ScriptQuery object) 11483 + * @param queryParams the query parameters. 11484 + *### 11485 +#macro(bindQueryParameters $queryObj $queryParams) 11486 + #set ($output = $queryObj) 11487 + #foreach( $key in $queryParams.keySet() ) 11488 + #set($output = $queryObj.bindValue($key, $queryParams.get($key))) 11489 + #end 11490 + #setVariable(~"$queryObj~" $output) 11491 +#end 11492 +## 11493 +## 11494 +## 11495 +#** 11496 + * Determines the blogposts layout: Default layout (Calendar), Image thumbnail 11497 + * 11498 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>postsLayout</tt> 11499 + * property set. 11500 + * @param postsLayout The resulting string. If the blog object does not define anything, it is considered full (default). 11501 + *### 11502 +#macro(getBlogPostsLayout $blogDoc $postsLayout) 11503 + #getBlogProperty($blogDoc 'postsLayout' 'full' $res) 11504 + #set ($postsLayout = $NULL) 11505 + #setVariable (~"$postsLayout~" $res) 11506 +#end 11507 +## 11508 +## 11509 +## 11510 +#** 11511 + * Retrieve the blog document of a categories space, which is the blog descriptor that have its categoriesLocation property set to the categories space 11512 + * 11513 + * @param categoriesSpace A <tt>String</tt>, the name of the space. 11514 + * @param blogDoc The resulting XDocument. 11515 + *### 11516 +#macro(getBlogDocumentForCategoriesSpace $categoriesSpace $blogDoc) 11517 + #set($macro.blogDocs = $services.query.hql(~", BaseObject obj, StringProperty as categoriesLocationProp where obj.name = doc.fullName and obj.className = '$blogClassname' and obj.id=categoriesLocationProp.id.id and categoriesLocationProp.id.name='categoriesLocation' and categoriesLocationProp.value=:categoriesLocation order by doc.creationDate~").setLimit(1).setOffset(0).bindValue('categoriesLocation', $categoriesSpace).execute()) 11518 + #if($macro.blogDocs.size() > 0 && $categoriesSpace != $defaultBlogSpace) 11519 + #set($macro.result = $xwiki.getDocument($macro.blogDocs.get(0))) 11520 + #else 11521 + ## Fallback to Blog.WebHome, the default blog 11522 + #set($macro.result = $xwiki.getDocument('Blog.WebHome')) 11523 + #end 11524 + #set ($blogDoc = $NULL) 11525 + #setVariable (~"$blogDoc~" $macro.result) 11526 +#end" %) 11527 +((( 11528 +(% class="macro-placeholder hidden" %) 11529 +((( 11530 +macro:velocity 11531 +))) 11532 +))) 11533 +))) 11534 + 11535 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogPostLayoutMacros~"" %) 11536 +((( 11537 +(% class="macro-placeholder hidden" %) 11538 +((( 11539 +macro:include 11540 +))) 11541 + 11542 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|#** 11543 + * Extract the layout parameters from a string. 11544 + * 11545 + * @param layoutParamsString The string representation of the layout parameters. 11546 + * It should contain a String following this format ~"paramName1=Value1|paramName2=Value2|...|paramNameK=ValueK~" 11547 + * @param layoutsParameters The resulting layout parameters Map. 11548 + *### 11549 +#macro(extractLayoutParametersFromString $layoutParamsString $layoutsParameters) 11550 + #set ($layoutsParameters = $NULL) 11551 + #set ($macro.layoutParams = \{}) 11552 + #if (~"$!layoutParamsString~" != '') 11553 + #set ($macro.paramsArr = $layoutParamsString.split('\\|')) 11554 + #foreach ($item in $macro.paramsArr) 11555 + #set ($itemSplit = $item.split('=')) 11556 + #if ($itemSplit.size() == 2) 11557 + #set ($discard = $macro.layoutParams.put($itemSplit[0].trim(), $itemSplit[1].trim())) 11558 + #end 11559 + #end 11560 + #end 11561 + #setVariable(~"$layoutsParameters~" $macro.layoutParams) 11562 +#end" %) 11563 +((( 11564 +(% class="macro-placeholder hidden" %) 11565 +((( 11566 +macro:velocity 11567 +))) 11568 +))) 11569 +))) 11570 + 11571 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-| #macro(displaySmallPost $pDoc $pObj) 11572 + #initLayoutVars($pDoc $pObj) 11573 + <div class=~"col-xs-12 col-sm-6 next_blog~"> 11574 + <a href=~"$pDoc.uRL~" class=~"thumbnail~"> 11575 + <div class=~"row~"> 11576 + <div class=~"col-xs-4~"> 11577 + #if ($imageAtt) 11578 + <img src=~"$escapetool.xml($pDoc.getAttachmentURL($pObj.getValue('image'),'download',$imgQs))~" /> 11579 + #end 11580 + </div> 11581 + <div class=~"col-xs-8 art_det~"> 11582 + <p class=~"text-left~"> 11583 + #if($displayTitle)$!postTitle #end 11584 + <br/><span class=~"date_info~"> $!dateStr </span> 11585 + </p> 11586 + #displayPostDetails($pDoc) 11587 + </div> 11588 + </div> 11589 + </a> 11590 + </div> 11591 + #end 11592 + ## 11593 + ## 11594 + ## 11595 + #macro(displayPinnedPost $pDoc $pObj) 11596 + #initLayoutVars($pDoc $pObj) 11597 + <div class=~"col-xs-12 col-sm-4 col-md-4 blog_pin~"> 11598 + <a href=~"$pDoc.uRL~" class=~"all-post-link~"> 11599 + <div class=~"thumbnail~"> 11600 + #if ($displayTitle && $displayTitleFirstOnPinnedPosts) 11601 + <h5 class=~"title-top~">$!postTitle <br/> <span class=~"date_info~"> $!dateStr </span> </h5> 11602 + #end 11603 + #if ($imageAtt) 11604 + <img src=~"$escapetool.xml($pDoc.getAttachmentURL($pObj.getValue('image'),'download',$imgQs))~" /> 11605 + #end 11606 + <div class=~"caption~"> 11607 + #if ($displayTitle && !$displayTitleFirstOnPinnedPosts) 11608 + <h5 class=~"text-left~">$!postTitle <br/> <span class=~"date_info~"> $!dateStr </span> </h5> 11609 + #end 11610 + #if ($displaySummaryOnPinnedPosts) 11611 + <div class=~"text-left post-summary~"> 11612 + #set ($postContent = $pObj.getProperty('extract').value) 11613 + $!pDoc.getRenderedContent($postContent, $pDoc.syntax.toIdString()) 11614 + </div> 11615 + #end 11616 + #displayPostDetails($pDoc) 11617 + </div> 11618 + </div> 11619 + </a> 11620 + </div> 11621 + #end 11622 + ## 11623 + ## 11624 + ## 11625 + #macro(formatPostDate $pDoc $pObj) 11626 + #set ($formattedDate = $NULL) 11627 + #set($dateStr = $datetool.format('medium', $postDate, $xcontext.locale)) 11628 + #if (~"$!dateStr~" != '') 11629 + #set ($dateArr = $dateStr.split(' ')) 11630 + #if ($dateArr.size() > 3) 11631 + #set ($dateStr = ~"~") 11632 + #foreach($s in $dateArr.subList(0, 3)) 11633 + #set ($dateStr = ~"$\{dateStr}$\{s} ~") 11634 + #end 11635 + #end 11636 + #end 11637 + #setVariable(~"$formattedDate~" $dateStr) 11638 + #end 11639 + ## 11640 + ## 11641 + ## 11642 + #macro(displayPostDetails $pDoc) 11643 + <div class=~"row post-details~"> 11644 + <div class=~"col-xs-8 detail~"> 11645 + #if ($services.like.displayButton($pDoc.documentReference)) 11646 + #set ($optLikeRecord = $services.like.getLikes($pDoc.documentReference)) 11647 + ## Retrieve the likes number in XWiki 12.9+ 11648 + #set ($likeNumber = $optLikeRecord.get()) 11649 + #if (!$stringtool.isNumeric($likeNumber.toString())) 11650 + ## Retrieve the likes number in XWiki versions greater than 12.7 and lower than 12.9. 11651 + #set ($likeNumber = $optLikeRecord.get().likeNumber) 11652 + #end 11653 + #if ($stringtool.isNumeric($likeNumber.toString())) 11654 + <div class=~"post-likes btn btn-default disabled badge~" 11655 + title=~"$escapetool.xml($services.localization.render('like.button.title', [$likeNumber]))~"> 11656 + $services.icon.renderHTML('heart') <span class=~"like-number~">$likeNumber</span> 11657 + </div> 11658 + #end 11659 + #elseif ($services.ratings) 11660 + #set ($ratingsConfigDoc = $services.ratings.getConfigurationDocument($pDoc.documentReference)) 11661 + #if ($ratingsConfigDoc.getValue('displayRatings') == 1) 11662 + <ul class=~"pull-left note list-inline~"> 11663 + #foreach ($x in [1..5]) 11664 + #set ($cls = ~"~") 11665 + #if ($x > $averageVote) 11666 + #set ($cls = ~"-o~") 11667 + #end 11668 + <li><span class=~"fa fa-star$\{cls}~"></span></li> 11669 + #end 11670 + </ul> 11671 + #end 11672 + #end 11673 + </div> 11674 + #if ($showPostComments) 11675 + <div class=~"col-xs-4 com_det~"> 11676 + <p class=~"text-right~">$services.icon.renderHTML('comments') ($!nbComments)</p> 11677 + </div> 11678 + #end 11679 + </div> 11680 + #end 11681 + ## 11682 + ## 11683 + ## 11684 + #macro(initLayoutVars $pDoc $pObj) 11685 + #set ($postTitle = $pDoc.display('title', 'view', $pObj)) 11686 + #set ($imageAtt = $pDoc.getAttachment($pObj.getValue('image'))) 11687 + #isPublished($pObj $isPublished) 11688 + #isHidden($pObj $isHidden) 11689 + #getEntryDate($pDoc $pObj $postDate) 11690 + #formatPostDate($postDate $dateStr) 11691 + #set ($averageVote = $mathtool.round($services.ratings.getAverageRating($pDoc.documentReference).getAverageVote())) 11692 + #set ($nbComments = $pDoc.getComments().size()) 11693 + #set ($showPostComments = true) 11694 + #if ($xwiki.getSpacePreferenceFor('showcomments', $pDoc.documentReference.lastSpaceReference, '1') == '0') 11695 + #set ($showPostComments = false) 11696 + #end 11697 + #end 11698 + ## 11699 + ## 11700 + ## 11701 + #macro(displayEditPinnedPostsButton) 11702 + #if (~"$!macroAdditionalParams.get('postIndex')~" == '0' && $services.security.authorization.hasAccess('edit', $pinnedPostsSourceDoc.fullName)) 11703 + #set ($discard = $xwiki.ssx.use('Blog.PinnedPostsSheet')) 11704 + #set ($discard = $xwiki.jsx.use('Blog.PinnedPostsSheet')) 11705 + #set ($editPinnedPostsURL = $xwiki.getURL($pinnedPostsSourceDoc.fullName, 'get', 'sheet=Blog.PinnedPostsSheet')) 11706 + #set ($newPinnedPostsObjectURL = $xwiki.getURL($pinnedPostsSourceDoc.fullName, 'objectadd', ~"classname=Blog.PinnedPostsClass&form_token=$!\{services.csrf.getToken()}~")) 11707 + <div class=~"edit-pinned-posts-container~"> 11708 + <div class=~"col-xs-12 col-sm-12 col-md-12 text-right edit-pinned-posts~"> 11709 + <a href=~"javascript:;~" class=~"editPinnedPosts~"> 11710 + <span class=~"glyphicon glyphicon-edit~" aria-hidden=~"true~"></span> $services.localization.render('blog.post.layout.cards.pinnedposts.edit') 11711 + </a> 11712 + <input type=~"hidden~" id=~"editPinnedPostsURL~" value=~"$escapetool.xml($editPinnedPostsURL)~"/> 11713 + <input type=~"hidden~" id=~"newPinnedPostsObjectURL~" value=~"$escapetool.xml($newPinnedPostsObjectURL)~"/> 11714 + </div> 11715 + </div> 11716 + #if (~"$!pinnedPostsObj~" == '') 11717 + <input type=~"hidden~" name=~"createPinnedPostsObject~" value=~"1~"/> 11718 + #end 11719 + #end 11720 + #end" %) 11721 +((( 11722 +(% class="macro-placeholder hidden" %) 11723 +((( 11724 +macro:velocity 11725 +))) 11726 +))) 11727 + 11728 +(% data-macro="startmacro:velocity|-||-| #if ($services.security.authorization.hasAccess('view', $xcontext.macro.params.reference)) 11729 + #set ($postDoc = $xwiki.getDocument($xcontext.macro.params.reference)) 11730 + #getEntryObject($postDoc $postObj) 11731 + #if (~"$!postObj~" != '') 11732 + #extractLayoutParametersFromString($xcontext.macro.params.params $macroAdditionalParams) 11733 + ## 11734 + #set ($displayTitle = true) 11735 + #if (~"$!macroAdditionalParams.get('displayTitle')~" == 'false') 11736 + #set ($displayTitle = false) 11737 + #end 11738 + ## 11739 + #set ($displayTitleFirstOnPinnedPosts = false) 11740 + #if (~"$!macroAdditionalParams.get('displayTitleFirstOnPinnedPosts')~" == 'true') 11741 + #set ($displayTitleFirstOnPinnedPosts = true) 11742 + #end 11743 + ## 11744 + #set ($displaySummaryOnPinnedPosts = true) 11745 + #if (~"$!macroAdditionalParams.get('displaySummaryOnPinnedPosts')~" == 'false') 11746 + #set ($displaySummaryOnPinnedPosts = false) 11747 + #end 11748 + ## 11749 + #set ($postIndex = $numbertool.toNumber($macroAdditionalParams.get('postIndex')).intValue()) 11750 + #if ($postIndex == 0) 11751 + #set ($stopBlogPostsDisplay = false) 11752 + #end 11753 + #set ($listResultsCount = $numbertool.toNumber($macroAdditionalParams.get('list-results-count')).intValue()) 11754 + #set($discard = $xwiki.ssx.use('Blog.BlogPostLayoutCards')) 11755 + #set($scaleWidth = 600) 11756 + #set($imgQs=~"width=$scaleWidth~") 11757 + ## Display pinned posts 11758 + ## Get the list of pinned posts 11759 + #set ($pinnedPosts = []) 11760 + #set ($pinnedPostsSourceDoc = $NULL) 11761 + #if (~"$!macroAdditionalParams.get('forceDisplayPinnedPosts')~" == 'true' && ~"$!macroAdditionalParams.get('list-blog')~" != '') 11762 + #set ($pinnedPostsSourceDoc = $xwiki.getDocument($macroAdditionalParams.get('list-blog'))) 11763 + ## Note that the 'list-blog' parameter contains the value of the 'blog' parameter of the blogpostlist macro passed to the current layout macro 11764 + #end 11765 + #if (~"$!pinnedPostsSourceDoc~" != '') 11766 + #set ($pinnedPostsObj = $pinnedPostsSourceDoc.getObject('Blog.PinnedPostsClass')) 11767 + #if (~"$!pinnedPostsObj~" != '') 11768 + #set ($orderedPinnedPostsJSON = $pinnedPostsObj.getProperty('orderedPosts').value) 11769 + #if (~"$!orderedPinnedPostsJSON~" != '') 11770 + #set ($pinnedPosts = $jsontool.parse($orderedPinnedPostsJSON)) 11771 + #else 11772 + #set ($pinnedPosts = $pinnedPostsObj.getProperty('posts').value) 11773 + #end 11774 + #end 11775 + #if ($postIndex == 0 && $pinnedPosts.size() > 0) 11776 + \{\{html clean=~"false~"}} 11777 + #set ($x = 0) 11778 + #set ($showPinnedPostsButton = true) 11779 + #foreach ($pinnedPost in $pinnedPosts) 11780 + #if ($x == 0) 11781 + <div class=~"row flex-container~"> 11782 + #if ($showPinnedPostsButton) 11783 + #displayEditPinnedPostsButton() 11784 + #set($showPinnedPostsButton = false) 11785 + #end 11786 + #end 11787 + #set ($pinnedPostDoc = $xwiki.getDocument($pinnedPost)) 11788 + #getEntryObject($pinnedPostDoc $pinnedPostObj) 11789 + #if (~"$!pinnedPostObj~" != '') 11790 + #displayPinnedPost($pinnedPostDoc $pinnedPostObj) 11791 + #end 11792 + #set ($x = $mathtool.add($x, 1)) 11793 + #if ($x == 3) 11794 + #set ($x = 0) 11795 + </div> 11796 + #end 11797 + #end 11798 + #if ($mathtool.mod($x, 3) != 0) 11799 + </div> 11800 + #end 11801 + \{\{/html}} 11802 + 11803 + ## If the first post is a pinned post : this means that all the posts are pinned 11804 + ## In this case, avoid displaying the posts again after the pinned posts section. 11805 + #if ($pinnedPosts.contains($xcontext.macro.params.reference)) 11806 + #set ($stopBlogPostsDisplay = true) 11807 + #end 11808 + #elseif($postIndex == 0 && $pinnedPosts.size() == 0) 11809 + 11810 + \{\{html}} 11811 + <div class=~"row no-pinnded-posts~"> 11812 + #displayEditPinnedPostsButton() 11813 + </div> 11814 + \{\{/html}} 11815 + 11816 + #end 11817 + #end 11818 + #if (!$stopBlogPostsDisplay) 11819 + \{\{html clean=~"false~"}} 11820 + #if (~"$!nbDisplayedPosts~" == '' || $postIndex == 0) 11821 + #set ($nbDisplayedPosts = 0) 11822 + #end 11823 + #if($mathtool.mod($nbDisplayedPosts, 2) == 0) 11824 + #if ($nbDisplayedPosts != 0) 11825 + </div> 11826 + #set ($lastHtmlTag = 'c') 11827 + #end 11828 + #if ($nbDisplayedPosts == $mathtool.sub($listResultsCount, 1)) 11829 + <div class=~"row~"> 11830 + #set ($lastHtmlTag = 'o') 11831 + #set ($lastRowClass = ~"~") 11832 + #else 11833 + <div class=~"row flex-container~"> 11834 + #set ($lastHtmlTag = 'o') 11835 + #set ($lastRowClass = ~"flex~") 11836 + #end 11837 + #end 11838 + #displaySmallPost($postDoc $postObj) 11839 + #if ($nbDisplayedPosts == $mathtool.sub($listResultsCount, 1)) 11840 + </div> 11841 + #set ($lastHtmlTag = 'c') 11842 + #end 11843 + #set ($nbDisplayedPosts = $mathtool.add($nbDisplayedPosts, 1)) 11844 + #if ($postIndex == $mathtool.sub($listResultsCount, 1)) 11845 + #if (~"$!lastHtmlTag~" == 'o') 11846 + #if ($mathtool.mod($nbDisplayedPosts, 2) == 1 && $lastRowClass == ~"flex~") 11847 + <div class=~"col-xs-12 col-sm-6~"></div> 11848 + #end 11849 + </div> 11850 + #end 11851 + #end 11852 + \{\{/html}} 11853 + #end 11854 + #else 11855 + \{\{error}}$services.localization.render('blog.blogpostlayout.notpost', [$xcontext.macro.params.reference])\{\{/error}} 11856 + #end 11857 + #else 11858 + \{\{error}}$services.localization.render('blog.blogpostlayout.post_view_not_allowed', [$xcontext.macro.params.reference])\{\{/error}} 11859 + #end" class="macro hidden macro-placeholder" %)macro:velocity(% data-macro="startmacro:html|-|clean=~"false~"|-|</div> 11860 +<div class=~"row flex-container~"> 11861 +<div class=~"col-xs-12 col-sm-6 next_blog~"> 11862 +<a href=~"/Nieuws/Aankondiging%20update%20Syntess%206.9.0181~" class=~"thumbnail~"> 11863 +<div class=~"row~"> 11864 +<div class=~"col-xs-4~"> 11865 +<img src=~"/download/Nieuws/Aankondiging%20update%20Syntess%206.9.0181/05-12-37-113_512.gif?width=600&rev=1.1~" /> 11866 +</div> 11867 +<div class=~"col-xs-8 art_det~"> 11868 +<p class=~"text-left~"> 11869 +Aankondiging update Syntess 6.9.0181 <br/><span class=~"date_info~"> Dec 12, 2023, </span> 11870 +</p> 11871 +<div class=~"row post-details~"> 11872 +<div class=~"col-xs-8 detail~"> 11873 +<div class=~"post-likes btn btn-default disabled badge~" 11874 +title=~"Number of likes on this page: 0~"> 11875 +<span class=~"fa fa-heart~"></span> <span class=~"like-number~">0</span> 11876 +</div> 11877 +</div> 11878 +<div class=~"col-xs-4 com_det~"> 11879 +<p class=~"text-right~"><span class=~"fa fa-comments~"></span> (0)</p> 11880 +</div> 11881 +</div> 11882 +</div> 11883 +</div> 11884 +</a> 11885 +</div>" class="macro macro hidden macro-placeholder" %)macro:html 11886 +))) 11887 +))) 11888 + 11889 +(% class="row flex-container" %) 11890 +((( 11891 +(% class="col-xs-12 col-sm-6 next_blog" %) 11892 +((( 11893 +(% class="row" %) 11894 +((( 11895 +(% class="col-xs-4" %) 11896 +((( 11897 +[[~[~[image:/download/Nieuws/Aankondiging%20update%20Syntess%206.9.0181/05-12-37-113_512.gif?width=600&rev=1.1~]~]>>path:/Nieuws/Aankondiging%20update%20Syntess%206.9.0181||class="thumbnail"]] 11898 +))) 11899 + 11900 +(% class="col-xs-8 art_det" %) 11901 +((( 11902 +(% class="text-left" %) 11903 +[[Aankondiging update Syntess 6.9.0181 11904 +(% class="date_info" %) Dec 12, 2023,>>path:/Nieuws/Aankondiging%20update%20Syntess%206.9.0181||class="thumbnail"]] 11905 + 11906 +(% class="row post-details" %) 11907 +((( 11908 +(% class="col-xs-8 detail" %) 11909 +((( 11910 +(% class="post-likes btn btn-default disabled badge" title="Number of likes on this page: 0" %) 11911 +((( 11912 +[[(% class="like-number" %)0>>path:/Nieuws/Aankondiging%20update%20Syntess%206.9.0181||class="thumbnail"]] 11913 +))) 11914 +))) 11915 + 11916 +(% class="col-xs-4 com_det" %) 11917 +((( 11918 +(% class="text-right" %) 11919 +[[(0)>>path:/Nieuws/Aankondiging%20update%20Syntess%206.9.0181||class="thumbnail"]] 11920 +))) 11921 +))) 11922 +))) 11923 +))) 11924 +))) 11925 + 11926 +(% class="macro" data-macro="startmacro:blogPostLayoutCards|-|reference=~"Nieuws.Nieuwsbrief oktober 2023~" params=~"displayTitle=true|useSummary=true|list-limit=10|list-layout=cards|list-blog=Nieuws.WebHome|list-paginated=no|list-results-count=10|postIndex=0|previousPostReference=|postIndex=1|previousPostReference=Nieuws.Nieuwsbrief januari 2024|postIndex=2|previousPostReference=Nieuws.Audit BMI en BORG|postIndex=3|previousPostReference=Nieuws.Nieuwsbrief december 2023|postIndex=4|previousPostReference=Nieuws.Nieuwsbrief november 2023|postIndex=5|previousPostReference=Nieuws.Feestelijke afsluiting, keuring materieel en tellen voorraden|postIndex=6|previousPostReference=Nieuws.RI&E 2023|postIndex=7|previousPostReference=Nieuws.Aankondiging update Syntess 6\\\\.9\\\\.0181~"" %) 11927 +((( 11928 +(% class="macro-placeholder hidden" %) 11929 +((( 11930 +macro:blogPostLayoutCards 11931 +))) 11932 + 11933 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogCode~"" %) 11934 +((( 11935 +(% class="macro-placeholder hidden" %) 11936 +((( 11937 +macro:include 11938 +))) 11939 + 11940 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogParameters~"" %) 11941 +((( 11942 +(% class="macro-placeholder hidden" %) 11943 +((( 11944 +macro:include 11945 +))) 11946 + 11947 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|## Blog 11948 +#set($blogClassname = 'Blog.BlogClass') 11949 +#set($blogTemplate = 'Blog.BlogTemplate') 11950 +#set($blogSheet = 'Blog.BlogSheet') 11951 +## Blog entries 11952 +#set($blogPostClassname = 'Blog.BlogPostClass') 11953 +#set($blogPostTemplate = 'Blog.BlogPostTemplate') 11954 +#set($blogPostSheet = 'Blog.BlogPostSheet') 11955 +#set($blogPostObjectNumber = $xwiki.getDocument($blogPostTemplate).getObject($blogPostClassname).number) 11956 +#set($oldArticleClassname = 'XWiki.ArticleClass') 11957 +## Categories 11958 +#set($blogCategoryClassname = 'Blog.CategoryClass') 11959 +#set($blogCategoryTemplate = 'Blog.CategoryTemplate') 11960 +#set($blogCategorySheet = 'Blog.CategorySheet') 11961 +#set($blogCategoriesSheet = 'Blog.CategoriesSheet') 11962 +#set($oldBlogCategoryClassname = 'Blog.Categories') 11963 +#set($defaultCategoryParent = 'Blog.Categories') 11964 +## Style 11965 +#set($blogStyleDocumentName = 'Blog.BlogStyle') 11966 +#set($blogStyleDocument = $xwiki.getDocument($blogStyleDocumentName)) 11967 +## Clientside scripts 11968 +#set($blogScriptsDocumentName = 'Blog.BlogScripts') 11969 +#set($blogPublisher = $xwiki.getDocument('Blog.Publisher')) 11970 +## Misc 11971 +#set($thisURL = $doc.getURL($xcontext.action, $request.queryString)) 11972 +#set($isBlogPost = $doc.getObject($blogPostClassname)) 11973 +#set($defaultBlogSpace = 'Blog') 11974 +#set($isCategoryPage = $doc.getObject($blogCategoryClassname)) 11975 +#set($isCategoriesHomePage = $doc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 11976 +## 11977 +## 11978 +## 11979 +#** 11980 + * Displays an image, taken from the blog style document. 11981 + * 11982 + * @param $imgName The name of the icon from icons set to use. 11983 + *# 11984 +#macro(toolImage $imgName)(% class=~"icon-manage~"~%)$services.icon.render($imgName)(%~%)#end" %) 11985 +((( 11986 +(% class="macro-placeholder hidden" %) 11987 +((( 11988 +macro:velocity 11989 +))) 11990 +))) 11991 +))) 11992 + 11993 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|## 11994 +## 11995 +## 11996 +## Import the blog skin and javascripts. 11997 +$!xwiki.ssx.use($blogStyleDocumentName)## 11998 +$!xwiki.jsx.use($blogScriptsDocumentName)## 11999 +## 12000 +## import the hierarchy for the path the blog post footer (in displayEntryBlogLocation) 12001 +#template('hierarchy_macros.vm')## 12002 +## 12003 +## 12004 +#** 12005 + * Prints a blog. This is the main macro used in the BlogSheet. 12006 + * 12007 + * @param blogDoc the XDocument holding the blog definition object. 12008 + *### 12009 +#macro(printBlog $blogDoc) 12010 + \{\{include reference='Blog.CreatePost'/}} 12011 + 12012 + ## Use the blogPostList macro to display the blogposts 12013 + ## The blogPostList is used only in case of 'all' or 'paged' blog display type because the blogPostList macro 12014 + ## do not support FTM the monthly and weekly blog display types 12015 + #getBlogDisplayType($blogDoc $displayType) 12016 + #if ($displayType == 'weekly' || $displayType == 'monthly') 12017 + #getBlogEntries($blogDoc $entries) 12018 + #displayBlog($entries 'index' true true) 12019 + #displayNavigationLinks($blogDoc) 12020 + #else 12021 + #getBlogDisplayType($blogDoc $displayType) 12022 + #set ($paginated = 'no') 12023 + #if ($displayType == 'paginated') 12024 + #set ($paginated = 'yes') 12025 + #end 12026 + #getBlogPostsLayout($blogDoc $postsLayout) 12027 + (% class=~"hfeed index~" ~%)(((\{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$paginated~" layout=~"$!postsLayout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" /}}))) 12028 + #end 12029 +#end 12030 +## 12031 +## 12032 +## 12033 +#** 12034 + * Shows blog information. In view mode, the description is printed. In edit mode, allows changing blog settings: title, 12035 + * description, blog type (global or in-space), index display type (fixed size pagination, weekly index, monthly index, 12036 + * all entries). 12037 + * 12038 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 12039 + *### 12040 +#macro(showBlogInfo $blogDoc) 12041 + #if($blogDoc.getObject($blogClassname)) 12042 + ## Keep testing for inline action for backward compatibility with older blogs. 12043 + #if($xcontext.action == 'edit' || $xcontext.action == 'inline') 12044 + #macro(displayProperty $blogDoc $propname) 12045 + ; #displayPropName($xwiki.getClass($blogClassname).get($propname)): 12046 + : $blogDoc.display($propname) 12047 + #end 12048 + #displayProperty($blogDoc 'title') 12049 + #displayProperty($blogDoc 'description') 12050 + #displayProperty($blogDoc 'displayType') 12051 + #displayProperty($blogDoc 'itemsPerPage') 12052 + #displayProperty($blogDoc 'postsLayout') 12053 + #displayProperty($blogDoc 'postsLayoutParameters') 12054 + #else 12055 + $blogDoc.display('description') 12056 + #end 12057 + #elseif($doc.fullName == $blogSheet) 12058 += $services.localization.render('blog.code.blogsheet') = 12059 + \{\{translation key='blog.code.sheetexplanation'/}} 12060 + #else 12061 + \{\{warning}}\{\{translation key='blog.code.notblog'/}}\{\{/warning}} 12062 + #end 12063 +#end 12064 +## 12065 +## 12066 +## 12067 +#** 12068 + * Retrieve the blog document, which usually is either <tt><Space>.WebHome</tt> for whole-spaces blogs, or 12069 + * <tt><Space>.Blog</tt> for in-space blogs. If none of these documents contains a blog object, then the first 12070 + * (alphabetically) document in the target space that contains one is returned. Finally, if no document in the current 12071 + * space contains a blog object, then <tt>Blog.WebHome</tt> is returned as the default blog. 12072 + * 12073 + * @param space A <tt>String</tt>, the name of the space where to search. 12074 + * @param blogDoc The resulting XDocument. 12075 + *### 12076 +#macro(getBlogDocument $space $blogDoc) 12077 + #set ($result = $NULL) 12078 + ## Check if the current space has a WebPreferences page that contains a Blog.EnablePanelsConfigurationClass object. 12079 + ## It is possible to display a blog's panels info in a page that is not inside a blog and in that case we need to 12080 + ## identify the right blog based on a configuration object in a WebPreferences page. 12081 + #set ($spaceReference = $services.model.resolveSpace($space)) 12082 + #set ($preferencesDocRef = $services.model.createDocumentReference('WebPreferences', $spaceReference)) 12083 + #set ($preferencesDoc = $xwiki.getDocument($preferencesDocRef)) 12084 + #set ($preferencesObj = $preferencesDoc.getObject('Blog.EnablePanelsConfigurationClass')) 12085 + #if ($preferencesObj) 12086 + #set ($result = $xwiki.getDocument( $preferencesObj.getValue('blog'))) 12087 + #end 12088 + ## If no special case occurs (like the previous one), identify a blog by checking the current location from its hierarchy. 12089 + #if (~"$!result~" == '') 12090 + ## First, try the Space.WebHome, for a whole-space blog 12091 + #set($result = $xwiki.getDocument(~"$\{space}.WebHome~")) 12092 + #if(!$result.getObject($blogClassname)) 12093 + ## Second, try the Space.Blog document 12094 + #set($result = $xwiki.getDocument(~"$\{space}.Blog~")) 12095 + #if(!$result.getObject($blogClassname)) 12096 + ## Third, try searching for a blog document in the current space 12097 + ## Prevent the query fail when the space contains dots '.' 12098 + #set($blogDocs = $services.query.hql(~", BaseObject obj where doc.space = :space and obj.name = doc.fullName and obj.className = '$blogClassname' order by doc.name~").setLimit(1).setOffset(0).bindValue('space', $space).execute()) 12099 + #if($blogDocs.size() > 0) 12100 + #set($result = $xwiki.getDocument($blogDocs.get(0))) 12101 + #else 12102 + ## Fourth, try searching for a blog document that have the its 'postsLocation' set to the current space 12103 + #set($blogDocs = $services.query.hql(~", BaseObject obj, StringProperty as postsLocationProp where obj.name = doc.fullName and obj.className = '$blogClassname' and obj.id=postsLocationProp.id.id and postsLocationProp.id.name='postsLocation' and postsLocationProp.value=:postsLocation order by doc.name~").setLimit(1).setOffset(0).bindValue('postsLocation', $space).execute()) 12104 + #if($blogDocs.size() > 0) 12105 + #set($result = $xwiki.getDocument($blogDocs.get(0))) 12106 + #else 12107 + ## Last, fallback to Blog.WebHome, the default blog 12108 + #set($result = $xwiki.getDocument('Blog.WebHome')) 12109 + #end 12110 + #end 12111 + #end 12112 + #end 12113 + #end 12114 + #set ($blogDoc = $NULL) 12115 + #setVariable (~"$blogDoc~" $result) 12116 +#end 12117 +## 12118 +## 12119 +## 12120 +#** 12121 + * Retrieve the blog title. 12122 + * 12123 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass<tt> object with the <tt>title</tt> property set. 12124 + * @param title The resulting title. 12125 + *### 12126 +#macro(getBlogTitle $blogDoc $title) 12127 + ## Titles can contain velocity code (most commonly translations), so we should evaluate them. 12128 + #set ($title = $NULL) 12129 + #setVariable (~"$title~" $!blogDoc.displayTitle) 12130 +#end 12131 +## 12132 +## 12133 +## 12134 +#** 12135 + * Retrieve the blog description. 12136 + * 12137 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>description</tt> 12138 + * property set. 12139 + * @param description The resulting description. 12140 + *### 12141 +#macro(getBlogDescription $blogDoc $description) 12142 + #getBlogProperty($blogDoc 'description' '' $result) 12143 + #set ($description = $NULL) 12144 + #setVariable (~"$description~" $result) 12145 +#end 12146 +## 12147 +## 12148 +## 12149 +#** 12150 + * Retrieves a list of entries to be displayed. The entries are either part of the blog's space, or have the blog 12151 + * document set as a parent. The number and range of entries returned (from all those belonging to this blog) depends on 12152 + * the blog display type: paginated (fixed number of entries), weekly (all entries in a week), monthly (all entries in a 12153 + * month), or all. 12154 + * 12155 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 12156 + * @param entries The resulting list of entries to display, a list of XDocument names. 12157 + *### 12158 +#macro(getBlogEntries $blogDoc $entries) 12159 + #if (!$entries) 12160 + #setVariable (~"$entries~" []) 12161 + #end 12162 + #getAllBlogPostsQuery($query) 12163 + #isDefaultBlog($blogDoc $isDefault) 12164 + #set($queryParams = \{}) 12165 + #if ($isDefault) 12166 + #getCategoryAllBlogPostsQuery($query) 12167 + #set($query = ~"$\{query} and (doc.creator = :creator or (isPublished.value = 1 and hidden.value = 0)) and (doc.space = :space or catList like :catList escape '!')~") 12168 + #set($discard = $queryParams.put('creator', $xcontext.user)) 12169 + #set($discard = $queryParams.put('space', $blogDoc.space)) 12170 + #set($sanitizedBlogSpace = $blogDoc.space.replaceAll('([%_!])', '!$1').concat('.%')) 12171 + #set($discard = $queryParams.put('catList', $sanitizedBlogSpace)) 12172 + #else 12173 + #set($query = ~"$\{query} and (doc.space = :space or doc.parent = :parent)~") 12174 + #getBlogPostsLocation($blogDoc.space $blogPostsLocation) 12175 + #set($discard = $queryParams.put('space', $blogPostsLocation)) 12176 + #set($discard = $queryParams.put('parent', $blogDoc.space)) 12177 + #end 12178 + #getBlogDisplayType($blogDoc $displayType) 12179 + #if($displayType == 'weekly') 12180 + #getWeeklyBlogEntries($blogDoc $query $entries $queryParams) 12181 + #elseif($displayType == 'monthly') 12182 + #getMonthlyBlogEntries($blogDoc $query $entries $queryParams) 12183 + #elseif($displayType == 'all') 12184 + #getAllBlogEntries($blogDoc $query $entries $queryParams) 12185 + #else 12186 + #getPagedBlogEntries($blogDoc $query $entries $queryParams) 12187 + #end 12188 +#end 12189 +## 12190 +## 12191 +## 12192 +#** 12193 + * Retrieves a list of entries to be displayed. The entries are taken from a ~"page~" of the blog, a sequence of documents 12194 + * defined by the request parameters <tt>ipp</tt> (items per page) and <tt>page</tt> (the current page). Initially the 12195 + * first page is displayed, with the number of entries defined in the blog object in the <tt>itemsPerPage</tt> property 12196 + * (10 if not defined). 12197 + * 12198 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 12199 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 12200 + * refined to restrict to a given space, or to a given search criteria, etc. 12201 + * @param entries The resulting list of entries to display, a list of XDocument names. 12202 + * @param queryParams The parameters to bind with the query. 12203 + *### 12204 +#macro(getPagedBlogEntries $blogDoc $query $entries $queryParams) 12205 + #if (!$entries) 12206 + #setVariable (~"$entries~" []) 12207 + #end 12208 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 12209 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~")) 12210 + #bindQueryParameters($countQueryObj $queryParams) 12211 + #bindQueryParameters($queryObj $queryParams) 12212 + #set($totalEntries = $countQueryObj.count()) 12213 + #getBlogProperty($blogDoc 'itemsPerPage' '10' $defaultItemsPerPage) 12214 + #set($defaultItemsPerPage = $numbertool.toNumber($defaultItemsPerPage).intValue()) 12215 + ## This macro is defined in the default macros.vm library. It also sets $itemsPerPage and $startAt. 12216 + #preparePagedViewParams($totalEntries $defaultItemsPerPage) 12217 + #set($discard = $entries.addAll($queryObj.setLimit($itemsPerPage).setOffset($startAt).addFilter(~"unique~").execute())) 12218 +#end 12219 +## 12220 +## 12221 +## 12222 +#** 12223 + * Retrieves a list of entries to be displayed. The entries are taken from a week of the blog. The target week is 12224 + * defined by the request parameters <tt>week</tt> (the week number in the year, from 1 to 52) and <tt>year</tt> (4 12225 + * digit year). Initially the current week is displayed. 12226 + * 12227 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 12228 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 12229 + * refined to restrict to a given space, or to a given search criteria, etc. 12230 + * @param entries The resulting list of entries to display, a list of XDocument names. 12231 + * @param queryParams The parameters to bind with the query. 12232 + *### 12233 +#macro(getWeeklyBlogEntries $blogDoc $query $entries $queryParams) 12234 + #if (!$entries) 12235 + #setVariable (~"$entries~" []) 12236 + #end 12237 + #getRequestedWeek($weekDate) 12238 + #set($dateFormatter = $xwiki.jodatime.getDateTimeFormatterForPattern('yyyy-MM-dd')) 12239 + #set($minDay = $dateFormatter.print($weekDate.toMutableDateTime().weekOfWeekyear().roundFloor())) 12240 + #set($maxDay = $dateFormatter.print($weekDate.toMutableDateTime().weekOfWeekyear().roundCeiling())) 12241 + #set($query = ~"$\{query} and publishDate.value >= '$minDay' and publishDate.value < '$maxDay'~") 12242 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 12243 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 12244 + #bindQueryParameters($countQueryObj $queryParams) 12245 + #bindQueryParameters($queryObj $queryParams) 12246 + #set($totalEntries = $countQueryObj.count()) 12247 + #set($discard = $entries.addAll($queryObj.execute())) 12248 +#end 12249 +## 12250 +## 12251 +## 12252 +#** 12253 + * Retrieves a list of entries to be displayed. The entries are taken from a month of the blog. The target month is 12254 + * defined by the request parameters <tt>month</tt> (the month number, from 1 to 12) and <tt>year</tt> (4 12255 + * digit year). Initially the current month is displayed. 12256 + * 12257 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 12258 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 12259 + * refined to restrict to a given space, or to a given search criteria, etc. 12260 + * @param entries The resulting list of entries to display, a list of XDocument names. 12261 + * @param queryParams The parameters to bind with the query. 12262 + *### 12263 +#macro(getMonthlyBlogEntries $blogDoc $query $entries) 12264 + #if (!$entries) 12265 + #setVariable (~"$entries~" []) 12266 + #end 12267 + #getRequestedMonth($monthDate) 12268 + #set($dateFormatter = $xwiki.jodatime.getDateTimeFormatterForPattern('yyyy-MM-dd')) 12269 + #set($minDay = $dateFormatter.print($monthDate.toMutableDateTime().monthOfYear().roundFloor())) 12270 + #set($maxDay = $dateFormatter.print($monthDate.toMutableDateTime().monthOfYear().roundCeiling())) 12271 + #set($query = ~"$\{query} and publishDate.value >= '$minDay' and publishDate.value < '$maxDay'~") 12272 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 12273 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 12274 + #bindQueryParameters($countQueryObj $queryParams) 12275 + #bindQueryParameters($queryObj $queryParams) 12276 + #set($totalEntries = $countQueryObj.count()) 12277 + #set($discard = $entries.addAll($queryObj.execute())) 12278 +#end 12279 +## 12280 +## 12281 +## 12282 +#** 12283 + * Retrieves a list of entries to be displayed. All entries belonging to the current blog are returned. 12284 + * 12285 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 12286 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 12287 + * refined to restrict to a given space, or to a given search criteria, etc. 12288 + * @param entries The resulting list of entries to display, a list of XDocument names. 12289 + * @param queryParams The parameters to bind with the query. 12290 + *### 12291 +#macro(getAllBlogEntries $blogDoc $query $entries $queryParams) 12292 + #if (!$entries) 12293 + #setVariable (~"$entries~" []) 12294 + #end 12295 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 12296 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 12297 + #bindQueryParameters($countQueryObj $queryParams) 12298 + #bindQueryParameters($queryObj $queryParams) 12299 + #set($totalEntries = $countQueryObj.count()) 12300 + #set($discard = $entries.addAll($queryObj.execute())) 12301 +#end 12302 +## 12303 +## 12304 +## 12305 +#** 12306 + * Retrieves a list of entries to be displayed. Only (and all) unpublished entries are returned. 12307 + * 12308 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 12309 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 12310 + * refined to restrict to a given space, or to a given search criteria, etc. 12311 + * @param queryParams The parameters to bind with the query. 12312 + * @param entries The resulting list of entries to display, a list of XDocument names. 12313 + *### 12314 +#macro(getUnpublishedBlogEntries $blogDoc $query $queryParams $entries) 12315 + #if (!$entries) 12316 + #setVariable (~"$entries~" []) 12317 + #end 12318 + #set($query = ~"$\{query} and isPublished.value = 0~") 12319 + #set ($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 12320 + #set ($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 12321 + #bindQueryParameters($countQueryObj $queryParams) 12322 + #bindQueryParameters($queryObj $queryParams) 12323 + #set($totalEntries = $countQueryObj.count()) 12324 + #set($discard = $entries.addAll($queryObj.execute())) 12325 +#end 12326 +## 12327 +## 12328 +## 12329 +#** 12330 + * Retrieves a list of entries to be displayed. The entries are taken from all the wiki, and not from a specific blog. 12331 + * 12332 + * @param entries The resulting list of entries to display, a list of XDocument names. 12333 + *### 12334 +#macro(getGlobalBlogEntries $entries) 12335 + #if (!$entries) 12336 + #setVariable (~"$entries~" []) 12337 + #end 12338 + #getAllBlogPostsQuery($query) 12339 + #set($totalEntries = $services.query.hql($query).count()) 12340 + #set($defaultItemsPerPage = 20) 12341 + ## This macro is defined in the default macros.vm library. It also sets $itemsPerPage and $startAt. 12342 + #preparePagedViewParams($totalEntries $defaultItemsPerPage) 12343 + #set($discard = $entries.addAll($services.query.hql(~"$\{query} order by publishDate.value desc~").setLimit($itemsPerPage).setOffset($startAt).execute())) 12344 +#end 12345 +#** 12346 + * Return the base query for selecting blog entries. It filters only visible entries, but does not bind to a specific 12347 + * blog, nor specify a range or an ordering criteria. 12348 + * 12349 + * This macro is a duplicate of #getAllBlogPostsQuery($query). Keep it in case it's used outside, in custom code. 12350 + * 12351 + * @param query The basic query for selecting blog entries. 12352 + *# 12353 +#macro(getBlogEntriesBaseQuery $query) 12354 + #getAllBlogPostsQuery($query) 12355 +#end 12356 +#** 12357 + * Return the Query for selecting the all wiki blog posts without filtering 12358 + * 12359 + * @param query The basic query for selecting blog entries. 12360 + *# 12361 +#macro(getAllBlogPostsQuery $query) 12362 + #set ($query = $NULL) 12363 + #setVariable(~"$query~" ~", BaseObject as obj, IntegerProperty isPublished, 12364 + IntegerProperty hidden, DateProperty publishDate 12365 + where doc.fullName <> '$blogPostTemplate' and 12366 + obj.name=doc.fullName and obj.className='$blogPostClassname' and 12367 + isPublished.id.id = obj.id and isPublished.id.name = 'published' and 12368 + hidden.id.id = obj.id and hidden.id.name='hidden' and 12369 + publishDate.id.id = obj.id and publishDate.id.name='publishDate' and 12370 + (doc.creator = '$escapetool.sql($xcontext.user)' or (isPublished.value = 1 and hidden.value = 0))~") 12371 +#end 12372 +## 12373 +## 12374 +## 12375 +###** 12376 + * Return the Query for selecting the all wiki blog posts with categories filtering 12377 + * 12378 + * @param query The basic query for selecting blog entries. 12379 + *### 12380 +#macro(getCategoryAllBlogPostsQuery $query) 12381 + #set ($query = $NULL) 12382 + #getAllBlogPostsQuery($baseQuery) 12383 + #set ($baseQuery = $baseQuery.replace('DateProperty publishDate', 'DateProperty publishDate, DBStringListProperty as category left join category.list catList')) 12384 + #setVariable(~"$query~" ~"$\{baseQuery} and obj.id=category.id.id and category.id.name='category'~") 12385 +#end 12386 +## 12387 +## 12388 +## 12389 +#** 12390 + * Determines how is the blog index split into pages: paginated (fixed number of entries), weekly (all entries in a 12391 + * week), monthly (all entries in a month), or all. 12392 + * 12393 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>displayType</tt> 12394 + * property set. 12395 + * @param displayType The resulting string. If the blog object does not define anything, it is considered paginated. 12396 + *### 12397 +#macro(getBlogDisplayType $blogDoc $displayType) 12398 + #getBlogProperty($blogDoc 'displayType' 'paginated' $result) 12399 + #set ($displayType = $NULL) 12400 + #setVariable (~"$displayType~" $result) 12401 +#end 12402 +## 12403 +## 12404 +## 12405 +#** 12406 + * Displays a list of entries. 12407 + * 12408 + * @param entries The entries to display, a list of XDocument names. 12409 + * @param displaying What exactly is displayed: blog index, a single blog entry, a blog category, search results, 12410 + * unpublished entries, etc. This will be used as the classname(s) for the container div (hfeed). Currently 12411 + * used values: index, single, category, search, unpublished, hidden. 12412 + * @param onlyExtract If <tt>true</tt>, only display the extract of articles where available, otherwise display the full content. 12413 + * @param shouldDisplayTitles If <tt>true</tt>, display the blog title (blog posts shouldn't display the title when they're 12414 + * displayed alone on their page since it's the page title which is used in this case) 12415 + *### 12416 +#macro(displayBlog $entries $displaying $onlyExtract $shouldDisplayTitles) 12417 + #set($blogDay = '') 12418 + (% class=~"hfeed $!\{displaying}~" ~%)((( 12419 + (% class=~"blogDay~" ~%)((( 12420 + #foreach ($entryDoc in $xwiki.wrapDocs($entries)) 12421 + #getEntryObject($entryDoc $entryObj) 12422 + ## Although all entries should have one of the two objects, better check to be sure. 12423 + #if(~"$!\{entryObj}~" != '') 12424 + #getEntryDate($entryDoc $entryObj $entryDate) 12425 + ## Display a ~"calendar sheet~" for each day. All entries posted on the same day share one such sheet. 12426 + #set($entryDateStr = $xwiki.formatDate($entryDate, 'yyyyMMMMdd')) 12427 + #if($blogDay != $entryDateStr) 12428 + #if($blogDay != '') 12429 + ))) 12430 + (% class=~"blogDay~" ~%)((( 12431 + #end 12432 + #displayBlogDate($entryDate) 12433 + #set ($blogDay = $entryDateStr) 12434 + #end 12435 + ## Finally, display the entry. 12436 + #displayEntry($entryDoc $entryObj $onlyExtract $shouldDisplayTitles true) 12437 + #end 12438 + #end 12439 + )))## blogDay 12440 + )))## hfeed 12441 +#end 12442 +## 12443 +## 12444 +## 12445 +#** 12446 + * Get the entry object, either a new BlogPost or an old Article. 12447 + * 12448 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 12449 + * @param entryObj The resulting xobject of the blog post. 12450 + *### 12451 +#macro(getEntryObject $entryDoc $__entryObj) 12452 + #set($result = $entryDoc.getObject(~"$\{blogPostClassname}~")) 12453 + #if(!$result) 12454 + #set($result = $entryDoc.getObject(~"$\{oldArticleClassname}~")) 12455 + #end 12456 + ## NOTE: The reason we put an underscore in front of the variable name is to prevent the following line from 12457 + ## overwriting the $entryObj variable that may be defined before this macro is called. Of course, $__entryObj may be 12458 + ## overwritten in this case but it's less likely to have such a variable defined before. 12459 + #set ($__entryObj = $NULL) 12460 + #setVariable (~"$__entryObj~" $result) 12461 +#end 12462 +## 12463 +## 12464 +## 12465 +#** 12466 + * Gets the date associated with a blog entry. This is the publication date. For unpublished entries, initially this is 12467 + * the document creation date, but can be edited by the user. 12468 + * 12469 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 12470 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 12471 + * @param result The resulting date, an instance of <tt>java.util.Date</tt>. 12472 + *### 12473 +#macro(getEntryDate $entryDoc $entryObj $result) 12474 + #set ($result = $NULL) 12475 + #setVariable (~"$result~" $entryObj.getProperty('publishDate').value) 12476 +#end 12477 +## 12478 +## 12479 +## 12480 +#** 12481 + * Displays a date, nicely formatted as a calendar page. 12482 + * 12483 + * @param date The date to display, an instance of <tt>java.util.Date</tt>. 12484 + *### 12485 +#macro(displayBlogDate $date) 12486 + #set($year = $xwiki.formatDate($date, 'yyyy')) 12487 + ## 3 letter month name, like Jan, Dec. 12488 + #set($month = $xwiki.formatDate($date, 'MMM')) 12489 + ## Uncomment to get a full length month name, like January, December. 12490 + ## TODO: this could be defined somewhere in the blog style. 12491 + ## #set($month = $xwiki.formatDate($date, 'MMMM')) 12492 + #set($day = $xwiki.formatDate($date, 'dd')) 12493 + (% class=~"blogdate~" ~%) 12494 + == (% class=~"month~" ~%)$month(%~%) (% class=~"day~" ~%)$day(%~%) (% class=~"year~" ~%)$year(%~%) == 12495 +#end 12496 +## 12497 +## 12498 +## 12499 +#** 12500 + * Displays a blog article: management tools, header, content, footer. 12501 + * 12502 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 12503 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 12504 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 12505 + * @param shouldDisplayTitle If <tt>true</tt>, display the blog title (blog posts shouldn't display the title 12506 + * when they're displayed alone on their page since it's the page title which is used in this case) 12507 + * @param shouldDisplayActions If <tt>true</tt>, display the blog post actions buttons 12508 + *### 12509 +#macro(displayEntry $entryDoc $entryObj $onlyExtract $shouldDisplayTitle $shouldDisplayActions) 12510 + ## Only articles with an explicit hidden setting or an explicit unpublished setting are hidden 12511 + #isPublished($entryObj $isPublished) 12512 + #isHidden($entryObj $isHidden) 12513 + #if($doc.fullName == $entryDoc.fullName) 12514 + (% class=~"hentry single-article~" ~%)((( 12515 + #else 12516 + (% class=~"hentry#if(!$isPublished) unpublished-article#elseif($isHidden) hidden-article#end~" ~%)((( 12517 + #end 12518 + #if ($shouldDisplayActions) 12519 + #displayEntryTools($entryDoc $entryObj) 12520 + #end 12521 + #if($shouldDisplayTitle) 12522 + #displayEntryTitle($entryDoc $entryObj) 12523 + #end 12524 + #if($doc.fullName == $entryDoc.fullName) 12525 + #if(!$isPublished) 12526 + \{\{warning}}\{\{translation key='blog.code.published'/}}\{\{/warning}} 12527 + #elseif($isHidden) 12528 + \{\{warning}}\{\{translation key='blog.code.hidden'/}}\{\{/warning}} 12529 + #end 12530 + #end 12531 + #displayEntryContent($entryDoc $entryObj $onlyExtract) 12532 + #displayEntryFooter($entryDoc $entryObj) 12533 + )))## hentry 12534 +#end 12535 +## 12536 +## 12537 +## 12538 +#** 12539 + * Checks if the provided blog is published or not. 12540 + * 12541 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 12542 + * @param isPublished The resulting boolean, true if the entry is considered published. 12543 + *### 12544 +#macro(isPublished $entryObj $isPublished) 12545 + #set ($isPublished = $NULL) 12546 + ## This should work for both old articles, which don't have the 'published' property at all, and 12547 + ## are considered published by default, and new entries, that should have 1 if published. 12548 + #if (~"$!\{entryObj.getProperty('published').value}~" != '0') 12549 + #setVariable (~"$isPublished~" true) 12550 + #else 12551 + #setVariable (~"$isPublished~" false) 12552 + #end 12553 +#end 12554 +## 12555 +## 12556 +## 12557 +#** 12558 + * Checks if the provided blog is hidden or not. 12559 + * 12560 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass<tt> xclass. 12561 + * @param isHiddel The resulting boolean, true if the entry is considered hidden. 12562 + *### 12563 +#macro(isHidden $entryObj $isHidden) 12564 + #set ($isHidden = $NULL) 12565 + ## This should work for both old articles, which don't have the 'hidden' property at all, and 12566 + ## are considered visible by default, and new entries, that should have 1 if hidden. 12567 + #if (~"$!\{entryObj.getProperty('hidden').value}~" == '1') 12568 + #setVariable (~"$isHidden~" true) 12569 + #else 12570 + #setVariable (~"$isHidden~" false) 12571 + #end 12572 +#end 12573 +## 12574 +## 12575 +## 12576 +#** 12577 + * Displays several ~"tools~" for manipulating blog posts: hide/show, publish, edit. 12578 + * 12579 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 12580 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 12581 + *### 12582 +#macro(displayEntryTools $entryDoc $entryObj) 12583 + #if($xcontext.action == 'view') 12584 + (% class=~"blog-entry-toolbox~" ~%)((( 12585 + #displayPublishButton($entryDoc $entryObj) 12586 + #displayHideShowButton($entryDoc $entryObj) 12587 + #displayEditButton($entryDoc $entryObj) 12588 + #displayDeleteButton($entryDoc $entryObj) 12589 + ))) 12590 + #end 12591 +#end 12592 +## 12593 +## 12594 +## 12595 +#** 12596 + * Displays the publish button to the entry <strong>creator</strong>, if the article is not published yet. 12597 + * 12598 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 12599 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 12600 + * @todo AJAX calls. 12601 + *### 12602 +#macro(displayPublishButton $entryDoc $entryObj) 12603 + #isPublished($entryObj $isPublished) 12604 + #if(!$isPublished && $entryDoc.creator == $xcontext.user && $xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 12605 + [[#toolImage('world')>>path:$blogPublisher.getURL('view', ~"entryName=$\{escapetool.url($entryDoc.fullName)}&xredirect=$\{escapetool.url($thisURL)}&form_token=$!\{services.csrf.getToken()}~")||title=~"$services.localization.render('blog.code.notpublished')~"]]## 12606 + #end 12607 +#end 12608 +## 12609 +## 12610 +## 12611 +#** 12612 + * Displays the hide or show button to the entry <strong>creator</strong>, if the article is already published. 12613 + * 12614 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 12615 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 12616 + *### 12617 +#macro(displayHideShowButton $entryDoc $entryObj) 12618 + #isPublished($entryObj $isPublished) 12619 + #isHidden($entryObj $isHidden) 12620 + ## Only published articles can be hidden. Unpublished articles are considered already hidden. 12621 + #if($isPublished && $entryDoc.creator == $xcontext.user && $xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 12622 + #set ($queryString = \{ 12623 + 'xredirect' : $thisURL, 12624 + 'form_token' : $services.csrf.getToken() 12625 + }) 12626 + #if ($isHidden) 12627 + #set ($discard = $queryString.putAll(\{ 12628 + ~"$\{entryObj.getxWikiClass().getName()}_$\{entryObj.number}_hidden~" : 0, 12629 + 'comment' : $services.localization.render('blog.code.madevisible') 12630 + })) 12631 + #set ($lockURL = $entryDoc.getURL('save', $escapetool.url($queryString))) 12632 + [[#toolImage('unlock')>>path:$lockURL||class=~"blog-tool-show~" title=~"$services.localization.render('blog.code.makevisible')~"]]## 12633 + #else 12634 + #set ($discard = $queryString.putAll(\{ 12635 + ~"$\{entryObj.getxWikiClass().getName()}_$\{entryObj.number}_hidden~" : 1, 12636 + 'comment' : $services.localization.render('blog.code.hid') 12637 + })) 12638 + #set ($lockURL = $entryDoc.getURL('save', $escapetool.url($queryString))) 12639 + [[#toolImage('lock')>>path:$lockURL||class=~"blog-tool-hide~" title=~"$services.localization.render('blog.code.hide')~"]]## 12640 + #end 12641 + #end 12642 +#end 12643 +## 12644 +## 12645 +## 12646 +#** 12647 + * Displays the edit button to those that can edit the article. 12648 + * 12649 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 12650 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 12651 + *### 12652 +#macro(displayEditButton $entryDoc $entryObj) 12653 + #if($xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 12654 + ## Call getDefaultEditMode() for backward compatibility with older blog posts. 12655 + [[#toolImage('pencil')>>path:$entryDoc.getURL('edit')||title=~"$services.localization.render('blog.code.editpost')~"]]## 12656 + #end 12657 +#end 12658 +## 12659 +## 12660 +## 12661 +#** 12662 + * Displays the delete button to those that can edit the article. 12663 + * 12664 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 12665 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 12666 + * @todo AJAX calls. 12667 + *### 12668 +#macro(displayDeleteButton $entryDoc $entryObj) 12669 + #if($xwiki.hasAccessLevel('delete', $xcontext.user, $entryDoc.fullName)) 12670 + [[#toolImage('cross')>>path:$entryDoc.getURL('delete')||title=~"$services.localization.render('blog.code.deletepost')~"]]## 12671 + #end 12672 +#end 12673 +## 12674 +## 12675 +## 12676 +#** 12677 + * Displays the title of the entry. 12678 + * 12679 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 12680 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 12681 + *### 12682 +#macro(displayEntryTitle $entryDoc $entryObj) 12683 + #if($doc.fullName == $entryDoc.fullName) 12684 + (% class=~"entry-title~" ~%) 12685 + = $services.rendering.escape($entryDoc.display('title', 'view', $entryObj), $xwiki.getCurrentContentSyntaxId()) = 12686 + #else 12687 + (% class=~"entry-title~" ~%) 12688 + === [[$services.rendering.escape($entryDoc.display('title', 'view', $entryObj),$xwiki.getCurrentContentSyntaxId())>>doc:$services.rendering.escape($services.model.serialize($entryDoc.getDocumentReference(),'default'),$xwiki.getCurrentContentSyntaxId())]] === 12689 + #end 12690 +#end 12691 +## 12692 +## 12693 +## 12694 +#** 12695 + * Displays the body of the entry. 12696 + * 12697 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 12698 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 12699 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 12700 + *### 12701 +#macro(displayEntryContent $entryDoc $entryObj $onlyExtract) 12702 + (% class=~"#if($onlyExtract)entry-summary#\{else}entry-content#end~" ~%)((( 12703 + #getEntryContent($entryDoc $entryObj $onlyExtract $entryContent) 12704 + ## FIXME: This causes the blog's content to not be annotatable. See http://jira.xwiki.org/browse/XWIKI-6328 12705 + ## Should probably be replaced by a display macro call with a reference to the object property holding the post's content 12706 + \{\{html wiki=~"false~"}}$entryDoc.getRenderedContent($entryContent, $entryDoc.syntax.toIdString())\{\{/html}} 12707 + ))) ## entry-content 12708 + (% class=~"clearfloats~" ~%)((())) 12709 +#end 12710 +## 12711 +## 12712 +## 12713 +#** 12714 + * Extracts the body of the entry that should be displayed. If <tt>onlyExtract</tt> is <tt>true</tt>, display the content 12715 + * of the <tt>extract</tt> field (if not empty). 12716 + * 12717 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 12718 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 12719 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 12720 + * @param entryContent The resulting content. 12721 + * @param removeEllipsis If <tt>true</tt>, then don't display an ellipsis at the end of the content (only used when 12722 + * <tt>onlyExtract</tt> is <tt>true</tt>) 12723 + *### 12724 +#macro(getEntryContent $entryDoc $entryObj $onlyExtract $entryContent $removeEllipsis) 12725 + #if ($onlyExtract) 12726 + ## Note: We trim the summary so that if there's some white space it won't cause the summary to be used instead 12727 + ## of the content. 12728 + #set ($macro.result = $entryObj.getProperty('extract').value.trim()) 12729 + #end 12730 + #if(~"$!macro.result~" == '') 12731 + #set($macro.result = $entryObj.getProperty('content').value) 12732 +#* Disabled until the content can be cleanly cut. 12733 +* #if($onlyExtract && $result.length()>$maxchars) 12734 +* #set($i = $macro.result.lastIndexOf(~" ~", $maxchars)) 12735 +* #set($i = $i + 1) 12736 +* #set($macro.result = ~"$\{macro.result.substring(0,$i)} *[...>$\{entryDoc.fullName}]*~") 12737 +* #end 12738 +## *### 12739 + #elseif (!$removeEllipsis) 12740 + #if($entryDoc.syntax.toIdString() == 'xwiki/1.0') 12741 + #set($macro.result = ~"$\{macro.result} <a href='$\{entryDoc.getExternalURL()}' title='$services.localization.render('blog.code.readpost')'>...</a>~") 12742 + #elseif($entryDoc.syntax.toIdString() == 'xwiki/2.0' || $entryDoc.syntax.toIdString() == 'xwiki/2.1') 12743 + #set($macro.result = ~"$\{macro.result} [[...>>$\{services.rendering.escape($entryDoc.getFullName(), $entryDoc.getSyntax())}||title='$services.localization.render('blog.code.readpost')']]~") 12744 + #end 12745 + #end 12746 + #set ($entryContent = $NULL) 12747 + #setVariable (~"$entryContent~" $macro.result) 12748 +#end 12749 +## 12750 +## 12751 +## 12752 +#** 12753 + * Displays the footer of the entry. 12754 + * 12755 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 12756 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 12757 + *### 12758 +#macro(displayEntryFooter $entryDoc $entryObj) 12759 + (% class=~"entry-footer~" ~%)((( 12760 + #isPublished($entryObj $isPublished) 12761 + (% class='entry-author-label' ~%) 12762 + #if($isPublished) 12763 + \{\{translation key='blog.code.postedby'/}} ## 12764 + #else 12765 + \{\{translation key='blog.code.createdby'/}} ## 12766 + #end 12767 + \{\{html wiki=~"false~" clean=~"false~"}}<span class='author vcard'>#userfn($entryDoc.creator)</span>\{\{/html}} ## 12768 + #getEntryDate($entryDoc $entryObj $entryDate) 12769 + #listCategories($entryObj) #* 12770 + ## Since the publish date and update date are not set at the exact same time, there could be a small difference that 12771 + ## we assume cannot be more than 3 seconds. 12772 + *#(% class=~"separator~" ~%)·(%~%) [[\{\{translation key='blog.code.permalink'/}}>>$services.rendering.escape($entryDoc,'xwiki/2.0')||rel=~"bookmark~"]] ## 12773 + #if ($showcomments) 12774 + (% class=~"separator~" ~%)·(%~%) [[\{\{translation key='blog.code.comments'/}}>>$services.rendering.escape($entryDoc,'xwiki/2.0')||anchor=~"Comments~"]] (% class=~"itemCount~" ~%)($entryDoc.comments.size())(%~%) ## 12775 + #end ## 12776 + #if($entryDoc != $doc) ## 12777 + #displayEntryBlogLocation($entryDoc $entryObj) ## 12778 + #end 12779 + )))## entry-footer 12780 +#end 12781 +## 12782 +## 12783 +#** 12784 + * Display the blog for the entry (if it is not the currently displayed blog) 12785 + * 12786 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 12787 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 12788 + *### 12789 +#macro(displayEntryBlogLocation $entryDoc $entryObj) 12790 + #getBlogPostsLocation($entryDoc.getSpace() $blogPostsLocation) ## 12791 + #if(~"$!blogPostsLocation~" != ~"~") ## 12792 + #set($blogSpaceRef = $services.model.resolveSpace($blogPostsLocation)) ## 12793 + #set($blogDocRef = $services.model.createDocumentReference($services.model.getEntityReference('DOCUMENT', 'default').getName(), $blogSpaceRef)) ## 12794 + #if($doc.documentReference != $blogDocRef) ## 12795 + #set($blogDoc = $xwiki.getDocument($blogDocRef)) ## 12796 + #set($blogObj = $blogDoc.getObject('Blog.BlogClass')) ## 12797 + #if($blogObj) (% class=~"blogrefpath~" ~%)((( ## 12798 + $services.localization.render('blog.code.postedin') \{\{html clean=~"false~" wiki=~"false~"}} #hierarchy($blogDocRef, \{'selfIsActive' : true, 'local': true}) \{\{/html}} ## 12799 + )))(%~%)## 12800 + #end 12801 + #end 12802 + #end 12803 +#end 12804 +## 12805 +## 12806 +## 12807 +## 12808 +#** 12809 + * List the categories an entry belongs to. Used in the footer. The categories are instances of <tt>Blog.CategoryClass</tt>. 12810 + * 12811 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 12812 + *### 12813 +#macro(listCategories $entryObj) 12814 + #if($entryObj.getxWikiClass().getName() == $blogPostClassname) 12815 + #set($categories = $entryObj.getProperty('category').value) 12816 + #set($first = true) 12817 + #if($categories.size() > 0) 12818 + #foreach($category in $categories) 12819 + #set($categoryDoc = $!xwiki.getDocument($category)) 12820 + #if(!$categoryDoc.isNew() && $categoryDoc.getObject($\{blogCategoryClassname})) 12821 + #if($foreach.count == 1) 12822 + (% class='separator' ~%)·(%~%) $services.localization.render('blog.code.categories') ## 12823 + #else 12824 + , ## 12825 + #end## 12826 + [[$!\{categoryDoc.getObject($blogCategoryClassname).getValue('name')}>>$\{category}||rel='tag']]## 12827 + #end## 12828 + #end## 12829 + #end 12830 + #end 12831 +#end 12832 +## 12833 +## 12834 +## 12835 +#** 12836 + * Displays blog pagination links (older and newer entries). 12837 + * 12838 + * @param blogDoc the XDocument holding the blog definition object. 12839 + *### 12840 +#macro(displayNavigationLinks $blogDoc) 12841 + (% class=~"clearfloats~" ~%)((())) 12842 + #getBlogDisplayType($blogDoc $displayType) 12843 + #if($displayType == 'weekly') 12844 + (% class=~"pagingLinks~" ~%)((( 12845 + #getRequestedWeek($weekDate) 12846 + $weekDate.addWeeks(-1)## 12847 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.previousweek'/}}>>$doc.documentReference.name||queryString=~"year=$weekDate.weekyear&week=$weekDate.weekOfWeekyear~" class=~"button~"]]**(%~%) 12848 + #sep() 12849 + $weekDate.addWeeks(2)## 2 because we already subtracted 1 above 12850 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.nextweek'/}} »>>$doc.documentReference.name||queryString=~"year=$weekDate.weekyear&week=$weekDate.weekOfWeekyear~" class=~"button~"]]**(%~%) 12851 + ))) 12852 + #elseif($displayType == 'monthly') 12853 + (% class=~"pagingLinks~" ~%)((( 12854 + #getRequestedMonth($monthDate) 12855 + $monthDate.addMonths(-1)## 12856 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.previousmonth'/}}>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"year=$monthDate.year&month=$monthDate.monthOfYear~" class=~"button~"]]**(%~%) 12857 + #sep() 12858 + $monthDate.addMonths(2)## 2 because we already subtracted 1 above 12859 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.nextmonth'/}} »>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"year=$monthDate.year&month=$monthDate.monthOfYear~" class=~"button~"]]**(%~%) 12860 + ))) 12861 + #elseif($displayType == 'all') 12862 + #else 12863 + ## Paginated 12864 + #if(($totalPages > 1)) 12865 + #set($queryString = '') 12866 + #foreach($p in $request.getParameterNames()) 12867 + #if($p != 'page' && $p != 'ipp') 12868 + #foreach($v in $request.getParameterValues($p)) 12869 + #set($queryString = ~"$\{queryString}&$\{escapetool.url($p)}=$\{escapetool.url($v)}~") 12870 + #end 12871 + #end 12872 + #end 12873 + (% class=~"pagingLinks~" ~%)((( 12874 + #if ($currentPageNumber < $totalPages) 12875 + #set($currentPageNumber = $currentPageNumber + 1) 12876 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.olderposts'/}}>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"page=$\{currentPageNumber}&ipp=$\{itemsPerPage}$queryString~" class=~"button~"]]**(%~%) 12877 + #set($currentPageNumber = $currentPageNumber - 1) 12878 + #end 12879 + #if ($currentPageNumber > 1) 12880 + #if ($currentPageNumber < $totalPages) 12881 + #sep() 12882 + #end 12883 + #set($currentPageNumber = $currentPageNumber - 1) 12884 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.newerposts'/}} »>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"page=$\{currentPageNumber}&ipp=$\{itemsPerPage}$queryString~" class=~"button~"]]**(%~%) 12885 + #set($currentPageNumber = $currentPageNumber + 1) 12886 + #end 12887 + (% class=~"clear~" ~%)(%~%) 12888 + )))## pagingLinks 12889 + #end 12890 + #end 12891 +#end 12892 +## 12893 +## 12894 +## 12895 +#** 12896 + * Displays a message box with ~"publish~" icon. 12897 + * 12898 + * @param message A text message concerning blog article publishing 12899 + *### 12900 +#macro(publishMessageBox $message) 12901 +(% class=~"plainmessage publish-message~" ~%)((($services.icon.render('world') $message))) 12902 +#end 12903 +#** 12904 + * Displays a message box with ~"show/hide~" icon. 12905 + * 12906 + * @param message A text message concerning blog article hiding 12907 + *### 12908 +#macro(hideMessageBox $message) 12909 +(% class=~"plainmessage hide-message~" ~%)((($services.icon.render('unlock') $message))) 12910 +#end 12911 +## 12912 +## 12913 +## 12914 +#** 12915 + * Determine the requested week, for using in a weekly-indexed blog. The relevant request parameters are 12916 + * <tt>year</tt> and <tt>week</tt>. By default, the current week is used. 12917 + * 12918 + * @param monthDate The resulting week, a JODATime MutableDateTime. 12919 + *### 12920 +#macro(getRequestedWeek $weekDate) 12921 + #set ($weekDate = $NULL) 12922 + #setVariable (~"$weekDate~" $xwiki.jodatime.mutableDateTime) 12923 + #if(~"$!\{request.year}~" != '') 12924 + #set ($discard = $weekDate.setYear($numbertool.toNumber($request.year).intValue())) 12925 + #end 12926 + #if(~"$!\{request.week}~" != '') 12927 + #set ($discard = $weekDate.setWeekOfWeekyear($numbertool.toNumber($request.week).intValue())) 12928 + #end 12929 +#end 12930 +## 12931 +## 12932 +## 12933 +#** 12934 + * Determine the requested month, for using in a monthly-indexed blog. The relevant request parameters are 12935 + * <tt>year</tt> and <tt>month</tt>. By default, the current month is used. 12936 + * 12937 + * @param monthDate The resulting month, a JODATime MutableDateTime. 12938 + *### 12939 +#macro(getRequestedMonth $monthDate) 12940 + #set ($monthDate = $NULL) 12941 + #setVariable (~"$monthDate~" $xwiki.jodatime.mutableDateTime) 12942 + #if(~"$!\{request.year}~" != '') 12943 + #set ($discard = $monthDate.setYear($numbertool.toNumber($request.year).intValue())) 12944 + #end 12945 + #if(~"$!\{request.month}~" != '') 12946 + #set ($discard = $monthDate.setMonthOfYear($numbertool.toNumber($request.month).intValue())) 12947 + #end 12948 +#end 12949 +## 12950 +## 12951 +## 12952 +#** 12953 + * Retrieve a blog property (title, display type, etc). 12954 + * 12955 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 12956 + * @param propertyName The name of the property to be retrieved. One of the <tt>Blog.BlogClass</tt>'s properties. 12957 + * @param defaultValue The default value to use in case the blog object does not define one. 12958 + * @param propertyValue The resulting value. 12959 + *### 12960 +#macro(getBlogProperty $blogDoc $propertyName $defaultValue $propertyValue) 12961 + #set($result = ~"$!\{blogDoc.getObject($\{blogClassname}).getProperty($propertyName).value}~") 12962 + #if($result == '') 12963 + #set($result = $defaultValue) 12964 + #end 12965 + #set ($propertyValue = $NULL) 12966 + #setVariable (~"$propertyValue~" $result) 12967 +#end 12968 + 12969 +#** 12970 + * If an error occurs when executing an action, set a specific response status and display an error message. 12971 + * 12972 + * @param status The response status. 12973 + * @param text The user readable error to be displayed. Can be a translation key. 12974 + * @param parameters The parameters to use when decoding the translation key. 12975 + *### 12976 +#macro(blog__actionResponseError $status $text $parameters) 12977 + $response.setStatus($status) 12978 + #if($request.ajax) 12979 + $services.localization.render($text, $!parameters) 12980 + #else 12981 + \{\{error}}$services.localization.render($text, $!parameters)\{\{/error}} 12982 + #end 12983 +#end 12984 +## 12985 +## 12986 +## 12987 +#** 12988 + * Check if a blog is the Default blog (The one in the 'Blog' space). 12989 + * 12990 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 12991 + * @param isDefault The resulting boolean. 12992 + *### 12993 +#macro(isDefaultBlog $blogDoc $isDefault) 12994 + #set ($result = false) 12995 + #if ($blogDoc.space == 'Blog') 12996 + #set ($result = true) 12997 + #end 12998 + #setVariable(~"$isDefault~" $result) 12999 +#end 13000 +## 13001 +## 13002 +## 13003 +#** 13004 + * Retrieve the blog posts location (space). 13005 + * 13006 + * @param blogSpace The blog space. It should contain a document with a <tt>Blog.BlogClass</tt> object 13007 + * @param postsLocation The resulting location. 13008 + *### 13009 +#macro(getBlogPostsLocation $blogSpace $postsLocation) 13010 + #getBlogDocument($blogSpace $blogDoc) 13011 + #getBlogProperty($blogDoc 'postsLocation' $blogDoc.space $result) 13012 + #set ($postsLocation = $NULL) 13013 + #setVariable (~"$postsLocation~" $result) 13014 +#end 13015 +## 13016 +## 13017 +## 13018 +#** 13019 + * Retrieve the blog categories location (space). 13020 + * 13021 + * @param blogSpace The blog space. It should contain a document with a <tt>Blog.BlogClass</tt> object 13022 + * @param categoriesLocation The resulting location. 13023 + *### 13024 +#macro(getBlogCategoriesLocation $blogSpace $categoriesLocation) 13025 + #getBlogDocument($blogSpace $blogDoc) 13026 + #getBlogProperty($blogDoc 'categoriesLocation' 'Blog' $result) 13027 + #set ($postsLocation = $NULL) 13028 + #setVariable (~"$categoriesLocation~" $result) 13029 +#end 13030 +###** 13031 + * Return the Query for selecting the blog posts based on the context where the posts are displayed. 13032 + * for example there is 4 different panel contexts: 13033 + * aBlog.aPost or aBlog.WebHome 13034 + * aCategorySpace.aCategory 13035 + * aCategorySpace.WebHome 13036 + * Blog.aPost or Blog.WebHome 13037 + * 13038 + * @param query The query for selecting blog entries. 13039 + * @param queryParams The parameters to bind with the generated query. 13040 + * @param targetDoc The document in which the articles will be displayed. 13041 + *### 13042 +#macro(getAllBlogPostsQueryBasedOnDisplayContext $targetDoc $query $queryParams) 13043 + #set ($query = $NULL) 13044 + #set ($queryParams = $NULL) 13045 + #getCategoryAllBlogPostsQuery($resultQuery) 13046 + #set ($resultQueryParams = \{}) 13047 + #set($sanitizedSpace = $targetDoc.space.replaceAll('([%_!])', '!$1').concat('.%')) 13048 + #if ($targetDoc.space == $defaultBlogSpace && !$targetDoc.getObject($blogCategoryClassname)) 13049 + ## Get all posts that are in the default blog space('Blog') or in category 'Blog.%' 13050 + #set ($discard = $resultQueryParams.put('space', $targetDoc.space)) 13051 + #set ($discard = $resultQueryParams.put('catList', $sanitizedSpace)) 13052 + #set($resultQuery = ~"$\{resultQuery} and (doc.space = :space or catList like :catList escape '!')~") 13053 + #elseif($targetDoc.getObject($blogCategoryClassname)) 13054 + ## Get all posts that are in a category aCategorySpace.aCategory 13055 + #set ($discard = $resultQueryParams.put('catList', $targetDoc.fullName)) 13056 + #set($resultQuery = ~"$\{resultQuery} and (catList=:catList)~") 13057 + #elseif($targetDoc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 13058 + ## Get all posts that are in a category aCategorySpace.% 13059 + #set ($discard = $resultQueryParams.put('catList', $sanitizedSpace)) 13060 + ## Exclude incategorized posts 13061 + #set ($excludedCategory = ~"$\{targetDoc.space}.WebHome~") 13062 + #if ($targetDoc.space == $defaultBlogSpace) 13063 + #set ($excludedCategory = ~"Blog.Categories.WebHome~") 13064 + #end 13065 + #set ($discard = $resultQueryParams.put('excludedCategory', $excludedCategory)) 13066 + #set($resultQuery = ~"$\{resultQuery} and (catList<>:excludedCategory) and (catList like :catList escape '!')~") 13067 + #else 13068 + ## Get all posts in blog space aBlog 13069 + #getAllBlogPostsQuery($resultQuery) 13070 + #getBlogPostsLocation($targetDoc.space $postsLocation) 13071 + #set ($discard = $resultQueryParams.put('space', $postsLocation)) 13072 + #set($resultQuery = ~"$\{resultQuery} and doc.space = :space~") 13073 + #end 13074 + #setVariable(~"$query~" $resultQuery) 13075 + #setVariable(~"$queryParams~" $resultQueryParams) 13076 +#end 13077 +## 13078 +## 13079 +## 13080 +###** 13081 + * Display blog posts based on the context where the posts are displayed. 13082 + * for example there is 4 different panel contexts: 13083 + * aBlog.aPost or aBlog.WebHome 13084 + * aCategorySpace.aCategory 13085 + * aCategorySpace.WebHome 13086 + * Blog.aPost or Blog.WebHome 13087 + * 13088 + * @param targetDoc The document in which the articles will be displayed. 13089 + * @param postsVisiblity The visibility of the posts to display: recent, unpublished, ... 13090 + * @param layout Layout of the the posts to display 13091 + * @param layoutParams additional layout parameters, it is a string like 'param1=val1|...|paramN=valueN' 13092 + * @param limit the number of posts to display 13093 + *### 13094 +#macro(displayBlogPostsBasedOnContext $targetDoc $postsVisiblity $layout $layoutParams $limit) 13095 + #if ($postLayout == 'full') 13096 + #set ($macro.paginated = 'yes') 13097 + #end 13098 + #if ($targetDoc.space == $defaultBlogSpace && !$targetDoc.getObject($blogCategoryClassname)) 13099 + ## Display all posts that are in the default blog space('Blog') or in category 'Blog.%' 13100 + #getBlogPostsLayout($xwiki.getDocument(~"$\{defaultBlogSpace}.WebHome~") $postsLayout) 13101 + #if (~"$!layout~" == '') 13102 + #set ($layout = $postsLayout) 13103 + #end 13104 + #if ($postsVisiblity == 'recent') 13105 + \{\{blogpostlist blog=~"$\{defaultBlogSpace.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')}.WebHome~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 13106 + #elseif($postsVisiblity == 'unpublished') 13107 + \{\{blogpostlist blog=~"$\{defaultBlogSpace.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')}.WebHome~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 13108 + #end 13109 + #elseif($targetDoc.getObject($blogCategoryClassname)) 13110 + ## Display all posts that are in a category aCategorySpace.aCategory 13111 + #getBlogDocumentForCategoriesSpace($targetDoc.space $blogDoc) 13112 + #getBlogPostsLayout($blogDoc $postsLayout) 13113 + #if (~"$!layout~" == '') 13114 + #set ($layout = $postsLayout) 13115 + #end 13116 + #if ($postsVisiblity == 'recent') 13117 + \{\{blogpostlist category=~"$targetDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 13118 + #elseif($postsVisiblity == 'unpublished') 13119 + \{\{blogpostlist category=~"$targetDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 13120 + #end 13121 + #elseif($targetDoc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 13122 + ## Display all posts that are in a category aCategorySpace.% 13123 + #getBlogDocumentForCategoriesSpace($targetDoc.space $blogDoc) 13124 + #getBlogPostsLayout($blogDoc $postsLayout) 13125 + #if (~"$!layout~" == '') 13126 + #set ($layout = $postsLayout) 13127 + #end 13128 + #if ($postsVisiblity == 'recent') 13129 + \{\{blogpostlist category=~"$targetDoc.space.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 13130 + #elseif($postsVisiblity == 'unpublished') 13131 + \{\{blogpostlist category=~"$targetDoc.space.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 13132 + #end 13133 + #else 13134 + ## Display all posts in blog space aBlog 13135 + #getBlogDocument($targetDoc.space $blogDoc) 13136 + #getBlogPostsLayout($blogDoc $postsLayout) 13137 + #if (~"$!layout~" == '') 13138 + #set ($layout = $postsLayout) 13139 + #end 13140 + #if ($postsVisiblity == 'recent') 13141 + \{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 13142 + #elseif($postsVisiblity == 'unpublished') 13143 + \{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 13144 + #end 13145 + #end 13146 +#end 13147 +## 13148 +## 13149 +## 13150 +#** 13151 + * Bind parameters to a query object. 13152 + * 13153 + * @param queryObj The query object (org.xwiki.query.internal.ScriptQuery object) 13154 + * @param queryParams the query parameters. 13155 + *### 13156 +#macro(bindQueryParameters $queryObj $queryParams) 13157 + #set ($output = $queryObj) 13158 + #foreach( $key in $queryParams.keySet() ) 13159 + #set($output = $queryObj.bindValue($key, $queryParams.get($key))) 13160 + #end 13161 + #setVariable(~"$queryObj~" $output) 13162 +#end 13163 +## 13164 +## 13165 +## 13166 +#** 13167 + * Determines the blogposts layout: Default layout (Calendar), Image thumbnail 13168 + * 13169 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>postsLayout</tt> 13170 + * property set. 13171 + * @param postsLayout The resulting string. If the blog object does not define anything, it is considered full (default). 13172 + *### 13173 +#macro(getBlogPostsLayout $blogDoc $postsLayout) 13174 + #getBlogProperty($blogDoc 'postsLayout' 'full' $res) 13175 + #set ($postsLayout = $NULL) 13176 + #setVariable (~"$postsLayout~" $res) 13177 +#end 13178 +## 13179 +## 13180 +## 13181 +#** 13182 + * Retrieve the blog document of a categories space, which is the blog descriptor that have its categoriesLocation property set to the categories space 13183 + * 13184 + * @param categoriesSpace A <tt>String</tt>, the name of the space. 13185 + * @param blogDoc The resulting XDocument. 13186 + *### 13187 +#macro(getBlogDocumentForCategoriesSpace $categoriesSpace $blogDoc) 13188 + #set($macro.blogDocs = $services.query.hql(~", BaseObject obj, StringProperty as categoriesLocationProp where obj.name = doc.fullName and obj.className = '$blogClassname' and obj.id=categoriesLocationProp.id.id and categoriesLocationProp.id.name='categoriesLocation' and categoriesLocationProp.value=:categoriesLocation order by doc.creationDate~").setLimit(1).setOffset(0).bindValue('categoriesLocation', $categoriesSpace).execute()) 13189 + #if($macro.blogDocs.size() > 0 && $categoriesSpace != $defaultBlogSpace) 13190 + #set($macro.result = $xwiki.getDocument($macro.blogDocs.get(0))) 13191 + #else 13192 + ## Fallback to Blog.WebHome, the default blog 13193 + #set($macro.result = $xwiki.getDocument('Blog.WebHome')) 13194 + #end 13195 + #set ($blogDoc = $NULL) 13196 + #setVariable (~"$blogDoc~" $macro.result) 13197 +#end" %) 13198 +((( 13199 +(% class="macro-placeholder hidden" %) 13200 +((( 13201 +macro:velocity 13202 +))) 13203 +))) 13204 +))) 13205 + 13206 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogPostLayoutMacros~"" %) 13207 +((( 13208 +(% class="macro-placeholder hidden" %) 13209 +((( 13210 +macro:include 13211 +))) 13212 + 13213 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|#** 13214 + * Extract the layout parameters from a string. 13215 + * 13216 + * @param layoutParamsString The string representation of the layout parameters. 13217 + * It should contain a String following this format ~"paramName1=Value1|paramName2=Value2|...|paramNameK=ValueK~" 13218 + * @param layoutsParameters The resulting layout parameters Map. 13219 + *### 13220 +#macro(extractLayoutParametersFromString $layoutParamsString $layoutsParameters) 13221 + #set ($layoutsParameters = $NULL) 13222 + #set ($macro.layoutParams = \{}) 13223 + #if (~"$!layoutParamsString~" != '') 13224 + #set ($macro.paramsArr = $layoutParamsString.split('\\|')) 13225 + #foreach ($item in $macro.paramsArr) 13226 + #set ($itemSplit = $item.split('=')) 13227 + #if ($itemSplit.size() == 2) 13228 + #set ($discard = $macro.layoutParams.put($itemSplit[0].trim(), $itemSplit[1].trim())) 13229 + #end 13230 + #end 13231 + #end 13232 + #setVariable(~"$layoutsParameters~" $macro.layoutParams) 13233 +#end" %) 13234 +((( 13235 +(% class="macro-placeholder hidden" %) 13236 +((( 13237 +macro:velocity 13238 +))) 13239 +))) 13240 +))) 13241 + 13242 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-| #macro(displaySmallPost $pDoc $pObj) 13243 + #initLayoutVars($pDoc $pObj) 13244 + <div class=~"col-xs-12 col-sm-6 next_blog~"> 13245 + <a href=~"$pDoc.uRL~" class=~"thumbnail~"> 13246 + <div class=~"row~"> 13247 + <div class=~"col-xs-4~"> 13248 + #if ($imageAtt) 13249 + <img src=~"$escapetool.xml($pDoc.getAttachmentURL($pObj.getValue('image'),'download',$imgQs))~" /> 13250 + #end 13251 + </div> 13252 + <div class=~"col-xs-8 art_det~"> 13253 + <p class=~"text-left~"> 13254 + #if($displayTitle)$!postTitle #end 13255 + <br/><span class=~"date_info~"> $!dateStr </span> 13256 + </p> 13257 + #displayPostDetails($pDoc) 13258 + </div> 13259 + </div> 13260 + </a> 13261 + </div> 13262 + #end 13263 + ## 13264 + ## 13265 + ## 13266 + #macro(displayPinnedPost $pDoc $pObj) 13267 + #initLayoutVars($pDoc $pObj) 13268 + <div class=~"col-xs-12 col-sm-4 col-md-4 blog_pin~"> 13269 + <a href=~"$pDoc.uRL~" class=~"all-post-link~"> 13270 + <div class=~"thumbnail~"> 13271 + #if ($displayTitle && $displayTitleFirstOnPinnedPosts) 13272 + <h5 class=~"title-top~">$!postTitle <br/> <span class=~"date_info~"> $!dateStr </span> </h5> 13273 + #end 13274 + #if ($imageAtt) 13275 + <img src=~"$escapetool.xml($pDoc.getAttachmentURL($pObj.getValue('image'),'download',$imgQs))~" /> 13276 + #end 13277 + <div class=~"caption~"> 13278 + #if ($displayTitle && !$displayTitleFirstOnPinnedPosts) 13279 + <h5 class=~"text-left~">$!postTitle <br/> <span class=~"date_info~"> $!dateStr </span> </h5> 13280 + #end 13281 + #if ($displaySummaryOnPinnedPosts) 13282 + <div class=~"text-left post-summary~"> 13283 + #set ($postContent = $pObj.getProperty('extract').value) 13284 + $!pDoc.getRenderedContent($postContent, $pDoc.syntax.toIdString()) 13285 + </div> 13286 + #end 13287 + #displayPostDetails($pDoc) 13288 + </div> 13289 + </div> 13290 + </a> 13291 + </div> 13292 + #end 13293 + ## 13294 + ## 13295 + ## 13296 + #macro(formatPostDate $pDoc $pObj) 13297 + #set ($formattedDate = $NULL) 13298 + #set($dateStr = $datetool.format('medium', $postDate, $xcontext.locale)) 13299 + #if (~"$!dateStr~" != '') 13300 + #set ($dateArr = $dateStr.split(' ')) 13301 + #if ($dateArr.size() > 3) 13302 + #set ($dateStr = ~"~") 13303 + #foreach($s in $dateArr.subList(0, 3)) 13304 + #set ($dateStr = ~"$\{dateStr}$\{s} ~") 13305 + #end 13306 + #end 13307 + #end 13308 + #setVariable(~"$formattedDate~" $dateStr) 13309 + #end 13310 + ## 13311 + ## 13312 + ## 13313 + #macro(displayPostDetails $pDoc) 13314 + <div class=~"row post-details~"> 13315 + <div class=~"col-xs-8 detail~"> 13316 + #if ($services.like.displayButton($pDoc.documentReference)) 13317 + #set ($optLikeRecord = $services.like.getLikes($pDoc.documentReference)) 13318 + ## Retrieve the likes number in XWiki 12.9+ 13319 + #set ($likeNumber = $optLikeRecord.get()) 13320 + #if (!$stringtool.isNumeric($likeNumber.toString())) 13321 + ## Retrieve the likes number in XWiki versions greater than 12.7 and lower than 12.9. 13322 + #set ($likeNumber = $optLikeRecord.get().likeNumber) 13323 + #end 13324 + #if ($stringtool.isNumeric($likeNumber.toString())) 13325 + <div class=~"post-likes btn btn-default disabled badge~" 13326 + title=~"$escapetool.xml($services.localization.render('like.button.title', [$likeNumber]))~"> 13327 + $services.icon.renderHTML('heart') <span class=~"like-number~">$likeNumber</span> 13328 + </div> 13329 + #end 13330 + #elseif ($services.ratings) 13331 + #set ($ratingsConfigDoc = $services.ratings.getConfigurationDocument($pDoc.documentReference)) 13332 + #if ($ratingsConfigDoc.getValue('displayRatings') == 1) 13333 + <ul class=~"pull-left note list-inline~"> 13334 + #foreach ($x in [1..5]) 13335 + #set ($cls = ~"~") 13336 + #if ($x > $averageVote) 13337 + #set ($cls = ~"-o~") 13338 + #end 13339 + <li><span class=~"fa fa-star$\{cls}~"></span></li> 13340 + #end 13341 + </ul> 13342 + #end 13343 + #end 13344 + </div> 13345 + #if ($showPostComments) 13346 + <div class=~"col-xs-4 com_det~"> 13347 + <p class=~"text-right~">$services.icon.renderHTML('comments') ($!nbComments)</p> 13348 + </div> 13349 + #end 13350 + </div> 13351 + #end 13352 + ## 13353 + ## 13354 + ## 13355 + #macro(initLayoutVars $pDoc $pObj) 13356 + #set ($postTitle = $pDoc.display('title', 'view', $pObj)) 13357 + #set ($imageAtt = $pDoc.getAttachment($pObj.getValue('image'))) 13358 + #isPublished($pObj $isPublished) 13359 + #isHidden($pObj $isHidden) 13360 + #getEntryDate($pDoc $pObj $postDate) 13361 + #formatPostDate($postDate $dateStr) 13362 + #set ($averageVote = $mathtool.round($services.ratings.getAverageRating($pDoc.documentReference).getAverageVote())) 13363 + #set ($nbComments = $pDoc.getComments().size()) 13364 + #set ($showPostComments = true) 13365 + #if ($xwiki.getSpacePreferenceFor('showcomments', $pDoc.documentReference.lastSpaceReference, '1') == '0') 13366 + #set ($showPostComments = false) 13367 + #end 13368 + #end 13369 + ## 13370 + ## 13371 + ## 13372 + #macro(displayEditPinnedPostsButton) 13373 + #if (~"$!macroAdditionalParams.get('postIndex')~" == '0' && $services.security.authorization.hasAccess('edit', $pinnedPostsSourceDoc.fullName)) 13374 + #set ($discard = $xwiki.ssx.use('Blog.PinnedPostsSheet')) 13375 + #set ($discard = $xwiki.jsx.use('Blog.PinnedPostsSheet')) 13376 + #set ($editPinnedPostsURL = $xwiki.getURL($pinnedPostsSourceDoc.fullName, 'get', 'sheet=Blog.PinnedPostsSheet')) 13377 + #set ($newPinnedPostsObjectURL = $xwiki.getURL($pinnedPostsSourceDoc.fullName, 'objectadd', ~"classname=Blog.PinnedPostsClass&form_token=$!\{services.csrf.getToken()}~")) 13378 + <div class=~"edit-pinned-posts-container~"> 13379 + <div class=~"col-xs-12 col-sm-12 col-md-12 text-right edit-pinned-posts~"> 13380 + <a href=~"javascript:;~" class=~"editPinnedPosts~"> 13381 + <span class=~"glyphicon glyphicon-edit~" aria-hidden=~"true~"></span> $services.localization.render('blog.post.layout.cards.pinnedposts.edit') 13382 + </a> 13383 + <input type=~"hidden~" id=~"editPinnedPostsURL~" value=~"$escapetool.xml($editPinnedPostsURL)~"/> 13384 + <input type=~"hidden~" id=~"newPinnedPostsObjectURL~" value=~"$escapetool.xml($newPinnedPostsObjectURL)~"/> 13385 + </div> 13386 + </div> 13387 + #if (~"$!pinnedPostsObj~" == '') 13388 + <input type=~"hidden~" name=~"createPinnedPostsObject~" value=~"1~"/> 13389 + #end 13390 + #end 13391 + #end" %) 13392 +((( 13393 +(% class="macro-placeholder hidden" %) 13394 +((( 13395 +macro:velocity 13396 +))) 13397 +))) 13398 + 13399 +(% class="macro" data-macro="startmacro:velocity|-||-| #if ($services.security.authorization.hasAccess('view', $xcontext.macro.params.reference)) 13400 + #set ($postDoc = $xwiki.getDocument($xcontext.macro.params.reference)) 13401 + #getEntryObject($postDoc $postObj) 13402 + #if (~"$!postObj~" != '') 13403 + #extractLayoutParametersFromString($xcontext.macro.params.params $macroAdditionalParams) 13404 + ## 13405 + #set ($displayTitle = true) 13406 + #if (~"$!macroAdditionalParams.get('displayTitle')~" == 'false') 13407 + #set ($displayTitle = false) 13408 + #end 13409 + ## 13410 + #set ($displayTitleFirstOnPinnedPosts = false) 13411 + #if (~"$!macroAdditionalParams.get('displayTitleFirstOnPinnedPosts')~" == 'true') 13412 + #set ($displayTitleFirstOnPinnedPosts = true) 13413 + #end 13414 + ## 13415 + #set ($displaySummaryOnPinnedPosts = true) 13416 + #if (~"$!macroAdditionalParams.get('displaySummaryOnPinnedPosts')~" == 'false') 13417 + #set ($displaySummaryOnPinnedPosts = false) 13418 + #end 13419 + ## 13420 + #set ($postIndex = $numbertool.toNumber($macroAdditionalParams.get('postIndex')).intValue()) 13421 + #if ($postIndex == 0) 13422 + #set ($stopBlogPostsDisplay = false) 13423 + #end 13424 + #set ($listResultsCount = $numbertool.toNumber($macroAdditionalParams.get('list-results-count')).intValue()) 13425 + #set($discard = $xwiki.ssx.use('Blog.BlogPostLayoutCards')) 13426 + #set($scaleWidth = 600) 13427 + #set($imgQs=~"width=$scaleWidth~") 13428 + ## Display pinned posts 13429 + ## Get the list of pinned posts 13430 + #set ($pinnedPosts = []) 13431 + #set ($pinnedPostsSourceDoc = $NULL) 13432 + #if (~"$!macroAdditionalParams.get('forceDisplayPinnedPosts')~" == 'true' && ~"$!macroAdditionalParams.get('list-blog')~" != '') 13433 + #set ($pinnedPostsSourceDoc = $xwiki.getDocument($macroAdditionalParams.get('list-blog'))) 13434 + ## Note that the 'list-blog' parameter contains the value of the 'blog' parameter of the blogpostlist macro passed to the current layout macro 13435 + #end 13436 + #if (~"$!pinnedPostsSourceDoc~" != '') 13437 + #set ($pinnedPostsObj = $pinnedPostsSourceDoc.getObject('Blog.PinnedPostsClass')) 13438 + #if (~"$!pinnedPostsObj~" != '') 13439 + #set ($orderedPinnedPostsJSON = $pinnedPostsObj.getProperty('orderedPosts').value) 13440 + #if (~"$!orderedPinnedPostsJSON~" != '') 13441 + #set ($pinnedPosts = $jsontool.parse($orderedPinnedPostsJSON)) 13442 + #else 13443 + #set ($pinnedPosts = $pinnedPostsObj.getProperty('posts').value) 13444 + #end 13445 + #end 13446 + #if ($postIndex == 0 && $pinnedPosts.size() > 0) 13447 + \{\{html clean=~"false~"}} 13448 + #set ($x = 0) 13449 + #set ($showPinnedPostsButton = true) 13450 + #foreach ($pinnedPost in $pinnedPosts) 13451 + #if ($x == 0) 13452 + <div class=~"row flex-container~"> 13453 + #if ($showPinnedPostsButton) 13454 + #displayEditPinnedPostsButton() 13455 + #set($showPinnedPostsButton = false) 13456 + #end 13457 + #end 13458 + #set ($pinnedPostDoc = $xwiki.getDocument($pinnedPost)) 13459 + #getEntryObject($pinnedPostDoc $pinnedPostObj) 13460 + #if (~"$!pinnedPostObj~" != '') 13461 + #displayPinnedPost($pinnedPostDoc $pinnedPostObj) 13462 + #end 13463 + #set ($x = $mathtool.add($x, 1)) 13464 + #if ($x == 3) 13465 + #set ($x = 0) 13466 + </div> 13467 + #end 13468 + #end 13469 + #if ($mathtool.mod($x, 3) != 0) 13470 + </div> 13471 + #end 13472 + \{\{/html}} 13473 + 13474 + ## If the first post is a pinned post : this means that all the posts are pinned 13475 + ## In this case, avoid displaying the posts again after the pinned posts section. 13476 + #if ($pinnedPosts.contains($xcontext.macro.params.reference)) 13477 + #set ($stopBlogPostsDisplay = true) 13478 + #end 13479 + #elseif($postIndex == 0 && $pinnedPosts.size() == 0) 13480 + 13481 + \{\{html}} 13482 + <div class=~"row no-pinnded-posts~"> 13483 + #displayEditPinnedPostsButton() 13484 + </div> 13485 + \{\{/html}} 13486 + 13487 + #end 13488 + #end 13489 + #if (!$stopBlogPostsDisplay) 13490 + \{\{html clean=~"false~"}} 13491 + #if (~"$!nbDisplayedPosts~" == '' || $postIndex == 0) 13492 + #set ($nbDisplayedPosts = 0) 13493 + #end 13494 + #if($mathtool.mod($nbDisplayedPosts, 2) == 0) 13495 + #if ($nbDisplayedPosts != 0) 13496 + </div> 13497 + #set ($lastHtmlTag = 'c') 13498 + #end 13499 + #if ($nbDisplayedPosts == $mathtool.sub($listResultsCount, 1)) 13500 + <div class=~"row~"> 13501 + #set ($lastHtmlTag = 'o') 13502 + #set ($lastRowClass = ~"~") 13503 + #else 13504 + <div class=~"row flex-container~"> 13505 + #set ($lastHtmlTag = 'o') 13506 + #set ($lastRowClass = ~"flex~") 13507 + #end 13508 + #end 13509 + #displaySmallPost($postDoc $postObj) 13510 + #if ($nbDisplayedPosts == $mathtool.sub($listResultsCount, 1)) 13511 + </div> 13512 + #set ($lastHtmlTag = 'c') 13513 + #end 13514 + #set ($nbDisplayedPosts = $mathtool.add($nbDisplayedPosts, 1)) 13515 + #if ($postIndex == $mathtool.sub($listResultsCount, 1)) 13516 + #if (~"$!lastHtmlTag~" == 'o') 13517 + #if ($mathtool.mod($nbDisplayedPosts, 2) == 1 && $lastRowClass == ~"flex~") 13518 + <div class=~"col-xs-12 col-sm-6~"></div> 13519 + #end 13520 + </div> 13521 + #end 13522 + #end 13523 + \{\{/html}} 13524 + #end 13525 + #else 13526 + \{\{error}}$services.localization.render('blog.blogpostlayout.notpost', [$xcontext.macro.params.reference])\{\{/error}} 13527 + #end 13528 + #else 13529 + \{\{error}}$services.localization.render('blog.blogpostlayout.post_view_not_allowed', [$xcontext.macro.params.reference])\{\{/error}} 13530 + #end" %) 13531 +((( 13532 +(% class="macro-placeholder hidden" %) 13533 +((( 13534 +macro:velocity 13535 +))) 13536 + 13537 +(% class="macro" data-macro="startmacro:html|-|clean=~"false~"|-|<div class=~"col-xs-12 col-sm-6 next_blog~"> 13538 +<a href=~"/Nieuws/Nieuwsbrief%20oktober%202023~" class=~"thumbnail~"> 13539 +<div class=~"row~"> 13540 +<div class=~"col-xs-4~"> 13541 +<img src=~"/download/Nieuws/Nieuwsbrief%20oktober%202023/fall-flower-orange-food-harvest-produce-858531-pxhere.com.jpg?width=600&rev=1.1~" /> 13542 +</div> 13543 +<div class=~"col-xs-8 art_det~"> 13544 +<p class=~"text-left~"> 13545 +Nieuwsbrief oktober 2023 <br/><span class=~"date_info~"> Nov 28, 2023, </span> 13546 +</p> 13547 +<div class=~"row post-details~"> 13548 +<div class=~"col-xs-8 detail~"> 13549 +<div class=~"post-likes btn btn-default disabled badge~" 13550 +title=~"Number of likes on this page: 1~"> 13551 +<span class=~"fa fa-heart~"></span> <span class=~"like-number~">1</span> 13552 +</div> 13553 +</div> 13554 +<div class=~"col-xs-4 com_det~"> 13555 +<p class=~"text-right~"><span class=~"fa fa-comments~"></span> (0)</p> 13556 +</div> 13557 +</div> 13558 +</div> 13559 +</div> 13560 +</a> 13561 +</div>" %) 13562 +((( 13563 +(% class="macro-placeholder hidden" %) 13564 +((( 13565 +macro:html 13566 +))) 13567 + 13568 +(% class="col-xs-12 col-sm-6 next_blog" %) 13569 +((( 13570 +(% class="row" %) 13571 +((( 13572 +(% class="col-xs-4" %) 13573 +((( 13574 +[[~[~[image:/download/Nieuws/Nieuwsbrief%20oktober%202023/fall-flower-orange-food-harvest-produce-858531-pxhere.com.jpg?width=600&rev=1.1~]~]>>path:/Nieuws/Nieuwsbrief%20oktober%202023||class="thumbnail"]] 13575 +))) 13576 + 13577 +(% class="col-xs-8 art_det" %) 13578 +((( 13579 +(% class="text-left" %) 13580 +[[Nieuwsbrief oktober 2023 13581 +(% class="date_info" %) Nov 28, 2023,>>path:/Nieuws/Nieuwsbrief%20oktober%202023||class="thumbnail"]] 13582 + 13583 +(% class="row post-details" %) 13584 +((( 13585 +(% class="col-xs-8 detail" %) 13586 +((( 13587 +(% class="post-likes btn btn-default disabled badge" title="Number of likes on this page: 1" %) 13588 +((( 13589 +[[(% class="like-number" %)1>>path:/Nieuws/Nieuwsbrief%20oktober%202023||class="thumbnail"]] 13590 +))) 13591 +))) 13592 + 13593 +(% class="col-xs-4 com_det" %) 13594 +((( 13595 +(% class="text-right" %) 13596 +[[(0)>>path:/Nieuws/Nieuwsbrief%20oktober%202023||class="thumbnail"]] 13597 +))) 13598 +))) 13599 +))) 13600 +))) 13601 +))) 13602 +))) 13603 +))) 13604 +))) 13605 + 13606 +(% class="macro" data-macro="startmacro:blogPostLayoutCards|-|reference=~"Nieuws.Nieuw beveiligingspaneel Axon van Aritech~" params=~"displayTitle=true|useSummary=true|list-limit=10|list-layout=cards|list-blog=Nieuws.WebHome|list-paginated=no|list-results-count=10|postIndex=0|previousPostReference=|postIndex=1|previousPostReference=Nieuws.Nieuwsbrief januari 2024|postIndex=2|previousPostReference=Nieuws.Audit BMI en BORG|postIndex=3|previousPostReference=Nieuws.Nieuwsbrief december 2023|postIndex=4|previousPostReference=Nieuws.Nieuwsbrief november 2023|postIndex=5|previousPostReference=Nieuws.Feestelijke afsluiting, keuring materieel en tellen voorraden|postIndex=6|previousPostReference=Nieuws.RI&E 2023|postIndex=7|previousPostReference=Nieuws.Aankondiging update Syntess 6\\\\.9\\\\.0181|postIndex=8|previousPostReference=Nieuws.Nieuwsbrief oktober 2023~"" %) 13607 +((( 13608 +(% class="macro-placeholder hidden" %) 13609 +((( 13610 +macro:blogPostLayoutCards 13611 +))) 13612 + 13613 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogCode~"" %) 13614 +((( 13615 +(% class="macro-placeholder hidden" %) 13616 +((( 13617 +macro:include 13618 +))) 13619 + 13620 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogParameters~"" %) 13621 +((( 13622 +(% class="macro-placeholder hidden" %) 13623 +((( 13624 +macro:include 13625 +))) 13626 + 13627 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|## Blog 13628 +#set($blogClassname = 'Blog.BlogClass') 13629 +#set($blogTemplate = 'Blog.BlogTemplate') 13630 +#set($blogSheet = 'Blog.BlogSheet') 13631 +## Blog entries 13632 +#set($blogPostClassname = 'Blog.BlogPostClass') 13633 +#set($blogPostTemplate = 'Blog.BlogPostTemplate') 13634 +#set($blogPostSheet = 'Blog.BlogPostSheet') 13635 +#set($blogPostObjectNumber = $xwiki.getDocument($blogPostTemplate).getObject($blogPostClassname).number) 13636 +#set($oldArticleClassname = 'XWiki.ArticleClass') 13637 +## Categories 13638 +#set($blogCategoryClassname = 'Blog.CategoryClass') 13639 +#set($blogCategoryTemplate = 'Blog.CategoryTemplate') 13640 +#set($blogCategorySheet = 'Blog.CategorySheet') 13641 +#set($blogCategoriesSheet = 'Blog.CategoriesSheet') 13642 +#set($oldBlogCategoryClassname = 'Blog.Categories') 13643 +#set($defaultCategoryParent = 'Blog.Categories') 13644 +## Style 13645 +#set($blogStyleDocumentName = 'Blog.BlogStyle') 13646 +#set($blogStyleDocument = $xwiki.getDocument($blogStyleDocumentName)) 13647 +## Clientside scripts 13648 +#set($blogScriptsDocumentName = 'Blog.BlogScripts') 13649 +#set($blogPublisher = $xwiki.getDocument('Blog.Publisher')) 13650 +## Misc 13651 +#set($thisURL = $doc.getURL($xcontext.action, $request.queryString)) 13652 +#set($isBlogPost = $doc.getObject($blogPostClassname)) 13653 +#set($defaultBlogSpace = 'Blog') 13654 +#set($isCategoryPage = $doc.getObject($blogCategoryClassname)) 13655 +#set($isCategoriesHomePage = $doc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 13656 +## 13657 +## 13658 +## 13659 +#** 13660 + * Displays an image, taken from the blog style document. 13661 + * 13662 + * @param $imgName The name of the icon from icons set to use. 13663 + *# 13664 +#macro(toolImage $imgName)(% class=~"icon-manage~"~%)$services.icon.render($imgName)(%~%)#end" %) 13665 +((( 13666 +(% class="macro-placeholder hidden" %) 13667 +((( 13668 +macro:velocity 13669 +))) 13670 +))) 13671 +))) 13672 + 13673 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|## 13674 +## 13675 +## 13676 +## Import the blog skin and javascripts. 13677 +$!xwiki.ssx.use($blogStyleDocumentName)## 13678 +$!xwiki.jsx.use($blogScriptsDocumentName)## 13679 +## 13680 +## import the hierarchy for the path the blog post footer (in displayEntryBlogLocation) 13681 +#template('hierarchy_macros.vm')## 13682 +## 13683 +## 13684 +#** 13685 + * Prints a blog. This is the main macro used in the BlogSheet. 13686 + * 13687 + * @param blogDoc the XDocument holding the blog definition object. 13688 + *### 13689 +#macro(printBlog $blogDoc) 13690 + \{\{include reference='Blog.CreatePost'/}} 13691 + 13692 + ## Use the blogPostList macro to display the blogposts 13693 + ## The blogPostList is used only in case of 'all' or 'paged' blog display type because the blogPostList macro 13694 + ## do not support FTM the monthly and weekly blog display types 13695 + #getBlogDisplayType($blogDoc $displayType) 13696 + #if ($displayType == 'weekly' || $displayType == 'monthly') 13697 + #getBlogEntries($blogDoc $entries) 13698 + #displayBlog($entries 'index' true true) 13699 + #displayNavigationLinks($blogDoc) 13700 + #else 13701 + #getBlogDisplayType($blogDoc $displayType) 13702 + #set ($paginated = 'no') 13703 + #if ($displayType == 'paginated') 13704 + #set ($paginated = 'yes') 13705 + #end 13706 + #getBlogPostsLayout($blogDoc $postsLayout) 13707 + (% class=~"hfeed index~" ~%)(((\{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$paginated~" layout=~"$!postsLayout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" /}}))) 13708 + #end 13709 +#end 13710 +## 13711 +## 13712 +## 13713 +#** 13714 + * Shows blog information. In view mode, the description is printed. In edit mode, allows changing blog settings: title, 13715 + * description, blog type (global or in-space), index display type (fixed size pagination, weekly index, monthly index, 13716 + * all entries). 13717 + * 13718 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 13719 + *### 13720 +#macro(showBlogInfo $blogDoc) 13721 + #if($blogDoc.getObject($blogClassname)) 13722 + ## Keep testing for inline action for backward compatibility with older blogs. 13723 + #if($xcontext.action == 'edit' || $xcontext.action == 'inline') 13724 + #macro(displayProperty $blogDoc $propname) 13725 + ; #displayPropName($xwiki.getClass($blogClassname).get($propname)): 13726 + : $blogDoc.display($propname) 13727 + #end 13728 + #displayProperty($blogDoc 'title') 13729 + #displayProperty($blogDoc 'description') 13730 + #displayProperty($blogDoc 'displayType') 13731 + #displayProperty($blogDoc 'itemsPerPage') 13732 + #displayProperty($blogDoc 'postsLayout') 13733 + #displayProperty($blogDoc 'postsLayoutParameters') 13734 + #else 13735 + $blogDoc.display('description') 13736 + #end 13737 + #elseif($doc.fullName == $blogSheet) 13738 += $services.localization.render('blog.code.blogsheet') = 13739 + \{\{translation key='blog.code.sheetexplanation'/}} 13740 + #else 13741 + \{\{warning}}\{\{translation key='blog.code.notblog'/}}\{\{/warning}} 13742 + #end 13743 +#end 13744 +## 13745 +## 13746 +## 13747 +#** 13748 + * Retrieve the blog document, which usually is either <tt><Space>.WebHome</tt> for whole-spaces blogs, or 13749 + * <tt><Space>.Blog</tt> for in-space blogs. If none of these documents contains a blog object, then the first 13750 + * (alphabetically) document in the target space that contains one is returned. Finally, if no document in the current 13751 + * space contains a blog object, then <tt>Blog.WebHome</tt> is returned as the default blog. 13752 + * 13753 + * @param space A <tt>String</tt>, the name of the space where to search. 13754 + * @param blogDoc The resulting XDocument. 13755 + *### 13756 +#macro(getBlogDocument $space $blogDoc) 13757 + #set ($result = $NULL) 13758 + ## Check if the current space has a WebPreferences page that contains a Blog.EnablePanelsConfigurationClass object. 13759 + ## It is possible to display a blog's panels info in a page that is not inside a blog and in that case we need to 13760 + ## identify the right blog based on a configuration object in a WebPreferences page. 13761 + #set ($spaceReference = $services.model.resolveSpace($space)) 13762 + #set ($preferencesDocRef = $services.model.createDocumentReference('WebPreferences', $spaceReference)) 13763 + #set ($preferencesDoc = $xwiki.getDocument($preferencesDocRef)) 13764 + #set ($preferencesObj = $preferencesDoc.getObject('Blog.EnablePanelsConfigurationClass')) 13765 + #if ($preferencesObj) 13766 + #set ($result = $xwiki.getDocument( $preferencesObj.getValue('blog'))) 13767 + #end 13768 + ## If no special case occurs (like the previous one), identify a blog by checking the current location from its hierarchy. 13769 + #if (~"$!result~" == '') 13770 + ## First, try the Space.WebHome, for a whole-space blog 13771 + #set($result = $xwiki.getDocument(~"$\{space}.WebHome~")) 13772 + #if(!$result.getObject($blogClassname)) 13773 + ## Second, try the Space.Blog document 13774 + #set($result = $xwiki.getDocument(~"$\{space}.Blog~")) 13775 + #if(!$result.getObject($blogClassname)) 13776 + ## Third, try searching for a blog document in the current space 13777 + ## Prevent the query fail when the space contains dots '.' 13778 + #set($blogDocs = $services.query.hql(~", BaseObject obj where doc.space = :space and obj.name = doc.fullName and obj.className = '$blogClassname' order by doc.name~").setLimit(1).setOffset(0).bindValue('space', $space).execute()) 13779 + #if($blogDocs.size() > 0) 13780 + #set($result = $xwiki.getDocument($blogDocs.get(0))) 13781 + #else 13782 + ## Fourth, try searching for a blog document that have the its 'postsLocation' set to the current space 13783 + #set($blogDocs = $services.query.hql(~", BaseObject obj, StringProperty as postsLocationProp where obj.name = doc.fullName and obj.className = '$blogClassname' and obj.id=postsLocationProp.id.id and postsLocationProp.id.name='postsLocation' and postsLocationProp.value=:postsLocation order by doc.name~").setLimit(1).setOffset(0).bindValue('postsLocation', $space).execute()) 13784 + #if($blogDocs.size() > 0) 13785 + #set($result = $xwiki.getDocument($blogDocs.get(0))) 13786 + #else 13787 + ## Last, fallback to Blog.WebHome, the default blog 13788 + #set($result = $xwiki.getDocument('Blog.WebHome')) 13789 + #end 13790 + #end 13791 + #end 13792 + #end 13793 + #end 13794 + #set ($blogDoc = $NULL) 13795 + #setVariable (~"$blogDoc~" $result) 13796 +#end 13797 +## 13798 +## 13799 +## 13800 +#** 13801 + * Retrieve the blog title. 13802 + * 13803 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass<tt> object with the <tt>title</tt> property set. 13804 + * @param title The resulting title. 13805 + *### 13806 +#macro(getBlogTitle $blogDoc $title) 13807 + ## Titles can contain velocity code (most commonly translations), so we should evaluate them. 13808 + #set ($title = $NULL) 13809 + #setVariable (~"$title~" $!blogDoc.displayTitle) 13810 +#end 13811 +## 13812 +## 13813 +## 13814 +#** 13815 + * Retrieve the blog description. 13816 + * 13817 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>description</tt> 13818 + * property set. 13819 + * @param description The resulting description. 13820 + *### 13821 +#macro(getBlogDescription $blogDoc $description) 13822 + #getBlogProperty($blogDoc 'description' '' $result) 13823 + #set ($description = $NULL) 13824 + #setVariable (~"$description~" $result) 13825 +#end 13826 +## 13827 +## 13828 +## 13829 +#** 13830 + * Retrieves a list of entries to be displayed. The entries are either part of the blog's space, or have the blog 13831 + * document set as a parent. The number and range of entries returned (from all those belonging to this blog) depends on 13832 + * the blog display type: paginated (fixed number of entries), weekly (all entries in a week), monthly (all entries in a 13833 + * month), or all. 13834 + * 13835 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 13836 + * @param entries The resulting list of entries to display, a list of XDocument names. 13837 + *### 13838 +#macro(getBlogEntries $blogDoc $entries) 13839 + #if (!$entries) 13840 + #setVariable (~"$entries~" []) 13841 + #end 13842 + #getAllBlogPostsQuery($query) 13843 + #isDefaultBlog($blogDoc $isDefault) 13844 + #set($queryParams = \{}) 13845 + #if ($isDefault) 13846 + #getCategoryAllBlogPostsQuery($query) 13847 + #set($query = ~"$\{query} and (doc.creator = :creator or (isPublished.value = 1 and hidden.value = 0)) and (doc.space = :space or catList like :catList escape '!')~") 13848 + #set($discard = $queryParams.put('creator', $xcontext.user)) 13849 + #set($discard = $queryParams.put('space', $blogDoc.space)) 13850 + #set($sanitizedBlogSpace = $blogDoc.space.replaceAll('([%_!])', '!$1').concat('.%')) 13851 + #set($discard = $queryParams.put('catList', $sanitizedBlogSpace)) 13852 + #else 13853 + #set($query = ~"$\{query} and (doc.space = :space or doc.parent = :parent)~") 13854 + #getBlogPostsLocation($blogDoc.space $blogPostsLocation) 13855 + #set($discard = $queryParams.put('space', $blogPostsLocation)) 13856 + #set($discard = $queryParams.put('parent', $blogDoc.space)) 13857 + #end 13858 + #getBlogDisplayType($blogDoc $displayType) 13859 + #if($displayType == 'weekly') 13860 + #getWeeklyBlogEntries($blogDoc $query $entries $queryParams) 13861 + #elseif($displayType == 'monthly') 13862 + #getMonthlyBlogEntries($blogDoc $query $entries $queryParams) 13863 + #elseif($displayType == 'all') 13864 + #getAllBlogEntries($blogDoc $query $entries $queryParams) 13865 + #else 13866 + #getPagedBlogEntries($blogDoc $query $entries $queryParams) 13867 + #end 13868 +#end 13869 +## 13870 +## 13871 +## 13872 +#** 13873 + * Retrieves a list of entries to be displayed. The entries are taken from a ~"page~" of the blog, a sequence of documents 13874 + * defined by the request parameters <tt>ipp</tt> (items per page) and <tt>page</tt> (the current page). Initially the 13875 + * first page is displayed, with the number of entries defined in the blog object in the <tt>itemsPerPage</tt> property 13876 + * (10 if not defined). 13877 + * 13878 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 13879 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 13880 + * refined to restrict to a given space, or to a given search criteria, etc. 13881 + * @param entries The resulting list of entries to display, a list of XDocument names. 13882 + * @param queryParams The parameters to bind with the query. 13883 + *### 13884 +#macro(getPagedBlogEntries $blogDoc $query $entries $queryParams) 13885 + #if (!$entries) 13886 + #setVariable (~"$entries~" []) 13887 + #end 13888 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 13889 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~")) 13890 + #bindQueryParameters($countQueryObj $queryParams) 13891 + #bindQueryParameters($queryObj $queryParams) 13892 + #set($totalEntries = $countQueryObj.count()) 13893 + #getBlogProperty($blogDoc 'itemsPerPage' '10' $defaultItemsPerPage) 13894 + #set($defaultItemsPerPage = $numbertool.toNumber($defaultItemsPerPage).intValue()) 13895 + ## This macro is defined in the default macros.vm library. It also sets $itemsPerPage and $startAt. 13896 + #preparePagedViewParams($totalEntries $defaultItemsPerPage) 13897 + #set($discard = $entries.addAll($queryObj.setLimit($itemsPerPage).setOffset($startAt).addFilter(~"unique~").execute())) 13898 +#end 13899 +## 13900 +## 13901 +## 13902 +#** 13903 + * Retrieves a list of entries to be displayed. The entries are taken from a week of the blog. The target week is 13904 + * defined by the request parameters <tt>week</tt> (the week number in the year, from 1 to 52) and <tt>year</tt> (4 13905 + * digit year). Initially the current week is displayed. 13906 + * 13907 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 13908 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 13909 + * refined to restrict to a given space, or to a given search criteria, etc. 13910 + * @param entries The resulting list of entries to display, a list of XDocument names. 13911 + * @param queryParams The parameters to bind with the query. 13912 + *### 13913 +#macro(getWeeklyBlogEntries $blogDoc $query $entries $queryParams) 13914 + #if (!$entries) 13915 + #setVariable (~"$entries~" []) 13916 + #end 13917 + #getRequestedWeek($weekDate) 13918 + #set($dateFormatter = $xwiki.jodatime.getDateTimeFormatterForPattern('yyyy-MM-dd')) 13919 + #set($minDay = $dateFormatter.print($weekDate.toMutableDateTime().weekOfWeekyear().roundFloor())) 13920 + #set($maxDay = $dateFormatter.print($weekDate.toMutableDateTime().weekOfWeekyear().roundCeiling())) 13921 + #set($query = ~"$\{query} and publishDate.value >= '$minDay' and publishDate.value < '$maxDay'~") 13922 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 13923 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 13924 + #bindQueryParameters($countQueryObj $queryParams) 13925 + #bindQueryParameters($queryObj $queryParams) 13926 + #set($totalEntries = $countQueryObj.count()) 13927 + #set($discard = $entries.addAll($queryObj.execute())) 13928 +#end 13929 +## 13930 +## 13931 +## 13932 +#** 13933 + * Retrieves a list of entries to be displayed. The entries are taken from a month of the blog. The target month is 13934 + * defined by the request parameters <tt>month</tt> (the month number, from 1 to 12) and <tt>year</tt> (4 13935 + * digit year). Initially the current month is displayed. 13936 + * 13937 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 13938 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 13939 + * refined to restrict to a given space, or to a given search criteria, etc. 13940 + * @param entries The resulting list of entries to display, a list of XDocument names. 13941 + * @param queryParams The parameters to bind with the query. 13942 + *### 13943 +#macro(getMonthlyBlogEntries $blogDoc $query $entries) 13944 + #if (!$entries) 13945 + #setVariable (~"$entries~" []) 13946 + #end 13947 + #getRequestedMonth($monthDate) 13948 + #set($dateFormatter = $xwiki.jodatime.getDateTimeFormatterForPattern('yyyy-MM-dd')) 13949 + #set($minDay = $dateFormatter.print($monthDate.toMutableDateTime().monthOfYear().roundFloor())) 13950 + #set($maxDay = $dateFormatter.print($monthDate.toMutableDateTime().monthOfYear().roundCeiling())) 13951 + #set($query = ~"$\{query} and publishDate.value >= '$minDay' and publishDate.value < '$maxDay'~") 13952 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 13953 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 13954 + #bindQueryParameters($countQueryObj $queryParams) 13955 + #bindQueryParameters($queryObj $queryParams) 13956 + #set($totalEntries = $countQueryObj.count()) 13957 + #set($discard = $entries.addAll($queryObj.execute())) 13958 +#end 13959 +## 13960 +## 13961 +## 13962 +#** 13963 + * Retrieves a list of entries to be displayed. All entries belonging to the current blog are returned. 13964 + * 13965 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 13966 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 13967 + * refined to restrict to a given space, or to a given search criteria, etc. 13968 + * @param entries The resulting list of entries to display, a list of XDocument names. 13969 + * @param queryParams The parameters to bind with the query. 13970 + *### 13971 +#macro(getAllBlogEntries $blogDoc $query $entries $queryParams) 13972 + #if (!$entries) 13973 + #setVariable (~"$entries~" []) 13974 + #end 13975 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 13976 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 13977 + #bindQueryParameters($countQueryObj $queryParams) 13978 + #bindQueryParameters($queryObj $queryParams) 13979 + #set($totalEntries = $countQueryObj.count()) 13980 + #set($discard = $entries.addAll($queryObj.execute())) 13981 +#end 13982 +## 13983 +## 13984 +## 13985 +#** 13986 + * Retrieves a list of entries to be displayed. Only (and all) unpublished entries are returned. 13987 + * 13988 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 13989 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 13990 + * refined to restrict to a given space, or to a given search criteria, etc. 13991 + * @param queryParams The parameters to bind with the query. 13992 + * @param entries The resulting list of entries to display, a list of XDocument names. 13993 + *### 13994 +#macro(getUnpublishedBlogEntries $blogDoc $query $queryParams $entries) 13995 + #if (!$entries) 13996 + #setVariable (~"$entries~" []) 13997 + #end 13998 + #set($query = ~"$\{query} and isPublished.value = 0~") 13999 + #set ($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 14000 + #set ($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 14001 + #bindQueryParameters($countQueryObj $queryParams) 14002 + #bindQueryParameters($queryObj $queryParams) 14003 + #set($totalEntries = $countQueryObj.count()) 14004 + #set($discard = $entries.addAll($queryObj.execute())) 14005 +#end 14006 +## 14007 +## 14008 +## 14009 +#** 14010 + * Retrieves a list of entries to be displayed. The entries are taken from all the wiki, and not from a specific blog. 14011 + * 14012 + * @param entries The resulting list of entries to display, a list of XDocument names. 14013 + *### 14014 +#macro(getGlobalBlogEntries $entries) 14015 + #if (!$entries) 14016 + #setVariable (~"$entries~" []) 14017 + #end 14018 + #getAllBlogPostsQuery($query) 14019 + #set($totalEntries = $services.query.hql($query).count()) 14020 + #set($defaultItemsPerPage = 20) 14021 + ## This macro is defined in the default macros.vm library. It also sets $itemsPerPage and $startAt. 14022 + #preparePagedViewParams($totalEntries $defaultItemsPerPage) 14023 + #set($discard = $entries.addAll($services.query.hql(~"$\{query} order by publishDate.value desc~").setLimit($itemsPerPage).setOffset($startAt).execute())) 14024 +#end 14025 +#** 14026 + * Return the base query for selecting blog entries. It filters only visible entries, but does not bind to a specific 14027 + * blog, nor specify a range or an ordering criteria. 14028 + * 14029 + * This macro is a duplicate of #getAllBlogPostsQuery($query). Keep it in case it's used outside, in custom code. 14030 + * 14031 + * @param query The basic query for selecting blog entries. 14032 + *# 14033 +#macro(getBlogEntriesBaseQuery $query) 14034 + #getAllBlogPostsQuery($query) 14035 +#end 14036 +#** 14037 + * Return the Query for selecting the all wiki blog posts without filtering 14038 + * 14039 + * @param query The basic query for selecting blog entries. 14040 + *# 14041 +#macro(getAllBlogPostsQuery $query) 14042 + #set ($query = $NULL) 14043 + #setVariable(~"$query~" ~", BaseObject as obj, IntegerProperty isPublished, 14044 + IntegerProperty hidden, DateProperty publishDate 14045 + where doc.fullName <> '$blogPostTemplate' and 14046 + obj.name=doc.fullName and obj.className='$blogPostClassname' and 14047 + isPublished.id.id = obj.id and isPublished.id.name = 'published' and 14048 + hidden.id.id = obj.id and hidden.id.name='hidden' and 14049 + publishDate.id.id = obj.id and publishDate.id.name='publishDate' and 14050 + (doc.creator = '$escapetool.sql($xcontext.user)' or (isPublished.value = 1 and hidden.value = 0))~") 14051 +#end 14052 +## 14053 +## 14054 +## 14055 +###** 14056 + * Return the Query for selecting the all wiki blog posts with categories filtering 14057 + * 14058 + * @param query The basic query for selecting blog entries. 14059 + *### 14060 +#macro(getCategoryAllBlogPostsQuery $query) 14061 + #set ($query = $NULL) 14062 + #getAllBlogPostsQuery($baseQuery) 14063 + #set ($baseQuery = $baseQuery.replace('DateProperty publishDate', 'DateProperty publishDate, DBStringListProperty as category left join category.list catList')) 14064 + #setVariable(~"$query~" ~"$\{baseQuery} and obj.id=category.id.id and category.id.name='category'~") 14065 +#end 14066 +## 14067 +## 14068 +## 14069 +#** 14070 + * Determines how is the blog index split into pages: paginated (fixed number of entries), weekly (all entries in a 14071 + * week), monthly (all entries in a month), or all. 14072 + * 14073 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>displayType</tt> 14074 + * property set. 14075 + * @param displayType The resulting string. If the blog object does not define anything, it is considered paginated. 14076 + *### 14077 +#macro(getBlogDisplayType $blogDoc $displayType) 14078 + #getBlogProperty($blogDoc 'displayType' 'paginated' $result) 14079 + #set ($displayType = $NULL) 14080 + #setVariable (~"$displayType~" $result) 14081 +#end 14082 +## 14083 +## 14084 +## 14085 +#** 14086 + * Displays a list of entries. 14087 + * 14088 + * @param entries The entries to display, a list of XDocument names. 14089 + * @param displaying What exactly is displayed: blog index, a single blog entry, a blog category, search results, 14090 + * unpublished entries, etc. This will be used as the classname(s) for the container div (hfeed). Currently 14091 + * used values: index, single, category, search, unpublished, hidden. 14092 + * @param onlyExtract If <tt>true</tt>, only display the extract of articles where available, otherwise display the full content. 14093 + * @param shouldDisplayTitles If <tt>true</tt>, display the blog title (blog posts shouldn't display the title when they're 14094 + * displayed alone on their page since it's the page title which is used in this case) 14095 + *### 14096 +#macro(displayBlog $entries $displaying $onlyExtract $shouldDisplayTitles) 14097 + #set($blogDay = '') 14098 + (% class=~"hfeed $!\{displaying}~" ~%)((( 14099 + (% class=~"blogDay~" ~%)((( 14100 + #foreach ($entryDoc in $xwiki.wrapDocs($entries)) 14101 + #getEntryObject($entryDoc $entryObj) 14102 + ## Although all entries should have one of the two objects, better check to be sure. 14103 + #if(~"$!\{entryObj}~" != '') 14104 + #getEntryDate($entryDoc $entryObj $entryDate) 14105 + ## Display a ~"calendar sheet~" for each day. All entries posted on the same day share one such sheet. 14106 + #set($entryDateStr = $xwiki.formatDate($entryDate, 'yyyyMMMMdd')) 14107 + #if($blogDay != $entryDateStr) 14108 + #if($blogDay != '') 14109 + ))) 14110 + (% class=~"blogDay~" ~%)((( 14111 + #end 14112 + #displayBlogDate($entryDate) 14113 + #set ($blogDay = $entryDateStr) 14114 + #end 14115 + ## Finally, display the entry. 14116 + #displayEntry($entryDoc $entryObj $onlyExtract $shouldDisplayTitles true) 14117 + #end 14118 + #end 14119 + )))## blogDay 14120 + )))## hfeed 14121 +#end 14122 +## 14123 +## 14124 +## 14125 +#** 14126 + * Get the entry object, either a new BlogPost or an old Article. 14127 + * 14128 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 14129 + * @param entryObj The resulting xobject of the blog post. 14130 + *### 14131 +#macro(getEntryObject $entryDoc $__entryObj) 14132 + #set($result = $entryDoc.getObject(~"$\{blogPostClassname}~")) 14133 + #if(!$result) 14134 + #set($result = $entryDoc.getObject(~"$\{oldArticleClassname}~")) 14135 + #end 14136 + ## NOTE: The reason we put an underscore in front of the variable name is to prevent the following line from 14137 + ## overwriting the $entryObj variable that may be defined before this macro is called. Of course, $__entryObj may be 14138 + ## overwritten in this case but it's less likely to have such a variable defined before. 14139 + #set ($__entryObj = $NULL) 14140 + #setVariable (~"$__entryObj~" $result) 14141 +#end 14142 +## 14143 +## 14144 +## 14145 +#** 14146 + * Gets the date associated with a blog entry. This is the publication date. For unpublished entries, initially this is 14147 + * the document creation date, but can be edited by the user. 14148 + * 14149 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 14150 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 14151 + * @param result The resulting date, an instance of <tt>java.util.Date</tt>. 14152 + *### 14153 +#macro(getEntryDate $entryDoc $entryObj $result) 14154 + #set ($result = $NULL) 14155 + #setVariable (~"$result~" $entryObj.getProperty('publishDate').value) 14156 +#end 14157 +## 14158 +## 14159 +## 14160 +#** 14161 + * Displays a date, nicely formatted as a calendar page. 14162 + * 14163 + * @param date The date to display, an instance of <tt>java.util.Date</tt>. 14164 + *### 14165 +#macro(displayBlogDate $date) 14166 + #set($year = $xwiki.formatDate($date, 'yyyy')) 14167 + ## 3 letter month name, like Jan, Dec. 14168 + #set($month = $xwiki.formatDate($date, 'MMM')) 14169 + ## Uncomment to get a full length month name, like January, December. 14170 + ## TODO: this could be defined somewhere in the blog style. 14171 + ## #set($month = $xwiki.formatDate($date, 'MMMM')) 14172 + #set($day = $xwiki.formatDate($date, 'dd')) 14173 + (% class=~"blogdate~" ~%) 14174 + == (% class=~"month~" ~%)$month(%~%) (% class=~"day~" ~%)$day(%~%) (% class=~"year~" ~%)$year(%~%) == 14175 +#end 14176 +## 14177 +## 14178 +## 14179 +#** 14180 + * Displays a blog article: management tools, header, content, footer. 14181 + * 14182 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 14183 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 14184 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 14185 + * @param shouldDisplayTitle If <tt>true</tt>, display the blog title (blog posts shouldn't display the title 14186 + * when they're displayed alone on their page since it's the page title which is used in this case) 14187 + * @param shouldDisplayActions If <tt>true</tt>, display the blog post actions buttons 14188 + *### 14189 +#macro(displayEntry $entryDoc $entryObj $onlyExtract $shouldDisplayTitle $shouldDisplayActions) 14190 + ## Only articles with an explicit hidden setting or an explicit unpublished setting are hidden 14191 + #isPublished($entryObj $isPublished) 14192 + #isHidden($entryObj $isHidden) 14193 + #if($doc.fullName == $entryDoc.fullName) 14194 + (% class=~"hentry single-article~" ~%)((( 14195 + #else 14196 + (% class=~"hentry#if(!$isPublished) unpublished-article#elseif($isHidden) hidden-article#end~" ~%)((( 14197 + #end 14198 + #if ($shouldDisplayActions) 14199 + #displayEntryTools($entryDoc $entryObj) 14200 + #end 14201 + #if($shouldDisplayTitle) 14202 + #displayEntryTitle($entryDoc $entryObj) 14203 + #end 14204 + #if($doc.fullName == $entryDoc.fullName) 14205 + #if(!$isPublished) 14206 + \{\{warning}}\{\{translation key='blog.code.published'/}}\{\{/warning}} 14207 + #elseif($isHidden) 14208 + \{\{warning}}\{\{translation key='blog.code.hidden'/}}\{\{/warning}} 14209 + #end 14210 + #end 14211 + #displayEntryContent($entryDoc $entryObj $onlyExtract) 14212 + #displayEntryFooter($entryDoc $entryObj) 14213 + )))## hentry 14214 +#end 14215 +## 14216 +## 14217 +## 14218 +#** 14219 + * Checks if the provided blog is published or not. 14220 + * 14221 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 14222 + * @param isPublished The resulting boolean, true if the entry is considered published. 14223 + *### 14224 +#macro(isPublished $entryObj $isPublished) 14225 + #set ($isPublished = $NULL) 14226 + ## This should work for both old articles, which don't have the 'published' property at all, and 14227 + ## are considered published by default, and new entries, that should have 1 if published. 14228 + #if (~"$!\{entryObj.getProperty('published').value}~" != '0') 14229 + #setVariable (~"$isPublished~" true) 14230 + #else 14231 + #setVariable (~"$isPublished~" false) 14232 + #end 14233 +#end 14234 +## 14235 +## 14236 +## 14237 +#** 14238 + * Checks if the provided blog is hidden or not. 14239 + * 14240 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass<tt> xclass. 14241 + * @param isHiddel The resulting boolean, true if the entry is considered hidden. 14242 + *### 14243 +#macro(isHidden $entryObj $isHidden) 14244 + #set ($isHidden = $NULL) 14245 + ## This should work for both old articles, which don't have the 'hidden' property at all, and 14246 + ## are considered visible by default, and new entries, that should have 1 if hidden. 14247 + #if (~"$!\{entryObj.getProperty('hidden').value}~" == '1') 14248 + #setVariable (~"$isHidden~" true) 14249 + #else 14250 + #setVariable (~"$isHidden~" false) 14251 + #end 14252 +#end 14253 +## 14254 +## 14255 +## 14256 +#** 14257 + * Displays several ~"tools~" for manipulating blog posts: hide/show, publish, edit. 14258 + * 14259 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 14260 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 14261 + *### 14262 +#macro(displayEntryTools $entryDoc $entryObj) 14263 + #if($xcontext.action == 'view') 14264 + (% class=~"blog-entry-toolbox~" ~%)((( 14265 + #displayPublishButton($entryDoc $entryObj) 14266 + #displayHideShowButton($entryDoc $entryObj) 14267 + #displayEditButton($entryDoc $entryObj) 14268 + #displayDeleteButton($entryDoc $entryObj) 14269 + ))) 14270 + #end 14271 +#end 14272 +## 14273 +## 14274 +## 14275 +#** 14276 + * Displays the publish button to the entry <strong>creator</strong>, if the article is not published yet. 14277 + * 14278 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 14279 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 14280 + * @todo AJAX calls. 14281 + *### 14282 +#macro(displayPublishButton $entryDoc $entryObj) 14283 + #isPublished($entryObj $isPublished) 14284 + #if(!$isPublished && $entryDoc.creator == $xcontext.user && $xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 14285 + [[#toolImage('world')>>path:$blogPublisher.getURL('view', ~"entryName=$\{escapetool.url($entryDoc.fullName)}&xredirect=$\{escapetool.url($thisURL)}&form_token=$!\{services.csrf.getToken()}~")||title=~"$services.localization.render('blog.code.notpublished')~"]]## 14286 + #end 14287 +#end 14288 +## 14289 +## 14290 +## 14291 +#** 14292 + * Displays the hide or show button to the entry <strong>creator</strong>, if the article is already published. 14293 + * 14294 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 14295 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 14296 + *### 14297 +#macro(displayHideShowButton $entryDoc $entryObj) 14298 + #isPublished($entryObj $isPublished) 14299 + #isHidden($entryObj $isHidden) 14300 + ## Only published articles can be hidden. Unpublished articles are considered already hidden. 14301 + #if($isPublished && $entryDoc.creator == $xcontext.user && $xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 14302 + #set ($queryString = \{ 14303 + 'xredirect' : $thisURL, 14304 + 'form_token' : $services.csrf.getToken() 14305 + }) 14306 + #if ($isHidden) 14307 + #set ($discard = $queryString.putAll(\{ 14308 + ~"$\{entryObj.getxWikiClass().getName()}_$\{entryObj.number}_hidden~" : 0, 14309 + 'comment' : $services.localization.render('blog.code.madevisible') 14310 + })) 14311 + #set ($lockURL = $entryDoc.getURL('save', $escapetool.url($queryString))) 14312 + [[#toolImage('unlock')>>path:$lockURL||class=~"blog-tool-show~" title=~"$services.localization.render('blog.code.makevisible')~"]]## 14313 + #else 14314 + #set ($discard = $queryString.putAll(\{ 14315 + ~"$\{entryObj.getxWikiClass().getName()}_$\{entryObj.number}_hidden~" : 1, 14316 + 'comment' : $services.localization.render('blog.code.hid') 14317 + })) 14318 + #set ($lockURL = $entryDoc.getURL('save', $escapetool.url($queryString))) 14319 + [[#toolImage('lock')>>path:$lockURL||class=~"blog-tool-hide~" title=~"$services.localization.render('blog.code.hide')~"]]## 14320 + #end 14321 + #end 14322 +#end 14323 +## 14324 +## 14325 +## 14326 +#** 14327 + * Displays the edit button to those that can edit the article. 14328 + * 14329 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 14330 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 14331 + *### 14332 +#macro(displayEditButton $entryDoc $entryObj) 14333 + #if($xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 14334 + ## Call getDefaultEditMode() for backward compatibility with older blog posts. 14335 + [[#toolImage('pencil')>>path:$entryDoc.getURL('edit')||title=~"$services.localization.render('blog.code.editpost')~"]]## 14336 + #end 14337 +#end 14338 +## 14339 +## 14340 +## 14341 +#** 14342 + * Displays the delete button to those that can edit the article. 14343 + * 14344 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 14345 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 14346 + * @todo AJAX calls. 14347 + *### 14348 +#macro(displayDeleteButton $entryDoc $entryObj) 14349 + #if($xwiki.hasAccessLevel('delete', $xcontext.user, $entryDoc.fullName)) 14350 + [[#toolImage('cross')>>path:$entryDoc.getURL('delete')||title=~"$services.localization.render('blog.code.deletepost')~"]]## 14351 + #end 14352 +#end 14353 +## 14354 +## 14355 +## 14356 +#** 14357 + * Displays the title of the entry. 14358 + * 14359 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 14360 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 14361 + *### 14362 +#macro(displayEntryTitle $entryDoc $entryObj) 14363 + #if($doc.fullName == $entryDoc.fullName) 14364 + (% class=~"entry-title~" ~%) 14365 + = $services.rendering.escape($entryDoc.display('title', 'view', $entryObj), $xwiki.getCurrentContentSyntaxId()) = 14366 + #else 14367 + (% class=~"entry-title~" ~%) 14368 + === [[$services.rendering.escape($entryDoc.display('title', 'view', $entryObj),$xwiki.getCurrentContentSyntaxId())>>doc:$services.rendering.escape($services.model.serialize($entryDoc.getDocumentReference(),'default'),$xwiki.getCurrentContentSyntaxId())]] === 14369 + #end 14370 +#end 14371 +## 14372 +## 14373 +## 14374 +#** 14375 + * Displays the body of the entry. 14376 + * 14377 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 14378 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 14379 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 14380 + *### 14381 +#macro(displayEntryContent $entryDoc $entryObj $onlyExtract) 14382 + (% class=~"#if($onlyExtract)entry-summary#\{else}entry-content#end~" ~%)((( 14383 + #getEntryContent($entryDoc $entryObj $onlyExtract $entryContent) 14384 + ## FIXME: This causes the blog's content to not be annotatable. See http://jira.xwiki.org/browse/XWIKI-6328 14385 + ## Should probably be replaced by a display macro call with a reference to the object property holding the post's content 14386 + \{\{html wiki=~"false~"}}$entryDoc.getRenderedContent($entryContent, $entryDoc.syntax.toIdString())\{\{/html}} 14387 + ))) ## entry-content 14388 + (% class=~"clearfloats~" ~%)((())) 14389 +#end 14390 +## 14391 +## 14392 +## 14393 +#** 14394 + * Extracts the body of the entry that should be displayed. If <tt>onlyExtract</tt> is <tt>true</tt>, display the content 14395 + * of the <tt>extract</tt> field (if not empty). 14396 + * 14397 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 14398 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 14399 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 14400 + * @param entryContent The resulting content. 14401 + * @param removeEllipsis If <tt>true</tt>, then don't display an ellipsis at the end of the content (only used when 14402 + * <tt>onlyExtract</tt> is <tt>true</tt>) 14403 + *### 14404 +#macro(getEntryContent $entryDoc $entryObj $onlyExtract $entryContent $removeEllipsis) 14405 + #if ($onlyExtract) 14406 + ## Note: We trim the summary so that if there's some white space it won't cause the summary to be used instead 14407 + ## of the content. 14408 + #set ($macro.result = $entryObj.getProperty('extract').value.trim()) 14409 + #end 14410 + #if(~"$!macro.result~" == '') 14411 + #set($macro.result = $entryObj.getProperty('content').value) 14412 +#* Disabled until the content can be cleanly cut. 14413 +* #if($onlyExtract && $result.length()>$maxchars) 14414 +* #set($i = $macro.result.lastIndexOf(~" ~", $maxchars)) 14415 +* #set($i = $i + 1) 14416 +* #set($macro.result = ~"$\{macro.result.substring(0,$i)} *[...>$\{entryDoc.fullName}]*~") 14417 +* #end 14418 +## *### 14419 + #elseif (!$removeEllipsis) 14420 + #if($entryDoc.syntax.toIdString() == 'xwiki/1.0') 14421 + #set($macro.result = ~"$\{macro.result} <a href='$\{entryDoc.getExternalURL()}' title='$services.localization.render('blog.code.readpost')'>...</a>~") 14422 + #elseif($entryDoc.syntax.toIdString() == 'xwiki/2.0' || $entryDoc.syntax.toIdString() == 'xwiki/2.1') 14423 + #set($macro.result = ~"$\{macro.result} [[...>>$\{services.rendering.escape($entryDoc.getFullName(), $entryDoc.getSyntax())}||title='$services.localization.render('blog.code.readpost')']]~") 14424 + #end 14425 + #end 14426 + #set ($entryContent = $NULL) 14427 + #setVariable (~"$entryContent~" $macro.result) 14428 +#end 14429 +## 14430 +## 14431 +## 14432 +#** 14433 + * Displays the footer of the entry. 14434 + * 14435 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 14436 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 14437 + *### 14438 +#macro(displayEntryFooter $entryDoc $entryObj) 14439 + (% class=~"entry-footer~" ~%)((( 14440 + #isPublished($entryObj $isPublished) 14441 + (% class='entry-author-label' ~%) 14442 + #if($isPublished) 14443 + \{\{translation key='blog.code.postedby'/}} ## 14444 + #else 14445 + \{\{translation key='blog.code.createdby'/}} ## 14446 + #end 14447 + \{\{html wiki=~"false~" clean=~"false~"}}<span class='author vcard'>#userfn($entryDoc.creator)</span>\{\{/html}} ## 14448 + #getEntryDate($entryDoc $entryObj $entryDate) 14449 + #listCategories($entryObj) #* 14450 + ## Since the publish date and update date are not set at the exact same time, there could be a small difference that 14451 + ## we assume cannot be more than 3 seconds. 14452 + *#(% class=~"separator~" ~%)·(%~%) [[\{\{translation key='blog.code.permalink'/}}>>$services.rendering.escape($entryDoc,'xwiki/2.0')||rel=~"bookmark~"]] ## 14453 + #if ($showcomments) 14454 + (% class=~"separator~" ~%)·(%~%) [[\{\{translation key='blog.code.comments'/}}>>$services.rendering.escape($entryDoc,'xwiki/2.0')||anchor=~"Comments~"]] (% class=~"itemCount~" ~%)($entryDoc.comments.size())(%~%) ## 14455 + #end ## 14456 + #if($entryDoc != $doc) ## 14457 + #displayEntryBlogLocation($entryDoc $entryObj) ## 14458 + #end 14459 + )))## entry-footer 14460 +#end 14461 +## 14462 +## 14463 +#** 14464 + * Display the blog for the entry (if it is not the currently displayed blog) 14465 + * 14466 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 14467 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 14468 + *### 14469 +#macro(displayEntryBlogLocation $entryDoc $entryObj) 14470 + #getBlogPostsLocation($entryDoc.getSpace() $blogPostsLocation) ## 14471 + #if(~"$!blogPostsLocation~" != ~"~") ## 14472 + #set($blogSpaceRef = $services.model.resolveSpace($blogPostsLocation)) ## 14473 + #set($blogDocRef = $services.model.createDocumentReference($services.model.getEntityReference('DOCUMENT', 'default').getName(), $blogSpaceRef)) ## 14474 + #if($doc.documentReference != $blogDocRef) ## 14475 + #set($blogDoc = $xwiki.getDocument($blogDocRef)) ## 14476 + #set($blogObj = $blogDoc.getObject('Blog.BlogClass')) ## 14477 + #if($blogObj) (% class=~"blogrefpath~" ~%)((( ## 14478 + $services.localization.render('blog.code.postedin') \{\{html clean=~"false~" wiki=~"false~"}} #hierarchy($blogDocRef, \{'selfIsActive' : true, 'local': true}) \{\{/html}} ## 14479 + )))(%~%)## 14480 + #end 14481 + #end 14482 + #end 14483 +#end 14484 +## 14485 +## 14486 +## 14487 +## 14488 +#** 14489 + * List the categories an entry belongs to. Used in the footer. The categories are instances of <tt>Blog.CategoryClass</tt>. 14490 + * 14491 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 14492 + *### 14493 +#macro(listCategories $entryObj) 14494 + #if($entryObj.getxWikiClass().getName() == $blogPostClassname) 14495 + #set($categories = $entryObj.getProperty('category').value) 14496 + #set($first = true) 14497 + #if($categories.size() > 0) 14498 + #foreach($category in $categories) 14499 + #set($categoryDoc = $!xwiki.getDocument($category)) 14500 + #if(!$categoryDoc.isNew() && $categoryDoc.getObject($\{blogCategoryClassname})) 14501 + #if($foreach.count == 1) 14502 + (% class='separator' ~%)·(%~%) $services.localization.render('blog.code.categories') ## 14503 + #else 14504 + , ## 14505 + #end## 14506 + [[$!\{categoryDoc.getObject($blogCategoryClassname).getValue('name')}>>$\{category}||rel='tag']]## 14507 + #end## 14508 + #end## 14509 + #end 14510 + #end 14511 +#end 14512 +## 14513 +## 14514 +## 14515 +#** 14516 + * Displays blog pagination links (older and newer entries). 14517 + * 14518 + * @param blogDoc the XDocument holding the blog definition object. 14519 + *### 14520 +#macro(displayNavigationLinks $blogDoc) 14521 + (% class=~"clearfloats~" ~%)((())) 14522 + #getBlogDisplayType($blogDoc $displayType) 14523 + #if($displayType == 'weekly') 14524 + (% class=~"pagingLinks~" ~%)((( 14525 + #getRequestedWeek($weekDate) 14526 + $weekDate.addWeeks(-1)## 14527 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.previousweek'/}}>>$doc.documentReference.name||queryString=~"year=$weekDate.weekyear&week=$weekDate.weekOfWeekyear~" class=~"button~"]]**(%~%) 14528 + #sep() 14529 + $weekDate.addWeeks(2)## 2 because we already subtracted 1 above 14530 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.nextweek'/}} »>>$doc.documentReference.name||queryString=~"year=$weekDate.weekyear&week=$weekDate.weekOfWeekyear~" class=~"button~"]]**(%~%) 14531 + ))) 14532 + #elseif($displayType == 'monthly') 14533 + (% class=~"pagingLinks~" ~%)((( 14534 + #getRequestedMonth($monthDate) 14535 + $monthDate.addMonths(-1)## 14536 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.previousmonth'/}}>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"year=$monthDate.year&month=$monthDate.monthOfYear~" class=~"button~"]]**(%~%) 14537 + #sep() 14538 + $monthDate.addMonths(2)## 2 because we already subtracted 1 above 14539 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.nextmonth'/}} »>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"year=$monthDate.year&month=$monthDate.monthOfYear~" class=~"button~"]]**(%~%) 14540 + ))) 14541 + #elseif($displayType == 'all') 14542 + #else 14543 + ## Paginated 14544 + #if(($totalPages > 1)) 14545 + #set($queryString = '') 14546 + #foreach($p in $request.getParameterNames()) 14547 + #if($p != 'page' && $p != 'ipp') 14548 + #foreach($v in $request.getParameterValues($p)) 14549 + #set($queryString = ~"$\{queryString}&$\{escapetool.url($p)}=$\{escapetool.url($v)}~") 14550 + #end 14551 + #end 14552 + #end 14553 + (% class=~"pagingLinks~" ~%)((( 14554 + #if ($currentPageNumber < $totalPages) 14555 + #set($currentPageNumber = $currentPageNumber + 1) 14556 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.olderposts'/}}>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"page=$\{currentPageNumber}&ipp=$\{itemsPerPage}$queryString~" class=~"button~"]]**(%~%) 14557 + #set($currentPageNumber = $currentPageNumber - 1) 14558 + #end 14559 + #if ($currentPageNumber > 1) 14560 + #if ($currentPageNumber < $totalPages) 14561 + #sep() 14562 + #end 14563 + #set($currentPageNumber = $currentPageNumber - 1) 14564 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.newerposts'/}} »>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"page=$\{currentPageNumber}&ipp=$\{itemsPerPage}$queryString~" class=~"button~"]]**(%~%) 14565 + #set($currentPageNumber = $currentPageNumber + 1) 14566 + #end 14567 + (% class=~"clear~" ~%)(%~%) 14568 + )))## pagingLinks 14569 + #end 14570 + #end 14571 +#end 14572 +## 14573 +## 14574 +## 14575 +#** 14576 + * Displays a message box with ~"publish~" icon. 14577 + * 14578 + * @param message A text message concerning blog article publishing 14579 + *### 14580 +#macro(publishMessageBox $message) 14581 +(% class=~"plainmessage publish-message~" ~%)((($services.icon.render('world') $message))) 14582 +#end 14583 +#** 14584 + * Displays a message box with ~"show/hide~" icon. 14585 + * 14586 + * @param message A text message concerning blog article hiding 14587 + *### 14588 +#macro(hideMessageBox $message) 14589 +(% class=~"plainmessage hide-message~" ~%)((($services.icon.render('unlock') $message))) 14590 +#end 14591 +## 14592 +## 14593 +## 14594 +#** 14595 + * Determine the requested week, for using in a weekly-indexed blog. The relevant request parameters are 14596 + * <tt>year</tt> and <tt>week</tt>. By default, the current week is used. 14597 + * 14598 + * @param monthDate The resulting week, a JODATime MutableDateTime. 14599 + *### 14600 +#macro(getRequestedWeek $weekDate) 14601 + #set ($weekDate = $NULL) 14602 + #setVariable (~"$weekDate~" $xwiki.jodatime.mutableDateTime) 14603 + #if(~"$!\{request.year}~" != '') 14604 + #set ($discard = $weekDate.setYear($numbertool.toNumber($request.year).intValue())) 14605 + #end 14606 + #if(~"$!\{request.week}~" != '') 14607 + #set ($discard = $weekDate.setWeekOfWeekyear($numbertool.toNumber($request.week).intValue())) 14608 + #end 14609 +#end 14610 +## 14611 +## 14612 +## 14613 +#** 14614 + * Determine the requested month, for using in a monthly-indexed blog. The relevant request parameters are 14615 + * <tt>year</tt> and <tt>month</tt>. By default, the current month is used. 14616 + * 14617 + * @param monthDate The resulting month, a JODATime MutableDateTime. 14618 + *### 14619 +#macro(getRequestedMonth $monthDate) 14620 + #set ($monthDate = $NULL) 14621 + #setVariable (~"$monthDate~" $xwiki.jodatime.mutableDateTime) 14622 + #if(~"$!\{request.year}~" != '') 14623 + #set ($discard = $monthDate.setYear($numbertool.toNumber($request.year).intValue())) 14624 + #end 14625 + #if(~"$!\{request.month}~" != '') 14626 + #set ($discard = $monthDate.setMonthOfYear($numbertool.toNumber($request.month).intValue())) 14627 + #end 14628 +#end 14629 +## 14630 +## 14631 +## 14632 +#** 14633 + * Retrieve a blog property (title, display type, etc). 14634 + * 14635 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 14636 + * @param propertyName The name of the property to be retrieved. One of the <tt>Blog.BlogClass</tt>'s properties. 14637 + * @param defaultValue The default value to use in case the blog object does not define one. 14638 + * @param propertyValue The resulting value. 14639 + *### 14640 +#macro(getBlogProperty $blogDoc $propertyName $defaultValue $propertyValue) 14641 + #set($result = ~"$!\{blogDoc.getObject($\{blogClassname}).getProperty($propertyName).value}~") 14642 + #if($result == '') 14643 + #set($result = $defaultValue) 14644 + #end 14645 + #set ($propertyValue = $NULL) 14646 + #setVariable (~"$propertyValue~" $result) 14647 +#end 14648 + 14649 +#** 14650 + * If an error occurs when executing an action, set a specific response status and display an error message. 14651 + * 14652 + * @param status The response status. 14653 + * @param text The user readable error to be displayed. Can be a translation key. 14654 + * @param parameters The parameters to use when decoding the translation key. 14655 + *### 14656 +#macro(blog__actionResponseError $status $text $parameters) 14657 + $response.setStatus($status) 14658 + #if($request.ajax) 14659 + $services.localization.render($text, $!parameters) 14660 + #else 14661 + \{\{error}}$services.localization.render($text, $!parameters)\{\{/error}} 14662 + #end 14663 +#end 14664 +## 14665 +## 14666 +## 14667 +#** 14668 + * Check if a blog is the Default blog (The one in the 'Blog' space). 14669 + * 14670 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 14671 + * @param isDefault The resulting boolean. 14672 + *### 14673 +#macro(isDefaultBlog $blogDoc $isDefault) 14674 + #set ($result = false) 14675 + #if ($blogDoc.space == 'Blog') 14676 + #set ($result = true) 14677 + #end 14678 + #setVariable(~"$isDefault~" $result) 14679 +#end 14680 +## 14681 +## 14682 +## 14683 +#** 14684 + * Retrieve the blog posts location (space). 14685 + * 14686 + * @param blogSpace The blog space. It should contain a document with a <tt>Blog.BlogClass</tt> object 14687 + * @param postsLocation The resulting location. 14688 + *### 14689 +#macro(getBlogPostsLocation $blogSpace $postsLocation) 14690 + #getBlogDocument($blogSpace $blogDoc) 14691 + #getBlogProperty($blogDoc 'postsLocation' $blogDoc.space $result) 14692 + #set ($postsLocation = $NULL) 14693 + #setVariable (~"$postsLocation~" $result) 14694 +#end 14695 +## 14696 +## 14697 +## 14698 +#** 14699 + * Retrieve the blog categories location (space). 14700 + * 14701 + * @param blogSpace The blog space. It should contain a document with a <tt>Blog.BlogClass</tt> object 14702 + * @param categoriesLocation The resulting location. 14703 + *### 14704 +#macro(getBlogCategoriesLocation $blogSpace $categoriesLocation) 14705 + #getBlogDocument($blogSpace $blogDoc) 14706 + #getBlogProperty($blogDoc 'categoriesLocation' 'Blog' $result) 14707 + #set ($postsLocation = $NULL) 14708 + #setVariable (~"$categoriesLocation~" $result) 14709 +#end 14710 +###** 14711 + * Return the Query for selecting the blog posts based on the context where the posts are displayed. 14712 + * for example there is 4 different panel contexts: 14713 + * aBlog.aPost or aBlog.WebHome 14714 + * aCategorySpace.aCategory 14715 + * aCategorySpace.WebHome 14716 + * Blog.aPost or Blog.WebHome 14717 + * 14718 + * @param query The query for selecting blog entries. 14719 + * @param queryParams The parameters to bind with the generated query. 14720 + * @param targetDoc The document in which the articles will be displayed. 14721 + *### 14722 +#macro(getAllBlogPostsQueryBasedOnDisplayContext $targetDoc $query $queryParams) 14723 + #set ($query = $NULL) 14724 + #set ($queryParams = $NULL) 14725 + #getCategoryAllBlogPostsQuery($resultQuery) 14726 + #set ($resultQueryParams = \{}) 14727 + #set($sanitizedSpace = $targetDoc.space.replaceAll('([%_!])', '!$1').concat('.%')) 14728 + #if ($targetDoc.space == $defaultBlogSpace && !$targetDoc.getObject($blogCategoryClassname)) 14729 + ## Get all posts that are in the default blog space('Blog') or in category 'Blog.%' 14730 + #set ($discard = $resultQueryParams.put('space', $targetDoc.space)) 14731 + #set ($discard = $resultQueryParams.put('catList', $sanitizedSpace)) 14732 + #set($resultQuery = ~"$\{resultQuery} and (doc.space = :space or catList like :catList escape '!')~") 14733 + #elseif($targetDoc.getObject($blogCategoryClassname)) 14734 + ## Get all posts that are in a category aCategorySpace.aCategory 14735 + #set ($discard = $resultQueryParams.put('catList', $targetDoc.fullName)) 14736 + #set($resultQuery = ~"$\{resultQuery} and (catList=:catList)~") 14737 + #elseif($targetDoc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 14738 + ## Get all posts that are in a category aCategorySpace.% 14739 + #set ($discard = $resultQueryParams.put('catList', $sanitizedSpace)) 14740 + ## Exclude incategorized posts 14741 + #set ($excludedCategory = ~"$\{targetDoc.space}.WebHome~") 14742 + #if ($targetDoc.space == $defaultBlogSpace) 14743 + #set ($excludedCategory = ~"Blog.Categories.WebHome~") 14744 + #end 14745 + #set ($discard = $resultQueryParams.put('excludedCategory', $excludedCategory)) 14746 + #set($resultQuery = ~"$\{resultQuery} and (catList<>:excludedCategory) and (catList like :catList escape '!')~") 14747 + #else 14748 + ## Get all posts in blog space aBlog 14749 + #getAllBlogPostsQuery($resultQuery) 14750 + #getBlogPostsLocation($targetDoc.space $postsLocation) 14751 + #set ($discard = $resultQueryParams.put('space', $postsLocation)) 14752 + #set($resultQuery = ~"$\{resultQuery} and doc.space = :space~") 14753 + #end 14754 + #setVariable(~"$query~" $resultQuery) 14755 + #setVariable(~"$queryParams~" $resultQueryParams) 14756 +#end 14757 +## 14758 +## 14759 +## 14760 +###** 14761 + * Display blog posts based on the context where the posts are displayed. 14762 + * for example there is 4 different panel contexts: 14763 + * aBlog.aPost or aBlog.WebHome 14764 + * aCategorySpace.aCategory 14765 + * aCategorySpace.WebHome 14766 + * Blog.aPost or Blog.WebHome 14767 + * 14768 + * @param targetDoc The document in which the articles will be displayed. 14769 + * @param postsVisiblity The visibility of the posts to display: recent, unpublished, ... 14770 + * @param layout Layout of the the posts to display 14771 + * @param layoutParams additional layout parameters, it is a string like 'param1=val1|...|paramN=valueN' 14772 + * @param limit the number of posts to display 14773 + *### 14774 +#macro(displayBlogPostsBasedOnContext $targetDoc $postsVisiblity $layout $layoutParams $limit) 14775 + #if ($postLayout == 'full') 14776 + #set ($macro.paginated = 'yes') 14777 + #end 14778 + #if ($targetDoc.space == $defaultBlogSpace && !$targetDoc.getObject($blogCategoryClassname)) 14779 + ## Display all posts that are in the default blog space('Blog') or in category 'Blog.%' 14780 + #getBlogPostsLayout($xwiki.getDocument(~"$\{defaultBlogSpace}.WebHome~") $postsLayout) 14781 + #if (~"$!layout~" == '') 14782 + #set ($layout = $postsLayout) 14783 + #end 14784 + #if ($postsVisiblity == 'recent') 14785 + \{\{blogpostlist blog=~"$\{defaultBlogSpace.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')}.WebHome~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 14786 + #elseif($postsVisiblity == 'unpublished') 14787 + \{\{blogpostlist blog=~"$\{defaultBlogSpace.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')}.WebHome~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 14788 + #end 14789 + #elseif($targetDoc.getObject($blogCategoryClassname)) 14790 + ## Display all posts that are in a category aCategorySpace.aCategory 14791 + #getBlogDocumentForCategoriesSpace($targetDoc.space $blogDoc) 14792 + #getBlogPostsLayout($blogDoc $postsLayout) 14793 + #if (~"$!layout~" == '') 14794 + #set ($layout = $postsLayout) 14795 + #end 14796 + #if ($postsVisiblity == 'recent') 14797 + \{\{blogpostlist category=~"$targetDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 14798 + #elseif($postsVisiblity == 'unpublished') 14799 + \{\{blogpostlist category=~"$targetDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 14800 + #end 14801 + #elseif($targetDoc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 14802 + ## Display all posts that are in a category aCategorySpace.% 14803 + #getBlogDocumentForCategoriesSpace($targetDoc.space $blogDoc) 14804 + #getBlogPostsLayout($blogDoc $postsLayout) 14805 + #if (~"$!layout~" == '') 14806 + #set ($layout = $postsLayout) 14807 + #end 14808 + #if ($postsVisiblity == 'recent') 14809 + \{\{blogpostlist category=~"$targetDoc.space.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 14810 + #elseif($postsVisiblity == 'unpublished') 14811 + \{\{blogpostlist category=~"$targetDoc.space.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 14812 + #end 14813 + #else 14814 + ## Display all posts in blog space aBlog 14815 + #getBlogDocument($targetDoc.space $blogDoc) 14816 + #getBlogPostsLayout($blogDoc $postsLayout) 14817 + #if (~"$!layout~" == '') 14818 + #set ($layout = $postsLayout) 14819 + #end 14820 + #if ($postsVisiblity == 'recent') 14821 + \{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 14822 + #elseif($postsVisiblity == 'unpublished') 14823 + \{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 14824 + #end 14825 + #end 14826 +#end 14827 +## 14828 +## 14829 +## 14830 +#** 14831 + * Bind parameters to a query object. 14832 + * 14833 + * @param queryObj The query object (org.xwiki.query.internal.ScriptQuery object) 14834 + * @param queryParams the query parameters. 14835 + *### 14836 +#macro(bindQueryParameters $queryObj $queryParams) 14837 + #set ($output = $queryObj) 14838 + #foreach( $key in $queryParams.keySet() ) 14839 + #set($output = $queryObj.bindValue($key, $queryParams.get($key))) 14840 + #end 14841 + #setVariable(~"$queryObj~" $output) 14842 +#end 14843 +## 14844 +## 14845 +## 14846 +#** 14847 + * Determines the blogposts layout: Default layout (Calendar), Image thumbnail 14848 + * 14849 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>postsLayout</tt> 14850 + * property set. 14851 + * @param postsLayout The resulting string. If the blog object does not define anything, it is considered full (default). 14852 + *### 14853 +#macro(getBlogPostsLayout $blogDoc $postsLayout) 14854 + #getBlogProperty($blogDoc 'postsLayout' 'full' $res) 14855 + #set ($postsLayout = $NULL) 14856 + #setVariable (~"$postsLayout~" $res) 14857 +#end 14858 +## 14859 +## 14860 +## 14861 +#** 14862 + * Retrieve the blog document of a categories space, which is the blog descriptor that have its categoriesLocation property set to the categories space 14863 + * 14864 + * @param categoriesSpace A <tt>String</tt>, the name of the space. 14865 + * @param blogDoc The resulting XDocument. 14866 + *### 14867 +#macro(getBlogDocumentForCategoriesSpace $categoriesSpace $blogDoc) 14868 + #set($macro.blogDocs = $services.query.hql(~", BaseObject obj, StringProperty as categoriesLocationProp where obj.name = doc.fullName and obj.className = '$blogClassname' and obj.id=categoriesLocationProp.id.id and categoriesLocationProp.id.name='categoriesLocation' and categoriesLocationProp.value=:categoriesLocation order by doc.creationDate~").setLimit(1).setOffset(0).bindValue('categoriesLocation', $categoriesSpace).execute()) 14869 + #if($macro.blogDocs.size() > 0 && $categoriesSpace != $defaultBlogSpace) 14870 + #set($macro.result = $xwiki.getDocument($macro.blogDocs.get(0))) 14871 + #else 14872 + ## Fallback to Blog.WebHome, the default blog 14873 + #set($macro.result = $xwiki.getDocument('Blog.WebHome')) 14874 + #end 14875 + #set ($blogDoc = $NULL) 14876 + #setVariable (~"$blogDoc~" $macro.result) 14877 +#end" %) 14878 +((( 14879 +(% class="macro-placeholder hidden" %) 14880 +((( 14881 +macro:velocity 14882 +))) 14883 +))) 14884 +))) 14885 + 14886 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogPostLayoutMacros~"" %) 14887 +((( 14888 +(% class="macro-placeholder hidden" %) 14889 +((( 14890 +macro:include 14891 +))) 14892 + 14893 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|#** 14894 + * Extract the layout parameters from a string. 14895 + * 14896 + * @param layoutParamsString The string representation of the layout parameters. 14897 + * It should contain a String following this format ~"paramName1=Value1|paramName2=Value2|...|paramNameK=ValueK~" 14898 + * @param layoutsParameters The resulting layout parameters Map. 14899 + *### 14900 +#macro(extractLayoutParametersFromString $layoutParamsString $layoutsParameters) 14901 + #set ($layoutsParameters = $NULL) 14902 + #set ($macro.layoutParams = \{}) 14903 + #if (~"$!layoutParamsString~" != '') 14904 + #set ($macro.paramsArr = $layoutParamsString.split('\\|')) 14905 + #foreach ($item in $macro.paramsArr) 14906 + #set ($itemSplit = $item.split('=')) 14907 + #if ($itemSplit.size() == 2) 14908 + #set ($discard = $macro.layoutParams.put($itemSplit[0].trim(), $itemSplit[1].trim())) 14909 + #end 14910 + #end 14911 + #end 14912 + #setVariable(~"$layoutsParameters~" $macro.layoutParams) 14913 +#end" %) 14914 +((( 14915 +(% class="macro-placeholder hidden" %) 14916 +((( 14917 +macro:velocity 14918 +))) 14919 +))) 14920 +))) 14921 + 14922 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-| #macro(displaySmallPost $pDoc $pObj) 14923 + #initLayoutVars($pDoc $pObj) 14924 + <div class=~"col-xs-12 col-sm-6 next_blog~"> 14925 + <a href=~"$pDoc.uRL~" class=~"thumbnail~"> 14926 + <div class=~"row~"> 14927 + <div class=~"col-xs-4~"> 14928 + #if ($imageAtt) 14929 + <img src=~"$escapetool.xml($pDoc.getAttachmentURL($pObj.getValue('image'),'download',$imgQs))~" /> 14930 + #end 14931 + </div> 14932 + <div class=~"col-xs-8 art_det~"> 14933 + <p class=~"text-left~"> 14934 + #if($displayTitle)$!postTitle #end 14935 + <br/><span class=~"date_info~"> $!dateStr </span> 14936 + </p> 14937 + #displayPostDetails($pDoc) 14938 + </div> 14939 + </div> 14940 + </a> 14941 + </div> 14942 + #end 14943 + ## 14944 + ## 14945 + ## 14946 + #macro(displayPinnedPost $pDoc $pObj) 14947 + #initLayoutVars($pDoc $pObj) 14948 + <div class=~"col-xs-12 col-sm-4 col-md-4 blog_pin~"> 14949 + <a href=~"$pDoc.uRL~" class=~"all-post-link~"> 14950 + <div class=~"thumbnail~"> 14951 + #if ($displayTitle && $displayTitleFirstOnPinnedPosts) 14952 + <h5 class=~"title-top~">$!postTitle <br/> <span class=~"date_info~"> $!dateStr </span> </h5> 14953 + #end 14954 + #if ($imageAtt) 14955 + <img src=~"$escapetool.xml($pDoc.getAttachmentURL($pObj.getValue('image'),'download',$imgQs))~" /> 14956 + #end 14957 + <div class=~"caption~"> 14958 + #if ($displayTitle && !$displayTitleFirstOnPinnedPosts) 14959 + <h5 class=~"text-left~">$!postTitle <br/> <span class=~"date_info~"> $!dateStr </span> </h5> 14960 + #end 14961 + #if ($displaySummaryOnPinnedPosts) 14962 + <div class=~"text-left post-summary~"> 14963 + #set ($postContent = $pObj.getProperty('extract').value) 14964 + $!pDoc.getRenderedContent($postContent, $pDoc.syntax.toIdString()) 14965 + </div> 14966 + #end 14967 + #displayPostDetails($pDoc) 14968 + </div> 14969 + </div> 14970 + </a> 14971 + </div> 14972 + #end 14973 + ## 14974 + ## 14975 + ## 14976 + #macro(formatPostDate $pDoc $pObj) 14977 + #set ($formattedDate = $NULL) 14978 + #set($dateStr = $datetool.format('medium', $postDate, $xcontext.locale)) 14979 + #if (~"$!dateStr~" != '') 14980 + #set ($dateArr = $dateStr.split(' ')) 14981 + #if ($dateArr.size() > 3) 14982 + #set ($dateStr = ~"~") 14983 + #foreach($s in $dateArr.subList(0, 3)) 14984 + #set ($dateStr = ~"$\{dateStr}$\{s} ~") 14985 + #end 14986 + #end 14987 + #end 14988 + #setVariable(~"$formattedDate~" $dateStr) 14989 + #end 14990 + ## 14991 + ## 14992 + ## 14993 + #macro(displayPostDetails $pDoc) 14994 + <div class=~"row post-details~"> 14995 + <div class=~"col-xs-8 detail~"> 14996 + #if ($services.like.displayButton($pDoc.documentReference)) 14997 + #set ($optLikeRecord = $services.like.getLikes($pDoc.documentReference)) 14998 + ## Retrieve the likes number in XWiki 12.9+ 14999 + #set ($likeNumber = $optLikeRecord.get()) 15000 + #if (!$stringtool.isNumeric($likeNumber.toString())) 15001 + ## Retrieve the likes number in XWiki versions greater than 12.7 and lower than 12.9. 15002 + #set ($likeNumber = $optLikeRecord.get().likeNumber) 15003 + #end 15004 + #if ($stringtool.isNumeric($likeNumber.toString())) 15005 + <div class=~"post-likes btn btn-default disabled badge~" 15006 + title=~"$escapetool.xml($services.localization.render('like.button.title', [$likeNumber]))~"> 15007 + $services.icon.renderHTML('heart') <span class=~"like-number~">$likeNumber</span> 15008 + </div> 15009 + #end 15010 + #elseif ($services.ratings) 15011 + #set ($ratingsConfigDoc = $services.ratings.getConfigurationDocument($pDoc.documentReference)) 15012 + #if ($ratingsConfigDoc.getValue('displayRatings') == 1) 15013 + <ul class=~"pull-left note list-inline~"> 15014 + #foreach ($x in [1..5]) 15015 + #set ($cls = ~"~") 15016 + #if ($x > $averageVote) 15017 + #set ($cls = ~"-o~") 15018 + #end 15019 + <li><span class=~"fa fa-star$\{cls}~"></span></li> 15020 + #end 15021 + </ul> 15022 + #end 15023 + #end 15024 + </div> 15025 + #if ($showPostComments) 15026 + <div class=~"col-xs-4 com_det~"> 15027 + <p class=~"text-right~">$services.icon.renderHTML('comments') ($!nbComments)</p> 15028 + </div> 15029 + #end 15030 + </div> 15031 + #end 15032 + ## 15033 + ## 15034 + ## 15035 + #macro(initLayoutVars $pDoc $pObj) 15036 + #set ($postTitle = $pDoc.display('title', 'view', $pObj)) 15037 + #set ($imageAtt = $pDoc.getAttachment($pObj.getValue('image'))) 15038 + #isPublished($pObj $isPublished) 15039 + #isHidden($pObj $isHidden) 15040 + #getEntryDate($pDoc $pObj $postDate) 15041 + #formatPostDate($postDate $dateStr) 15042 + #set ($averageVote = $mathtool.round($services.ratings.getAverageRating($pDoc.documentReference).getAverageVote())) 15043 + #set ($nbComments = $pDoc.getComments().size()) 15044 + #set ($showPostComments = true) 15045 + #if ($xwiki.getSpacePreferenceFor('showcomments', $pDoc.documentReference.lastSpaceReference, '1') == '0') 15046 + #set ($showPostComments = false) 15047 + #end 15048 + #end 15049 + ## 15050 + ## 15051 + ## 15052 + #macro(displayEditPinnedPostsButton) 15053 + #if (~"$!macroAdditionalParams.get('postIndex')~" == '0' && $services.security.authorization.hasAccess('edit', $pinnedPostsSourceDoc.fullName)) 15054 + #set ($discard = $xwiki.ssx.use('Blog.PinnedPostsSheet')) 15055 + #set ($discard = $xwiki.jsx.use('Blog.PinnedPostsSheet')) 15056 + #set ($editPinnedPostsURL = $xwiki.getURL($pinnedPostsSourceDoc.fullName, 'get', 'sheet=Blog.PinnedPostsSheet')) 15057 + #set ($newPinnedPostsObjectURL = $xwiki.getURL($pinnedPostsSourceDoc.fullName, 'objectadd', ~"classname=Blog.PinnedPostsClass&form_token=$!\{services.csrf.getToken()}~")) 15058 + <div class=~"edit-pinned-posts-container~"> 15059 + <div class=~"col-xs-12 col-sm-12 col-md-12 text-right edit-pinned-posts~"> 15060 + <a href=~"javascript:;~" class=~"editPinnedPosts~"> 15061 + <span class=~"glyphicon glyphicon-edit~" aria-hidden=~"true~"></span> $services.localization.render('blog.post.layout.cards.pinnedposts.edit') 15062 + </a> 15063 + <input type=~"hidden~" id=~"editPinnedPostsURL~" value=~"$escapetool.xml($editPinnedPostsURL)~"/> 15064 + <input type=~"hidden~" id=~"newPinnedPostsObjectURL~" value=~"$escapetool.xml($newPinnedPostsObjectURL)~"/> 15065 + </div> 15066 + </div> 15067 + #if (~"$!pinnedPostsObj~" == '') 15068 + <input type=~"hidden~" name=~"createPinnedPostsObject~" value=~"1~"/> 15069 + #end 15070 + #end 15071 + #end" %) 15072 +((( 15073 +(% class="macro-placeholder hidden" %) 15074 +((( 15075 +macro:velocity 15076 +))) 15077 +))) 15078 + 15079 +(% data-macro="startmacro:velocity|-||-| #if ($services.security.authorization.hasAccess('view', $xcontext.macro.params.reference)) 15080 + #set ($postDoc = $xwiki.getDocument($xcontext.macro.params.reference)) 15081 + #getEntryObject($postDoc $postObj) 15082 + #if (~"$!postObj~" != '') 15083 + #extractLayoutParametersFromString($xcontext.macro.params.params $macroAdditionalParams) 15084 + ## 15085 + #set ($displayTitle = true) 15086 + #if (~"$!macroAdditionalParams.get('displayTitle')~" == 'false') 15087 + #set ($displayTitle = false) 15088 + #end 15089 + ## 15090 + #set ($displayTitleFirstOnPinnedPosts = false) 15091 + #if (~"$!macroAdditionalParams.get('displayTitleFirstOnPinnedPosts')~" == 'true') 15092 + #set ($displayTitleFirstOnPinnedPosts = true) 15093 + #end 15094 + ## 15095 + #set ($displaySummaryOnPinnedPosts = true) 15096 + #if (~"$!macroAdditionalParams.get('displaySummaryOnPinnedPosts')~" == 'false') 15097 + #set ($displaySummaryOnPinnedPosts = false) 15098 + #end 15099 + ## 15100 + #set ($postIndex = $numbertool.toNumber($macroAdditionalParams.get('postIndex')).intValue()) 15101 + #if ($postIndex == 0) 15102 + #set ($stopBlogPostsDisplay = false) 15103 + #end 15104 + #set ($listResultsCount = $numbertool.toNumber($macroAdditionalParams.get('list-results-count')).intValue()) 15105 + #set($discard = $xwiki.ssx.use('Blog.BlogPostLayoutCards')) 15106 + #set($scaleWidth = 600) 15107 + #set($imgQs=~"width=$scaleWidth~") 15108 + ## Display pinned posts 15109 + ## Get the list of pinned posts 15110 + #set ($pinnedPosts = []) 15111 + #set ($pinnedPostsSourceDoc = $NULL) 15112 + #if (~"$!macroAdditionalParams.get('forceDisplayPinnedPosts')~" == 'true' && ~"$!macroAdditionalParams.get('list-blog')~" != '') 15113 + #set ($pinnedPostsSourceDoc = $xwiki.getDocument($macroAdditionalParams.get('list-blog'))) 15114 + ## Note that the 'list-blog' parameter contains the value of the 'blog' parameter of the blogpostlist macro passed to the current layout macro 15115 + #end 15116 + #if (~"$!pinnedPostsSourceDoc~" != '') 15117 + #set ($pinnedPostsObj = $pinnedPostsSourceDoc.getObject('Blog.PinnedPostsClass')) 15118 + #if (~"$!pinnedPostsObj~" != '') 15119 + #set ($orderedPinnedPostsJSON = $pinnedPostsObj.getProperty('orderedPosts').value) 15120 + #if (~"$!orderedPinnedPostsJSON~" != '') 15121 + #set ($pinnedPosts = $jsontool.parse($orderedPinnedPostsJSON)) 15122 + #else 15123 + #set ($pinnedPosts = $pinnedPostsObj.getProperty('posts').value) 15124 + #end 15125 + #end 15126 + #if ($postIndex == 0 && $pinnedPosts.size() > 0) 15127 + \{\{html clean=~"false~"}} 15128 + #set ($x = 0) 15129 + #set ($showPinnedPostsButton = true) 15130 + #foreach ($pinnedPost in $pinnedPosts) 15131 + #if ($x == 0) 15132 + <div class=~"row flex-container~"> 15133 + #if ($showPinnedPostsButton) 15134 + #displayEditPinnedPostsButton() 15135 + #set($showPinnedPostsButton = false) 15136 + #end 15137 + #end 15138 + #set ($pinnedPostDoc = $xwiki.getDocument($pinnedPost)) 15139 + #getEntryObject($pinnedPostDoc $pinnedPostObj) 15140 + #if (~"$!pinnedPostObj~" != '') 15141 + #displayPinnedPost($pinnedPostDoc $pinnedPostObj) 15142 + #end 15143 + #set ($x = $mathtool.add($x, 1)) 15144 + #if ($x == 3) 15145 + #set ($x = 0) 15146 + </div> 15147 + #end 15148 + #end 15149 + #if ($mathtool.mod($x, 3) != 0) 15150 + </div> 15151 + #end 15152 + \{\{/html}} 15153 + 15154 + ## If the first post is a pinned post : this means that all the posts are pinned 15155 + ## In this case, avoid displaying the posts again after the pinned posts section. 15156 + #if ($pinnedPosts.contains($xcontext.macro.params.reference)) 15157 + #set ($stopBlogPostsDisplay = true) 15158 + #end 15159 + #elseif($postIndex == 0 && $pinnedPosts.size() == 0) 15160 + 15161 + \{\{html}} 15162 + <div class=~"row no-pinnded-posts~"> 15163 + #displayEditPinnedPostsButton() 15164 + </div> 15165 + \{\{/html}} 15166 + 15167 + #end 15168 + #end 15169 + #if (!$stopBlogPostsDisplay) 15170 + \{\{html clean=~"false~"}} 15171 + #if (~"$!nbDisplayedPosts~" == '' || $postIndex == 0) 15172 + #set ($nbDisplayedPosts = 0) 15173 + #end 15174 + #if($mathtool.mod($nbDisplayedPosts, 2) == 0) 15175 + #if ($nbDisplayedPosts != 0) 15176 + </div> 15177 + #set ($lastHtmlTag = 'c') 15178 + #end 15179 + #if ($nbDisplayedPosts == $mathtool.sub($listResultsCount, 1)) 15180 + <div class=~"row~"> 15181 + #set ($lastHtmlTag = 'o') 15182 + #set ($lastRowClass = ~"~") 15183 + #else 15184 + <div class=~"row flex-container~"> 15185 + #set ($lastHtmlTag = 'o') 15186 + #set ($lastRowClass = ~"flex~") 15187 + #end 15188 + #end 15189 + #displaySmallPost($postDoc $postObj) 15190 + #if ($nbDisplayedPosts == $mathtool.sub($listResultsCount, 1)) 15191 + </div> 15192 + #set ($lastHtmlTag = 'c') 15193 + #end 15194 + #set ($nbDisplayedPosts = $mathtool.add($nbDisplayedPosts, 1)) 15195 + #if ($postIndex == $mathtool.sub($listResultsCount, 1)) 15196 + #if (~"$!lastHtmlTag~" == 'o') 15197 + #if ($mathtool.mod($nbDisplayedPosts, 2) == 1 && $lastRowClass == ~"flex~") 15198 + <div class=~"col-xs-12 col-sm-6~"></div> 15199 + #end 15200 + </div> 15201 + #end 15202 + #end 15203 + \{\{/html}} 15204 + #end 15205 + #else 15206 + \{\{error}}$services.localization.render('blog.blogpostlayout.notpost', [$xcontext.macro.params.reference])\{\{/error}} 15207 + #end 15208 + #else 15209 + \{\{error}}$services.localization.render('blog.blogpostlayout.post_view_not_allowed', [$xcontext.macro.params.reference])\{\{/error}} 15210 + #end" class="macro hidden macro-placeholder" %)macro:velocity(% data-macro="startmacro:html|-|clean=~"false~"|-|</div> 15211 +<div class=~"row flex-container~"> 15212 +<div class=~"col-xs-12 col-sm-6 next_blog~"> 15213 +<a href=~"/Nieuws/Nieuw%20beveiligingspaneel%20Axon%20van%20Aritech~" class=~"thumbnail~"> 15214 +<div class=~"row~"> 15215 +<div class=~"col-xs-4~"> 15216 +<img src=~"/download/Nieuws/Nieuw%20beveiligingspaneel%20Axon%20van%20Aritech/axon.png?width=600&rev=1.1~" /> 15217 +</div> 15218 +<div class=~"col-xs-8 art_det~"> 15219 +<p class=~"text-left~"> 15220 +Nieuw beveiligingspaneel Axon van Aritech <br/><span class=~"date_info~"> Nov 22, 2023, </span> 15221 +</p> 15222 +<div class=~"row post-details~"> 15223 +<div class=~"col-xs-8 detail~"> 15224 +<div class=~"post-likes btn btn-default disabled badge~" 15225 +title=~"Number of likes on this page: 1~"> 15226 +<span class=~"fa fa-heart~"></span> <span class=~"like-number~">1</span> 15227 +</div> 15228 +</div> 15229 +<div class=~"col-xs-4 com_det~"> 15230 +<p class=~"text-right~"><span class=~"fa fa-comments~"></span> (1)</p> 15231 +</div> 15232 +</div> 15233 +</div> 15234 +</div> 15235 +</a> 15236 +</div>" class="macro macro hidden macro-placeholder" %)macro:html 15237 +))) 15238 +))) 15239 + 15240 +(% class="row flex-container" %) 15241 +((( 15242 +(% class="col-xs-12 col-sm-6 next_blog" %) 15243 +((( 15244 +(% class="row" %) 15245 +((( 15246 +(% class="col-xs-4" %) 15247 +((( 15248 +[[~[~[image:/download/Nieuws/Nieuw%20beveiligingspaneel%20Axon%20van%20Aritech/axon.png?width=600&rev=1.1~]~]>>path:/Nieuws/Nieuw%20beveiligingspaneel%20Axon%20van%20Aritech||class="thumbnail"]] 15249 +))) 15250 + 15251 +(% class="col-xs-8 art_det" %) 15252 +((( 15253 +(% class="text-left" %) 15254 +[[Nieuw beveiligingspaneel Axon van Aritech 15255 +(% class="date_info" %) Nov 22, 2023,>>path:/Nieuws/Nieuw%20beveiligingspaneel%20Axon%20van%20Aritech||class="thumbnail"]] 15256 + 15257 +(% class="row post-details" %) 15258 +((( 15259 +(% class="col-xs-8 detail" %) 15260 +((( 15261 +(% class="post-likes btn btn-default disabled badge" title="Number of likes on this page: 1" %) 15262 +((( 15263 +[[(% class="like-number" %)1>>path:/Nieuws/Nieuw%20beveiligingspaneel%20Axon%20van%20Aritech||class="thumbnail"]] 15264 +))) 15265 +))) 15266 + 15267 +(% class="col-xs-4 com_det" %) 15268 +((( 15269 +(% class="text-right" %) 15270 +[[(1)>>path:/Nieuws/Nieuw%20beveiligingspaneel%20Axon%20van%20Aritech||class="thumbnail"]] 15271 +))) 15272 +))) 15273 +))) 15274 +))) 15275 +))) 15276 + 15277 +(% class="macro" data-macro="startmacro:blogPostLayoutCards|-|reference=~"Nieuws.Aankondiging update Syntess 6\\\\.9\\\\.0\\\\.0166~" params=~"displayTitle=true|useSummary=true|list-limit=10|list-layout=cards|list-blog=Nieuws.WebHome|list-paginated=no|list-results-count=10|postIndex=0|previousPostReference=|postIndex=1|previousPostReference=Nieuws.Nieuwsbrief januari 2024|postIndex=2|previousPostReference=Nieuws.Audit BMI en BORG|postIndex=3|previousPostReference=Nieuws.Nieuwsbrief december 2023|postIndex=4|previousPostReference=Nieuws.Nieuwsbrief november 2023|postIndex=5|previousPostReference=Nieuws.Feestelijke afsluiting, keuring materieel en tellen voorraden|postIndex=6|previousPostReference=Nieuws.RI&E 2023|postIndex=7|previousPostReference=Nieuws.Aankondiging update Syntess 6\\\\.9\\\\.0181|postIndex=8|previousPostReference=Nieuws.Nieuwsbrief oktober 2023|postIndex=9|previousPostReference=Nieuws.Nieuw beveiligingspaneel Axon van Aritech~"" %) 15278 +((( 15279 +(% class="macro-placeholder hidden" %) 15280 +((( 15281 +macro:blogPostLayoutCards 15282 +))) 15283 + 15284 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogCode~"" %) 15285 +((( 15286 +(% class="macro-placeholder hidden" %) 15287 +((( 15288 +macro:include 15289 +))) 15290 + 15291 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogParameters~"" %) 15292 +((( 15293 +(% class="macro-placeholder hidden" %) 15294 +((( 15295 +macro:include 15296 +))) 15297 + 15298 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|## Blog 15299 +#set($blogClassname = 'Blog.BlogClass') 15300 +#set($blogTemplate = 'Blog.BlogTemplate') 15301 +#set($blogSheet = 'Blog.BlogSheet') 15302 +## Blog entries 15303 +#set($blogPostClassname = 'Blog.BlogPostClass') 15304 +#set($blogPostTemplate = 'Blog.BlogPostTemplate') 15305 +#set($blogPostSheet = 'Blog.BlogPostSheet') 15306 +#set($blogPostObjectNumber = $xwiki.getDocument($blogPostTemplate).getObject($blogPostClassname).number) 15307 +#set($oldArticleClassname = 'XWiki.ArticleClass') 15308 +## Categories 15309 +#set($blogCategoryClassname = 'Blog.CategoryClass') 15310 +#set($blogCategoryTemplate = 'Blog.CategoryTemplate') 15311 +#set($blogCategorySheet = 'Blog.CategorySheet') 15312 +#set($blogCategoriesSheet = 'Blog.CategoriesSheet') 15313 +#set($oldBlogCategoryClassname = 'Blog.Categories') 15314 +#set($defaultCategoryParent = 'Blog.Categories') 15315 +## Style 15316 +#set($blogStyleDocumentName = 'Blog.BlogStyle') 15317 +#set($blogStyleDocument = $xwiki.getDocument($blogStyleDocumentName)) 15318 +## Clientside scripts 15319 +#set($blogScriptsDocumentName = 'Blog.BlogScripts') 15320 +#set($blogPublisher = $xwiki.getDocument('Blog.Publisher')) 15321 +## Misc 15322 +#set($thisURL = $doc.getURL($xcontext.action, $request.queryString)) 15323 +#set($isBlogPost = $doc.getObject($blogPostClassname)) 15324 +#set($defaultBlogSpace = 'Blog') 15325 +#set($isCategoryPage = $doc.getObject($blogCategoryClassname)) 15326 +#set($isCategoriesHomePage = $doc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 15327 +## 15328 +## 15329 +## 15330 +#** 15331 + * Displays an image, taken from the blog style document. 15332 + * 15333 + * @param $imgName The name of the icon from icons set to use. 15334 + *# 15335 +#macro(toolImage $imgName)(% class=~"icon-manage~"~%)$services.icon.render($imgName)(%~%)#end" %) 15336 +((( 15337 +(% class="macro-placeholder hidden" %) 15338 +((( 15339 +macro:velocity 15340 +))) 15341 +))) 15342 +))) 15343 + 15344 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|## 15345 +## 15346 +## 15347 +## Import the blog skin and javascripts. 15348 +$!xwiki.ssx.use($blogStyleDocumentName)## 15349 +$!xwiki.jsx.use($blogScriptsDocumentName)## 15350 +## 15351 +## import the hierarchy for the path the blog post footer (in displayEntryBlogLocation) 15352 +#template('hierarchy_macros.vm')## 15353 +## 15354 +## 15355 +#** 15356 + * Prints a blog. This is the main macro used in the BlogSheet. 15357 + * 15358 + * @param blogDoc the XDocument holding the blog definition object. 15359 + *### 15360 +#macro(printBlog $blogDoc) 15361 + \{\{include reference='Blog.CreatePost'/}} 15362 + 15363 + ## Use the blogPostList macro to display the blogposts 15364 + ## The blogPostList is used only in case of 'all' or 'paged' blog display type because the blogPostList macro 15365 + ## do not support FTM the monthly and weekly blog display types 15366 + #getBlogDisplayType($blogDoc $displayType) 15367 + #if ($displayType == 'weekly' || $displayType == 'monthly') 15368 + #getBlogEntries($blogDoc $entries) 15369 + #displayBlog($entries 'index' true true) 15370 + #displayNavigationLinks($blogDoc) 15371 + #else 15372 + #getBlogDisplayType($blogDoc $displayType) 15373 + #set ($paginated = 'no') 15374 + #if ($displayType == 'paginated') 15375 + #set ($paginated = 'yes') 15376 + #end 15377 + #getBlogPostsLayout($blogDoc $postsLayout) 15378 + (% class=~"hfeed index~" ~%)(((\{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$paginated~" layout=~"$!postsLayout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" /}}))) 15379 + #end 15380 +#end 15381 +## 15382 +## 15383 +## 15384 +#** 15385 + * Shows blog information. In view mode, the description is printed. In edit mode, allows changing blog settings: title, 15386 + * description, blog type (global or in-space), index display type (fixed size pagination, weekly index, monthly index, 15387 + * all entries). 15388 + * 15389 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 15390 + *### 15391 +#macro(showBlogInfo $blogDoc) 15392 + #if($blogDoc.getObject($blogClassname)) 15393 + ## Keep testing for inline action for backward compatibility with older blogs. 15394 + #if($xcontext.action == 'edit' || $xcontext.action == 'inline') 15395 + #macro(displayProperty $blogDoc $propname) 15396 + ; #displayPropName($xwiki.getClass($blogClassname).get($propname)): 15397 + : $blogDoc.display($propname) 15398 + #end 15399 + #displayProperty($blogDoc 'title') 15400 + #displayProperty($blogDoc 'description') 15401 + #displayProperty($blogDoc 'displayType') 15402 + #displayProperty($blogDoc 'itemsPerPage') 15403 + #displayProperty($blogDoc 'postsLayout') 15404 + #displayProperty($blogDoc 'postsLayoutParameters') 15405 + #else 15406 + $blogDoc.display('description') 15407 + #end 15408 + #elseif($doc.fullName == $blogSheet) 15409 += $services.localization.render('blog.code.blogsheet') = 15410 + \{\{translation key='blog.code.sheetexplanation'/}} 15411 + #else 15412 + \{\{warning}}\{\{translation key='blog.code.notblog'/}}\{\{/warning}} 15413 + #end 15414 +#end 15415 +## 15416 +## 15417 +## 15418 +#** 15419 + * Retrieve the blog document, which usually is either <tt><Space>.WebHome</tt> for whole-spaces blogs, or 15420 + * <tt><Space>.Blog</tt> for in-space blogs. If none of these documents contains a blog object, then the first 15421 + * (alphabetically) document in the target space that contains one is returned. Finally, if no document in the current 15422 + * space contains a blog object, then <tt>Blog.WebHome</tt> is returned as the default blog. 15423 + * 15424 + * @param space A <tt>String</tt>, the name of the space where to search. 15425 + * @param blogDoc The resulting XDocument. 15426 + *### 15427 +#macro(getBlogDocument $space $blogDoc) 15428 + #set ($result = $NULL) 15429 + ## Check if the current space has a WebPreferences page that contains a Blog.EnablePanelsConfigurationClass object. 15430 + ## It is possible to display a blog's panels info in a page that is not inside a blog and in that case we need to 15431 + ## identify the right blog based on a configuration object in a WebPreferences page. 15432 + #set ($spaceReference = $services.model.resolveSpace($space)) 15433 + #set ($preferencesDocRef = $services.model.createDocumentReference('WebPreferences', $spaceReference)) 15434 + #set ($preferencesDoc = $xwiki.getDocument($preferencesDocRef)) 15435 + #set ($preferencesObj = $preferencesDoc.getObject('Blog.EnablePanelsConfigurationClass')) 15436 + #if ($preferencesObj) 15437 + #set ($result = $xwiki.getDocument( $preferencesObj.getValue('blog'))) 15438 + #end 15439 + ## If no special case occurs (like the previous one), identify a blog by checking the current location from its hierarchy. 15440 + #if (~"$!result~" == '') 15441 + ## First, try the Space.WebHome, for a whole-space blog 15442 + #set($result = $xwiki.getDocument(~"$\{space}.WebHome~")) 15443 + #if(!$result.getObject($blogClassname)) 15444 + ## Second, try the Space.Blog document 15445 + #set($result = $xwiki.getDocument(~"$\{space}.Blog~")) 15446 + #if(!$result.getObject($blogClassname)) 15447 + ## Third, try searching for a blog document in the current space 15448 + ## Prevent the query fail when the space contains dots '.' 15449 + #set($blogDocs = $services.query.hql(~", BaseObject obj where doc.space = :space and obj.name = doc.fullName and obj.className = '$blogClassname' order by doc.name~").setLimit(1).setOffset(0).bindValue('space', $space).execute()) 15450 + #if($blogDocs.size() > 0) 15451 + #set($result = $xwiki.getDocument($blogDocs.get(0))) 15452 + #else 15453 + ## Fourth, try searching for a blog document that have the its 'postsLocation' set to the current space 15454 + #set($blogDocs = $services.query.hql(~", BaseObject obj, StringProperty as postsLocationProp where obj.name = doc.fullName and obj.className = '$blogClassname' and obj.id=postsLocationProp.id.id and postsLocationProp.id.name='postsLocation' and postsLocationProp.value=:postsLocation order by doc.name~").setLimit(1).setOffset(0).bindValue('postsLocation', $space).execute()) 15455 + #if($blogDocs.size() > 0) 15456 + #set($result = $xwiki.getDocument($blogDocs.get(0))) 15457 + #else 15458 + ## Last, fallback to Blog.WebHome, the default blog 15459 + #set($result = $xwiki.getDocument('Blog.WebHome')) 15460 + #end 15461 + #end 15462 + #end 15463 + #end 15464 + #end 15465 + #set ($blogDoc = $NULL) 15466 + #setVariable (~"$blogDoc~" $result) 15467 +#end 15468 +## 15469 +## 15470 +## 15471 +#** 15472 + * Retrieve the blog title. 15473 + * 15474 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass<tt> object with the <tt>title</tt> property set. 15475 + * @param title The resulting title. 15476 + *### 15477 +#macro(getBlogTitle $blogDoc $title) 15478 + ## Titles can contain velocity code (most commonly translations), so we should evaluate them. 15479 + #set ($title = $NULL) 15480 + #setVariable (~"$title~" $!blogDoc.displayTitle) 15481 +#end 15482 +## 15483 +## 15484 +## 15485 +#** 15486 + * Retrieve the blog description. 15487 + * 15488 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>description</tt> 15489 + * property set. 15490 + * @param description The resulting description. 15491 + *### 15492 +#macro(getBlogDescription $blogDoc $description) 15493 + #getBlogProperty($blogDoc 'description' '' $result) 15494 + #set ($description = $NULL) 15495 + #setVariable (~"$description~" $result) 15496 +#end 15497 +## 15498 +## 15499 +## 15500 +#** 15501 + * Retrieves a list of entries to be displayed. The entries are either part of the blog's space, or have the blog 15502 + * document set as a parent. The number and range of entries returned (from all those belonging to this blog) depends on 15503 + * the blog display type: paginated (fixed number of entries), weekly (all entries in a week), monthly (all entries in a 15504 + * month), or all. 15505 + * 15506 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 15507 + * @param entries The resulting list of entries to display, a list of XDocument names. 15508 + *### 15509 +#macro(getBlogEntries $blogDoc $entries) 15510 + #if (!$entries) 15511 + #setVariable (~"$entries~" []) 15512 + #end 15513 + #getAllBlogPostsQuery($query) 15514 + #isDefaultBlog($blogDoc $isDefault) 15515 + #set($queryParams = \{}) 15516 + #if ($isDefault) 15517 + #getCategoryAllBlogPostsQuery($query) 15518 + #set($query = ~"$\{query} and (doc.creator = :creator or (isPublished.value = 1 and hidden.value = 0)) and (doc.space = :space or catList like :catList escape '!')~") 15519 + #set($discard = $queryParams.put('creator', $xcontext.user)) 15520 + #set($discard = $queryParams.put('space', $blogDoc.space)) 15521 + #set($sanitizedBlogSpace = $blogDoc.space.replaceAll('([%_!])', '!$1').concat('.%')) 15522 + #set($discard = $queryParams.put('catList', $sanitizedBlogSpace)) 15523 + #else 15524 + #set($query = ~"$\{query} and (doc.space = :space or doc.parent = :parent)~") 15525 + #getBlogPostsLocation($blogDoc.space $blogPostsLocation) 15526 + #set($discard = $queryParams.put('space', $blogPostsLocation)) 15527 + #set($discard = $queryParams.put('parent', $blogDoc.space)) 15528 + #end 15529 + #getBlogDisplayType($blogDoc $displayType) 15530 + #if($displayType == 'weekly') 15531 + #getWeeklyBlogEntries($blogDoc $query $entries $queryParams) 15532 + #elseif($displayType == 'monthly') 15533 + #getMonthlyBlogEntries($blogDoc $query $entries $queryParams) 15534 + #elseif($displayType == 'all') 15535 + #getAllBlogEntries($blogDoc $query $entries $queryParams) 15536 + #else 15537 + #getPagedBlogEntries($blogDoc $query $entries $queryParams) 15538 + #end 15539 +#end 15540 +## 15541 +## 15542 +## 15543 +#** 15544 + * Retrieves a list of entries to be displayed. The entries are taken from a ~"page~" of the blog, a sequence of documents 15545 + * defined by the request parameters <tt>ipp</tt> (items per page) and <tt>page</tt> (the current page). Initially the 15546 + * first page is displayed, with the number of entries defined in the blog object in the <tt>itemsPerPage</tt> property 15547 + * (10 if not defined). 15548 + * 15549 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 15550 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 15551 + * refined to restrict to a given space, or to a given search criteria, etc. 15552 + * @param entries The resulting list of entries to display, a list of XDocument names. 15553 + * @param queryParams The parameters to bind with the query. 15554 + *### 15555 +#macro(getPagedBlogEntries $blogDoc $query $entries $queryParams) 15556 + #if (!$entries) 15557 + #setVariable (~"$entries~" []) 15558 + #end 15559 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 15560 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~")) 15561 + #bindQueryParameters($countQueryObj $queryParams) 15562 + #bindQueryParameters($queryObj $queryParams) 15563 + #set($totalEntries = $countQueryObj.count()) 15564 + #getBlogProperty($blogDoc 'itemsPerPage' '10' $defaultItemsPerPage) 15565 + #set($defaultItemsPerPage = $numbertool.toNumber($defaultItemsPerPage).intValue()) 15566 + ## This macro is defined in the default macros.vm library. It also sets $itemsPerPage and $startAt. 15567 + #preparePagedViewParams($totalEntries $defaultItemsPerPage) 15568 + #set($discard = $entries.addAll($queryObj.setLimit($itemsPerPage).setOffset($startAt).addFilter(~"unique~").execute())) 15569 +#end 15570 +## 15571 +## 15572 +## 15573 +#** 15574 + * Retrieves a list of entries to be displayed. The entries are taken from a week of the blog. The target week is 15575 + * defined by the request parameters <tt>week</tt> (the week number in the year, from 1 to 52) and <tt>year</tt> (4 15576 + * digit year). Initially the current week is displayed. 15577 + * 15578 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 15579 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 15580 + * refined to restrict to a given space, or to a given search criteria, etc. 15581 + * @param entries The resulting list of entries to display, a list of XDocument names. 15582 + * @param queryParams The parameters to bind with the query. 15583 + *### 15584 +#macro(getWeeklyBlogEntries $blogDoc $query $entries $queryParams) 15585 + #if (!$entries) 15586 + #setVariable (~"$entries~" []) 15587 + #end 15588 + #getRequestedWeek($weekDate) 15589 + #set($dateFormatter = $xwiki.jodatime.getDateTimeFormatterForPattern('yyyy-MM-dd')) 15590 + #set($minDay = $dateFormatter.print($weekDate.toMutableDateTime().weekOfWeekyear().roundFloor())) 15591 + #set($maxDay = $dateFormatter.print($weekDate.toMutableDateTime().weekOfWeekyear().roundCeiling())) 15592 + #set($query = ~"$\{query} and publishDate.value >= '$minDay' and publishDate.value < '$maxDay'~") 15593 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 15594 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 15595 + #bindQueryParameters($countQueryObj $queryParams) 15596 + #bindQueryParameters($queryObj $queryParams) 15597 + #set($totalEntries = $countQueryObj.count()) 15598 + #set($discard = $entries.addAll($queryObj.execute())) 15599 +#end 15600 +## 15601 +## 15602 +## 15603 +#** 15604 + * Retrieves a list of entries to be displayed. The entries are taken from a month of the blog. The target month is 15605 + * defined by the request parameters <tt>month</tt> (the month number, from 1 to 12) and <tt>year</tt> (4 15606 + * digit year). Initially the current month is displayed. 15607 + * 15608 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 15609 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 15610 + * refined to restrict to a given space, or to a given search criteria, etc. 15611 + * @param entries The resulting list of entries to display, a list of XDocument names. 15612 + * @param queryParams The parameters to bind with the query. 15613 + *### 15614 +#macro(getMonthlyBlogEntries $blogDoc $query $entries) 15615 + #if (!$entries) 15616 + #setVariable (~"$entries~" []) 15617 + #end 15618 + #getRequestedMonth($monthDate) 15619 + #set($dateFormatter = $xwiki.jodatime.getDateTimeFormatterForPattern('yyyy-MM-dd')) 15620 + #set($minDay = $dateFormatter.print($monthDate.toMutableDateTime().monthOfYear().roundFloor())) 15621 + #set($maxDay = $dateFormatter.print($monthDate.toMutableDateTime().monthOfYear().roundCeiling())) 15622 + #set($query = ~"$\{query} and publishDate.value >= '$minDay' and publishDate.value < '$maxDay'~") 15623 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 15624 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 15625 + #bindQueryParameters($countQueryObj $queryParams) 15626 + #bindQueryParameters($queryObj $queryParams) 15627 + #set($totalEntries = $countQueryObj.count()) 15628 + #set($discard = $entries.addAll($queryObj.execute())) 15629 +#end 15630 +## 15631 +## 15632 +## 15633 +#** 15634 + * Retrieves a list of entries to be displayed. All entries belonging to the current blog are returned. 15635 + * 15636 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 15637 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 15638 + * refined to restrict to a given space, or to a given search criteria, etc. 15639 + * @param entries The resulting list of entries to display, a list of XDocument names. 15640 + * @param queryParams The parameters to bind with the query. 15641 + *### 15642 +#macro(getAllBlogEntries $blogDoc $query $entries $queryParams) 15643 + #if (!$entries) 15644 + #setVariable (~"$entries~" []) 15645 + #end 15646 + #set($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 15647 + #set($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 15648 + #bindQueryParameters($countQueryObj $queryParams) 15649 + #bindQueryParameters($queryObj $queryParams) 15650 + #set($totalEntries = $countQueryObj.count()) 15651 + #set($discard = $entries.addAll($queryObj.execute())) 15652 +#end 15653 +## 15654 +## 15655 +## 15656 +#** 15657 + * Retrieves a list of entries to be displayed. Only (and all) unpublished entries are returned. 15658 + * 15659 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 15660 + * @param query The base query for selecting entries. Apart from the base query that selects entries, it can further be 15661 + * refined to restrict to a given space, or to a given search criteria, etc. 15662 + * @param queryParams The parameters to bind with the query. 15663 + * @param entries The resulting list of entries to display, a list of XDocument names. 15664 + *### 15665 +#macro(getUnpublishedBlogEntries $blogDoc $query $queryParams $entries) 15666 + #if (!$entries) 15667 + #setVariable (~"$entries~" []) 15668 + #end 15669 + #set($query = ~"$\{query} and isPublished.value = 0~") 15670 + #set ($countQueryObj = $services.query.hql($query).addFilter(~"unique~")) 15671 + #set ($queryObj = $services.query.hql(~"$\{query} order by publishDate.value desc~").addFilter(~"unique~")) 15672 + #bindQueryParameters($countQueryObj $queryParams) 15673 + #bindQueryParameters($queryObj $queryParams) 15674 + #set($totalEntries = $countQueryObj.count()) 15675 + #set($discard = $entries.addAll($queryObj.execute())) 15676 +#end 15677 +## 15678 +## 15679 +## 15680 +#** 15681 + * Retrieves a list of entries to be displayed. The entries are taken from all the wiki, and not from a specific blog. 15682 + * 15683 + * @param entries The resulting list of entries to display, a list of XDocument names. 15684 + *### 15685 +#macro(getGlobalBlogEntries $entries) 15686 + #if (!$entries) 15687 + #setVariable (~"$entries~" []) 15688 + #end 15689 + #getAllBlogPostsQuery($query) 15690 + #set($totalEntries = $services.query.hql($query).count()) 15691 + #set($defaultItemsPerPage = 20) 15692 + ## This macro is defined in the default macros.vm library. It also sets $itemsPerPage and $startAt. 15693 + #preparePagedViewParams($totalEntries $defaultItemsPerPage) 15694 + #set($discard = $entries.addAll($services.query.hql(~"$\{query} order by publishDate.value desc~").setLimit($itemsPerPage).setOffset($startAt).execute())) 15695 +#end 15696 +#** 15697 + * Return the base query for selecting blog entries. It filters only visible entries, but does not bind to a specific 15698 + * blog, nor specify a range or an ordering criteria. 15699 + * 15700 + * This macro is a duplicate of #getAllBlogPostsQuery($query). Keep it in case it's used outside, in custom code. 15701 + * 15702 + * @param query The basic query for selecting blog entries. 15703 + *# 15704 +#macro(getBlogEntriesBaseQuery $query) 15705 + #getAllBlogPostsQuery($query) 15706 +#end 15707 +#** 15708 + * Return the Query for selecting the all wiki blog posts without filtering 15709 + * 15710 + * @param query The basic query for selecting blog entries. 15711 + *# 15712 +#macro(getAllBlogPostsQuery $query) 15713 + #set ($query = $NULL) 15714 + #setVariable(~"$query~" ~", BaseObject as obj, IntegerProperty isPublished, 15715 + IntegerProperty hidden, DateProperty publishDate 15716 + where doc.fullName <> '$blogPostTemplate' and 15717 + obj.name=doc.fullName and obj.className='$blogPostClassname' and 15718 + isPublished.id.id = obj.id and isPublished.id.name = 'published' and 15719 + hidden.id.id = obj.id and hidden.id.name='hidden' and 15720 + publishDate.id.id = obj.id and publishDate.id.name='publishDate' and 15721 + (doc.creator = '$escapetool.sql($xcontext.user)' or (isPublished.value = 1 and hidden.value = 0))~") 15722 +#end 15723 +## 15724 +## 15725 +## 15726 +###** 15727 + * Return the Query for selecting the all wiki blog posts with categories filtering 15728 + * 15729 + * @param query The basic query for selecting blog entries. 15730 + *### 15731 +#macro(getCategoryAllBlogPostsQuery $query) 15732 + #set ($query = $NULL) 15733 + #getAllBlogPostsQuery($baseQuery) 15734 + #set ($baseQuery = $baseQuery.replace('DateProperty publishDate', 'DateProperty publishDate, DBStringListProperty as category left join category.list catList')) 15735 + #setVariable(~"$query~" ~"$\{baseQuery} and obj.id=category.id.id and category.id.name='category'~") 15736 +#end 15737 +## 15738 +## 15739 +## 15740 +#** 15741 + * Determines how is the blog index split into pages: paginated (fixed number of entries), weekly (all entries in a 15742 + * week), monthly (all entries in a month), or all. 15743 + * 15744 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>displayType</tt> 15745 + * property set. 15746 + * @param displayType The resulting string. If the blog object does not define anything, it is considered paginated. 15747 + *### 15748 +#macro(getBlogDisplayType $blogDoc $displayType) 15749 + #getBlogProperty($blogDoc 'displayType' 'paginated' $result) 15750 + #set ($displayType = $NULL) 15751 + #setVariable (~"$displayType~" $result) 15752 +#end 15753 +## 15754 +## 15755 +## 15756 +#** 15757 + * Displays a list of entries. 15758 + * 15759 + * @param entries The entries to display, a list of XDocument names. 15760 + * @param displaying What exactly is displayed: blog index, a single blog entry, a blog category, search results, 15761 + * unpublished entries, etc. This will be used as the classname(s) for the container div (hfeed). Currently 15762 + * used values: index, single, category, search, unpublished, hidden. 15763 + * @param onlyExtract If <tt>true</tt>, only display the extract of articles where available, otherwise display the full content. 15764 + * @param shouldDisplayTitles If <tt>true</tt>, display the blog title (blog posts shouldn't display the title when they're 15765 + * displayed alone on their page since it's the page title which is used in this case) 15766 + *### 15767 +#macro(displayBlog $entries $displaying $onlyExtract $shouldDisplayTitles) 15768 + #set($blogDay = '') 15769 + (% class=~"hfeed $!\{displaying}~" ~%)((( 15770 + (% class=~"blogDay~" ~%)((( 15771 + #foreach ($entryDoc in $xwiki.wrapDocs($entries)) 15772 + #getEntryObject($entryDoc $entryObj) 15773 + ## Although all entries should have one of the two objects, better check to be sure. 15774 + #if(~"$!\{entryObj}~" != '') 15775 + #getEntryDate($entryDoc $entryObj $entryDate) 15776 + ## Display a ~"calendar sheet~" for each day. All entries posted on the same day share one such sheet. 15777 + #set($entryDateStr = $xwiki.formatDate($entryDate, 'yyyyMMMMdd')) 15778 + #if($blogDay != $entryDateStr) 15779 + #if($blogDay != '') 15780 + ))) 15781 + (% class=~"blogDay~" ~%)((( 15782 + #end 15783 + #displayBlogDate($entryDate) 15784 + #set ($blogDay = $entryDateStr) 15785 + #end 15786 + ## Finally, display the entry. 15787 + #displayEntry($entryDoc $entryObj $onlyExtract $shouldDisplayTitles true) 15788 + #end 15789 + #end 15790 + )))## blogDay 15791 + )))## hfeed 15792 +#end 15793 +## 15794 +## 15795 +## 15796 +#** 15797 + * Get the entry object, either a new BlogPost or an old Article. 15798 + * 15799 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 15800 + * @param entryObj The resulting xobject of the blog post. 15801 + *### 15802 +#macro(getEntryObject $entryDoc $__entryObj) 15803 + #set($result = $entryDoc.getObject(~"$\{blogPostClassname}~")) 15804 + #if(!$result) 15805 + #set($result = $entryDoc.getObject(~"$\{oldArticleClassname}~")) 15806 + #end 15807 + ## NOTE: The reason we put an underscore in front of the variable name is to prevent the following line from 15808 + ## overwriting the $entryObj variable that may be defined before this macro is called. Of course, $__entryObj may be 15809 + ## overwritten in this case but it's less likely to have such a variable defined before. 15810 + #set ($__entryObj = $NULL) 15811 + #setVariable (~"$__entryObj~" $result) 15812 +#end 15813 +## 15814 +## 15815 +## 15816 +#** 15817 + * Gets the date associated with a blog entry. This is the publication date. For unpublished entries, initially this is 15818 + * the document creation date, but can be edited by the user. 15819 + * 15820 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 15821 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 15822 + * @param result The resulting date, an instance of <tt>java.util.Date</tt>. 15823 + *### 15824 +#macro(getEntryDate $entryDoc $entryObj $result) 15825 + #set ($result = $NULL) 15826 + #setVariable (~"$result~" $entryObj.getProperty('publishDate').value) 15827 +#end 15828 +## 15829 +## 15830 +## 15831 +#** 15832 + * Displays a date, nicely formatted as a calendar page. 15833 + * 15834 + * @param date The date to display, an instance of <tt>java.util.Date</tt>. 15835 + *### 15836 +#macro(displayBlogDate $date) 15837 + #set($year = $xwiki.formatDate($date, 'yyyy')) 15838 + ## 3 letter month name, like Jan, Dec. 15839 + #set($month = $xwiki.formatDate($date, 'MMM')) 15840 + ## Uncomment to get a full length month name, like January, December. 15841 + ## TODO: this could be defined somewhere in the blog style. 15842 + ## #set($month = $xwiki.formatDate($date, 'MMMM')) 15843 + #set($day = $xwiki.formatDate($date, 'dd')) 15844 + (% class=~"blogdate~" ~%) 15845 + == (% class=~"month~" ~%)$month(%~%) (% class=~"day~" ~%)$day(%~%) (% class=~"year~" ~%)$year(%~%) == 15846 +#end 15847 +## 15848 +## 15849 +## 15850 +#** 15851 + * Displays a blog article: management tools, header, content, footer. 15852 + * 15853 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 15854 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 15855 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 15856 + * @param shouldDisplayTitle If <tt>true</tt>, display the blog title (blog posts shouldn't display the title 15857 + * when they're displayed alone on their page since it's the page title which is used in this case) 15858 + * @param shouldDisplayActions If <tt>true</tt>, display the blog post actions buttons 15859 + *### 15860 +#macro(displayEntry $entryDoc $entryObj $onlyExtract $shouldDisplayTitle $shouldDisplayActions) 15861 + ## Only articles with an explicit hidden setting or an explicit unpublished setting are hidden 15862 + #isPublished($entryObj $isPublished) 15863 + #isHidden($entryObj $isHidden) 15864 + #if($doc.fullName == $entryDoc.fullName) 15865 + (% class=~"hentry single-article~" ~%)((( 15866 + #else 15867 + (% class=~"hentry#if(!$isPublished) unpublished-article#elseif($isHidden) hidden-article#end~" ~%)((( 15868 + #end 15869 + #if ($shouldDisplayActions) 15870 + #displayEntryTools($entryDoc $entryObj) 15871 + #end 15872 + #if($shouldDisplayTitle) 15873 + #displayEntryTitle($entryDoc $entryObj) 15874 + #end 15875 + #if($doc.fullName == $entryDoc.fullName) 15876 + #if(!$isPublished) 15877 + \{\{warning}}\{\{translation key='blog.code.published'/}}\{\{/warning}} 15878 + #elseif($isHidden) 15879 + \{\{warning}}\{\{translation key='blog.code.hidden'/}}\{\{/warning}} 15880 + #end 15881 + #end 15882 + #displayEntryContent($entryDoc $entryObj $onlyExtract) 15883 + #displayEntryFooter($entryDoc $entryObj) 15884 + )))## hentry 15885 +#end 15886 +## 15887 +## 15888 +## 15889 +#** 15890 + * Checks if the provided blog is published or not. 15891 + * 15892 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 15893 + * @param isPublished The resulting boolean, true if the entry is considered published. 15894 + *### 15895 +#macro(isPublished $entryObj $isPublished) 15896 + #set ($isPublished = $NULL) 15897 + ## This should work for both old articles, which don't have the 'published' property at all, and 15898 + ## are considered published by default, and new entries, that should have 1 if published. 15899 + #if (~"$!\{entryObj.getProperty('published').value}~" != '0') 15900 + #setVariable (~"$isPublished~" true) 15901 + #else 15902 + #setVariable (~"$isPublished~" false) 15903 + #end 15904 +#end 15905 +## 15906 +## 15907 +## 15908 +#** 15909 + * Checks if the provided blog is hidden or not. 15910 + * 15911 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass<tt> xclass. 15912 + * @param isHiddel The resulting boolean, true if the entry is considered hidden. 15913 + *### 15914 +#macro(isHidden $entryObj $isHidden) 15915 + #set ($isHidden = $NULL) 15916 + ## This should work for both old articles, which don't have the 'hidden' property at all, and 15917 + ## are considered visible by default, and new entries, that should have 1 if hidden. 15918 + #if (~"$!\{entryObj.getProperty('hidden').value}~" == '1') 15919 + #setVariable (~"$isHidden~" true) 15920 + #else 15921 + #setVariable (~"$isHidden~" false) 15922 + #end 15923 +#end 15924 +## 15925 +## 15926 +## 15927 +#** 15928 + * Displays several ~"tools~" for manipulating blog posts: hide/show, publish, edit. 15929 + * 15930 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 15931 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 15932 + *### 15933 +#macro(displayEntryTools $entryDoc $entryObj) 15934 + #if($xcontext.action == 'view') 15935 + (% class=~"blog-entry-toolbox~" ~%)((( 15936 + #displayPublishButton($entryDoc $entryObj) 15937 + #displayHideShowButton($entryDoc $entryObj) 15938 + #displayEditButton($entryDoc $entryObj) 15939 + #displayDeleteButton($entryDoc $entryObj) 15940 + ))) 15941 + #end 15942 +#end 15943 +## 15944 +## 15945 +## 15946 +#** 15947 + * Displays the publish button to the entry <strong>creator</strong>, if the article is not published yet. 15948 + * 15949 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 15950 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 15951 + * @todo AJAX calls. 15952 + *### 15953 +#macro(displayPublishButton $entryDoc $entryObj) 15954 + #isPublished($entryObj $isPublished) 15955 + #if(!$isPublished && $entryDoc.creator == $xcontext.user && $xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 15956 + [[#toolImage('world')>>path:$blogPublisher.getURL('view', ~"entryName=$\{escapetool.url($entryDoc.fullName)}&xredirect=$\{escapetool.url($thisURL)}&form_token=$!\{services.csrf.getToken()}~")||title=~"$services.localization.render('blog.code.notpublished')~"]]## 15957 + #end 15958 +#end 15959 +## 15960 +## 15961 +## 15962 +#** 15963 + * Displays the hide or show button to the entry <strong>creator</strong>, if the article is already published. 15964 + * 15965 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 15966 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 15967 + *### 15968 +#macro(displayHideShowButton $entryDoc $entryObj) 15969 + #isPublished($entryObj $isPublished) 15970 + #isHidden($entryObj $isHidden) 15971 + ## Only published articles can be hidden. Unpublished articles are considered already hidden. 15972 + #if($isPublished && $entryDoc.creator == $xcontext.user && $xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 15973 + #set ($queryString = \{ 15974 + 'xredirect' : $thisURL, 15975 + 'form_token' : $services.csrf.getToken() 15976 + }) 15977 + #if ($isHidden) 15978 + #set ($discard = $queryString.putAll(\{ 15979 + ~"$\{entryObj.getxWikiClass().getName()}_$\{entryObj.number}_hidden~" : 0, 15980 + 'comment' : $services.localization.render('blog.code.madevisible') 15981 + })) 15982 + #set ($lockURL = $entryDoc.getURL('save', $escapetool.url($queryString))) 15983 + [[#toolImage('unlock')>>path:$lockURL||class=~"blog-tool-show~" title=~"$services.localization.render('blog.code.makevisible')~"]]## 15984 + #else 15985 + #set ($discard = $queryString.putAll(\{ 15986 + ~"$\{entryObj.getxWikiClass().getName()}_$\{entryObj.number}_hidden~" : 1, 15987 + 'comment' : $services.localization.render('blog.code.hid') 15988 + })) 15989 + #set ($lockURL = $entryDoc.getURL('save', $escapetool.url($queryString))) 15990 + [[#toolImage('lock')>>path:$lockURL||class=~"blog-tool-hide~" title=~"$services.localization.render('blog.code.hide')~"]]## 15991 + #end 15992 + #end 15993 +#end 15994 +## 15995 +## 15996 +## 15997 +#** 15998 + * Displays the edit button to those that can edit the article. 15999 + * 16000 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 16001 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 16002 + *### 16003 +#macro(displayEditButton $entryDoc $entryObj) 16004 + #if($xwiki.hasAccessLevel('edit', $xcontext.user, $entryDoc.fullName)) 16005 + ## Call getDefaultEditMode() for backward compatibility with older blog posts. 16006 + [[#toolImage('pencil')>>path:$entryDoc.getURL('edit')||title=~"$services.localization.render('blog.code.editpost')~"]]## 16007 + #end 16008 +#end 16009 +## 16010 +## 16011 +## 16012 +#** 16013 + * Displays the delete button to those that can edit the article. 16014 + * 16015 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 16016 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 16017 + * @todo AJAX calls. 16018 + *### 16019 +#macro(displayDeleteButton $entryDoc $entryObj) 16020 + #if($xwiki.hasAccessLevel('delete', $xcontext.user, $entryDoc.fullName)) 16021 + [[#toolImage('cross')>>path:$entryDoc.getURL('delete')||title=~"$services.localization.render('blog.code.deletepost')~"]]## 16022 + #end 16023 +#end 16024 +## 16025 +## 16026 +## 16027 +#** 16028 + * Displays the title of the entry. 16029 + * 16030 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 16031 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 16032 + *### 16033 +#macro(displayEntryTitle $entryDoc $entryObj) 16034 + #if($doc.fullName == $entryDoc.fullName) 16035 + (% class=~"entry-title~" ~%) 16036 + = $services.rendering.escape($entryDoc.display('title', 'view', $entryObj), $xwiki.getCurrentContentSyntaxId()) = 16037 + #else 16038 + (% class=~"entry-title~" ~%) 16039 + === [[$services.rendering.escape($entryDoc.display('title', 'view', $entryObj),$xwiki.getCurrentContentSyntaxId())>>doc:$services.rendering.escape($services.model.serialize($entryDoc.getDocumentReference(),'default'),$xwiki.getCurrentContentSyntaxId())]] === 16040 + #end 16041 +#end 16042 +## 16043 +## 16044 +## 16045 +#** 16046 + * Displays the body of the entry. 16047 + * 16048 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 16049 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 16050 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 16051 + *### 16052 +#macro(displayEntryContent $entryDoc $entryObj $onlyExtract) 16053 + (% class=~"#if($onlyExtract)entry-summary#\{else}entry-content#end~" ~%)((( 16054 + #getEntryContent($entryDoc $entryObj $onlyExtract $entryContent) 16055 + ## FIXME: This causes the blog's content to not be annotatable. See http://jira.xwiki.org/browse/XWIKI-6328 16056 + ## Should probably be replaced by a display macro call with a reference to the object property holding the post's content 16057 + \{\{html wiki=~"false~"}}$entryDoc.getRenderedContent($entryContent, $entryDoc.syntax.toIdString())\{\{/html}} 16058 + ))) ## entry-content 16059 + (% class=~"clearfloats~" ~%)((())) 16060 +#end 16061 +## 16062 +## 16063 +## 16064 +#** 16065 + * Extracts the body of the entry that should be displayed. If <tt>onlyExtract</tt> is <tt>true</tt>, display the content 16066 + * of the <tt>extract</tt> field (if not empty). 16067 + * 16068 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 16069 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 16070 + * @param onlyExtract If <tt>true</tt>, try to display only a summary of the entry, instead of the full content. 16071 + * @param entryContent The resulting content. 16072 + * @param removeEllipsis If <tt>true</tt>, then don't display an ellipsis at the end of the content (only used when 16073 + * <tt>onlyExtract</tt> is <tt>true</tt>) 16074 + *### 16075 +#macro(getEntryContent $entryDoc $entryObj $onlyExtract $entryContent $removeEllipsis) 16076 + #if ($onlyExtract) 16077 + ## Note: We trim the summary so that if there's some white space it won't cause the summary to be used instead 16078 + ## of the content. 16079 + #set ($macro.result = $entryObj.getProperty('extract').value.trim()) 16080 + #end 16081 + #if(~"$!macro.result~" == '') 16082 + #set($macro.result = $entryObj.getProperty('content').value) 16083 +#* Disabled until the content can be cleanly cut. 16084 +* #if($onlyExtract && $result.length()>$maxchars) 16085 +* #set($i = $macro.result.lastIndexOf(~" ~", $maxchars)) 16086 +* #set($i = $i + 1) 16087 +* #set($macro.result = ~"$\{macro.result.substring(0,$i)} *[...>$\{entryDoc.fullName}]*~") 16088 +* #end 16089 +## *### 16090 + #elseif (!$removeEllipsis) 16091 + #if($entryDoc.syntax.toIdString() == 'xwiki/1.0') 16092 + #set($macro.result = ~"$\{macro.result} <a href='$\{entryDoc.getExternalURL()}' title='$services.localization.render('blog.code.readpost')'>...</a>~") 16093 + #elseif($entryDoc.syntax.toIdString() == 'xwiki/2.0' || $entryDoc.syntax.toIdString() == 'xwiki/2.1') 16094 + #set($macro.result = ~"$\{macro.result} [[...>>$\{services.rendering.escape($entryDoc.getFullName(), $entryDoc.getSyntax())}||title='$services.localization.render('blog.code.readpost')']]~") 16095 + #end 16096 + #end 16097 + #set ($entryContent = $NULL) 16098 + #setVariable (~"$entryContent~" $macro.result) 16099 +#end 16100 +## 16101 +## 16102 +## 16103 +#** 16104 + * Displays the footer of the entry. 16105 + * 16106 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 16107 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 16108 + *### 16109 +#macro(displayEntryFooter $entryDoc $entryObj) 16110 + (% class=~"entry-footer~" ~%)((( 16111 + #isPublished($entryObj $isPublished) 16112 + (% class='entry-author-label' ~%) 16113 + #if($isPublished) 16114 + \{\{translation key='blog.code.postedby'/}} ## 16115 + #else 16116 + \{\{translation key='blog.code.createdby'/}} ## 16117 + #end 16118 + \{\{html wiki=~"false~" clean=~"false~"}}<span class='author vcard'>#userfn($entryDoc.creator)</span>\{\{/html}} ## 16119 + #getEntryDate($entryDoc $entryObj $entryDate) 16120 + #listCategories($entryObj) #* 16121 + ## Since the publish date and update date are not set at the exact same time, there could be a small difference that 16122 + ## we assume cannot be more than 3 seconds. 16123 + *#(% class=~"separator~" ~%)·(%~%) [[\{\{translation key='blog.code.permalink'/}}>>$services.rendering.escape($entryDoc,'xwiki/2.0')||rel=~"bookmark~"]] ## 16124 + #if ($showcomments) 16125 + (% class=~"separator~" ~%)·(%~%) [[\{\{translation key='blog.code.comments'/}}>>$services.rendering.escape($entryDoc,'xwiki/2.0')||anchor=~"Comments~"]] (% class=~"itemCount~" ~%)($entryDoc.comments.size())(%~%) ## 16126 + #end ## 16127 + #if($entryDoc != $doc) ## 16128 + #displayEntryBlogLocation($entryDoc $entryObj) ## 16129 + #end 16130 + )))## entry-footer 16131 +#end 16132 +## 16133 +## 16134 +#** 16135 + * Display the blog for the entry (if it is not the currently displayed blog) 16136 + * 16137 + * @param entryDoc The xdocument of the blog post. Each post resides in its own document. 16138 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 16139 + *### 16140 +#macro(displayEntryBlogLocation $entryDoc $entryObj) 16141 + #getBlogPostsLocation($entryDoc.getSpace() $blogPostsLocation) ## 16142 + #if(~"$!blogPostsLocation~" != ~"~") ## 16143 + #set($blogSpaceRef = $services.model.resolveSpace($blogPostsLocation)) ## 16144 + #set($blogDocRef = $services.model.createDocumentReference($services.model.getEntityReference('DOCUMENT', 'default').getName(), $blogSpaceRef)) ## 16145 + #if($doc.documentReference != $blogDocRef) ## 16146 + #set($blogDoc = $xwiki.getDocument($blogDocRef)) ## 16147 + #set($blogObj = $blogDoc.getObject('Blog.BlogClass')) ## 16148 + #if($blogObj) (% class=~"blogrefpath~" ~%)((( ## 16149 + $services.localization.render('blog.code.postedin') \{\{html clean=~"false~" wiki=~"false~"}} #hierarchy($blogDocRef, \{'selfIsActive' : true, 'local': true}) \{\{/html}} ## 16150 + )))(%~%)## 16151 + #end 16152 + #end 16153 + #end 16154 +#end 16155 +## 16156 +## 16157 +## 16158 +## 16159 +#** 16160 + * List the categories an entry belongs to. Used in the footer. The categories are instances of <tt>Blog.CategoryClass</tt>. 16161 + * 16162 + * @param entryObj The xobject of the blog post, an instance of the <tt>Blog.BlogPostClass</tt> xclass. 16163 + *### 16164 +#macro(listCategories $entryObj) 16165 + #if($entryObj.getxWikiClass().getName() == $blogPostClassname) 16166 + #set($categories = $entryObj.getProperty('category').value) 16167 + #set($first = true) 16168 + #if($categories.size() > 0) 16169 + #foreach($category in $categories) 16170 + #set($categoryDoc = $!xwiki.getDocument($category)) 16171 + #if(!$categoryDoc.isNew() && $categoryDoc.getObject($\{blogCategoryClassname})) 16172 + #if($foreach.count == 1) 16173 + (% class='separator' ~%)·(%~%) $services.localization.render('blog.code.categories') ## 16174 + #else 16175 + , ## 16176 + #end## 16177 + [[$!\{categoryDoc.getObject($blogCategoryClassname).getValue('name')}>>$\{category}||rel='tag']]## 16178 + #end## 16179 + #end## 16180 + #end 16181 + #end 16182 +#end 16183 +## 16184 +## 16185 +## 16186 +#** 16187 + * Displays blog pagination links (older and newer entries). 16188 + * 16189 + * @param blogDoc the XDocument holding the blog definition object. 16190 + *### 16191 +#macro(displayNavigationLinks $blogDoc) 16192 + (% class=~"clearfloats~" ~%)((())) 16193 + #getBlogDisplayType($blogDoc $displayType) 16194 + #if($displayType == 'weekly') 16195 + (% class=~"pagingLinks~" ~%)((( 16196 + #getRequestedWeek($weekDate) 16197 + $weekDate.addWeeks(-1)## 16198 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.previousweek'/}}>>$doc.documentReference.name||queryString=~"year=$weekDate.weekyear&week=$weekDate.weekOfWeekyear~" class=~"button~"]]**(%~%) 16199 + #sep() 16200 + $weekDate.addWeeks(2)## 2 because we already subtracted 1 above 16201 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.nextweek'/}} »>>$doc.documentReference.name||queryString=~"year=$weekDate.weekyear&week=$weekDate.weekOfWeekyear~" class=~"button~"]]**(%~%) 16202 + ))) 16203 + #elseif($displayType == 'monthly') 16204 + (% class=~"pagingLinks~" ~%)((( 16205 + #getRequestedMonth($monthDate) 16206 + $monthDate.addMonths(-1)## 16207 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.previousmonth'/}}>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"year=$monthDate.year&month=$monthDate.monthOfYear~" class=~"button~"]]**(%~%) 16208 + #sep() 16209 + $monthDate.addMonths(2)## 2 because we already subtracted 1 above 16210 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.nextmonth'/}} »>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"year=$monthDate.year&month=$monthDate.monthOfYear~" class=~"button~"]]**(%~%) 16211 + ))) 16212 + #elseif($displayType == 'all') 16213 + #else 16214 + ## Paginated 16215 + #if(($totalPages > 1)) 16216 + #set($queryString = '') 16217 + #foreach($p in $request.getParameterNames()) 16218 + #if($p != 'page' && $p != 'ipp') 16219 + #foreach($v in $request.getParameterValues($p)) 16220 + #set($queryString = ~"$\{queryString}&$\{escapetool.url($p)}=$\{escapetool.url($v)}~") 16221 + #end 16222 + #end 16223 + #end 16224 + (% class=~"pagingLinks~" ~%)((( 16225 + #if ($currentPageNumber < $totalPages) 16226 + #set($currentPageNumber = $currentPageNumber + 1) 16227 + (% class=~"prevPage~" ~%)**[[« \{\{translation key='blog.code.olderposts'/}}>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"page=$\{currentPageNumber}&ipp=$\{itemsPerPage}$queryString~" class=~"button~"]]**(%~%) 16228 + #set($currentPageNumber = $currentPageNumber - 1) 16229 + #end 16230 + #if ($currentPageNumber > 1) 16231 + #if ($currentPageNumber < $totalPages) 16232 + #sep() 16233 + #end 16234 + #set($currentPageNumber = $currentPageNumber - 1) 16235 + (% class=~"nextPage~" ~%)**[[\{\{translation key='blog.code.newerposts'/}} »>>$services.rendering.escape($doc.documentReference.name,$doc.syntax)||queryString=~"page=$\{currentPageNumber}&ipp=$\{itemsPerPage}$queryString~" class=~"button~"]]**(%~%) 16236 + #set($currentPageNumber = $currentPageNumber + 1) 16237 + #end 16238 + (% class=~"clear~" ~%)(%~%) 16239 + )))## pagingLinks 16240 + #end 16241 + #end 16242 +#end 16243 +## 16244 +## 16245 +## 16246 +#** 16247 + * Displays a message box with ~"publish~" icon. 16248 + * 16249 + * @param message A text message concerning blog article publishing 16250 + *### 16251 +#macro(publishMessageBox $message) 16252 +(% class=~"plainmessage publish-message~" ~%)((($services.icon.render('world') $message))) 16253 +#end 16254 +#** 16255 + * Displays a message box with ~"show/hide~" icon. 16256 + * 16257 + * @param message A text message concerning blog article hiding 16258 + *### 16259 +#macro(hideMessageBox $message) 16260 +(% class=~"plainmessage hide-message~" ~%)((($services.icon.render('unlock') $message))) 16261 +#end 16262 +## 16263 +## 16264 +## 16265 +#** 16266 + * Determine the requested week, for using in a weekly-indexed blog. The relevant request parameters are 16267 + * <tt>year</tt> and <tt>week</tt>. By default, the current week is used. 16268 + * 16269 + * @param monthDate The resulting week, a JODATime MutableDateTime. 16270 + *### 16271 +#macro(getRequestedWeek $weekDate) 16272 + #set ($weekDate = $NULL) 16273 + #setVariable (~"$weekDate~" $xwiki.jodatime.mutableDateTime) 16274 + #if(~"$!\{request.year}~" != '') 16275 + #set ($discard = $weekDate.setYear($numbertool.toNumber($request.year).intValue())) 16276 + #end 16277 + #if(~"$!\{request.week}~" != '') 16278 + #set ($discard = $weekDate.setWeekOfWeekyear($numbertool.toNumber($request.week).intValue())) 16279 + #end 16280 +#end 16281 +## 16282 +## 16283 +## 16284 +#** 16285 + * Determine the requested month, for using in a monthly-indexed blog. The relevant request parameters are 16286 + * <tt>year</tt> and <tt>month</tt>. By default, the current month is used. 16287 + * 16288 + * @param monthDate The resulting month, a JODATime MutableDateTime. 16289 + *### 16290 +#macro(getRequestedMonth $monthDate) 16291 + #set ($monthDate = $NULL) 16292 + #setVariable (~"$monthDate~" $xwiki.jodatime.mutableDateTime) 16293 + #if(~"$!\{request.year}~" != '') 16294 + #set ($discard = $monthDate.setYear($numbertool.toNumber($request.year).intValue())) 16295 + #end 16296 + #if(~"$!\{request.month}~" != '') 16297 + #set ($discard = $monthDate.setMonthOfYear($numbertool.toNumber($request.month).intValue())) 16298 + #end 16299 +#end 16300 +## 16301 +## 16302 +## 16303 +#** 16304 + * Retrieve a blog property (title, display type, etc). 16305 + * 16306 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 16307 + * @param propertyName The name of the property to be retrieved. One of the <tt>Blog.BlogClass</tt>'s properties. 16308 + * @param defaultValue The default value to use in case the blog object does not define one. 16309 + * @param propertyValue The resulting value. 16310 + *### 16311 +#macro(getBlogProperty $blogDoc $propertyName $defaultValue $propertyValue) 16312 + #set($result = ~"$!\{blogDoc.getObject($\{blogClassname}).getProperty($propertyName).value}~") 16313 + #if($result == '') 16314 + #set($result = $defaultValue) 16315 + #end 16316 + #set ($propertyValue = $NULL) 16317 + #setVariable (~"$propertyValue~" $result) 16318 +#end 16319 + 16320 +#** 16321 + * If an error occurs when executing an action, set a specific response status and display an error message. 16322 + * 16323 + * @param status The response status. 16324 + * @param text The user readable error to be displayed. Can be a translation key. 16325 + * @param parameters The parameters to use when decoding the translation key. 16326 + *### 16327 +#macro(blog__actionResponseError $status $text $parameters) 16328 + $response.setStatus($status) 16329 + #if($request.ajax) 16330 + $services.localization.render($text, $!parameters) 16331 + #else 16332 + \{\{error}}$services.localization.render($text, $!parameters)\{\{/error}} 16333 + #end 16334 +#end 16335 +## 16336 +## 16337 +## 16338 +#** 16339 + * Check if a blog is the Default blog (The one in the 'Blog' space). 16340 + * 16341 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object. 16342 + * @param isDefault The resulting boolean. 16343 + *### 16344 +#macro(isDefaultBlog $blogDoc $isDefault) 16345 + #set ($result = false) 16346 + #if ($blogDoc.space == 'Blog') 16347 + #set ($result = true) 16348 + #end 16349 + #setVariable(~"$isDefault~" $result) 16350 +#end 16351 +## 16352 +## 16353 +## 16354 +#** 16355 + * Retrieve the blog posts location (space). 16356 + * 16357 + * @param blogSpace The blog space. It should contain a document with a <tt>Blog.BlogClass</tt> object 16358 + * @param postsLocation The resulting location. 16359 + *### 16360 +#macro(getBlogPostsLocation $blogSpace $postsLocation) 16361 + #getBlogDocument($blogSpace $blogDoc) 16362 + #getBlogProperty($blogDoc 'postsLocation' $blogDoc.space $result) 16363 + #set ($postsLocation = $NULL) 16364 + #setVariable (~"$postsLocation~" $result) 16365 +#end 16366 +## 16367 +## 16368 +## 16369 +#** 16370 + * Retrieve the blog categories location (space). 16371 + * 16372 + * @param blogSpace The blog space. It should contain a document with a <tt>Blog.BlogClass</tt> object 16373 + * @param categoriesLocation The resulting location. 16374 + *### 16375 +#macro(getBlogCategoriesLocation $blogSpace $categoriesLocation) 16376 + #getBlogDocument($blogSpace $blogDoc) 16377 + #getBlogProperty($blogDoc 'categoriesLocation' 'Blog' $result) 16378 + #set ($postsLocation = $NULL) 16379 + #setVariable (~"$categoriesLocation~" $result) 16380 +#end 16381 +###** 16382 + * Return the Query for selecting the blog posts based on the context where the posts are displayed. 16383 + * for example there is 4 different panel contexts: 16384 + * aBlog.aPost or aBlog.WebHome 16385 + * aCategorySpace.aCategory 16386 + * aCategorySpace.WebHome 16387 + * Blog.aPost or Blog.WebHome 16388 + * 16389 + * @param query The query for selecting blog entries. 16390 + * @param queryParams The parameters to bind with the generated query. 16391 + * @param targetDoc The document in which the articles will be displayed. 16392 + *### 16393 +#macro(getAllBlogPostsQueryBasedOnDisplayContext $targetDoc $query $queryParams) 16394 + #set ($query = $NULL) 16395 + #set ($queryParams = $NULL) 16396 + #getCategoryAllBlogPostsQuery($resultQuery) 16397 + #set ($resultQueryParams = \{}) 16398 + #set($sanitizedSpace = $targetDoc.space.replaceAll('([%_!])', '!$1').concat('.%')) 16399 + #if ($targetDoc.space == $defaultBlogSpace && !$targetDoc.getObject($blogCategoryClassname)) 16400 + ## Get all posts that are in the default blog space('Blog') or in category 'Blog.%' 16401 + #set ($discard = $resultQueryParams.put('space', $targetDoc.space)) 16402 + #set ($discard = $resultQueryParams.put('catList', $sanitizedSpace)) 16403 + #set($resultQuery = ~"$\{resultQuery} and (doc.space = :space or catList like :catList escape '!')~") 16404 + #elseif($targetDoc.getObject($blogCategoryClassname)) 16405 + ## Get all posts that are in a category aCategorySpace.aCategory 16406 + #set ($discard = $resultQueryParams.put('catList', $targetDoc.fullName)) 16407 + #set($resultQuery = ~"$\{resultQuery} and (catList=:catList)~") 16408 + #elseif($targetDoc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 16409 + ## Get all posts that are in a category aCategorySpace.% 16410 + #set ($discard = $resultQueryParams.put('catList', $sanitizedSpace)) 16411 + ## Exclude incategorized posts 16412 + #set ($excludedCategory = ~"$\{targetDoc.space}.WebHome~") 16413 + #if ($targetDoc.space == $defaultBlogSpace) 16414 + #set ($excludedCategory = ~"Blog.Categories.WebHome~") 16415 + #end 16416 + #set ($discard = $resultQueryParams.put('excludedCategory', $excludedCategory)) 16417 + #set($resultQuery = ~"$\{resultQuery} and (catList<>:excludedCategory) and (catList like :catList escape '!')~") 16418 + #else 16419 + ## Get all posts in blog space aBlog 16420 + #getAllBlogPostsQuery($resultQuery) 16421 + #getBlogPostsLocation($targetDoc.space $postsLocation) 16422 + #set ($discard = $resultQueryParams.put('space', $postsLocation)) 16423 + #set($resultQuery = ~"$\{resultQuery} and doc.space = :space~") 16424 + #end 16425 + #setVariable(~"$query~" $resultQuery) 16426 + #setVariable(~"$queryParams~" $resultQueryParams) 16427 +#end 16428 +## 16429 +## 16430 +## 16431 +###** 16432 + * Display blog posts based on the context where the posts are displayed. 16433 + * for example there is 4 different panel contexts: 16434 + * aBlog.aPost or aBlog.WebHome 16435 + * aCategorySpace.aCategory 16436 + * aCategorySpace.WebHome 16437 + * Blog.aPost or Blog.WebHome 16438 + * 16439 + * @param targetDoc The document in which the articles will be displayed. 16440 + * @param postsVisiblity The visibility of the posts to display: recent, unpublished, ... 16441 + * @param layout Layout of the the posts to display 16442 + * @param layoutParams additional layout parameters, it is a string like 'param1=val1|...|paramN=valueN' 16443 + * @param limit the number of posts to display 16444 + *### 16445 +#macro(displayBlogPostsBasedOnContext $targetDoc $postsVisiblity $layout $layoutParams $limit) 16446 + #if ($postLayout == 'full') 16447 + #set ($macro.paginated = 'yes') 16448 + #end 16449 + #if ($targetDoc.space == $defaultBlogSpace && !$targetDoc.getObject($blogCategoryClassname)) 16450 + ## Display all posts that are in the default blog space('Blog') or in category 'Blog.%' 16451 + #getBlogPostsLayout($xwiki.getDocument(~"$\{defaultBlogSpace}.WebHome~") $postsLayout) 16452 + #if (~"$!layout~" == '') 16453 + #set ($layout = $postsLayout) 16454 + #end 16455 + #if ($postsVisiblity == 'recent') 16456 + \{\{blogpostlist blog=~"$\{defaultBlogSpace.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')}.WebHome~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 16457 + #elseif($postsVisiblity == 'unpublished') 16458 + \{\{blogpostlist blog=~"$\{defaultBlogSpace.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')}.WebHome~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 16459 + #end 16460 + #elseif($targetDoc.getObject($blogCategoryClassname)) 16461 + ## Display all posts that are in a category aCategorySpace.aCategory 16462 + #getBlogDocumentForCategoriesSpace($targetDoc.space $blogDoc) 16463 + #getBlogPostsLayout($blogDoc $postsLayout) 16464 + #if (~"$!layout~" == '') 16465 + #set ($layout = $postsLayout) 16466 + #end 16467 + #if ($postsVisiblity == 'recent') 16468 + \{\{blogpostlist category=~"$targetDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 16469 + #elseif($postsVisiblity == 'unpublished') 16470 + \{\{blogpostlist category=~"$targetDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 16471 + #end 16472 + #elseif($targetDoc.getObject('XWiki.DocumentSheetBinding').sheet == 'Blog.CategoriesSheet') 16473 + ## Display all posts that are in a category aCategorySpace.% 16474 + #getBlogDocumentForCategoriesSpace($targetDoc.space $blogDoc) 16475 + #getBlogPostsLayout($blogDoc $postsLayout) 16476 + #if (~"$!layout~" == '') 16477 + #set ($layout = $postsLayout) 16478 + #end 16479 + #if ($postsVisiblity == 'recent') 16480 + \{\{blogpostlist category=~"$targetDoc.space.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 16481 + #elseif($postsVisiblity == 'unpublished') 16482 + \{\{blogpostlist category=~"$targetDoc.space.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 16483 + #end 16484 + #else 16485 + ## Display all posts in blog space aBlog 16486 + #getBlogDocument($targetDoc.space $blogDoc) 16487 + #getBlogPostsLayout($blogDoc $postsLayout) 16488 + #if (~"$!layout~" == '') 16489 + #set ($layout = $postsLayout) 16490 + #end 16491 + #if ($postsVisiblity == 'recent') 16492 + \{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='yes' hidden='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" limit=~"$limit~" /}} 16493 + #elseif($postsVisiblity == 'unpublished') 16494 + \{\{blogpostlist blog=~"$blogDoc.fullName.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" published='no' layout=~"$!layout.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" layoutParams=~"$!layoutParams.replaceAll('~~', '~~~~').replaceAll('~"', '~~~"')~" paginated=~"$!macro.paginated~" limit=~"$limit~" /}} 16495 + #end 16496 + #end 16497 +#end 16498 +## 16499 +## 16500 +## 16501 +#** 16502 + * Bind parameters to a query object. 16503 + * 16504 + * @param queryObj The query object (org.xwiki.query.internal.ScriptQuery object) 16505 + * @param queryParams the query parameters. 16506 + *### 16507 +#macro(bindQueryParameters $queryObj $queryParams) 16508 + #set ($output = $queryObj) 16509 + #foreach( $key in $queryParams.keySet() ) 16510 + #set($output = $queryObj.bindValue($key, $queryParams.get($key))) 16511 + #end 16512 + #setVariable(~"$queryObj~" $output) 16513 +#end 16514 +## 16515 +## 16516 +## 16517 +#** 16518 + * Determines the blogposts layout: Default layout (Calendar), Image thumbnail 16519 + * 16520 + * @param blogDoc The blog document. It should contain a <tt>Blog.BlogClass</tt> object with the <tt>postsLayout</tt> 16521 + * property set. 16522 + * @param postsLayout The resulting string. If the blog object does not define anything, it is considered full (default). 16523 + *### 16524 +#macro(getBlogPostsLayout $blogDoc $postsLayout) 16525 + #getBlogProperty($blogDoc 'postsLayout' 'full' $res) 16526 + #set ($postsLayout = $NULL) 16527 + #setVariable (~"$postsLayout~" $res) 16528 +#end 16529 +## 16530 +## 16531 +## 16532 +#** 16533 + * Retrieve the blog document of a categories space, which is the blog descriptor that have its categoriesLocation property set to the categories space 16534 + * 16535 + * @param categoriesSpace A <tt>String</tt>, the name of the space. 16536 + * @param blogDoc The resulting XDocument. 16537 + *### 16538 +#macro(getBlogDocumentForCategoriesSpace $categoriesSpace $blogDoc) 16539 + #set($macro.blogDocs = $services.query.hql(~", BaseObject obj, StringProperty as categoriesLocationProp where obj.name = doc.fullName and obj.className = '$blogClassname' and obj.id=categoriesLocationProp.id.id and categoriesLocationProp.id.name='categoriesLocation' and categoriesLocationProp.value=:categoriesLocation order by doc.creationDate~").setLimit(1).setOffset(0).bindValue('categoriesLocation', $categoriesSpace).execute()) 16540 + #if($macro.blogDocs.size() > 0 && $categoriesSpace != $defaultBlogSpace) 16541 + #set($macro.result = $xwiki.getDocument($macro.blogDocs.get(0))) 16542 + #else 16543 + ## Fallback to Blog.WebHome, the default blog 16544 + #set($macro.result = $xwiki.getDocument('Blog.WebHome')) 16545 + #end 16546 + #set ($blogDoc = $NULL) 16547 + #setVariable (~"$blogDoc~" $macro.result) 16548 +#end" %) 16549 +((( 16550 +(% class="macro-placeholder hidden" %) 16551 +((( 16552 +macro:velocity 16553 +))) 16554 +))) 16555 +))) 16556 + 16557 +(% class="macro" data-macro="startmacro:include|-|reference=~"Blog.BlogPostLayoutMacros~"" %) 16558 +((( 16559 +(% class="macro-placeholder hidden" %) 16560 +((( 16561 +macro:include 16562 +))) 16563 + 16564 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-|#** 16565 + * Extract the layout parameters from a string. 16566 + * 16567 + * @param layoutParamsString The string representation of the layout parameters. 16568 + * It should contain a String following this format ~"paramName1=Value1|paramName2=Value2|...|paramNameK=ValueK~" 16569 + * @param layoutsParameters The resulting layout parameters Map. 16570 + *### 16571 +#macro(extractLayoutParametersFromString $layoutParamsString $layoutsParameters) 16572 + #set ($layoutsParameters = $NULL) 16573 + #set ($macro.layoutParams = \{}) 16574 + #if (~"$!layoutParamsString~" != '') 16575 + #set ($macro.paramsArr = $layoutParamsString.split('\\|')) 16576 + #foreach ($item in $macro.paramsArr) 16577 + #set ($itemSplit = $item.split('=')) 16578 + #if ($itemSplit.size() == 2) 16579 + #set ($discard = $macro.layoutParams.put($itemSplit[0].trim(), $itemSplit[1].trim())) 16580 + #end 16581 + #end 16582 + #end 16583 + #setVariable(~"$layoutsParameters~" $macro.layoutParams) 16584 +#end" %) 16585 +((( 16586 +(% class="macro-placeholder hidden" %) 16587 +((( 16588 +macro:velocity 16589 +))) 16590 +))) 16591 +))) 16592 + 16593 +(% class="macro" data-macro="startmacro:velocity|-|output=~"false~"|-| #macro(displaySmallPost $pDoc $pObj) 16594 + #initLayoutVars($pDoc $pObj) 16595 + <div class=~"col-xs-12 col-sm-6 next_blog~"> 16596 + <a href=~"$pDoc.uRL~" class=~"thumbnail~"> 16597 + <div class=~"row~"> 16598 + <div class=~"col-xs-4~"> 16599 + #if ($imageAtt) 16600 + <img src=~"$escapetool.xml($pDoc.getAttachmentURL($pObj.getValue('image'),'download',$imgQs))~" /> 16601 + #end 16602 + </div> 16603 + <div class=~"col-xs-8 art_det~"> 16604 + <p class=~"text-left~"> 16605 + #if($displayTitle)$!postTitle #end 16606 + <br/><span class=~"date_info~"> $!dateStr </span> 16607 + </p> 16608 + #displayPostDetails($pDoc) 16609 + </div> 16610 + </div> 16611 + </a> 16612 + </div> 16613 + #end 16614 + ## 16615 + ## 16616 + ## 16617 + #macro(displayPinnedPost $pDoc $pObj) 16618 + #initLayoutVars($pDoc $pObj) 16619 + <div class=~"col-xs-12 col-sm-4 col-md-4 blog_pin~"> 16620 + <a href=~"$pDoc.uRL~" class=~"all-post-link~"> 16621 + <div class=~"thumbnail~"> 16622 + #if ($displayTitle && $displayTitleFirstOnPinnedPosts) 16623 + <h5 class=~"title-top~">$!postTitle <br/> <span class=~"date_info~"> $!dateStr </span> </h5> 16624 + #end 16625 + #if ($imageAtt) 16626 + <img src=~"$escapetool.xml($pDoc.getAttachmentURL($pObj.getValue('image'),'download',$imgQs))~" /> 16627 + #end 16628 + <div class=~"caption~"> 16629 + #if ($displayTitle && !$displayTitleFirstOnPinnedPosts) 16630 + <h5 class=~"text-left~">$!postTitle <br/> <span class=~"date_info~"> $!dateStr </span> </h5> 16631 + #end 16632 + #if ($displaySummaryOnPinnedPosts) 16633 + <div class=~"text-left post-summary~"> 16634 + #set ($postContent = $pObj.getProperty('extract').value) 16635 + $!pDoc.getRenderedContent($postContent, $pDoc.syntax.toIdString()) 16636 + </div> 16637 + #end 16638 + #displayPostDetails($pDoc) 16639 + </div> 16640 + </div> 16641 + </a> 16642 + </div> 16643 + #end 16644 + ## 16645 + ## 16646 + ## 16647 + #macro(formatPostDate $pDoc $pObj) 16648 + #set ($formattedDate = $NULL) 16649 + #set($dateStr = $datetool.format('medium', $postDate, $xcontext.locale)) 16650 + #if (~"$!dateStr~" != '') 16651 + #set ($dateArr = $dateStr.split(' ')) 16652 + #if ($dateArr.size() > 3) 16653 + #set ($dateStr = ~"~") 16654 + #foreach($s in $dateArr.subList(0, 3)) 16655 + #set ($dateStr = ~"$\{dateStr}$\{s} ~") 16656 + #end 16657 + #end 16658 + #end 16659 + #setVariable(~"$formattedDate~" $dateStr) 16660 + #end 16661 + ## 16662 + ## 16663 + ## 16664 + #macro(displayPostDetails $pDoc) 16665 + <div class=~"row post-details~"> 16666 + <div class=~"col-xs-8 detail~"> 16667 + #if ($services.like.displayButton($pDoc.documentReference)) 16668 + #set ($optLikeRecord = $services.like.getLikes($pDoc.documentReference)) 16669 + ## Retrieve the likes number in XWiki 12.9+ 16670 + #set ($likeNumber = $optLikeRecord.get()) 16671 + #if (!$stringtool.isNumeric($likeNumber.toString())) 16672 + ## Retrieve the likes number in XWiki versions greater than 12.7 and lower than 12.9. 16673 + #set ($likeNumber = $optLikeRecord.get().likeNumber) 16674 + #end 16675 + #if ($stringtool.isNumeric($likeNumber.toString())) 16676 + <div class=~"post-likes btn btn-default disabled badge~" 16677 + title=~"$escapetool.xml($services.localization.render('like.button.title', [$likeNumber]))~"> 16678 + $services.icon.renderHTML('heart') <span class=~"like-number~">$likeNumber</span> 16679 + </div> 16680 + #end 16681 + #elseif ($services.ratings) 16682 + #set ($ratingsConfigDoc = $services.ratings.getConfigurationDocument($pDoc.documentReference)) 16683 + #if ($ratingsConfigDoc.getValue('displayRatings') == 1) 16684 + <ul class=~"pull-left note list-inline~"> 16685 + #foreach ($x in [1..5]) 16686 + #set ($cls = ~"~") 16687 + #if ($x > $averageVote) 16688 + #set ($cls = ~"-o~") 16689 + #end 16690 + <li><span class=~"fa fa-star$\{cls}~"></span></li> 16691 + #end 16692 + </ul> 16693 + #end 16694 + #end 16695 + </div> 16696 + #if ($showPostComments) 16697 + <div class=~"col-xs-4 com_det~"> 16698 + <p class=~"text-right~">$services.icon.renderHTML('comments') ($!nbComments)</p> 16699 + </div> 16700 + #end 16701 + </div> 16702 + #end 16703 + ## 16704 + ## 16705 + ## 16706 + #macro(initLayoutVars $pDoc $pObj) 16707 + #set ($postTitle = $pDoc.display('title', 'view', $pObj)) 16708 + #set ($imageAtt = $pDoc.getAttachment($pObj.getValue('image'))) 16709 + #isPublished($pObj $isPublished) 16710 + #isHidden($pObj $isHidden) 16711 + #getEntryDate($pDoc $pObj $postDate) 16712 + #formatPostDate($postDate $dateStr) 16713 + #set ($averageVote = $mathtool.round($services.ratings.getAverageRating($pDoc.documentReference).getAverageVote())) 16714 + #set ($nbComments = $pDoc.getComments().size()) 16715 + #set ($showPostComments = true) 16716 + #if ($xwiki.getSpacePreferenceFor('showcomments', $pDoc.documentReference.lastSpaceReference, '1') == '0') 16717 + #set ($showPostComments = false) 16718 + #end 16719 + #end 16720 + ## 16721 + ## 16722 + ## 16723 + #macro(displayEditPinnedPostsButton) 16724 + #if (~"$!macroAdditionalParams.get('postIndex')~" == '0' && $services.security.authorization.hasAccess('edit', $pinnedPostsSourceDoc.fullName)) 16725 + #set ($discard = $xwiki.ssx.use('Blog.PinnedPostsSheet')) 16726 + #set ($discard = $xwiki.jsx.use('Blog.PinnedPostsSheet')) 16727 + #set ($editPinnedPostsURL = $xwiki.getURL($pinnedPostsSourceDoc.fullName, 'get', 'sheet=Blog.PinnedPostsSheet')) 16728 + #set ($newPinnedPostsObjectURL = $xwiki.getURL($pinnedPostsSourceDoc.fullName, 'objectadd', ~"classname=Blog.PinnedPostsClass&form_token=$!\{services.csrf.getToken()}~")) 16729 + <div class=~"edit-pinned-posts-container~"> 16730 + <div class=~"col-xs-12 col-sm-12 col-md-12 text-right edit-pinned-posts~"> 16731 + <a href=~"javascript:;~" class=~"editPinnedPosts~"> 16732 + <span class=~"glyphicon glyphicon-edit~" aria-hidden=~"true~"></span> $services.localization.render('blog.post.layout.cards.pinnedposts.edit') 16733 + </a> 16734 + <input type=~"hidden~" id=~"editPinnedPostsURL~" value=~"$escapetool.xml($editPinnedPostsURL)~"/> 16735 + <input type=~"hidden~" id=~"newPinnedPostsObjectURL~" value=~"$escapetool.xml($newPinnedPostsObjectURL)~"/> 16736 + </div> 16737 + </div> 16738 + #if (~"$!pinnedPostsObj~" == '') 16739 + <input type=~"hidden~" name=~"createPinnedPostsObject~" value=~"1~"/> 16740 + #end 16741 + #end 16742 + #end" %) 16743 +((( 16744 +(% class="macro-placeholder hidden" %) 16745 +((( 16746 +macro:velocity 16747 +))) 16748 +))) 16749 + 16750 +(% class="macro" data-macro="startmacro:velocity|-||-| #if ($services.security.authorization.hasAccess('view', $xcontext.macro.params.reference)) 16751 + #set ($postDoc = $xwiki.getDocument($xcontext.macro.params.reference)) 16752 + #getEntryObject($postDoc $postObj) 16753 + #if (~"$!postObj~" != '') 16754 + #extractLayoutParametersFromString($xcontext.macro.params.params $macroAdditionalParams) 16755 + ## 16756 + #set ($displayTitle = true) 16757 + #if (~"$!macroAdditionalParams.get('displayTitle')~" == 'false') 16758 + #set ($displayTitle = false) 16759 + #end 16760 + ## 16761 + #set ($displayTitleFirstOnPinnedPosts = false) 16762 + #if (~"$!macroAdditionalParams.get('displayTitleFirstOnPinnedPosts')~" == 'true') 16763 + #set ($displayTitleFirstOnPinnedPosts = true) 16764 + #end 16765 + ## 16766 + #set ($displaySummaryOnPinnedPosts = true) 16767 + #if (~"$!macroAdditionalParams.get('displaySummaryOnPinnedPosts')~" == 'false') 16768 + #set ($displaySummaryOnPinnedPosts = false) 16769 + #end 16770 + ## 16771 + #set ($postIndex = $numbertool.toNumber($macroAdditionalParams.get('postIndex')).intValue()) 16772 + #if ($postIndex == 0) 16773 + #set ($stopBlogPostsDisplay = false) 16774 + #end 16775 + #set ($listResultsCount = $numbertool.toNumber($macroAdditionalParams.get('list-results-count')).intValue()) 16776 + #set($discard = $xwiki.ssx.use('Blog.BlogPostLayoutCards')) 16777 + #set($scaleWidth = 600) 16778 + #set($imgQs=~"width=$scaleWidth~") 16779 + ## Display pinned posts 16780 + ## Get the list of pinned posts 16781 + #set ($pinnedPosts = []) 16782 + #set ($pinnedPostsSourceDoc = $NULL) 16783 + #if (~"$!macroAdditionalParams.get('forceDisplayPinnedPosts')~" == 'true' && ~"$!macroAdditionalParams.get('list-blog')~" != '') 16784 + #set ($pinnedPostsSourceDoc = $xwiki.getDocument($macroAdditionalParams.get('list-blog'))) 16785 + ## Note that the 'list-blog' parameter contains the value of the 'blog' parameter of the blogpostlist macro passed to the current layout macro 16786 + #end 16787 + #if (~"$!pinnedPostsSourceDoc~" != '') 16788 + #set ($pinnedPostsObj = $pinnedPostsSourceDoc.getObject('Blog.PinnedPostsClass')) 16789 + #if (~"$!pinnedPostsObj~" != '') 16790 + #set ($orderedPinnedPostsJSON = $pinnedPostsObj.getProperty('orderedPosts').value) 16791 + #if (~"$!orderedPinnedPostsJSON~" != '') 16792 + #set ($pinnedPosts = $jsontool.parse($orderedPinnedPostsJSON)) 16793 + #else 16794 + #set ($pinnedPosts = $pinnedPostsObj.getProperty('posts').value) 16795 + #end 16796 + #end 16797 + #if ($postIndex == 0 && $pinnedPosts.size() > 0) 16798 + \{\{html clean=~"false~"}} 16799 + #set ($x = 0) 16800 + #set ($showPinnedPostsButton = true) 16801 + #foreach ($pinnedPost in $pinnedPosts) 16802 + #if ($x == 0) 16803 + <div class=~"row flex-container~"> 16804 + #if ($showPinnedPostsButton) 16805 + #displayEditPinnedPostsButton() 16806 + #set($showPinnedPostsButton = false) 16807 + #end 16808 + #end 16809 + #set ($pinnedPostDoc = $xwiki.getDocument($pinnedPost)) 16810 + #getEntryObject($pinnedPostDoc $pinnedPostObj) 16811 + #if (~"$!pinnedPostObj~" != '') 16812 + #displayPinnedPost($pinnedPostDoc $pinnedPostObj) 16813 + #end 16814 + #set ($x = $mathtool.add($x, 1)) 16815 + #if ($x == 3) 16816 + #set ($x = 0) 16817 + </div> 16818 + #end 16819 + #end 16820 + #if ($mathtool.mod($x, 3) != 0) 16821 + </div> 16822 + #end 16823 + \{\{/html}} 16824 + 16825 + ## If the first post is a pinned post : this means that all the posts are pinned 16826 + ## In this case, avoid displaying the posts again after the pinned posts section. 16827 + #if ($pinnedPosts.contains($xcontext.macro.params.reference)) 16828 + #set ($stopBlogPostsDisplay = true) 16829 + #end 16830 + #elseif($postIndex == 0 && $pinnedPosts.size() == 0) 16831 + 16832 + \{\{html}} 16833 + <div class=~"row no-pinnded-posts~"> 16834 + #displayEditPinnedPostsButton() 16835 + </div> 16836 + \{\{/html}} 16837 + 16838 + #end 16839 + #end 16840 + #if (!$stopBlogPostsDisplay) 16841 + \{\{html clean=~"false~"}} 16842 + #if (~"$!nbDisplayedPosts~" == '' || $postIndex == 0) 16843 + #set ($nbDisplayedPosts = 0) 16844 + #end 16845 + #if($mathtool.mod($nbDisplayedPosts, 2) == 0) 16846 + #if ($nbDisplayedPosts != 0) 16847 + </div> 16848 + #set ($lastHtmlTag = 'c') 16849 + #end 16850 + #if ($nbDisplayedPosts == $mathtool.sub($listResultsCount, 1)) 16851 + <div class=~"row~"> 16852 + #set ($lastHtmlTag = 'o') 16853 + #set ($lastRowClass = ~"~") 16854 + #else 16855 + <div class=~"row flex-container~"> 16856 + #set ($lastHtmlTag = 'o') 16857 + #set ($lastRowClass = ~"flex~") 16858 + #end 16859 + #end 16860 + #displaySmallPost($postDoc $postObj) 16861 + #if ($nbDisplayedPosts == $mathtool.sub($listResultsCount, 1)) 16862 + </div> 16863 + #set ($lastHtmlTag = 'c') 16864 + #end 16865 + #set ($nbDisplayedPosts = $mathtool.add($nbDisplayedPosts, 1)) 16866 + #if ($postIndex == $mathtool.sub($listResultsCount, 1)) 16867 + #if (~"$!lastHtmlTag~" == 'o') 16868 + #if ($mathtool.mod($nbDisplayedPosts, 2) == 1 && $lastRowClass == ~"flex~") 16869 + <div class=~"col-xs-12 col-sm-6~"></div> 16870 + #end 16871 + </div> 16872 + #end 16873 + #end 16874 + \{\{/html}} 16875 + #end 16876 + #else 16877 + \{\{error}}$services.localization.render('blog.blogpostlayout.notpost', [$xcontext.macro.params.reference])\{\{/error}} 16878 + #end 16879 + #else 16880 + \{\{error}}$services.localization.render('blog.blogpostlayout.post_view_not_allowed', [$xcontext.macro.params.reference])\{\{/error}} 16881 + #end" %) 16882 +((( 16883 +(% class="macro-placeholder hidden" %) 16884 +((( 16885 +macro:velocity 16886 +))) 16887 + 16888 +(% class="macro" data-macro="startmacro:html|-|clean=~"false~"|-|<div class=~"col-xs-12 col-sm-6 next_blog~"> 16889 +<a href=~"/Nieuws/Aankondiging%20update%20Syntess%206.9.0.0166~" class=~"thumbnail~"> 16890 +<div class=~"row~"> 16891 +<div class=~"col-xs-4~"> 16892 +<img src=~"/download/Nieuws/Aankondiging%20update%20Syntess%206.9.0.0166/giphy9.gif?width=600&rev=1.1~" /> 16893 +</div> 16894 +<div class=~"col-xs-8 art_det~"> 16895 +<p class=~"text-left~"> 16896 +Aankondiging update Syntess 6.9.0.0166 <br/><span class=~"date_info~"> Nov 21, 2023, </span> 16897 +</p> 16898 +<div class=~"row post-details~"> 16899 +<div class=~"col-xs-8 detail~"> 16900 +<div class=~"post-likes btn btn-default disabled badge~" 16901 +title=~"Number of likes on this page: 0~"> 16902 +<span class=~"fa fa-heart~"></span> <span class=~"like-number~">0</span> 16903 +</div> 16904 +</div> 16905 +<div class=~"col-xs-4 com_det~"> 16906 +<p class=~"text-right~"><span class=~"fa fa-comments~"></span> (0)</p> 16907 +</div> 16908 +</div> 16909 +</div> 16910 +</div> 16911 +</a> 16912 +</div> 16913 +</div>" %) 16914 +((( 16915 +(% class="macro-placeholder hidden" %) 16916 +((( 16917 +macro:html 16918 +))) 16919 + 16920 +(% class="col-xs-12 col-sm-6 next_blog" %) 16921 +((( 16922 +(% class="row" %) 16923 +((( 16924 +(% class="col-xs-4" %) 16925 +((( 16926 +[[~[~[image:/download/Nieuws/Aankondiging%20update%20Syntess%206.9.0.0166/giphy9.gif?width=600&rev=1.1~]~]>>path:/Nieuws/Aankondiging%20update%20Syntess%206.9.0.0166||class="thumbnail"]] 16927 +))) 16928 + 16929 +(% class="col-xs-8 art_det" %) 16930 +((( 16931 +(% class="text-left" %) 16932 +[[Aankondiging update Syntess 6.9.0.0166 16933 +(% class="date_info" %) Nov 21, 2023,>>path:/Nieuws/Aankondiging%20update%20Syntess%206.9.0.0166||class="thumbnail"]] 16934 + 16935 +(% class="row post-details" %) 16936 +((( 16937 +(% class="col-xs-8 detail" %) 16938 +((( 16939 +(% class="post-likes btn btn-default disabled badge" title="Number of likes on this page: 0" %) 16940 +((( 16941 +[[(% class="like-number" %)0>>path:/Nieuws/Aankondiging%20update%20Syntess%206.9.0.0166||class="thumbnail"]] 16942 +))) 16943 +))) 16944 + 16945 +(% class="col-xs-4 com_det" %) 16946 +((( 16947 +(% class="text-right" %) 16948 +[[(0)>>path:/Nieuws/Aankondiging%20update%20Syntess%206.9.0.0166||class="thumbnail"]] 16949 +))) 16950 +))) 16951 +))) 16952 +))) 16953 +))) 16954 +))) 16955 +))) 16956 +))) 16957 +))) 16958 +))) 16959 +))) 16960 +))) 16961 +))) 16962 +))) 16963 +))) 16964 + 16965 +(% style="background:url(~"https://kennis.wardenburg.nl/webjars/wiki%3Axwiki/xwiki-platform-ckeditor-webjar/15.8/plugins/widget/images/handle.png~") rgba(220, 220, 220, 0.5); left:0px; top:-15px" %)[[image:data:image/gif;base64,R0lGODlhAQABAPABAP///wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==||height="15" role="presentation" title="Click and drag to move" width="15"]] 16966 +))) 16967 + 16968 + 56 56 57 57 ))) 58 58 59 -{{velocity}} 60 -#if (!$hasEdit) 16972 +(% aria-label="macro:velocity widget" contenteditable="false" role="region" tabindex="-1" %) 16973 +((( 16974 +(% class="macro" data-macro="startmacro:velocity|-||-|#if (!$hasEdit) 61 61 ## Verstop tabs 62 62 #set($displayDocExtra = false) 63 63 ## En de balken onderin 64 64 #set ($displayShortcuts = false) 65 -#end 66 -{{/velocity}} 16979 +#end" data-widget="xwiki-macro" data-xwiki-dom-updated="true" %) 16980 +((( 16981 +(% class="macro-placeholder" %) 16982 +((( 16983 +macro:velocity 16984 +))) 16985 +))) 16986 + 16987 +(% style="background-color:rgba(220,220,220,0.5)" %)[[image:data:image/gif;base64,R0lGODlhAQABAPABAP///wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==||height="15" role="presentation" title="Click and drag to move" width="15"]] 16988 +))) 
 
- cheer.gif
-   - Author
-   ... ... @@ -1,1 +1,0 @@ 1 -XWiki.Jesse 
- Size
-   ... ... @@ -1,1 +1,0 @@ 1 -3.0 KB 
- Content
 
- logo.png
-   - Author
-   ... ... @@ -1,1 +1,0 @@ 1 -XWiki.Jesse 
- Size
-   ... ... @@ -1,1 +1,0 @@ 1 -423.5 KB 
- Content
 
- aanhetwerk.gif
-   - Author
-   ... ... @@ -1,0 +1,1 @@ 1 +XWiki.Jesse 
- Size
-   ... ... @@ -1,0 +1,1 @@ 1 +4.0 MB 
- Content
 
- favicon.ico
-   - Author
-   ... ... @@ -1,0 +1,1 @@ 1 +XWiki.Jesse 
- Size
-   ... ... @@ -1,0 +1,1 @@ 1 +14.7 KB 
- Content
 
- XWiki.StyleSheetExtension[0]
-   - Code
-   ... ... @@ -3,3 +3,106 @@ 3 3 height: 169px; 4 4 width: 100%; 5 5 } 6 + 7 + /* Styling for the button */ 8 + .login-btn { 9 + display: inline-block; 10 + padding: 24px 48px; /* Increased padding for a slightly bigger button */ 11 + background-color: #D9D9D9; /* Grijs (formulier - header) */ 12 + color: #000; /* Text color */ 13 + border: none; 14 + border-radius: 5px; 15 + font-size: 27px; /* Increased font size for a slightly larger button */ 16 + cursor: pointer; 17 + position: relative; 18 + overflow: hidden; 19 + transition: all 0.3s ease; 20 + } 21 + 22 + /* Styling for the button hover effect */ 23 + .login-btn:hover { 24 + background-color: #F2F2F2; /* Lighter shade of grey on hover */ 25 + } 26 + 27 + /* Styling for the button before animation */ 28 + .login-btn:before { 29 + content: ""; 30 + position: absolute; 31 + top: 50%; 32 + left: 50%; 33 + width: 300%; 34 + height: 300%; 35 + background-color: #878787; 36 + transition: all 0.3s ease; 37 + border-radius: 50%; 38 + z-index: 0; 39 + transform: translate(-50%, -50%); 40 + } 41 + 42 + /* Animation when button is clicked */ 43 + .login-btn.clicked:before { 44 + width: 50%%; 45 + height: 50%; 46 + opacity: 0; 47 + } 48 + 49 + /* Styling for the button text */ 50 + .login-btn span { 51 + position: relative; 52 + z-index: 1; 53 + } 54 + 55 +/* Styling for the smaller button */ 56 +.small-login-btn { 57 + display: inline-block; 58 + padding: 12px 24px; /* Reduced padding for a smaller button */ 59 + background-color: #D9D9D9; /* Grijs (formulier - header) */ 60 + color: #000; /* Text color */ 61 + border: none; 62 + border-radius: 5px; 63 + font-size: 18px; /* Reduced font size for a smaller button */ 64 + cursor: pointer; 65 + position: relative; 66 + overflow: hidden; 67 + transition: all 0.3s ease; 68 +} 69 + 70 +/* Styling for the smaller button hover effect */ 71 +.small-login-btn:hover { 72 + background-color: #F2F2F2; /* Lighter shade of grey on hover */ 73 +} 74 + 75 +/* Styling for the smaller button before animation */ 76 +.small-login-btn:before { 77 + content: ""; 78 + position: absolute; 79 + top: 50%; 80 + left: 50%; 81 + width: 300%; 82 + height: 300%; 83 + background-color: #878787; 84 + transition: all 0.3s ease; 85 + border-radius: 50%; 86 + z-index: 0; 87 + transform: translate(-50%, -50%); 88 +} 89 + 90 +/* Animation when smaller button is clicked */ 91 +.small-login-btn.clicked:before { 92 + width: 50%%; 93 + height: 50%; 94 + opacity: 0; 95 +} 96 + 97 +/* Styling for the smaller button text */ 98 +.small-login-btn span { 99 + position: relative; 100 + z-index: 1; 101 +} 102 + 103 +/* gehele container 104 +#xwikimaincontainer { 105 + width: 970px; 106 +} 107 +*/ 108 + 
 
- XWiki.XWikiRights[0]
-   - levels
-   ... ... @@ -1,1 +1,1 @@ 1 -view 1 +view,script 
 
- XWiki.XWikiRights[1]
-   - levels
-   ... ... @@ -1,1 +1,1 @@ 1 -view 1 +view,script 
 
- XWiki.XWikiRights[2]
-   - levels
-   ... ... @@ -1,1 +1,1 @@ 1 -view 1 +view,script 
 
- XWiki.XWikiRights[5]
-   - levels
-   ... ... @@ -1,1 +1,1 @@ 1 -view 1 +view,comment,edit,script,delete 
 
- XWiki.XWikiRights[9]
-   - allow
-   ... ... @@ -1,0 +1,1 @@ 1 +Allow 
- groups
-   ... ... @@ -1,0 +1,1 @@ 1 +XWiki.Axians 
- levels
-   ... ... @@ -1,0 +1,1 @@ 1 +view,script 
 
