Section menu for the lazy blogger
authorbep <bjorn.erik.pedersen@gmail.com>
Tue, 6 Jan 2015 17:11:06 +0000 (18:11 +0100)
committerspf13 <steve.francia@gmail.com>
Sat, 9 May 2015 01:28:19 +0000 (21:28 -0400)
The current menu system works great, but is too much work if all you want is a simple menu with the sections as menu items, and having these menu items connected to the pages in a way that enables setting the correct menu item as active for both the section lists and the pages itself.

This commit adds a new option `SectionPagesMenu' which, if set, will create a new menu with that name with all the sections as menu items. The pages in the sections will behave as "shadow members" of these section items as `blogpage.HasMenuCurrent "sectionmenu" $sectionmenuitem` will return true.

If a menu item with the same `identifier` is defined in site config, *that* item will take precedence.

commands/hugo.go
hugolib/menu_test.go
hugolib/page.go
hugolib/site.go

index f93e81982b52acbd50f72b73e01d5e3e0d74a6b5..0ad49bbe5e611550f4d6ec8b565767702430580e 100644 (file)
@@ -104,7 +104,7 @@ func init() {
 
        // This message will be shown to Windows users if Hugo is opened from explorer.exe
        cobra.MousetrapHelpText = `
-       
+
   Hugo is a command line tool
 
   You need to open cmd.exe and run it from there.`
@@ -153,6 +153,7 @@ func InitializeConfig() {
        viper.SetDefault("PaginatePath", "page")
        viper.SetDefault("Blackfriday", helpers.NewBlackfriday())
        viper.SetDefault("RSSUri", "index.xml")
+       viper.SetDefault("SectionPagesMenu", "")
 
        if hugoCmdV.PersistentFlags().Lookup("buildDrafts").Changed {
                viper.Set("BuildDrafts", Draft)
index d4d425d50ed45857dda35d876ff1565a8c0879bf..73aeb07ddf71cad713916716942a367c3466692f 100644 (file)
@@ -11,6 +11,7 @@ import (
        "github.com/spf13/hugo/source"
        "github.com/spf13/viper"
        "github.com/stretchr/testify/assert"
+       "path/filepath"
 )
 
 const (
@@ -90,9 +91,15 @@ weight = 3
 Front Matter with Menu Pages`)
 
 var MENU_PAGE_SOURCES = []source.ByteSource{
-       {"sect/doc1.md", MENU_PAGE_1},
-       {"sect/doc2.md", MENU_PAGE_2},
-       {"sect/doc3.md", MENU_PAGE_3},
+       {filepath.FromSlash("sect/doc1.md"), MENU_PAGE_1},
+       {filepath.FromSlash("sect/doc2.md"), MENU_PAGE_2},
+       {filepath.FromSlash("sect/doc3.md"), MENU_PAGE_3},
+}
+
+var MENU_PAGE_SECTIONS_SOURCES = []source.ByteSource{
+       {filepath.FromSlash("first/doc1.md"), MENU_PAGE_1},
+       {filepath.FromSlash("first/doc2.md"), MENU_PAGE_2},
+       {filepath.FromSlash("second-section/doc3.md"), MENU_PAGE_3},
 }
 
 func tstCreateMenuPageWithNameTOML(title, menu, name string) []byte {
index e53781b64eb3a82ffad7dc9f35eee1305f26bfab..3ed12af6baa38051789aed770b4a7eb47870320b 100644 (file)
@@ -556,6 +556,12 @@ func (p *Page) GetParam(key string) interface{} {
 
 func (p *Page) HasMenuCurrent(menu string, me *MenuEntry) bool {
        menus := p.Menus()
+       sectionPagesMenu := viper.GetString("SectionPagesMenu")
+
+       // page is labeled as "shadow-member" of the menu with the same identifier as the section
+       if sectionPagesMenu != "" && p.Section() != "" && sectionPagesMenu == menu && p.Section() == me.Identifier {
+               return true
+       }
 
        if m, ok := menus[menu]; ok {
                if me.HasChildren() {
index 32d74dff936e67a8deb11fea6d359be4627727ad..c52c89e6d53ab2c581e99a4b9737d3bf198b68e4 100644 (file)
@@ -762,8 +762,25 @@ func (s *Site) assembleMenus() {
                }
        }
 
+       sectionPagesMenu := viper.GetString("SectionPagesMenu")
+       sectionPagesMenus := make(map[string]interface{})
        //creating flat hash
        for _, p := range s.Pages {
+
+               if sectionPagesMenu != "" {
+                       if _, ok := sectionPagesMenus[p.Section()]; !ok {
+                               if p.Section() != "" {
+                                       me := MenuEntry{Identifier: p.Section(), Name: helpers.MakeTitle(p.Section()), URL: s.permalinkStr(p.Section())}
+                                       if _, ok := flat[twoD{sectionPagesMenu, me.KeyName()}]; ok {
+                                               // menu with same id defined in config, let that one win
+                                               continue
+                                       }
+                                       flat[twoD{sectionPagesMenu, me.KeyName()}] = &me
+                                       sectionPagesMenus[p.Section()] = true
+                               }
+                       }
+               }
+
                for name, me := range p.Menus() {
                        if _, ok := flat[twoD{name, me.KeyName()}]; ok {
                                jww.ERROR.Printf("Two or more menu items have the same name/identifier in %q Menu. Identified as %q.\n Rename or set a unique identifier. \n", name, me.KeyName())