This is a valid RSS feed.
This feed is valid, but interoperability with the widest range of feed readers could be improved by implementing the following recommendations.
line 95, column 0: (25 occurrences) [help]
line 95, column 0: (25 occurrences) [help]
line 210, column 0: (22 occurrences) [help]
<div class="video-container"><iframe loading="lazy" title="XPages JDBC Confi ...
<blockquote class="twitter-tweet" data-lang="en">
<p><script async src="//platform.twitter.com/widgets.js" charset="utf-8"></s ...
<p>It made me think a little about Swiper, and I thought I would give it ano ...
<?xml version="1.0" encoding="UTF-8"?><rss version="2.0"
xmlns:content="http://purl.org/rss/1.0/modules/content/"
xmlns:wfw="http://wellformedweb.org/CommentAPI/"
xmlns:dc="http://purl.org/dc/elements/1.1/"
xmlns:atom="http://www.w3.org/2005/Atom"
xmlns:sy="http://purl.org/rss/1.0/modules/syndication/"
xmlns:slash="http://purl.org/rss/1.0/modules/slash/"
>
<channel>
<title>Software Development – Cameron Gregor</title>
<atom:link href="https://camerongregor.com/category/software-development/feed/" rel="self" type="application/rss+xml" />
<link>https://camerongregor.com</link>
<description>Sharing Tips, Tools and Techniques for XPages Developers</description>
<lastBuildDate>Tue, 18 Jun 2019 23:34:28 +0000</lastBuildDate>
<language>en-AU</language>
<sy:updatePeriod>
hourly </sy:updatePeriod>
<sy:updateFrequency>
1 </sy:updateFrequency>
<generator>https://wordpress.org/?v=6.8.3</generator>
<item>
<title>Our current Deployment setup: Github + Jenkins + BuildXPages</title>
<link>https://camerongregor.com/2019/06/19/our-current-deployment-setup-github-jenkins-buildxpages/</link>
<comments>https://camerongregor.com/2019/06/19/our-current-deployment-setup-github-jenkins-buildxpages/#comments</comments>
<dc:creator><![CDATA[camerongregor]]></dc:creator>
<pubDate>Tue, 18 Jun 2019 14:17:32 +0000</pubDate>
<category><![CDATA[Software Development]]></category>
<category><![CDATA[buildxpages]]></category>
<category><![CDATA[jenkins]]></category>
<guid isPermaLink="false">http://camerongregor.com/?p=589</guid>
<description><![CDATA[I recently received a question from Patrick Kwinten about whether I am still using my BuildXPages project, and whether I have involved Jenkins in the setup. The answer is yes, I still use BuildXPages...]]></description>
<content:encoded><![CDATA[
<p>I recently received a question from Patrick Kwinten about whether I am still using my <a href="http://camac.github.io/BuildXPages/">BuildXPages </a>project, and whether I have involved Jenkins in the setup.</p>
<p>The answer is yes, I still use BuildXPages on a daily basis, and in regards to Jenkins, I have been using Jenkins for almost 5 years to build and deploy our XPages projects. I decided to keep the BuildXPages Documentation ‘agnostic’ on the choice of Build server, so as to focus just on the library and the tasks it can achieve. The point was that you should be able to use any build server you like. So this is the reason Jenkins was not really mentioned in the documentation.</p>
<p>It has been more than a year since I have done a blog post so I thought instead of responding to Patrick directly, it would be a good excuse to finally put another blog post out there! Maybe it will spark me into action a bit more and I might make some more posts after, well see. I have to be honest, the constant ‘XPages is dead’ mantra from here there and everywhere, combined with a recently developed addiction to online chess has a negative effect on my contributions to the community!</p>
<h2 class="wp-block-heading">What is our Current Setup?</h2>
<p>This will just be a rough overview of our current setup. You can see a more detailed post on our setup from about 5 years ago <a href="https://camerongregor.com/2014/08/09/build-system-for-xpages-and-osgi-plugins/">here</a>. Many of the details are the same but I will just go over it quickly here in a “Stream of consciousness” so I can get this blog post out quickly. </p>
<p>All our XPages related work is done in one monolithic repository, hosted as a private repository on Github. Initially we had multiple repositories but it was just way too much overhead to be switching branches / merging on multiple repositories. Moving to a single repository simplified everything greatly and I’m very glad we did.</p>
<p>Our repository consists of about 15+ different NSFs, and maybe 20+ custom java plugins. </p>
<p>All the NSFs are edited in Domino Designer and synchronised to the ODP using the Team Development NSF to ODP sync and of course they are using Swiper! </p>
<p>All our Java plugins are developed using Eclipse IDE. Our java plugins consist of a custom java framework (which has no dependency on XPages so it can ran as background tasks etc.) which provides the concept of a custom ‘application’ with ‘services’, ‘settings’, POJO Model configurations. The framework manages mapping the Models to the NotesDocuments and back etc. among a lot of other things.<br>Our plugins then also contain a custom XPages library which provides an additional layer of our framework that includes the XPages dependency. It has all the XPages Controls that we have developed, it has other extensions that we have made on top of the extension library (things like the <a href="https://camerongregor.com/2017/12/05/jdbcnotesdocuments/">Notes based JDBC provider</a>). It also contains our custom renderers for the SmartAdmin wrapbootstrap theme (v3), which is our base theme for our applications.</p>
<p>In addition to these base framework plugins, we have specific plugins for each custom ‘application’. These provide the custom java code which defines the actual business applications that we deliver.</p>
<h2 class="wp-block-heading">Branching strategy</h2>
<p>We initially went with git-flow because it had a pretty diagram that explained it, and it was built directly into Sourcetree which is our git client of choice. However it quickly became apparent that the git-flow branching model does not suit the way we deliver our product. </p>
<p>Instead we moved to the <a href="https://guides.github.com/introduction/flow/">github-flow</a> method, in which there is one ‘master’ branch, and everything else is just hanging off that. Read the link for a better explanation, but again this was a much simpler strategy that suits a ‘continuous deployment’ style of code release, as opposed to the git-flow which I believe better supports the situation that you might have to support multiple versions of your product, and need to retain the ability to apply hotfixes and patches to different versions.</p>
<h2 class="wp-block-heading">Branch ready for Production!</h2>
<p>So I have been working away on my branch and I am ready to deploy! I create a pull request on github, we have the chance to review it and decide if it is ready, if it is all ready to go then, I merge the pull request on github and now my code has made it into the master branch, ready to deploy.</p>
<figure class="wp-block-image"><img fetchpriority="high" decoding="async" width="787" height="791" src="https://camerongregor.com/wp-content/uploads/2019/06/image.png" alt="" class="wp-image-590" srcset="https://camerongregor.com/wp-content/uploads/2019/06/image.png 787w, https://camerongregor.com/wp-content/uploads/2019/06/image-150x150.png 150w, https://camerongregor.com/wp-content/uploads/2019/06/image-298x300.png 298w, https://camerongregor.com/wp-content/uploads/2019/06/image-768x772.png 768w, https://camerongregor.com/wp-content/uploads/2019/06/image-160x160.png 160w" sizes="(max-width: 787px) 100vw, 787px" /></figure>
<h2 class="wp-block-heading">Build It!</h2>
<p>As mentioned earlier, we use Jenkins to build and deploy our xpages applications and plugins. We have windows machine set up with Jenkins running, and also the ‘headless’ domino designer running, and also an eclipse installation too.</p>
<p>Our Jenkins server used to be publicly accessible, and therefore used to receive a ‘ping’ from github to automatically trigger a build whenever master branch was updated. But since our Jenkins machine is now only accessible from within our private network, it no longer receives the github notification. You could also set up the jenkins server to ‘poll’ github for changes, however I have found that it is not a problem to just manually trigger the production build because I usually just do it immediately after merging the pull-request anyway.</p>
<p>Here is a screenshot of our Jenkins homepage, there are a bunch of jobs on there but truth is we only really use 3 of them. 1 for building and 2 for deploying.</p>
<p>The build job is called ‘Horizon’ as this is our name for our system, I have set this one up as a multi-branch pipeline job, because I thought I might have a ‘production’ and a ‘staging’ building, but in truth we only ever do a production build so the multi-branch thing was unnecessary.</p>
<figure class="wp-block-image"><img decoding="async" width="1024" height="601" src="https://camerongregor.com/wp-content/uploads/2019/06/image-1-1024x601.png" alt="" class="wp-image-591" srcset="https://camerongregor.com/wp-content/uploads/2019/06/image-1-1024x601.png 1024w, https://camerongregor.com/wp-content/uploads/2019/06/image-1-300x176.png 300w, https://camerongregor.com/wp-content/uploads/2019/06/image-1-768x451.png 768w, https://camerongregor.com/wp-content/uploads/2019/06/image-1.png 1264w" sizes="(max-width: 1024px) 100vw, 1024px" /></figure>
<p>Looking at the page of the production branch build, you can see the jenkins ‘pipeline’ is split into checkout, build plugins, build nsfs.</p>
<p>Here you can see something that happens occasionally where it fails when I introduce some new properties to some XPages controls (which I only do maybe once a month). What happens is, the plugins build successfully with the new control properties, however the Headless Domino Designer NSF Build triggers immediately afterwards, and does not have these new plugins installed, so the NSFs fail. What you can’t see is that after build #387, I manually re-install the newly built XPages library to the Headless Domino designer, and then trigger another build. It usually returns to a successful build immediately afterwards but here I got a random error for build #388 for some reason, and running the build again #389 was successful. This is a little annoying but usually most deployments don’t involve new control properties so this is a rare occurrence.</p>
<figure class="wp-block-image"><img decoding="async" width="1024" height="674" src="https://camerongregor.com/wp-content/uploads/2019/06/image-2-1024x674.png" alt="" class="wp-image-592" srcset="https://camerongregor.com/wp-content/uploads/2019/06/image-2-1024x674.png 1024w, https://camerongregor.com/wp-content/uploads/2019/06/image-2-300x197.png 300w, https://camerongregor.com/wp-content/uploads/2019/06/image-2-768x505.png 768w, https://camerongregor.com/wp-content/uploads/2019/06/image-2.png 1251w" sizes="(max-width: 1024px) 100vw, 1024px" /></figure>
<h2 class="wp-block-heading">Jenkins Pipeline file</h2>
<p>The Jenkins ‘pipeline’ build is a newer way of defining the build. Here is the pipeline file that we are using. You can see it simply triggers 2 ant tasks, and reports the success / failure via email</p>
<pre class="urvanov-syntax-highlighter-plain-tag">pipeline{
agent any
stages {
stage('Build Plugins') {
steps {
echo "Build Plugins on ${env.BRANCH_NAME}"
withEnv( ["ANT_HOME=${tool 'Default'}"] ) {
bat '%ANT_HOME%/bin/ant.bat distplugins'
}
}
}
stage('Build NSFs') {
steps {
withEnv( ["ANT_HOME=${tool 'DefaultCam'}"] ) {
bat '%ANT_HOME%/bin/ant.bat buildNsfs'
}
}
}
stage('Results') {
steps {
archive '**/com.jord.*.jar'
}
}
}
post {
success {
emailext body: "HAPI Success ${currentBuild.currentResult}", subject: "BuildXPages ${currentBuild.currentResult}", to: 'xxxxxx@xxxxx'
}
changed {
emailext body: "HAPI ${currentBuild.currentResult}", subject: "BuildXPages ${currentBuild.currentResult}", to: 'xxxxx@xxxxx'
}
failure {
emailext body: 'HAPI Failed', subject: "BuildXPages Failed", to: 'xxxxxx@xxxxxxx'
}
}
}</pre>
<h2 class="wp-block-heading">Ant Tasks</h2>
<p>The two ant tasks triggered in the JenkinsFile (distplugins and buildNsfs) are defined in a custom ant build.xml script that is committed to the repository. This script makes use of the functionality of the BuildXPages project.</p>
<h3 class="wp-block-heading">Building Plugins</h3>
<p>The distplugins task is just used to sequence some other tasks in the script</p>
<figure class="wp-block-image"><img loading="lazy" decoding="async" width="771" height="58" src="https://camerongregor.com/wp-content/uploads/2019/06/image-3.png" alt="" class="wp-image-593" srcset="https://camerongregor.com/wp-content/uploads/2019/06/image-3.png 771w, https://camerongregor.com/wp-content/uploads/2019/06/image-3-300x23.png 300w, https://camerongregor.com/wp-content/uploads/2019/06/image-3-768x58.png 768w" sizes="auto, (max-width: 771px) 100vw, 771px" /></figure>
<figure class="wp-block-image"><img loading="lazy" decoding="async" width="534" height="234" src="https://camerongregor.com/wp-content/uploads/2019/06/image-4.png" alt="" class="wp-image-594" srcset="https://camerongregor.com/wp-content/uploads/2019/06/image-4.png 534w, https://camerongregor.com/wp-content/uploads/2019/06/image-4-300x131.png 300w" sizes="auto, (max-width: 534px) 100vw, 534px" /><figcaption>This shows the copyPlugins task which simply copies the source of the plugins to a working directory in preparation to be built by Eclipse PDE Build’s Headless plugin/feature building.</figcaption></figure>
<figure class="wp-block-image"><img loading="lazy" decoding="async" width="731" height="277" src="https://camerongregor.com/wp-content/uploads/2019/06/image-5.png" alt="" class="wp-image-595" srcset="https://camerongregor.com/wp-content/uploads/2019/06/image-5.png 731w, https://camerongregor.com/wp-content/uploads/2019/06/image-5-300x114.png 300w, https://camerongregor.com/wp-content/uploads/2019/06/image-5-1320x500.png 1320w" sizes="auto, (max-width: 731px) 100vw, 731px" /><figcaption>This is the buildPlugins custom task which calls the BuildXPages buildPlugins task, this is where Eclipse PDE Build does it’s thing.</figcaption></figure>
<figure class="wp-block-image"><img loading="lazy" decoding="async" width="396" height="53" src="https://camerongregor.com/wp-content/uploads/2019/06/image-6.png" alt="" class="wp-image-596" srcset="https://camerongregor.com/wp-content/uploads/2019/06/image-6.png 396w, https://camerongregor.com/wp-content/uploads/2019/06/image-6-300x40.png 300w" sizes="auto, (max-width: 396px) 100vw, 396px" /><figcaption>After building the plugins I unzip them because they are zipped up after PDE build does it’s thing.</figcaption></figure>
<figure class="wp-block-image"><img loading="lazy" decoding="async" width="508" height="166" src="https://camerongregor.com/wp-content/uploads/2019/06/image-7.png" alt="" class="wp-image-597" srcset="https://camerongregor.com/wp-content/uploads/2019/06/image-7.png 508w, https://camerongregor.com/wp-content/uploads/2019/06/image-7-300x98.png 300w" sizes="auto, (max-width: 508px) 100vw, 508px" /><figcaption>Lastly, I copy the newly build plugins to an ‘update site’ so I always have the latest plugins available in the same place on the file system. This is where I will deploy them from when we deploy</figcaption></figure>
<h3 class="wp-block-heading">Building NSFs</h3>
<p>The buildNsfs task coordinates the building of all the NSFs</p>
<figure class="wp-block-image"><img loading="lazy" decoding="async" width="579" height="377" src="https://camerongregor.com/wp-content/uploads/2019/06/image-8.png" alt="" class="wp-image-598" srcset="https://camerongregor.com/wp-content/uploads/2019/06/image-8.png 579w, https://camerongregor.com/wp-content/uploads/2019/06/image-8-300x195.png 300w" sizes="auto, (max-width: 579px) 100vw, 579px" /></figure>
<p>The ‘nsfbuild’ is a macro that is defined within the same custom build.xml ant script, this calls the ‘buildnsf’ task from the BuildXPages project.</p>
<figure class="wp-block-image"><img loading="lazy" decoding="async" width="995" height="411" src="https://camerongregor.com/wp-content/uploads/2019/06/image-9.png" alt="" class="wp-image-599" srcset="https://camerongregor.com/wp-content/uploads/2019/06/image-9.png 995w, https://camerongregor.com/wp-content/uploads/2019/06/image-9-300x124.png 300w, https://camerongregor.com/wp-content/uploads/2019/06/image-9-768x317.png 768w" sizes="auto, (max-width: 995px) 100vw, 995px" /></figure>
<p>If there are any errors in the NSF build, they will show in the console output of the Jenkins build</p>
<figure class="wp-block-image"><img loading="lazy" decoding="async" width="864" height="322" src="https://camerongregor.com/wp-content/uploads/2019/06/image-10.png" alt="" class="wp-image-600" srcset="https://camerongregor.com/wp-content/uploads/2019/06/image-10.png 864w, https://camerongregor.com/wp-content/uploads/2019/06/image-10-300x112.png 300w, https://camerongregor.com/wp-content/uploads/2019/06/image-10-768x286.png 768w" sizes="auto, (max-width: 864px) 100vw, 864px" /></figure>
<p>If there were no errors during plugins and NSF building, then we get a green status and we are ready to deploy!</p>
<figure class="wp-block-image"><img loading="lazy" decoding="async" width="886" height="213" src="https://camerongregor.com/wp-content/uploads/2019/06/image-11.png" alt="" class="wp-image-601" srcset="https://camerongregor.com/wp-content/uploads/2019/06/image-11.png 886w, https://camerongregor.com/wp-content/uploads/2019/06/image-11-300x72.png 300w, https://camerongregor.com/wp-content/uploads/2019/06/image-11-768x185.png 768w" sizes="auto, (max-width: 886px) 100vw, 886px" /></figure>
<p>As you can see, our build takes about 4 mins. Our NSFs’ are building incrementally thanks to the Headless Designer Plugin that comes with BuildXPages project. Otherwise, a full-build of every NSF would be performed every time, and this would take much longer.</p>
<h2 class="wp-block-heading">Deploying!</h2>
<p>We have 2 Jenkins jobs for deploying, one to upload our plugins to the NSF update site, and another one to run the design refresh of the templates on the production server.</p>
<p>Our servers are set to restart every morning at 6am, so we actually schedule these 2 tasks to run every morning at 5:15am for plugins and then 5:30am for templates refresh. </p>
<p>We used to just restart after the 1am design refresh but we had some people from other offices still using our server at that time, so 6am was a better time for our company.</p>
<h3 class="wp-block-heading">Import Master Plugins</h3>
<p>The Import master plugins is just a custom Jenkins job that is not linked to any source repository, it simply has one file in the job that is the ant script.</p>
<p>The ant script uses the BuildXPages <strong>importplugins </strong>task to import the plugins into an Update site nsf that is on the production server, from the plugins final location in the earlier build task.</p>
<h3 class="wp-block-heading">Refresh Data Templates</h3>
<p>All the NSFs that are ‘built’ in the Headless Domino Designer have a template name with a suffix of ‘_master’ to indicate they are the master branch design. These NSFs are considered to be ‘local’ to the Jenkins machine. </p>
<p>All the production templates are set to inherit from their templates with suffix ‘_master’.</p>
<p>So for refreshing the production templates, we simply use the BuildXPages refreshdbdesign task to refresh from our ‘local’ templates to the production server.</p>
<figure class="wp-block-image"><img loading="lazy" decoding="async" width="804" height="326" src="https://camerongregor.com/wp-content/uploads/2019/06/image-12.png" alt="" class="wp-image-602" srcset="https://camerongregor.com/wp-content/uploads/2019/06/image-12.png 804w, https://camerongregor.com/wp-content/uploads/2019/06/image-12-300x122.png 300w, https://camerongregor.com/wp-content/uploads/2019/06/image-12-768x311.png 768w" sizes="auto, (max-width: 804px) 100vw, 804px" /></figure>
<h2 class="wp-block-heading">Conclusion</h2>
<p>Hopefully that was a good overview on how we are managing and deploying our xpages applications using BuildXPages + Jenkins. I’m sure I missed something here or there so let me know if you have any questions!</p>
]]></content:encoded>
<wfw:commentRss>https://camerongregor.com/2019/06/19/our-current-deployment-setup-github-jenkins-buildxpages/feed/</wfw:commentRss>
<slash:comments>2</slash:comments>
</item>
<item>
<title>Minor enhancements to the XPages Table Control</title>
<link>https://camerongregor.com/2017/12/13/minor-enhancements-to-the-xpages-table-control/</link>
<comments>https://camerongregor.com/2017/12/13/minor-enhancements-to-the-xpages-table-control/#comments</comments>
<dc:creator><![CDATA[camerongregor]]></dc:creator>
<pubDate>Wed, 13 Dec 2017 10:44:04 +0000</pubDate>
<category><![CDATA[Software Development]]></category>
<category><![CDATA[control]]></category>
<category><![CDATA[extlibx]]></category>
<category><![CDATA[xpages]]></category>
<guid isPermaLink="false">http://camerongregor.com/?p=566</guid>
<description><![CDATA[As mentioned in my previous post, I have started up a fork of the ExtLibX project and have started contributing to it instead of my previous GregorbyteXspLibrary. Last week I shared the JDBC Configuration...]]></description>
<content:encoded><![CDATA[<p>As mentioned in <a href="https://camerongregor.com/2017/12/01/my-new-approach-for-sharing-xpages-controls-and-extensions/" target="_blank" rel="noopener">my previous post</a>, I have started up a fork of the ExtLibX project and have started contributing to it instead of my previous GregorbyteXspLibrary. Last week I shared the <a href="https://camerongregor.com/2017/12/05/jdbcnotesdocuments/" target="_blank" rel="noopener">JDBC Configuration Provider</a> which uses Notes Documents to store the configuration info, within this release was another small contribution with 2 minor enhancements for the XPages <strong>table</strong> control.</p>
<p>The core XPages table control does not provide support for <em>thead</em>, <em>tbody</em>, and <em>tfoot</em> elements. Also there isn’t an option for a <em>th</em> control.</p>
<p>There are some easy workarounds for this such as just <a href="http://www.dominoguru.com/pages/ibm_xpages_xptable_using_passthru_markup.html" target="_blank" rel="noopener">using passthrough html</a> as Chris Toohey demonstrated on his blog.</p>
<p>This passthru technique works fine but I just thought I would share my approach, which was to add support for these elements by creating an extra <strong>th</strong> control, and also extending the table renderer to support use of thead, tfoot and tbody.</p>
<p>It isn’t that exciting but I just thought it was a good example of being able to extend core controls, and add your own desired functionality to them, or modifying existing functionality.</p>
<p>If you would like to try these modifications out, then <a href="https://github.com/camac/XPagesExtLibX/releases" target="_blank" rel="noopener">download the latest release</a> from my camac/ExtLibX project and install to both domino designer and your target domino server.</p>
<p>If you don’t want to install plugins, you should also be able to take the table renderer and th component source code from within <a href="https://github.com/OpenNTF/XPagesExtLibX/commit/145e5ccffd4d49e184c5c9b7295560769dec29c6" target="_blank" rel="noopener">this commit</a> and install it to the java, faces-config and xsp-config files in your nsf</p>
<h2>Using the Table Header Cell control</h2>
<p>To use a th Cell, simply use the Table Header Cell control. It is exactly like a normal <code><xp:td></code> control, except it will render as <em>th</em> instead of <em>td</em></p><pre class="urvanov-syntax-highlighter-plain-tag"><xp:tr>
<!-- Using the th control -->
<xe:th>Last Name</xe:th>
<xe:th>First Name</xe:th>
</xp:tr></pre><p></p>
<h2>Using Table Header, Footer and Body sections</h2>
<p>Support for <thead>, <tfoot> and <tbody> elements is provided by an extended version of the Table Renderer, this renderer’s rendererType is: <strong>com.ibm.xsp.extlibx.Table</strong>.</p>
<p>You have 2 choices on how to enable the table renderer</p>
<ul>
<li>Make the renderer default for all tables in the application</li>
<li>Enable the renderer only for specific tables</li>
</ul>
<h3>Setting Table Renderer for All Tables</h3>
<p>You can set the renderer Type for all Tables in your application by including the following in your theme configuration file:</p><pre class="urvanov-syntax-highlighter-plain-tag"><control>
<name>HtmlTable</name>
<property>
<name>rendererType</name>
<value>com.ibm.xsp.extlibx.Table</value>
</property>
</control></pre><p></p>
<h3>Setting RendererType for a specific table</h3>
<p>To enable the renderer for specific tables, just set the rendererType to <strong>com.ibm.xsp.extlibx.Table</strong></p><pre class="urvanov-syntax-highlighter-plain-tag"><xp:table rendererType="com.ibm.xsp.extlibx.Table"></pre><p></p>
<h3>How to specify thead, tfoot and tbody</h3>
<p>The thead and tfoot elements can be used by providing a facet named thead or tfoot<br />
The tbody element will be rendered by default and contain all the child elements of the table</p><pre class="urvanov-syntax-highlighter-plain-tag"><?xml version="1.0" encoding="UTF-8"?>
<xp:view xmlns:xp="http://www.ibm.com/xsp/core" xmlns:xe="http://www.ibm.com/xsp/coreex">
<xp:table rendererType="com.ibm.xsp.extlibx.Table">
<!-- thead and tfoot are specified using facets -->
<xp:this.facets>
<xp:tr xp:key="thead">
<xe:th>Header 1</xe:th>
<xe:th>Header 2</xe:th>
</xp:tr>
<xp:tr xp:key="tfoot">
<xp:td>Footer 1</xp:td>
<xp:td>Footer 2</xp:td>
</xp:tr>
</xp:this.facets>
<!-- all the child controls will be automatically renderer within a tbody element -->
<xp:tr>
<xp:td>R1C1</xp:td>
<xp:td>R1C2</xp:td>
</xp:tr>
<xp:tr>
<xp:td>R2C1</xp:td>
<xp:td>R2C2</xp:td>
</xp:tr>
</xp:table>
</xp:view></pre><p></p>
<h3>Multiple rows within thead / tfoot</h3>
<p>Chris Toohey makes a good point in the comments, sometimes you want more than one row in the header or footer. You can achieve this by using a panel as the root component instead of the tr. You can then put multiple tr’s within that panel.</p>
<p>The panel’s <div> tags won’t be rendered unless it has an id. If it has an id and you don’t want the tags rendered you can also use <code>disableOutputTag="false"</code>.</p><pre class="urvanov-syntax-highlighter-plain-tag"><xp:table>
<xp:this.facets>
<!-- Multiple rows within tfoot / thead -->
<xp:panel xp:key="tfoot">
<xp:tr>
<xp:td></xp:td>
<xp:td></xp:td>
</xp:tr>
<xp:tr>
<xp:td></xp:td>
<xp:td></xp:td>
</xp:tr>
</xp:panel>
</xp:this.facets>
</xp:table></pre><p> </p>
<p>Let me know what you think. I know this one wasn’t very exciting but I should have some better controls coming out soon! I just did this one because it was quick and easy.</p>
]]></content:encoded>
<wfw:commentRss>https://camerongregor.com/2017/12/13/minor-enhancements-to-the-xpages-table-control/feed/</wfw:commentRss>
<slash:comments>2</slash:comments>
</item>
<item>
<title>XPages JDBC Configuration via NotesDocuments</title>
<link>https://camerongregor.com/2017/12/05/jdbcnotesdocuments/</link>
<comments>https://camerongregor.com/2017/12/05/jdbcnotesdocuments/#respond</comments>
<dc:creator><![CDATA[camerongregor]]></dc:creator>
<pubDate>Tue, 05 Dec 2017 11:34:46 +0000</pubDate>
<category><![CDATA[Software Development]]></category>
<category><![CDATA[extlibx]]></category>
<category><![CDATA[jdbc]]></category>
<category><![CDATA[xpages]]></category>
<guid isPermaLink="false">http://camerongregor.com/?p=550</guid>
<description><![CDATA[If you have used the Extension Library’s Relational controls, then you are probably familiar with the process of configuring your JDBC Connection details. You have the option of setting up a ‘named’ connection so...]]></description>
<content:encoded><![CDATA[<p>If you have used the Extension Library’s Relational controls, then you are probably familiar with the process of configuring your JDBC Connection details.</p>
<p>You have the option of setting up a ‘named’ connection so you can reference a database connection by name. This involves creating an xml file <strong><yourconnection>.jdbc</strong> in the <strong>WebContent\WEB-INF\jdbc\ </strong>directory. This xml file specifies the driver, url, username, password, and perhaps some information about connection pooling.</p>
<p>Alternatively your can just build your own connection url and use this directly in your Java code or XPages controls.</p>
<p>Named connections are a much nicer way to go, and they allow for connection pooling, however the method of using the xml file within the WEB-INF\jdbc directory is a little annoying.</p>
<p>This is not a very accessible or flexible way to manage your connection information. Often your local development SQL Database will have different username / password / url etc. So you often end up changing the config files back and forth everytime you want to develop and deploy.</p>
<p>So I decided to come up with a way to store JDBC Connection information somewhere else, and it seemed logical to store it in NotesDocuments. This way the JDBC connections are configurable at run-time and do not require a change to a design.</p>
<p>Additionally, these JDBC configuration documents can be protected from read-access so that only admins and database signer’s can access them. The session as signer should be able to access them with elevated rights.</p>
<p><strong>Short Demonstration Video</strong></p>
<p>If you would like to see the JDBC Provider in action please have a look at this video</p>
<div class="video-container"><iframe loading="lazy" title="XPages JDBC Configuration from Notes Documents" width="500" height="281" src="https://www.youtube.com/embed/7sassOh-wkw?feature=oembed&wmode=opaque" frameborder="0" allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture; web-share" referrerpolicy="strict-origin-when-cross-origin" allowfullscreen></iframe></div>
<h3></h3>
<h3>How can I use this JDBCNotesDocumentProvider system?</h3>
<p>Download the <a href="https://github.com/camac/XPagesExtLibX/releases/tag/v9.0.1.v00_17-jdbcprovideralpha" target="_blank" rel="noopener">Alpha Release</a> in my fork of the ExtLibX project and have a look at the documentation in the PDF in the release. You can also have a look in the demo video above.</p>
<p>As mentioned in my previous post, <a href="https://camerongregor.com/2017/12/01/my-new-approach-for-sharing-xpages-controls-and-extensions/" target="_blank" rel="noopener">I have decided to release all my future contributions through the ExtLibX project</a>.</p>
<p>If this solution is working for you then you should be fine to use it in production but I am just calling it a alpha release because I am looking to get some feedback before making a final ‘release’.</p>
<h3>How does the JDBCNotesDocumentProvider work?</h3>
<p>I basically copied the way the Extension Library initializes it’s JDBC Information from the xml files in the WEB-INF/jdbc folder, and modified it to instead look in NotesDocuments! Sounds simple but there was a little bit of fiddling around.</p>
<p>Basically:</p>
<ul>
<li>Someone Access the XPages application</li>
<li>The JDBC Configuration Initialization is triggered
<ul>
<li>If it is set to use ‘local’ configuration it will search the current database for JDBCConnection info</li>
<li>If it is set to use ‘global’ configuration it will search the central configuration database for JDBCConnection info</li>
</ul>
</li>
</ul>
<p>Actually in my original solution, I was using an ‘ApplicationInitializer’ however this runs <em>before </em>you have access to sessionAsSigner, so this was a bit of a problem. I had trouble getting a full-access session but managed to do it with some trickery, however I don’t think this ‘trickery’ is suitable to be shared, so I came up with an alternative.</p>
<p>I moved the initialization code to a PhaseListener, and set it up so the initialization will only run if it hasn’t been successfully done yet. I still don’t like this so much but it works.</p>
<h3>Feedback / Comments</h3>
<p>Would love to hear if this is something you think you would use. If you have any suggestions / improvements please let me know!</p>
<p> </p>
]]></content:encoded>
<wfw:commentRss>https://camerongregor.com/2017/12/05/jdbcnotesdocuments/feed/</wfw:commentRss>
<slash:comments>0</slash:comments>
</item>
<item>
<title>My New Approach for Sharing XPages Controls and Extensions</title>
<link>https://camerongregor.com/2017/12/01/my-new-approach-for-sharing-xpages-controls-and-extensions/</link>
<comments>https://camerongregor.com/2017/12/01/my-new-approach-for-sharing-xpages-controls-and-extensions/#comments</comments>
<dc:creator><![CDATA[camerongregor]]></dc:creator>
<pubDate>Fri, 01 Dec 2017 12:33:41 +0000</pubDate>
<category><![CDATA[Software Development]]></category>
<category><![CDATA[extlib]]></category>
<category><![CDATA[xpages]]></category>
<guid isPermaLink="false">http://camerongregor.com/?p=553</guid>
<description><![CDATA[Over the past few years I have shared a handful of controls and techniques on my blog, and most of these I have made available on github in one way or another. Originally I...]]></description>
<content:encoded><![CDATA[<p>Over the past few years I have shared a handful of controls and techniques on my blog, and most of these I have made available on github in one way or another. Originally I was creating a separate github project for each one such as <a href="https://github.com/camac/XPagesEmailValidatorPlugin" target="_blank" rel="noopener">EmailValidator</a>, <a href="https://github.com/camac/XPagesPhoneNumberControl" target="_blank" rel="noopener">PhoneNumberControl</a> and the <a href="https://github.com/camac/XPagesMessagesControl" target="_blank" rel="noopener">Messages Controls.</a> After these few I decided it would be more efficient to create a single project that I could put them all into, so I then began to contribute them via my <a href="https://github.com/camac/GregorbyteXspLibrary" target="_blank" rel="noopener">GregorbyteXspLibrary</a>.</p>
<p>From time to time I had a few comments like “This would be a nice addition to the Extension Library”. However whenever I found time to work on open source projects, it always seemed to be a higher priority to work on sharing something new instead of doing more work on something I had already shared.</p>
<p>It would mean the extra step of moving the work from GregorbyteXspLibrary into the ExtensionLibrary, refactoring package names, testing it all again, updating copyright licensing, and then creating a pull request to have it merged into the Extension Library.</p>
<p>Even after that there would be no guarantee of pull requests being accepted into the OpenNTF repository. So the work maybe would just sit there and seem like a wasted effort. I thought “well maybe people would be still be able to install my ‘fork’ of the Extension Library?”, but then I could see how this get messy when future IBM releases come out with newer Extension Library versions.</p>
<p>The short story: I just wasn’t feeling that clear on how I should approach it so I just left it as it was.</p>
<h3>What about ExtLibX?</h3>
<p>I had totally forgotten about ExtLibX.</p>
<p>ExtLib is the main Extension Library that we all know and love, but there is also the ExtLib<strong>X</strong> which is the ‘e<strong>X</strong>perimental’ Extension Library.</p>
<p>The ExtLibX project was created exactly for the purpose of submitting experimental controls. It is a bonus extension library that people have the option of installing alongside the main Extension Library.</p>
<p>The idea is that after the experimental controls have gained acceptance as a “good idea” they could be migrated into the main Extension Library.</p>
<p>The ExtLibX originally contained things like the Relational controls (which are now part of the core ExtLib), and currently has some work sitting in it for Bootstrap4.</p>
<p>So now that I remember about ExtLibX, I have finally come up with a strategy for sharing controls that I am confident in.</p>
<h2>My New Strategy for Sharing Reusable Controls and Techniques!</h2>
<ul>
<li>My existing <a href="https://github.com/camac/GregorbyteXspLibrary" target="_blank" rel="noopener">camac/GregorbyteXspLibrary</a> will remain as it stands now, with no further development</li>
<li>I’ve created my own fork of the experimental extension library: <a href="https://github.com/camac/XPagesExtLibX" target="_blank" rel="noopener">camac/XPagesExtLibX</a></li>
<li>I’ll share all my new controls and techniques through camac/XPagesExtLibX</li>
<li>I’ll prepare install-able releases so that people can use the controls straight away if they want to. This will hopefully help me get some feedback / bug reports / suggested improvements.</li>
<li>If I get feedback that a control or technique is useful and is working without problems, I will prepare a pull request for that control so that it can be reviewed for acceptance into the main <a href="https://github.com/OpenNTF/XPagesExtLibX" target="_blank" rel="noopener">OpenNTF/XPagesExtLibX</a> project.</li>
<li>If IBM decide that control is useful to everyone, they are then able to migrate it from ExtLibX to ExtLib at there own desire as they have access to both ExtLibX and ExtLib</li>
<li>When I get time I will move the controls from GregorbyteXspLibrary over to ExtLibX</li>
</ul>
<p>Overall I am pretty happy with this new strategy as enables me to share controls so that they are immediately usable, but also still have a pathway to <em>possibly</em> be included in the ExtensionLibrary.</p>
<p>I am hoping to share a few controls and techniques in the next few months that will be a bit more useful for day-to-day XPages work, and also a few extended versions of some standard controls that make them a little bit more developer friendly.</p>
<p>I’m hoping to release the first version in the next week which should contain</p>
<ul>
<li>Some modified Table Controls
<ul>
<li>Modified table renderer to have proper <code><thead> <tfoot> <tbody></code> structure.</li>
<li><code><th></code> control</li>
</ul>
</li>
<li>Master / Slave checkbox controls
<ul>
<li>easy bulk record selection within repeat controls</li>
<li>select all / deselect all functionality</li>
<li>automatic row-styling based on selected checkbox</li>
<li>ability to use the whole row as a click target for that row’s checkbox</li>
</ul>
</li>
<li>A JDBC Connection provider that loads configuration information from Notes Documents in a config database (either current database or centralised database)</li>
</ul>
<p>If your interested in any of these above please let me know in the comments below!<br />
Do you have any other ideas for other Controls? I have a few more in the pipeline. Let’s see if we can breathe a bit of life into the Extension Library!</p>
]]></content:encoded>
<wfw:commentRss>https://camerongregor.com/2017/12/01/my-new-approach-for-sharing-xpages-controls-and-extensions/feed/</wfw:commentRss>
<slash:comments>2</slash:comments>
</item>
<item>
<title>Embedded Experiences not rendering after IBM Notes FP9</title>
<link>https://camerongregor.com/2017/09/27/embedded-experiences-not-rendering-after-ibm-notes-fp9/</link>
<comments>https://camerongregor.com/2017/09/27/embedded-experiences-not-rendering-after-ibm-notes-fp9/#comments</comments>
<dc:creator><![CDATA[camerongregor]]></dc:creator>
<pubDate>Wed, 27 Sep 2017 05:36:43 +0000</pubDate>
<category><![CDATA[Software Development]]></category>
<category><![CDATA[embeddedexperience]]></category>
<category><![CDATA[FP9]]></category>
<category><![CDATA[notes]]></category>
<guid isPermaLink="false">http://camerongregor.com/?p=535</guid>
<description><![CDATA[Embedded Experiences are no longer rendering after upgrading to FP9. I suspect it is something to do with the changes to Embedded Browser, either removing it or upgrading it or something else. I haven’t...]]></description>
<content:encoded><![CDATA[<p>Embedded Experiences are no longer rendering after upgrading to FP9.</p>
<p>I suspect it is something to do with the changes to Embedded Browser, either removing it or upgrading it or something else.</p>
<p>I haven’t had a chance to investigate fully. Perhaps there may be a workaround, or perhaps this is specific to the way we are using embedded experiences.</p>
<p>We use embedded experiences quite a bit at our company so we have rolled back for now. If I figure anything out I will post an update but just be aware before upgrading if you are using embedded experiences to test FP9 in your situation before rolling out.</p>
<p>The trace log shows the following stack traces.</p>
<p><img loading="lazy" decoding="async" class="aligncenter size-full wp-image-541" src="https://camerongregor.com/wp-content/uploads/2017/09/EmbeddedExperienceBug1.png" alt="" width="798" height="573" srcset="https://camerongregor.com/wp-content/uploads/2017/09/EmbeddedExperienceBug1.png 798w, https://camerongregor.com/wp-content/uploads/2017/09/EmbeddedExperienceBug1-300x215.png 300w, https://camerongregor.com/wp-content/uploads/2017/09/EmbeddedExperienceBug1-768x551.png 768w" sizes="auto, (max-width: 798px) 100vw, 798px" /></p>
<p><img loading="lazy" decoding="async" class="aligncenter size-full wp-image-542" src="https://camerongregor.com/wp-content/uploads/2017/09/EmbeddedExperienceBug2.png" alt="" width="786" height="675" srcset="https://camerongregor.com/wp-content/uploads/2017/09/EmbeddedExperienceBug2.png 786w, https://camerongregor.com/wp-content/uploads/2017/09/EmbeddedExperienceBug2-300x258.png 300w, https://camerongregor.com/wp-content/uploads/2017/09/EmbeddedExperienceBug2-768x660.png 768w" sizes="auto, (max-width: 786px) 100vw, 786px" /></p>
]]></content:encoded>
<wfw:commentRss>https://camerongregor.com/2017/09/27/embedded-experiences-not-rendering-after-ibm-notes-fp9/feed/</wfw:commentRss>
<slash:comments>2</slash:comments>
</item>
<item>
<title>BuildXPages Deployment Automation Tools – v1.0.0</title>
<link>https://camerongregor.com/2017/09/21/buildxpages-deployment-automation-tools-v1-0-0/</link>
<comments>https://camerongregor.com/2017/09/21/buildxpages-deployment-automation-tools-v1-0-0/#comments</comments>
<dc:creator><![CDATA[camerongregor]]></dc:creator>
<pubDate>Thu, 21 Sep 2017 12:47:18 +0000</pubDate>
<category><![CDATA[Software Development]]></category>
<category><![CDATA[ant]]></category>
<category><![CDATA[deployment]]></category>
<category><![CDATA[designer]]></category>
<category><![CDATA[headless]]></category>
<category><![CDATA[plugin]]></category>
<guid isPermaLink="false">http://camerongregor.com/?p=529</guid>
<description><![CDATA[After quite a bit of work I have finally published the first release of BuildXPages! What is BuildXPages? BuildXPages is a project which is useful if you are interested in Automating tasks that are...]]></description>
<content:encoded><![CDATA[<p>After quite a bit of work I have finally published the <a href="https://github.com/camac/BuildXPages/releases/tag/v1.0.0" target="_blank" rel="noopener">first release of BuildXPages!</a></p>
<h2>What is BuildXPages?</h2>
<div>
<p>BuildXPages is a project which is useful if you are interested in Automating tasks that are involved in building XPages.</p>
</div>
<div>
<p>For example you may be interested in automating some of these tasks:</p>
</div>
<div>
<ul>
<li>Building NSFs from an On-Disk Project</li>
<li>Building Plugins and Features</li>
<li>Refreshing NSF Designs</li>
<li>Setting Template Inheritance of NSFs</li>
<li>Start, Stop and Restart Http server</li>
<li>Uploading Plugins to an NSF Update Site</li>
<li>Deploy Plugins to Notes/Designer</li>
</ul>
</div>
<div>
<p>BuildXPages provides tools which can be used to achieve these tasks. The main artifact of the project is a collection of tasks that can be used in an Ant build script, but the project also includes a Plugin for Domino Designer to facilitate the building of NSFs.</p>
<h2>Project Background</h2>
</div>
<p>Automating the build and deployment of any software application is a great idea to be able achieve the same result every time. Information and tools for doing this in XPages is relatively scarce but you can find blog posts and some tools out there.</p>
<p>I have written a few years ago about the <a href="https://camerongregor.com/2014/08/09/build-system-for-xpages-and-osgi-plugins/" target="_blank" rel="noopener">build system for XPages</a> we use at my workplace, and have shared my slides from the <a href="https://camerongregor.com/2015/06/11/build-automation-for-xpages-presentation-slides-ausluginform-2015/" target="_blank" rel="noopener">talk I gave at AUSLUG 2015</a> so have a look at those for some background (be aware these posts are old and I may be doing some things differently now), I decided to gather the tools that I have developed to support this and make them available under it’s own Open Source Project.</p>
<p>If you look you will see the project has actually been on Github since 2014! But the project never had any good documentation and some of the parts where not very friendly to use.</p>
<p>Over the last couple of months I have been working away on bringing the project up to a standard where it can be understood and used by others. This includes things like:</p>
<ul>
<li>making information and error messages more useful and understandable</li>
<li>making task names and arguments more user-friendly</li>
<li>replacing my hard-coded options with configurable options</li>
<li>most of all .. writing documentation!</li>
</ul>
<h2>Documentation</h2>
<p>You can always find the latest Documentation on the <a href="http://camac.github.io/BuildXPages/" target="_blank" rel="noopener">Github Project’s Documentation website</a>. The documentation is also contained as PDF and HTML in the release download.</p>
<p>A large part of writing the documentation was trying to figure out how to explain getting your environment setup. Whilst I think I have everything covered there is a every chance I have missed something that may be different for your environment, so if you do try it out and it doesn’t work please don’t hesitate to contact me.</p>
<h3>Asciidoctor FTW!</h3>
<p>Documentation for the Project is written Asciidoctor using AsciidoctFX and generated using the ant-asciidoctorj plugin. The Asciidoctor ‘toolchain’ is awesome and deserves it’s own blog post / video but that will have to be another day. There is already plenty of information out there about it. If you are going to write any documentation I highly recommend having a look at using it and feel free to ask me about it here or in the <a href="https://xpages-slack.herokuapp.com/" target="_blank" rel="noopener">XPages Slack chat</a>. You can see the ‘recipe’ I use for generating the documentation by looking at the build.xml file in the root of the project, and you can see all the documentation ‘source’ files in the doc folder.</p>
<h3>Demonstration Tutorials</h3>
<p>The Project contains some demonstration tutorials which are designed to verify that you have everything set up and to demonstrate the very basic tasks.</p>
<h2>Feedback / Problems / Suggestions</h2>
<p>I you have any problems with these let me know and I can double check what may be the problem either in your setup, or maybe a bug in the project.</p>
<h2>A note about Build Tools</h2>
<p>Part of your automation process is deciding what ‘build tool’ you will use, which is usually program that helps coordinate a sequence of tasks that need to be done in order to build / deploy your project. This could be as simple as a batch file, or use a well developed system specifically for this purpose.</p>
<p>I chose Ant because of it’s simplicity and availability of good documentation. It was very easy for me to get up and running and there were many examples out there to follow. I had also tried to give maven a go but found it hard to find good documentation and after a few dead ends just gave up and returned to making progress with ant.</p>
<p>So with Ant as my choice I have written all these custom tasks to work with Ant. Ant is written in Java and so all the tasks are java based, Maven and Gradle are both Java based and I know there are other XPages developers who prefer maven or gradle. If anyone is interested in translating these tasks to be used as a plugin for gradle or maven plugin then I would be happy to help you adapt them and hopefully include them in this project so that others can also use.</p>
<p>Also: if you are using maven be aware there is already the <a href="https://guedebyte.blog/2016/03/26/building-nsf-using-the-maven-headlessdesigner-plugin-from-openntf/" target="_blank" rel="noopener">headlessdesigner-maven-plugin</a> which is written about on Christian Guedemann’s blog</p>
]]></content:encoded>
<wfw:commentRss>https://camerongregor.com/2017/09/21/buildxpages-deployment-automation-tools-v1-0-0/feed/</wfw:commentRss>
<slash:comments>1</slash:comments>
</item>
<item>
<title>Generating and Downloading Files Using XPages’ Persistence Service</title>
<link>https://camerongregor.com/2017/09/13/generating-and-downloading-files-using-xpages-persistence-service/</link>
<comments>https://camerongregor.com/2017/09/13/generating-and-downloading-files-using-xpages-persistence-service/#comments</comments>
<dc:creator><![CDATA[camerongregor]]></dc:creator>
<pubDate>Wed, 13 Sep 2017 11:42:23 +0000</pubDate>
<category><![CDATA[Software Development]]></category>
<category><![CDATA[download]]></category>
<category><![CDATA[file]]></category>
<category><![CDATA[persistence]]></category>
<category><![CDATA[xpages]]></category>
<guid isPermaLink="false">http://camerongregor.com/?p=517</guid>
<description><![CDATA[When developing an XPages application you may run into the need to generate files to be downloaded by the user. For example: Generating a PDF Report Exporting Data to Excel Spreadsheet Creating a Zip...]]></description>
<content:encoded><![CDATA[<p>When developing an XPages application you may run into the need to generate files to be downloaded by the user.</p>
<p>For example:</p>
<ul>
<li>Generating a PDF Report</li>
<li>Exporting Data to Excel Spreadsheet</li>
<li>Creating a Zip File of several attachments</li>
</ul>
<p>Achieving these tasks usually raises 2 main problems to solve;</p>
<ol>
<li>What temporary place can I use to generate the files?</li>
<li>How do I allow the user to download the generated files?</li>
</ol>
<p>For the temporary place problem, some common solution is to either generate the files to some ‘in memory’ representation or alternatively use the Java Temporary Directory. For the downloading problem, a common solution is to provide a button which calls a SSJS or Java bean method that hijacks the HttpResponse and writes out the appropriate headers and file contents.</p>
<p>These solutions work fine, however built in to the XPages framework is the ‘Persistence Service’ which is a mechanism for temporarily Persisting files that need to be available for download . Although this persistence service was built primarily for the standard operations of XPages, there is nothing stopping us from also enjoying the way it solves these 2 problems.</p>
<h2>Why does the Persistence Service exist?</h2>
<p>An advantage of Domino Databases is how easy it is to store Files and Images inside ‘Rich Text’ Fields. When XPages needs to display these Embedded Images or allow Downloads of files from within the Document, then the user’s web browser will need to make request for these embedded objects using some sort of URL.</p>
<p>Domino provides URLs which provide direct access to these embedded files and images, however these only provide access to the ‘saved’ version of a document as it exists inside the Notes Database. These URLs are of no help for the ‘No mans land’ period of time in which someone is creating a new document, or when editing an existing document and uploads a new File or Image but has not yet saved the document.</p>
<p>When you open up Domino Document via XPages, the persistence service extracts the Document’s embedded images to a temporary directory in the filesystem, and also uses this directory as a place for newly uploaded files/images to live until they are finally decided to be saved into the document inside the Notes Database. The files in this temporary directory are made available to the user via a special URL.</p>
<h2>How does the Persistence Service Work?</h2>
<p>The XPages server has a server-wide property which determines the <strong>Base Persistence Directory.</strong></p>
<p>The default value for this is <strong><DominoDataDirectory>/xsppers </strong> but you can configure it to be anywhere you like using the xsp property <strong>xsp.state.persistence.directory</strong></p>
<p>When a new XPages Application starts up, it is assigned an ‘Application Id’ which is really just a sequential number, so the first application that starts up gets number 1, next number 2 etc.</p>
<p>When a user starts using an Application they are given a ‘session id’ which is a bit more like a Universal ID.</p>
<blockquote><p>Side Note: I’m not sure how this technique goes with Anonymous access as I have only been programming for authenticated users.</p></blockquote>
<p>The persistence service uses these 3 pieces of information to determine a user’s persistence directory for that application.</p>
<p>So lets assume, my Base persistence directory is <strong>C:\Domino\Data\xsppers</strong>, and the application id is <strong>1</strong> and my session id is <strong>9847239OENTHUDO</strong> then we can find my persistence directory for that application in the file system at <strong><xsppers>/<applicationid>/<sessionid></strong> or in this example:</p>
<p>C:\Domino\Data\xsppers\1\9847239OENTHUDO\</p>
<p>Now here is the handy part, XPages has a resource provider which maps a special URL <strong>/xsp/.ibmmodres/persistence</strong> to the user’s persistence directory for that application.</p>
<p>So lets assume we create a folder ‘myfolder’ under my user persistence directory and put a file ‘myfile.pdf’ in that folder::</p>
<p>C:\Domino\Data\xsppers\1\9847239OENTHUDO\myfolder\myfile.pdf</p>
<p>This file would then be available via the url:</p>
<p>https://yourserver/yourapp.nsf<strong>/xsp/.ibmmodres/persistence/myfolder/myfile.pdf</strong></p>
<p>If another user tried to use the same url it would not work because they would have a different session id, and would be mapping to a different directory</p>
<h3>Cleaning up files</h3>
<p>Another handy feature of the persistence service is that it cleans up after itself! So you can stress less about generating a massive zip file knowing that it won’t be left around for too long and end up filling up your server’s hard drive.</p>
<p>After a user session for an application times-out then the user’s persistence directory and all its contents will be deleted.</p>
<p>After the entire application times-out, then the application’s persistence directory e.g. <strong>C:\Domino\Data\xsppers\1</strong> and all it’s contents will be deleted.</p>
<h2>So how do I actually use this?</h2>
<p>So your main 2 weapons here are :</p>
<ul>
<li>You have a User’s Persistence Directory in which you can generate files (which will also clean up after itself)</li>
<li>You have a handy URL available for the user to access the User’s Persistence Directory</li>
</ul>
<p>To generate files into the User’s Persistence directory we can either just figure out the path of the user’s application directory and write files straight to it, or alternatively we can also ask the applications PersistenceService to do it for us.</p>
<h3>Doing it ourselves</h3>
<p>To do it ourselves we need to be able to figure out the User’s persistence directory. I have included a Utility class at the bottom of this post with some handy methods. Let’s have a look at 3 of the methods:</p><pre class="urvanov-syntax-highlighter-plain-tag">public static String getSessionId() {
return SessionUtil.getSessionId(FacesContext.getCurrentInstance());
}
public static String getBasePersistenceFolder() {
ApplicationEx ex = ApplicationEx.getInstance();
String propDir = ex.getProperty("xsp.state.persistence.directory", null);
if (StringUtil.isEmpty(propDir)) {
File defaultDir = SystemUtil.DEFAULT_PERSISTENCEDIR;
return defaultDir.getAbsolutePath();
} else {
return propDir;
}
}
public static String getPersistenceSessionFolder() {
ApplicationEx ex = ApplicationEx.getInstance();
return getBasePersistenceFolder() + File.separator + ex.getApplicationId() + File.separator + getSessionId();
}</pre><p>So the main one here is <strong>getPersistenceSessionFolder</strong> and you can see it is just building up that folder location using the 3 parts mentioned earlier. We get the BasePersistenceFolder (xsppers) by querying the xsp property, and if it has not been set using an xsp property, we fall back to the default for the current platform.<br />
We get the session id using the com.ibm.xsp.util.SessionUtil class</p>
<p>Once you have access to this folder, you can create folders and files directly in here. The convention of the PersistenceService is to place your generated files within a subfolder of the User Persistence Directory e.g. <strong><userpersistencedir/<somefolder>/<somefile></strong> instead of <userpersistencedir>/<somefile>. If you do not follow this convention then your download url will not work as it expects exactly this structure.</p>
<h3>Doing it using the PersistenceService</h3>
<p>An XPages Application has a PersistenceService which can be used for creating a ‘PersistedContent’ object. This persisted content object can be used to get an OutputStream to write to, and also the corresponding java ‘File’ object for that PersistedContent.</p>
<p>The utility class has a method which shows how to get the persistence service. Note the persistence service is marked as Deprecated but you can still use it.</p><pre class="urvanov-syntax-highlighter-plain-tag">@SuppressWarnings("deprecation")
public static PersistenceService getPersistenceService() {
ApplicationEx ex = ApplicationEx.getInstance();
return ex.getPersistenceService();
}</pre><p>Also the utility class has a method to help create PersistedContent. You can see all the Persistence service methods follow the convention of using a folder and a file name.</p><pre class="urvanov-syntax-highlighter-plain-tag">public static PersistedContent createPersistedContent(String folderName, String fileName)
throws IOException, NotesException {
String sessionId = getSessionId();
PersistenceService ps = getPersistenceService();
PersistedContent pd = ps.createPersistedContent(sessionId, folderName, fileName);
return pd;
}</pre><p>And you can see how this is used in the method persistInputStream</p>
<blockquote><p>Note: I am using Apache Commons IOUtils for copying a stream but if you cannot use this then you can replace with another stream copying algorithm or perhaps using IBM Commons StreamUtil.copy(inputstream, outputstream). I am using Apache commons because it automatically buffers the streams so that if you are dealing with large files you don’t use too much memory</p></blockquote>
<p></p><pre class="urvanov-syntax-highlighter-plain-tag">public static File persistInputStream(InputStream is, String folderName, String fileName)
throws NotesException, IOException {
String sessionId = getSessionId();
return persistInputStream(is, sessionId, folderName, fileName);
}
public static File persistInputStream(InputStream is, String sessionId, String folderName, String fileName)
throws IOException {
PersistenceService ps = getPersistenceService();
PersistedContent pd = ps.createPersistedContent(sessionId, folderName, fileName);
OutputStream fos = pd.getOutputStream();
org.apache.commons.io.IOUtils.copy(is, fos);
fos.close();
return pd.getContentAsFile();
}</pre><p></p>
<h2>Providing a Link to download the file</h2>
<p>This is the easy part. all you have to do is use the standard link control!</p><pre class="urvanov-syntax-highlighter-plain-tag"><xp:link escape="true" text="Download Me" id="link1"
value="/xsp/.ibmmodres/persistence/myfolder/myfile.pdf"></xp:link></pre><p></p>
<h2>Summary</h2>
<p>So I have found this way to be a very useful way to manage generating and downloading files, I hope it has been interesting / useful for you and if you have any questions or have found a mistake in this post please leave a comment and I promise to respond!</p>
<h2>Utility Class</h2>
<p>As promised here is the utility class which is also <a href="https://openntf.org/XSnippets.nsf/snippet.xsp?id=persistence-service-utility-class" target="_blank" rel="noopener">available as an XSnippet</a></p><pre class="urvanov-syntax-highlighter-plain-tag">package com.jord.xsp.file;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.util.Arrays;
import javax.faces.context.FacesContext;
import com.ibm.commons.util.StringUtil;
import com.ibm.xsp.application.ApplicationEx;
import com.ibm.xsp.component.UIFileuploadEx.UploadedFile;
import com.ibm.xsp.http.IUploadedFile;
import com.ibm.xsp.persistence.PersistedContent;
import com.ibm.xsp.persistence.PersistenceService;
import com.ibm.xsp.util.SessionUtil;
import com.ibm.xsp.util.SystemUtil;
import lotus.domino.NotesException;
public class PersistenceUtil {
@SuppressWarnings("deprecation")
public static PersistenceService getPersistenceService() {
ApplicationEx ex = ApplicationEx.getInstance();
return ex.getPersistenceService();
}
public static String getSessionId() {
return SessionUtil.getSessionId(FacesContext.getCurrentInstance());
}
public static String getBasePersistenceFolder() {
ApplicationEx ex = ApplicationEx.getInstance();
String propDir = ex.getProperty("xsp.state.persistence.directory", null);
if (StringUtil.isEmpty(propDir)) {
File defaultDir = SystemUtil.DEFAULT_PERSISTENCEDIR;
return defaultDir.getAbsolutePath();
} else {
return propDir;
}
}
public static String getPersistenceSessionFolder() {
ApplicationEx ex = ApplicationEx.getInstance();
return getBasePersistenceFolder() + File.separator + ex.getApplicationId() + File.separator + getSessionId();
}
public static File createdPath(File file) {
File dir = file;
if (!file.isDirectory())
dir = file.getParentFile();
dir.mkdirs();
return file;
}
public static File getAvailableFile(final File file) {
if (file.exists()) {
File testFile;
int counter = 0;
String extension = "";
String fileName = file.getAbsolutePath();
int dotIndex = fileName.lastIndexOf('.');
if (dotIndex > 0) {
extension = fileName.substring(dotIndex);
fileName = fileName.substring(0, dotIndex);
}
while ((testFile = new File(fileName + "_" + counter + extension)).exists())
counter++;
return testFile;
} else {
return file;
}
}
/**
* Returns a safe version of a given name Currently just prevents moving
* directories etc
*
* @return
*/
public static String safeFileName(String fileName) {
return fileName.replace("/", "").replace("\\", "");
}
public static File persistFile(File file, String folderName, String fileName) throws IOException, NotesException {
String sessionId = getSessionId();
return persistFile(file, sessionId, folderName, fileName);
}
public static File persistFile(File file, String sessionId, String folderName, String fileName) throws IOException {
PersistenceService ps = getPersistenceService();
PersistedContent pd = ps.createPersistedContent(sessionId, folderName, fileName);
FileInputStream fis = new FileInputStream(file);
OutputStream fos = pd.getOutputStream();
org.apache.commons.io.IOUtils.copy(fis, fos);
fos.close();
fis.close();
return pd.getContentAsFile();
}
public static File persistInputStream(InputStream is, String folderName, String fileName)
throws NotesException, IOException {
String sessionId = getSessionId();
return persistInputStream(is, sessionId, folderName, fileName);
}
public static File persistInputStream(InputStream is, String sessionId, String folderName, String fileName)
throws IOException {
PersistenceService ps = getPersistenceService();
PersistedContent pd = ps.createPersistedContent(sessionId, folderName, fileName);
OutputStream fos = pd.getOutputStream();
org.apache.commons.io.IOUtils.copy(is, fos);
fos.close();
return pd.getContentAsFile();
}
public static PersistedContent createPersistedContent(String folderName, String fileName)
throws IOException, NotesException {
String sessionId = getSessionId();
PersistenceService ps = getPersistenceService();
PersistedContent pd = ps.createPersistedContent(sessionId, folderName, fileName);
return pd;
}
public static PersistedContent[] getPersistedContents(String folderName) throws IOException {
String sessionId = getSessionId();
PersistenceService ps = getPersistenceService();
PersistedContent[] pcs = ps.getPersistedContents(sessionId, folderName);
return pcs;
}
public static PersistedContent getPersistedContent(String folderName, String fileName)
throws IOException, NotesException {
String sessionId = getSessionId();
PersistenceService ps = getPersistenceService();
PersistedContent pd = ps.getPersistedContent(sessionId, folderName, fileName);
return pd;
}
public static File persistUploadedFile(UploadedFile upload, String folderName) throws IOException, NotesException {
String sessionId = getSessionId();
return persistUploadedFile(upload, sessionId, folderName);
}
public static File persistUploadedFile(UploadedFile upload, String sessionId, String folderName)
throws IOException {
File file = upload.getUploadedFile().getServerFile();
String fileName = upload.getUploadedFile().getClientFileName();
return persistFile(file, sessionId, folderName, fileName);
}
public static File persistUploadedFile(IUploadedFile upload, String sessionId, String folderName)
throws IOException {
File file = upload.getServerFile();
String fileName = upload.getClientFileName();
return persistFile(file, sessionId, folderName, fileName);
}
public static void deletePersistenceFolder(String folderName) throws IOException {
String sessionId = getSessionId();
String[] folders = getPersistenceService().getFolders(sessionId);
if (!Arrays.asList(folders).contains(folderName)) {
return;
}
getPersistenceService().deleteFolder(sessionId, folderName);
}
}</pre><p> </p>
<p> </p>
]]></content:encoded>
<wfw:commentRss>https://camerongregor.com/2017/09/13/generating-and-downloading-files-using-xpages-persistence-service/feed/</wfw:commentRss>
<slash:comments>8</slash:comments>
</item>
<item>
<title>Bundle Inspector – Diagnosing XPages Plugin Resolution problems</title>
<link>https://camerongregor.com/2017/08/02/bundle-inspector-diagnosing-xpages-plugin-resolution-problems/</link>
<comments>https://camerongregor.com/2017/08/02/bundle-inspector-diagnosing-xpages-plugin-resolution-problems/#respond</comments>
<dc:creator><![CDATA[camerongregor]]></dc:creator>
<pubDate>Wed, 02 Aug 2017 12:26:56 +0000</pubDate>
<category><![CDATA[Software Development]]></category>
<category><![CDATA[debug]]></category>
<category><![CDATA[designer]]></category>
<category><![CDATA[osgi]]></category>
<category><![CDATA[plugin]]></category>
<category><![CDATA[xpages]]></category>
<guid isPermaLink="false">http://camerongregor.com/?p=492</guid>
<description><![CDATA[A common task with XPages development is to installing some plugins that provide additional functionality. If you do any plugin development of your own, you end up doing this task a lot more as...]]></description>
<content:encoded><![CDATA[<p>A common task with XPages development is to installing some plugins that provide additional functionality.<br />
If you do any plugin development of your own, you end up doing this task a lot more as you create new plugins, install new versions etc.</p>
<p>It is also common to encounter some problems after installing plugins to your Domino Server! Maybe your expected plugin doesn’t load up at all? maybe the wrong version is loading?</p>
<p>The standard method of diagnosing these problems is to use the OSGi console via the Domino Console (for the server) or by launching Designer with the -RPARAMS -console arguments.</p>
<p>However to provide a more user friendly way to diagnose problems, I have written a plugin for both the Domino Server and Domino Designer which provides a little more user friendly experience of diagnosing plugin problems than the standard method of using the OSGi console.</p>
<blockquote><p>Disclaimer: Throughout this I use the words Plugin and Bundle interchangeably. Technically in OSGi they are called Bundles, but Eclipse calls them plugins, so there is a weird overlap. But for the purpose of this they are the same thing.</p></blockquote>
<p>The Some common root causes are:</p>
<ul>
<li><strong>You plugin is not loading at all<br />
</strong>Maybe your Server/Domino designer is not actually configured to load plugins from where you think it is?</li>
<li><strong>Some extra versions of the plugin are being loaded from somewhere you didn’t realise<br />
</strong>Plugins can be loaded from more than one location, for example NSF Update Sites, File system, Eclipse Workspace Projects (when developing on a local Domino Server). So it is often very important to inspect exactly what plugins are loaded and where they are loading from.</li>
<li><strong>You plugin is loaded but some of it’s dependencies cannot be resolved<br />
</strong>Even when plugins do load correctly, they may have problems resolving their dependencies, and then they end up with the state ‘INSTALLED’ but not resolved, and effectively useless.</li>
</ul>
<h2>How to Install Bundle Inspector</h2>
<p>Download the <a href="https://github.com/camac/GregorbyteXspLibrary/releases">latest release of GregorbyteXspLibrary</a>, and Install the ‘Bundles Inspector’ feature to your Domino Server and Domino Designer.<br />
I am assuming if you have a need for the Bundle Inspector, then you must know how to install plugins to Domino Server or Designer, but if you don’t know then let me know via a comment and I will detail the process.</p>
<p>You can also install the other main feature of GregorbyteXspLibrary if you would like some other XPages controls that I have blogged about before.</p>
<h2>Using in Domino Designer</h2>
<p>After you have installed and restarted, you should see 2 new buttons in the toolbar / menu bar</p>
<p><img loading="lazy" decoding="async" class="aligncenter size-full wp-image-501" src="https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_ToolbarIcons.jpg" alt="" width="756" height="140" srcset="https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_ToolbarIcons.jpg 756w, https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_ToolbarIcons-300x56.jpg 300w" sizes="auto, (max-width: 756px) 100vw, 756px" /></p>
<p>The Puzzle piece button will launch the Bundle Inspector. You can then filter for a particular plugin that you are having a problem with.</p>
<p><img loading="lazy" decoding="async" class="aligncenter size-full wp-image-502" src="https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_Dialog.jpg" alt="" width="647" height="493" srcset="https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_Dialog.jpg 647w, https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_Dialog-300x229.jpg 300w" sizes="auto, (max-width: 647px) 100vw, 647px" /></p>
<p>Note that by default it is a ‘starts with’ match, but you can use wildcards such as * if you would like to do a ‘contains’ match as I am here for anything with ‘gregor’ in it.</p>
<ul>
<li>Plugins are sorted alphabetically ascending, and then descending by version, so the most recent version should be at the top.</li>
<li>Plugins that are Resolved will show the green puzzle piece icon, plugins that have not resolved will show the grey puzzle piece icon.</li>
<li>Beneath the list you can view the ‘location’ of the selected plugin e.g. where did the plugin load from.</li>
</ul>
<p>If you would like to see the reason why a plugin has not resolved, select it from the list and click OK, as you can see here, the libphonenumber is plugin is not loaded because there is a newer version of it loaded.</p>
<p><img loading="lazy" decoding="async" class="aligncenter size-full wp-image-503" src="https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_DialogDiagnosis.jpg" alt="" width="518" height="141" srcset="https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_DialogDiagnosis.jpg 518w, https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_DialogDiagnosis-300x82.jpg 300w" sizes="auto, (max-width: 518px) 100vw, 518px" /></p>
<h2>Using on a Domino Server</h2>
<p>The way Bundles Inspector works on the server is, by providing a ‘bundlesBean’ from an XspLibrary. To use this bean, you need to setup a particular page in an XPages Database.</p>
<p>For these steps, It is assumed that you have the Bundles Inspector plugin installed to both your server and your Domino Designer.</p>
<h3>Setup the XPage</h3>
<p>It is a good idea to put the BundleInspector XPage in a database that does not depend on any other Xpages libraries. This way, if there is a problem with any of the Xpages libraries that you usually use, it will not affect the BundlesInspector Xpage. You can also put access control on this database if you want to restrict who can use the Bundles Inspector.</p>
<p>Go to the XPages Properties page and enable the ‘com.gregorbyte.xsp.bundlesinspector’ library</p>
<p><img loading="lazy" decoding="async" class="aligncenter size-full wp-image-505" src="https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_EnableLibrary.jpg" alt="" width="412" height="120" srcset="https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_EnableLibrary.jpg 412w, https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_EnableLibrary-300x87.jpg 300w" sizes="auto, (max-width: 412px) 100vw, 412px" />Then Create a new xpage in your desired database</p>
<p>Select the ‘Copy Sample XPage to Clipboard’ button.</p>
<p><img loading="lazy" decoding="async" class="aligncenter size-full wp-image-501" src="https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_ToolbarIcons.jpg" alt="" width="756" height="140" srcset="https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_ToolbarIcons.jpg 756w, https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_ToolbarIcons-300x56.jpg 300w" sizes="auto, (max-width: 756px) 100vw, 756px" /></p>
<p>You should then see the confirmation message</p>
<p><img loading="lazy" decoding="async" class="aligncenter size-full wp-image-506" src="https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_SampleCopied.jpg" alt="" width="518" height="141" srcset="https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_SampleCopied.jpg 518w, https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_SampleCopied-300x82.jpg 300w" sizes="auto, (max-width: 518px) 100vw, 518px" /></p>
<p>Go to the Source tab of your XPage and paste the contents over the top</p>
<p><img loading="lazy" decoding="async" class="aligncenter size-full wp-image-507" src="https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_Pasted.jpg" alt="" width="597" height="373" srcset="https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_Pasted.jpg 597w, https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_Pasted-300x187.jpg 300w" sizes="auto, (max-width: 597px) 100vw, 597px" /></p>
<p>Feel free to customise this page to however you like, you could change the colours layout etc. The page is merely provided as a starting point.</p>
<h3>Using the Bundle Inspector for the Server</h3>
<p>Navigate to your XPage in your web browser. (The XPage should be on the Server that you are diagnosing bundles for)</p>
<ul>
<li>You can filter for the plugin your are interested in using the filter</li>
<li>There is also an example of including a quick link to a common search e.g. extlib</li>
<li>The bundle location is shown in the table so you can see where the plugin is loading from</li>
<li>The bundle state is shown on the Right, green means resolved, red means not resolved</li>
</ul>
<p><img loading="lazy" decoding="async" class="aligncenter size-full wp-image-508" src="https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_Web.jpg" alt="" width="1171" height="275" srcset="https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_Web.jpg 1171w, https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_Web-300x70.jpg 300w, https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_Web-768x180.jpg 768w, https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_Web-1024x240.jpg 1024w" sizes="auto, (max-width: 1171px) 100vw, 1171px" /></p>
<p>If a plugin is not resolved you can click the ‘Diagnose’ button and the diagnosis will show beneath the row of the plugin</p>
<p><img loading="lazy" decoding="async" class="aligncenter size-full wp-image-509" src="https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_WebDiagnose.jpg" alt="" width="1237" height="99" srcset="https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_WebDiagnose.jpg 1237w, https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_WebDiagnose-300x24.jpg 300w, https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_WebDiagnose-768x61.jpg 768w, https://camerongregor.com/wp-content/uploads/2017/08/BundleInspector_WebDiagnose-1024x82.jpg 1024w" sizes="auto, (max-width: 1237px) 100vw, 1237px" /></p>
<p> </p>
<h2>Conclusion</h2>
<p>So there you have it, the aim of the Bundle Inspector is to give a better experience for diagnosing plugin problems.</p>
<p>Let me know what you think, would this help you? any improvements? Can you think of a better name the Bundle Inspector? I don’t really like it but that is what I went with…</p>
<p>If you have any problems installing let me know!</p>
]]></content:encoded>
<wfw:commentRss>https://camerongregor.com/2017/08/02/bundle-inspector-diagnosing-xpages-plugin-resolution-problems/feed/</wfw:commentRss>
<slash:comments>0</slash:comments>
</item>
<item>
<title>Swiper Official Version 2 Release</title>
<link>https://camerongregor.com/2017/07/21/swiper-official-version-2-release/</link>
<comments>https://camerongregor.com/2017/07/21/swiper-official-version-2-release/#comments</comments>
<dc:creator><![CDATA[camerongregor]]></dc:creator>
<pubDate>Fri, 21 Jul 2017 13:52:52 +0000</pubDate>
<category><![CDATA[Software Development]]></category>
<category><![CDATA[designer]]></category>
<category><![CDATA[sourcecontrol]]></category>
<category><![CDATA[swiper]]></category>
<guid isPermaLink="false">http://camerongregor.com/?p=495</guid>
<description><![CDATA[So I have finally posted Swiper version 2 to OpenNTF! (and Github of course) Swiper OpenNTF Project Page Swiper Github Releases The latest version is 2.0.1, and is the same as 2.0.0beta but with...]]></description>
<content:encoded><![CDATA[<p>So I have finally posted Swiper version 2 to OpenNTF! (and Github of course)</p>
<ul>
<li><a href="https://www.openntf.org/main.nsf/project.xsp?r=project/Swiper/summary" target="_blank" rel="noopener">Swiper OpenNTF Project Page</a></li>
<li><a href="https://github.com/camac/Swiper/releases" target="_blank" rel="noopener">Swiper Github Releases</a></li>
</ul>
<p>The latest version is 2.0.1, and is the same as 2.0.0beta but with a bug fix for the toolbar buttons.</p>
<p>Swiper 2.0.0 beta has been available on the Github project site for a few months, and the core functionality of it works as planned. There was however a bug within the shortcut buttons that I added to the menu bar, and this is a bit of a nasty bug that can cause deletion of design elements.</p>
<p>Why has it taken me so long to fix this bug and prepare the final release? to be perfectly honest with you, I just couldn’t muster up the motivation to get this done. My development process for Swiper involves launching Domino Designer from my eclipse IDE. After I upgraded my IBM Notes to 9.0.1 FP8 my designer launch settings no longer worked. By this time frankly my tank was empty. I knew re-configuring the launch settings was destined to be a fiddly encounter, and entire nights can be wasted with no fruitful results. All my extra-curricular projects are done in my spare-time, and this usually means between 8:30pm-11:30pm, and after a full day of problem-solving at work, I just couldn’t muster up the courage to face a task that had a high-probability of failure. I was satisfied that the beta version was available, and the project would have to wait until I felt the motivation again.</p>
<p>Well this week, I noticed a tweet from Per Henrik Lausten about ‘Open Source Friday’.</p>
<blockquote class="twitter-tweet" data-lang="en">
<p dir="ltr" lang="en">Contribute to open source on <a href="https://twitter.com/hashtag/OpenSourceFriday?src=hash">#OpenSourceFriday</a> (by <a href="https://twitter.com/github">@github</a>):<a href="https://t.co/EeHpi9UphG">https://t.co/EeHpi9UphG</a></p>
<p>— Per Henrik Lausten (@perlausten) <a href="https://twitter.com/perlausten/status/887551759731871744">July 19, 2017</a></p></blockquote>
<p><script async src="//platform.twitter.com/widgets.js" charset="utf-8"></script></p>
<p>It made me think a little about Swiper, and I thought I would give it another go. I also felt validated for taking my little break from the project when I read the ‘<a href="https://opensource.guide/best-practices/#its-okay-to-hit-pause" target="_blank" rel="noopener">It’s ok to hit pause</a>‘ section of the Open Source best practices for Maintainers <img src="https://s.w.org/images/core/emoji/16.0.1/72x72/1f642.png" alt="🙂" class="wp-smiley" style="height: 1em; max-height: 1em;" /></p>
<p>So I needed to fix my Designer Launch settings, I really don’t know how to do this by myself and have relied on others for this information. Previously it was Mikkel Heisterberg, and then for the past few years I turned to <a href="https://twitter.com/RalfMPetter" target="_blank" rel="noopener">Ralf M Petter</a>. I was very sad to see recent news that Ralf passed away from a cancer related illness. I would just like to express my sympathy to Ralf’s family for their loss. Ralf’s blog posts on setting up Eclipse to launch Notes and Designer were extremely helpful to me whilst developing Swiper and other plugins that I have made for Notes and Designer. Ralf was always quick to respond to my questions and his expertise will be missed.</p>
<p>A couple of months ago I had asked Ralf if he knew how to launch Designer FP8 from eclipse and surely enough within a day he had provided me with updated instructions, I had also asked Gary Marjoram @ IBM for some advice and armed with all this advice I was able to get FP8 launching again after a couple of hours tweaking.</p>
<p>Tonight I was able to debug the toolbar buttons again and therefore I was able to release the official version 2.</p>
<p>Let me know if you have any problems or suggestions!</p>
]]></content:encoded>
<wfw:commentRss>https://camerongregor.com/2017/07/21/swiper-official-version-2-release/feed/</wfw:commentRss>
<slash:comments>5</slash:comments>
</item>
<item>
<title>Auto-width Bootstrap Column XPages Controls</title>
<link>https://camerongregor.com/2017/03/22/auto-width-bootstrap-column-xpages-controls/</link>
<comments>https://camerongregor.com/2017/03/22/auto-width-bootstrap-column-xpages-controls/#respond</comments>
<dc:creator><![CDATA[camerongregor]]></dc:creator>
<pubDate>Tue, 21 Mar 2017 13:13:18 +0000</pubDate>
<category><![CDATA[Software Development]]></category>
<category><![CDATA[bootstrap]]></category>
<category><![CDATA[control]]></category>
<category><![CDATA[xpages]]></category>
<guid isPermaLink="false">http://camerongregor.com/?p=479</guid>
<description><![CDATA[I’ve been stuck working with OneUI Version 3 for the past couple of years, due to a regretful decision made at the beginning of my major project. OneUI was better than nothing but very frustrating...]]></description>
<content:encoded><![CDATA[<p>I’ve been stuck working with <a href="http://infolib.lotus.com/resources/oneui/3.0/docPublic/examples.htm?content=2colGrid.htm" target="_blank">OneUI Version 3</a> for the past couple of years, due to a regretful decision made at the beginning of my major project. OneUI was better than nothing but very frustrating at times.<br />
Finally, I have moved on to my next project and I am now using bootstrap (version 3)</p>
<p>A common task when laying out a page using bootstrap is to divide sections up into rows and columns, and use the appropriate css styles to do so. I’m going to assume you are familiar with bootstrap’s grid system, and if not I recommend you <a href="http://getbootstrap.com/css/#grid" target="_blank">hearing it from the horse’s mouth</a> instead of me trying to explain it again.</p>
<p>I like creating XspLibrary controls, so I decided I would go ahead and create row and column controls. It could be seen as a little overkill and you can easily live without it, but I see the <a href="http://showcase.bootsfaces.net/layout/basic.jsf" target="_blank">Bootfaces project</a> did it, so I took inspiration from their container, row and column controls.</p>
<p>I then also added the ability for the columns to have their widths automatically distributed. Auto-widths is a feature that is <a href="https://v4-alpha.getbootstrap.com/layout/grid/#auto-layout-columns" target="_blank">coming with bootstrap 4</a>, but for now we need to do it ourselves!</p>
<h2>Example XPages Markup</h2>
<p>The following examples end up looking like this: (<strong>Note:</strong> the blue border is just added for demo purposes to show the outline of the columns)</p>
<p><img loading="lazy" decoding="async" class="aligncenter size-full wp-image-481" src="https://camerongregor.com/wp-content/uploads/2017/03/BootstrapDemoPage.jpg" alt="" width="981" height="140" srcset="https://camerongregor.com/wp-content/uploads/2017/03/BootstrapDemoPage.jpg 981w, https://camerongregor.com/wp-content/uploads/2017/03/BootstrapDemoPage-300x43.jpg 300w, https://camerongregor.com/wp-content/uploads/2017/03/BootstrapDemoPage-768x110.jpg 768w" sizes="auto, (max-width: 981px) 100vw, 981px" /></p>
<p>Here is a simple example of 2-column layout with equal width columns</p><pre class="urvanov-syntax-highlighter-plain-tag"><gb:row id="row1">
<gb:column id="column1">These two columns</gb:column>
<gb:column id="column2">are Equal Width</gb:column>
</gb:row></pre><p>Notice above how I haven’t specified any column widths? The columns will all become equal width, so in the example above, they will both be 6 columns wide (12 divided by 2 columns).</p>
<p>If there have 3 columns, then they would each automatically be 4 columns wide (12 divided by 3 columns).</p><pre class="urvanov-syntax-highlighter-plain-tag"><gb:row id="row5">
<gb:column id="column3">These three </gb:column>
<gb:column id="column10">columns are also</gb:column>
<gb:column id="column11">Equal Width</gb:column>
</gb:row></pre><p></p>
<h3>Specifying a column width</h3>
<p>If you want a column to be a specifc width, you use the span property to specify the width for that column, the remaining columns will receive their fair share of whatever is left over.</p><pre class="urvanov-syntax-highlighter-plain-tag"><gb:row id="row2">
<gb:column id="column4">These columns have</gb:column>
<gb:column id="column5" span="6">the middle with span="6"</gb:column>
<gb:column id="column6">and the outer columns are auto width</gb:column>
</gb:row></pre><p></p>
<h3>Specifying alignment</h3>
<p>I’ve also added an align property so I can quickly align the contents.</p><pre class="urvanov-syntax-highlighter-plain-tag"><gb:row id="row4">
<gb:column id="column7" align="left">This is align left (default)</gb:column>
<gb:column id="column8" align="center">This is align middle</gb:column>
<gb:column id="column9" align="right">This is align left</gb:column>
</gb:row></pre><p></p>
<h3>Full Row column</h3>
<p>If you want to use the entire row, then you can just leave the column definition out entirely and the contents will automatically be wrapped in a col-md-12 column</p><pre class="urvanov-syntax-highlighter-plain-tag"><gb:row id="row3">
If you want a Full Row, then just don't include columns
</gb:row></pre><p></p>
<h3>Using style and styleClass</h3>
<p>You can still use style and styleClass properties as you normally would.</p>
<h3>Container Control</h3>
<p>If you are using one of the extension library bootstrap applicationLayouts, then you won’t need to use the container control because it is already covered by that. But, If you are rolling your own you can use the container layout which has a fluid property to control whether is is a bootstrap fluid layout or not.</p><pre class="urvanov-syntax-highlighter-plain-tag"><gb:container id="container1" fluid="true">
<!-- rows, columns, your content -->
</gb:container></pre><p></p>
<h2>Overview of creating the controls</h2>
<p>Here is what we are going to do:</p>
<ol>
<li>Create the Container UIComponent</li>
<li>Create the Column UIComponent</li>
<li>Create the Row UIComponent</li>
<li>Provide Control Definitions via an xsp-config file so that designer knows how to use them</li>
<li>Program in the auto-width algorithm</li>
</ol>
<p>Notice we are not going to create a renderer. We will be extending the panel control, and the existing panel renderer will be used automatically.</p>
<p>For each component we are just going to <em>extend </em>the existing <strong><xp:panel></strong> control to re-use all the existing functionality, and just add a few extra properties.</p>
<h3>Creating the Container Component</h3>
<p>The container component is simply an extension of the panel control (UIPanelEx) with an additional property ‘fluid’. We add the getters and setters for the fluid property, and make sure it is included in the restore and save state operations.</p>
<p>We also override the getStyleClass method so that it will always concatenate ‘container’ or ‘container-fluid’ depending on the fluid property.</p><pre class="urvanov-syntax-highlighter-plain-tag">package com.gregorbyte.xsp.component.bootstrap;
import javax.faces.context.FacesContext;
import javax.faces.el.ValueBinding;
import com.gregorbyte.xsp.util.GregorbyteUtil;
import com.ibm.xsp.component.UIPanelEx;
public class UIContainer extends UIPanelEx {
public Boolean fluid;
public UIContainer() {
}
@Override
public String getStyleClass() {
String parent = super.getStyleClass();
if (isFluid()) {
return GregorbyteUtil.concatStyleClasses("container-fluid", parent);
} else {
return GregorbyteUtil.concatStyleClasses("container", parent);
}
}
public Boolean isFluid() {
if (this.fluid != null) {
return this.fluid;
}
ValueBinding vb = getValueBinding("fluid");
if (vb != null) {
return (Boolean) vb.getValue(getFacesContext());
}
return false;
}
public void setFluid(Boolean fluid) {
this.fluid = fluid;
}
@Override
public void restoreState(FacesContext context, Object state) {
Object[] values = (Object[]) state;
super.restoreState(context, values[0]);
fluid = (Boolean) values[1];
}
@Override
public Object saveState(FacesContext context) {
Object[] values = new Object[2];
values[0] = super.saveState(context);
values[1] = fluid;
return values;
}
}</pre><p></p>
<h3>Creating the Column Component</h3>
<p>The column component is a little more complex. We are also going to extend the UIPanelEx control but we will also add some control properties</p>
<ul>
<li><strong>span</strong> – This will be an integer that can be set to specify the column span of the bootstrap grid</li>
<li><strong>align</strong> – This will be a String which can be ‘left’ (Default), ‘center’ or ‘right’. This will then add the necessary bootstrap css class to perform aligning of the contents</li>
</ul>
<p>I am also going to add a property <strong>guessSpan </strong>which is not exposed to domino designer and will only be used to store our ‘guesstimate’ column span when calculating the auto widths.</p>
<p>Then, I create a private method ‘getEffectiveSpan’ which will check if a column span has been explicitly set with the <strong>span</strong> property and if so it will use that. If a span was not set, it will use the ‘guessSpan’ that was guessed via the algorithm</p>
<p>Then we override the ‘getStyleClass’ method, so that when the renderer comes along and asks for the styleClasses, we can inject some styleClasses of our own.</p>
<p>We are using the very handy method from Extension Library’s ExtLibUtil, which allows us to easily concatenate styles. I have copied the method into my own Util class so that there will be no dependency on the Extension Library.</p>
<p>First we ask for any styles that have been explicitly set. Then we concatenate our column style from the bootstrap grid system e.g. “col-md-8”, and then we concatenate an alignment style if one was set on the column.</p><pre class="urvanov-syntax-highlighter-plain-tag">package com.gregorbyte.xsp.component.bootstrap;
import javax.faces.context.FacesContext;
import javax.faces.el.ValueBinding;
import com.gregorbyte.xsp.util.GregorbyteUtil;
import com.ibm.commons.util.StringUtil;
import com.ibm.xsp.component.UIPanelEx;
public class UIColumn extends UIPanelEx {
private Integer guessSpan = null;
// Defaults for Medium Columns, use styleClass for specific layouts
private Integer span = null;
private String align = null;
public UIColumn() {
super();
}
private Integer getEffectiveSpan() {
if (getSpan() != null) {
return getSpan();
} else if (getGuessSpan() != null) {
return getGuessSpan();
}
return 12;
}
@Override
public String getStyleClass() {
String styleClass = super.getStyleClass();
styleClass = GregorbyteUtil.concatStyleClasses(styleClass, "col-md-" + getEffectiveSpan());
String align = getAlign();
if (StringUtil.equals(align, "left")) {
styleClass = GregorbyteUtil.concatStyleClasses(styleClass, "text-left");
} else if (StringUtil.equals(align, "right")) {
styleClass = GregorbyteUtil.concatStyleClasses(styleClass, "text-right");
} else if (StringUtil.equals(align, "center")) {
styleClass = GregorbyteUtil.concatStyleClasses(styleClass, "text-center");
}
return styleClass;
}
public Integer getGuessSpan() {
return guessSpan;
}
public void setGuessSpan(Integer guessSpan) {
this.guessSpan = guessSpan;
}
public Integer getSpan() {
if (this.span != null) {
return this.span;
}
ValueBinding vb = getValueBinding("span");
if (vb != null) {
return (Integer) vb.getValue(getFacesContext());
}
return null;
}
public void setSpan(Integer span) {
this.span = span;
}
public String getAlign() {
if (this.align != null) {
return this.align;
}
ValueBinding vb = getValueBinding("align");
if (vb != null) {
return (String) vb.getValue(getFacesContext());
}
return null;
}
public void setAlign(String align) {
this.align = align;
}
@Override
public void restoreState(FacesContext context, Object state) {
Object[] values = (Object[]) state;
super.restoreState(context, values[0]);
guessSpan = (Integer) values[1];
span = (Integer) values[2];
align = (String) values[3];
}
@Override
public Object saveState(FacesContext context) {
Object[] values = new Object[4];
values[0] = super.saveState(context);
values[1] = guessSpan;
values[2] = span;
values[3] = align;
return values;
}
}</pre><p></p>
<h3>Creating the Row Component</h3>
<p>The row component is also an extension of the panel control. But we are going to add some extra behaviour to it. Just before the Row’s children (the columns) are Rendered we are going to calculate the necessary spans for any columns that did not have a span. If we can’t find any columns as children, then we will assume that we are supposed to wrap all the contents into a single column and therefore we will render a start and end tag for a <div class=”col-md-12″></p>
<h3>Programming the auto-width algorithm</h3>
<p>The auto-width algorithm does the following</p>
<ol>
<li>Finds the total span of columns that have been explicitly allocated using the <strong>span</strong> property</li>
<li>Figures out the remaining span available (e.g. 12 – allocated)</li>
<li>Figures out the average span to allocate to remaining columns without a span</li>
<li>Allocates the ‘guessed’ span to the remaining columns</li>
</ol>
<p></p><pre class="urvanov-syntax-highlighter-plain-tag">package com.gregorbyte.xsp.component.bootstrap;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
import javax.faces.FacesException;
import javax.faces.component.UIComponent;
import javax.faces.context.FacesContext;
import com.gregorbyte.xsp.util.GregorbyteUtil;
import com.ibm.xsp.component.UIPanelEx;
import com.ibm.xsp.util.TypedUtil;
public class UIRow extends UIPanelEx {
private Boolean autoColumn = false;
public UIRow() {
super();
}
@Override
public String getStyleClass() {
String parent = super.getStyleClass();
return GregorbyteUtil.concatStyleClasses("row", parent);
}
@Override
public void encodeBegin(FacesContext context) throws IOException {
super.encodeBegin(context);
calculateColumnSpans(context);
if (autoColumn) {
context.getResponseWriter().startElement("div", null);
context.getResponseWriter().writeAttribute("class", "col-md-12", null);
}
}
@Override
public void encodeEnd(FacesContext context) throws IOException {
if (autoColumn) {
context.getResponseWriter().endElement("div");
}
super.encodeEnd(context);
}
public void calculateColumnSpans(FacesContext context) throws FacesException {
List<UIComponent> kids = TypedUtil.getChildren(this);
List<UIColumn> cols = new ArrayList<UIColumn>();
int totalMediumSpecified = 0;
for (UIComponent comp : kids) {
if (comp instanceof UIColumn) {
UIColumn col = (UIColumn) comp;
if (col.getSpan() != null) {
totalMediumSpecified += col.getSpan();
} else {
cols.add(col);
}
}
}
// If No Columns we will render
autoColumn = (cols.size() == 0);
if (autoColumn)
return;
int remaining = 12 - totalMediumSpecified;
Integer colGuess = remaining / cols.size();
if (colGuess != null) {
for (UIColumn uiColumn : cols) {
uiColumn.setGuessSpan(colGuess);
}
}
}
@Override
public void restoreState(FacesContext context, Object state) {
Object[] values = (Object[]) state;
super.restoreState(context, values[0]);
autoColumn = (Boolean) values[1];
}
@Override
public Object saveState(FacesContext context) {
Object[] values = new Object[2];
values[0] = super.saveState(context);
values[1] = autoColumn;
return values;
}
}</pre><p></p>
<h2>Provide Component Definition via xsp-config file</h2>
<p>The xsp-config file tells Domino Designer everything it needs to know to include it on an XPage so we need to specify the properties that can be set.</p>
<p>The file starts with namespace definition to say ‘all component definitions in this file are under this namespace’ and then it specifies the 2 components.</p>
<p>We provide the description, tagname, display name, corresponding java class (so it knows how to instantiate it) a component type (usually just the same as the class).</p>
<p>Then we specify the properties that can be set. You can see for the row there is no extra properties, but for the column we have added the <strong>span</strong> and <strong>align </strong>properties and have provided combo options Left, Center and Right for the <strong>align</strong> property. For the container component, we have specified the <strong>fluid</strong> property.</p><pre class="urvanov-syntax-highlighter-plain-tag"><?xml version="1.0" encoding="UTF-8"?>
<faces-config>
<faces-config-extension>
<namespace-uri>http://www.gregorbyte.com/xsp/</namespace-uri>
<default-prefix>gb</default-prefix>
</faces-config-extension>
<component>
<description>Row Container for Bootstrap Columns
</description>
<display-name>Bootstrap Row</display-name>
<component-type>com.gregorbyte.xsp.bootstrap.Row
</component-type>
<component-class>com.gregorbyte.xsp.component.bootstrap.UIRow
</component-class>
<component-extension>
<base-component-type>com.ibm.xsp.Panel
</base-component-type>
<component-family>javax.faces.Panel
</component-family>
<tag-name>row</tag-name>
<designer-extension>
<in-palette>true</in-palette>
<category>Gregorbyte Library</category>
</designer-extension>
</component-extension>
</component>
<component>
<description>Bootstrap Columns
</description>
<display-name>Bootstrap Column</display-name>
<component-type>com.gregorbyte.xsp.bootstrap.Column
</component-type>
<component-class>com.gregorbyte.xsp.component.bootstrap.UIColumn
</component-class>
<property>
<description>Adds an Alignment class to the column </description>
<display-name>Alignment</display-name>
<property-name>align</property-name>
<property-class>java.lang.String</property-class>
<property-extension>
<designer-extension>
<editor>com.ibm.workplace.designer.property.editors.comboParameterEditor</editor>
<editor-parameter>
left
center
right
</editor-parameter>
<category>basics</category>
</designer-extension>
</property-extension>
</property>
<property>
<description>Number of Columns to Span (Medium Screen size). Leave blank for auto calculation</description>
<display-name>Columns to Span</display-name>
<property-name>span</property-name>
<property-class>java.lang.Integer</property-class>
<property-extension>
<allow-run-time-binding>true</allow-run-time-binding>
<designer-extension>
<category>basics</category>
</designer-extension>
</property-extension>
</property>
<component-extension>
<base-component-type>com.ibm.xsp.Panel
</base-component-type>
<component-family>javax.faces.Panel
</component-family>
<tag-name>column</tag-name>
<designer-extension>
<in-palette>true</in-palette>
<category>Gregorbyte Library</category>
</designer-extension>
</component-extension>
</component>
<component>
<description>Container for Bootstrap Layout
</description>
<display-name>Bootstrap Container</display-name>
<component-type>com.gregorbyte.xsp.bootstrap.Container
</component-type>
<component-class>com.gregorbyte.xsp.component.bootstrap.UIContainer
</component-class>
<property>
<description>Use Fluid Layout
</description>
<display-name>Use Fluid Layout</display-name>
<property-name>fluid</property-name>
<property-class>java.lang.Boolean</property-class>
<property-extension>
<allow-run-time-binding>true</allow-run-time-binding>
<designer-extension>
<category>display</category>
</designer-extension>
</property-extension>
</property>
<component-extension>
<base-component-type>com.ibm.xsp.Panel
</base-component-type>
<component-family>javax.faces.Panel
</component-family>
<tag-name>container</tag-name>
<designer-extension>
<in-palette>true</in-palette>
<category>Gregorbyte Library</category>
</designer-extension>
</component-extension>
</component>
</faces-config></pre><p></p>
<h2>Installing</h2>
<p>If you would like to use these controls, you have 2 choices!</p>
<ol>
<li>Install the <a href="https://github.com/camac/GregorbyteXspLibrary/releases" target="_blank">latest version of my GregorbyteXspLibrary</a> to your Designer Client and Domino Server using the same method as Extension Library Installation</li>
<li>Scavenge the necessary files from my Github repository and include it your own NSF</li>
</ol>
<p>To Scavenge the necessary files from my <a href="https://github.com/camac/GregorbyteXspLibrary" target="_blank">Github Repo</a>:</p>
<ul>
<li>You will need to place these java files in your java design elements:
<ul>
<li>/com.gregorbyte.xsp.controls/src/com/gregorbyte/xsp/component/bootstrap/UIColumn.java</li>
<li>/com.gregorbyte.xsp.controls/src/com/gregorbyte/xsp/component/bootstrap/UIRow.java</li>
<li>/com.gregorbyte.xsp.controls/src/com/gregorbyte/xsp/component/bootstrap/UIContainer.java</li>
<li>/com.gregorbyte.xsp.core/src/com/gregorbyte/xsp/util/GregorbyteUtil.java</li>
</ul>
</li>
<li>You will need to place this xsp-config file in the WEB-INF directory
<ul>
<li>/com.gregorbyte.xsp.controls/src/com/gregorbyte/xsp/config/gregorbyte-bootstrap.xsp-config</li>
</ul>
</li>
</ul>
<h1>Conclusion</h1>
<p>You can easily get the job done without these controls, but so far I have really enjoyed having them in my palette.</p>
<p>I’ve only just started using these controls, so I expect to improve on them. Maybe by adding the ‘small’, ‘medium’, ‘large’, ‘extraLarge’ properties to the columns (but for now I can use styleClass=”col-sm-4″ etc.)</p>
<p>I’m also keen to take a look at other parts of the bootfaces project and see if any of the ideas might be effective in Xpages.</p>
<p>If you end up using these please let me know! I’d love to hear any suggestions and as always if any problems then please leave a comment or post an issue on my GregorbyteXspLibrary project.</p>
<p> </p>
<h3></h3>
]]></content:encoded>
<wfw:commentRss>https://camerongregor.com/2017/03/22/auto-width-bootstrap-column-xpages-controls/feed/</wfw:commentRss>
<slash:comments>0</slash:comments>
</item>
</channel>
</rss>
If you would like to create a banner that links to this page (i.e. this validation result), do the following:
Download the "valid RSS" banner.
Upload the image to your own server. (This step is important. Please do not link directly to the image on this server.)
Add this HTML to your page (change the image src
attribute if necessary):
If you would like to create a text link instead, here is the URL you can use: