Add Explicit Support for Multi-Section Pages

The ā€˜one-pagerā€™ tutorial shows us how Kirby can be used to setup a single-page site, where the page has multiple sections.

All sites I do have more than one page - the site itself is usually divided into at least a ā€˜homeā€™ page and a ā€˜blogā€™, and then it often might have other features required by the client, like a product catalog, event calendar, or forum. I donā€™t think Iā€™ve ever done a single-page site, however in all my sites, I do need to have a ā€˜multi-sectionā€™ page, exactly as described in the ā€˜one-pagerā€™ tutorial.

Iā€™m currently doing my first two sites in Kirby, and both are going to be essentially ā€˜one-pagerā€™ sites, with a blog. That is: the ā€˜homeā€™ page will be divided into sections, layed out with sub-pages and snippets, and then there will also be a blog. Iā€™ve managed to do the multi-sectioned homepage by slightly altering the instructions on the ā€˜one-pagerā€™ tutorial - in my version, the pages that represent the sections are subpages of ā€˜homeā€™, not at the top level of the site.

The whole process, however, made me feel as if ā€˜sectioned pagesā€™ like that are ā€œsecond-class citizensā€ in Kirby. The pages representing the different sub-sections of the parent are inserted as snippets in the parentā€™s template, and this has limitations - ie, snippets do not have their own controllers. Also, pagesā€™ URLs are by default made available by the router, so all page sections end up with their own URL. In order to avoid the user reaching these URLs, the developer has to either setup redirection manually in the ā€˜defaultā€™ template for the site, or - if the default is already being used for another purpose - setup redirection in a template for each section page. It feels like a workaround.

Considering that multi-section pages are such a common use-case, perhaps setting them up could be made easier in the future, with a few alterations/additions to Kirby. For instance:

  • we create ā€˜sectionā€™ objects: they are basically page objects, but they DO NOT ROUTE - ie, they route to the error page, or another route specified by the developer in a special config. In the filesystem, sections might be indicated by a name starting with an underscore - so that, for instance, a directory called ā€˜contactā€™ would indicate a page, but one called ā€˜_contactā€™ would indicate a section, and would not be accessible directly via http.

  • a ā€˜pageā€™ should then be able to have sections, sub-pages and files displayed in the panel.

  • sections can have their own templates, blueprints and controllers, perhaps stored in the siteā€™s templates/blueprints/controllers folder in a sub-directory with the same name as their parent pageā€™s.

  • the page object should be given a ā€˜sections()ā€™ method, to fetch a collection of the pageā€™s sections. The section object should be given a ā€˜render()ā€™ method, that would run the sectionā€™s controller, then output the sectionā€™s template. This would enable us to do:

foreach($page->sections() as $section) {
  $section->render($optionalData);
}

I havenā€™t looked at the Kirby core code, so I cannot gauge whether this would be an incredible amount of work, or whether it would be realisticā€¦ but itā€™s worthwhile dreaming! :wink:

2 Likes

Well, havenā€™t thought this through, but the first thing that comes to mind is that it all depends what you want to include on your section page, sometimes your sections will only be part of a page, e.g. have a section of the last 5 blog articles or some of your projects etc, but also have a blog page and a projects page. Then using the blog etc. templates for the section page would interfere with the blog template for the blog page. Also, in such a case it wouldnā€™t make sense to make the blog page a child of some other page.

What I have in mind is a site structure like this:

- Home
  - _hero-banner
  - _about_us
  - _contact
- Blog
  - article1
  - article2
  - article3
  - ...
- Catalog
  - _specialpromo
  - _featuredproduct
  - product1
  - product2
  - product3
  - ..

ā€œHomeā€, ā€œBlogā€ and ā€œCatalogā€ would be the top-level pages of the site.

The Home page is a multi-section page - its sections are: ā€˜hero-bannerā€™, ā€˜about-usā€™ and ā€˜contactā€™.

The Blog page is a simple page, with sub-pages - ā€˜article1ā€™, ā€˜article2ā€™, ā€˜article3ā€™, etc.

The Catalog page is a multi-section page (ā€˜specialpromoā€™ and ā€˜featuredproductā€™) with sub-pages (ā€˜product1ā€™, ā€˜product2ā€™, ā€˜product3ā€™, etc.).

While this is doable with Kirby at the moment, it does require jumping through hoops - because we must use pages as ā€˜sectionsā€™.

Iā€™ve felt this need a couple times too. I think there is only a problem when we use ā€œKirby pagesā€ as ā€œsectionsā€ and donā€™t need them as standalone ā€œweb pagesā€.

Iā€™m not sure we need as many changes as you explained, nor have a different kind of object. Technically almost all the pieces are already here. The ā€œsectionā€ object you describe is basically a ā€œpageā€ without a URL.
It could be a new page option:

pages:
  template: project
  url: false

Then you could filter with something like:

$pages->children()->visible()->noUrl()

+ a way to visually identify/separate them in the panel of course.

Or something different and actually clever. This is just a wild idea, I really have no clue how it could be done :wink:
Perhaps it could be a plugin.

I think there is yet another use case: very complex blueprints that result in loooong forms.
We could feel the need to split split such a form for convenience. In this topic we discussed a way to split a long form into tabs or collapse them as an accordion (with a link to github with an implementation as an accordion).
In your structure _about_us and _contact on the home sound like contents that are not displayed anywhere else, in this case the tabs or collapsible form could be a solution.

3 Likes

@Malvese I agree: I think that a ā€˜sectionā€™ is basically a page with no URL route, and they need to be handled differently in the Panel.

In fact, there are 3 things that we should be able to do with ā€˜sectionsā€™:

  1. ā€˜Sectionsā€™ should not be routable - no URL route should be created for sections.

  2. ā€˜Sectionsā€™ should have their own section in the Panel sidebar, like ā€˜pagesā€™ and ā€˜filesā€™.

  3. We should be able to include a pageā€™s sections in its template like a snippet. Unlike a snippet, however, ā€˜sectionsā€™ should also be able to have their own controller, that run when the section is rendered on the template.

I donā€™t believe that you can do all three things without creating a ā€˜sectionā€™ object/class. Nevertheless, as you pointed out, a ā€˜sectionā€™ is basically a ā€˜pageā€™, with very few changes. This may mean that a ā€˜sectionā€™ class could be relatively easy to write - it might be a simple extension to the existing ā€˜pageā€™ class. I donā€™t really know how much work it would be to write it, to be honest, because I havenā€™t looked at the Kirby code! :blush:

1 Like

I came to the forums to make the same suggestion. I think a little added functionality to the panel - to allow it to view subpages as sections - would make much more sense.

The only crucial change I can see would be the way pages and sections are displayed in the panel. Sections, structurally, would still be subfolders of that page. But, there would be no need to edit information on them (they are just there for organization) - better to have immediate access to their subpages.

I recently used Kirby to build a site for Tunica Magazine. The structure of the magazine content is something similar to:

Magazine/
  - Issue 3
     - Portfolios
         - Artist A
         - Artist B
         - Artist C
     - Interviews
         - Interview A
         - Interview B
         - Interview C
     - Articles
         - ...
     - Essays
         - ...

For each issue, all portfolios, interviews, articles and essays appear on each page. There is no issue-3/portfolios/ page, but there is a issue-3/portfolios/artist-a/. (there is no portfolios template, just issue and artist)

When explaining this to the client, I needed to explain to them that adding the Portfolios page to the issue wasnā€™t actually creating a page, but was a necessary step. This and the redundancy of seeing a Portfolios page with no editable content felt somewhat hack-y and unprofessional.

I think the panel would feel much better if this was possible. The actual structure of the files would not have to be changed - just the way they are displayed in the panel.

In Kirby a ā€œpageā€ is clearly some content + a URL. There are workarounds to emulate ā€œsectionsā€ but when you create a page in the panel a URL is required. It can be ignored, as it autocompletes with a sensible path, but itā€™s still there.
But is a useless URL that big a deal?

  • If we never link to these ā€œsectionsā€ nobody should land there, not even search engines because I believe they follow links. I donā€™t think they will ever try to index ā€œ/issue3/portfoliosā€ out of nowhere.
  • Even if they did, and for the few visitors who edit URLs manually, we can route these sections to another page or serve a 403 or 404. This is what is explained in the One pager tutorial in the Docs, so letā€™s call it the official way :wink:

@SQBiz Your pages like ā€œPortfoliosā€ are not really required, you could just add subpages to the issue itself (I assume they have separate blueprints, so you could still filter them). But I understand the way you used them creates a nicer display in the Panel.
Being able to distinguish different blueprint in the sidebar would be nice. Just listing them by blueprint type would be very useful. I think for me at least if would void the need for sections.
If you keep these ā€œsectionsā€ you could use the subpage builder to create them automatically every time an new issue is added, and avoid the hassle of creating them manually for your editors.

Following the one-page tutorial still creates URLs for the sections. If I have a ā€˜homeā€™ page with an ā€˜aboutā€™ section I end up, by default, with a ā€˜home/aboutā€™ URL. Custom routes are needed to actually route such requests to an error page. It is not a neat solution.

Most importantly: pages are conceptually different from sections. A page should be able to have BOTH sections and sub-pages, and both types of objects should be able to be displayed in the panel.

Hello! Some time ago I developed a basic support for Multi-Section pages for my own use. Iā€™ve taken some ideas from this post and Iā€™ve created the ā€˜Section Pluginā€™ for Kirby CMS, please visit the repo here.

The plugin cover almost all the features exposed here, but definitely would be great if @bastianallgeier offers native support for multi-sections pages.

Iā€™ve used and tested the plugin here and it works like a charm.

Note: this is in beta, Definitely Iā€™m a noob with the php performance and I donā€™t know if there is a better way to achieve this, ideas are welcome.

PS: Sorry for my english

2 Likes

Looks great! Love the rendering functions.
The only minor inconvenient I see is that the underscore before the template name is visible in the Panel. I can also be a problem if we already use the same naming scheme for some reason.
A possible solution may be a list of ā€œsectionā€ templates in config.php so the files can be named any way we like:

c::set('sections', array(
  'hero-header',
  'services',
  'team'
));

A remark on the routing, it would seem great if the page slug is prefixed by a # to indicate a section instead of a page. Letā€™s say you have /article and a child section #intro, the child could be routed to ./article/#intro. All the section template needs is an in attribute on the container element.

A late post, but to me it seems having a variety of sections is instrumental to get truly editorial webpages (like magazines offer)

You can already do this:

Letā€™s say your ā€œone-pagerā€ page is /article (with the template article.php) and it has a child page named /article/intro (with the template section.php). To change the URL to the section in all links, you can then create a page model named site/models/section.php with the following content:

<?php

class SectionPage extends Page {
  public function url() {
    return $this->parent()->url() . '#' . $this->uid();
  }
}
3 Likes

It amazes me what can be done out of the box, Iā€™m from a front end side so routing is not my strong suit, but this helps a lot. Thank you.

Still however, a specific ā€˜sectionsā€™ content type (besides files and pages) with automated routing (folders can start with a #) and specific blueprints/snippets would be a great addition.

Mainly because now the blueprint folder specifically will become somewhat messy* (and sections could use their own default). Also for clients the distinction between inline sections and child pages would become more clear. And if Iā€™m not mistaken, there needs to be a route for each section template.

* Iā€™m talking about several articles - like case studies - which are one pagers, yet part of a larger system which soon ends up with a plethora of blueprints and templates.

You might be interested in the kirby-builder plugin by @timoetting https://github.com/TimOetting/kirby-builder