14 for 14: Building a Navigation Button Bar

By Daniel Wood, 28 May 2015

Previous Navigation Articles and our own Navigation

This is the third navigation article we have done and it has been quite some time between drinks. The first one - Starting out Simple - covers building a basic navigation bar using repeating fields.  In the second article - Taking it Further - we extended upon the first by introducing selected screen highlights and a navigation script.

It may surprise you to know that we here at Digital Fusion still use this method for our navigation bar with some small tweaks. Repeating fields are still utilised and have been abstracted to the point the navigation bar setup can be customised per-user and all of the formatting can be done via a combination of styles and editing the repeating field contents.  It is robust, works well and highly flexible.

  

So why consider the button bar?

It's a great question. The button bar on its surface seems to be very similar to our navigation method in regards to how it can be styled to look, and how it functions. Each segment can run a script for navigation, and you have total control over how it looks - sounds like a perfect substitute!

And for most applications it is, but there are some small limitations as we'll get into later.

One of the biggest benefits of the button bar over other methods is that the button bar is simply a layout object without a fixed context. If properly setup, it can be built to be entirely generic and contextless, and therefore able to be copy/pasted across multiple layouts.

   

The components of a button bar we need to setup

So you've decided you want to use a button bar for your navigation - how do you now make it entirely generic and contextless? If we look at the button bar setup we can identify areas we'll need to abstract:

The first we will cover is the label.  Fortunately labels can be defined for a segment based on calculation so that is perfect, we can have a different calculation result for each segment.

Next we need to be able to run a script to carry out the navigation. The script needs to know which segment was pressed by the user in order to carry out the correct navigation. Fortunately we can pass a script parameter that differs for each segment, so that takes care of that.

What about if we wish to highlight the current screen segment the user is on? Fortunately once again we can specify what the active segment is via calculation so all good there.

The last thing is icons. Ahh, now we have a little issue.  To this point everything could be abstracted into a calculation, and thus alterable outside of the button bar object, but can we change the icon? The answer at this point is no.  If your navigation bar is to use built in button icons, then once you define an icon for a segment that's it. You cannot programmatically alter this icon in the future, meaning if you wanted to change it you would have to copy/paste the button bar across all your layouts again.

No icon? Perfect! The button bar is a great tool for a text based navigation bar.

  

So how do we abstract?

There are a number of ways in which you can abstract the information needed by the button bar. In our example we are going to build a simple navigation table to store our information. We'll then build a custom function to obtain this information for the button bar to use.

Above we have our navigation table. It has three fields:

  • Segment - just a number. It will correlate to a segment of the button bar.
  • Label - what we wish the label to be in the button bar
  • Layout - what layout will the navigation take the user to when clicked.

And here is our example navigation bar:

As you can see we have 5 segments. Each label correlates to the segment defined in the navigation table. If we were to for example swap the segment numbers for Penguin and Phone in the navigation table we end up with:

The button bar automatically responds to this change and reorders the Phone and Penguin segments.

Notice anything unusual though? The icons! We cannot have control over these so this type of dynamic reordering of a button bar is prone to problems if you use icons unfortunately :(

    

Getting the Navigation information to the button bar

It's all fine and well to setup this information in a table but we need to deliver it to the button bar in a way such that the button bar does not have to deal with context.

To achieve this we have built a simple custom function which we call @NAV_GetProperty

This custom function takes two parameters:

  • _theNumber: This is a segment number for which you want information.
  • _theProperty: This is a keyword that tells the function what information you need.

And that's it.  As an example, what if we needed the label for the third segment:

@NAV_GetProperty ( 3 ; "Label" )

The result of this function will be "Phone".

The custom function is used in three situations:

  • To define labels - @NAV_GetProperty ( 1 ; "Label" )
  • Navigation script to obtain layout - @NAV_GetProperty ( 1 ; "Layout" )
  • To determine active segment - @NAV_GetProperty ( "" ; "Active" )

The last situation there is a slightly different case where we do not pass a segment number. In this case the custom function returns a segment number that corresponds to which one should be the active segment.

  

Using ExecuteSQL to abstract

The custom function is able to return this information, and determine active segment, through the use of the ExecuteSQL function.

For obtaining the layout or label, it simply is carrying out a SELECT query based on the passed segment number, and property.  Note: The property corresponds to the field name in the navigation table - so if you decide to change these field names you must also update the custom function accordingly.

   

Defining what an Active Segment is

In order for us to show an active segment, we must first understand how FileMaker defines what the active segment is for a button bar.

We can either define the active segment explicitly, or via calculation. If defined explicitly we have no control over dynamically altering this, so it is useless for our navigation bar.

What needs to be returned by our calculation in order for the result to be treated as the active segment?  The answer is an object name.  Each segment can be assigned an object name via the Inspector, and FileMaker compares the calculation result to object names in order to determine active segment.

What are the implications of this? Well first you need to ensure every segment in your button bar has an object name. In our example we have given each the following:

nav_X

where X is a number from 1 to 5, in order.

Active segments are determined by first looking at what layout the user is on. The custom function will perform a SELECT query to obtain the Segment number for the navigation record whose layout matches the current layout. This tells us which segment number our layout is for. We can then simply prefix it with "nav_" to return an object name that corresponds to the button bar active segment.

    

Putting it all together

We have now a single custom function that we can use to define the active segment and segment labels and layouts.

Our active segment calculation is thus: 

@NAV_GetProperty ( "" ; "Active" ) 

Our label calculation is:

@NAV_GetProperty ( 1 ; "Label" )

and our Navigation script is passed a segment number from which it can obtain the layout:

@NAV_GetProperty ( Get ( ScriptParameter ) ; "Layout" )

Very elegant, very simple, and very flexible!

    

Taking it further

Because ExecuteSQL is used to obtain navigation information, you can do some pretty cool things with your navigation bar setup.

Perhaps you added a new table called "UserNavigation" which acts as a join table between Navigation, and your Users table. Each record has a user ID and a navigation ID. Basing your navigation bar on this table, you could have different navigation bars for users based on things such as their access privileges, or role.  Simply add a record for a user to give them screen access via the button bar.

Another cool feature of the button bar is that text formatting functions will override the styles. You can therefore conditionally format labels to appear disabled if you wish to prevent user access to certain screens as opposed to hiding that screen from the button bar. Using the TextColor function you can achieve this.

   

Button Bar limitations

We have already covered some limitations around abstracting icons, but there is another property of button bars that may mean they are not suitable to your needs. Each segment of a button bar is the same width as all other segments - you cannot define an individual segments width. This means segments with a long label will wrap its text as opposed to extending the width of the segment:

Sometimes this may not look that bad, but icons will be offset to accommodate.

  

In conclusion

Button bars are a great FileMaker 14 tool and can be used for a number of applications, one of which may be a navigation bar. As far as making them totally abstracted goes, they work best without icons but if you can live with the icons being fixed to the object then they look great with icons.

Compared to our older articles around navigation bars the button bar comprises of only a single object as opposed to multiple objects for the repeating field implementation. Button bars should also work great in WebDirect and FileMaker Go.

We are strongly looking at whether it is worth making the switch from our repeating field based navigation to button bars. At this point the jury is still out and icon abstraction is a big limitation that will likely see us keep to repeating fields for now.

   

Example File.

All the screenshots and examples in this article are from our demo file that you can download below. Obviously being a FIleMaker 14 feature this works best in 14, but you can still open it in 13 if you need to.

Click here to download the example file

Something to say? Post a comment...

Comments

  • Tom 18/01/2024 8:30am (1 year ago)

    Hi,

    I'm trying to impliment this idea in my own database. I have my navigation items in a separate table, and have added the custom function from your demo, but I can't seem to get the button labels to populate.

    I see your example file has a relationship between id_interface and id_navigation, is that something I need to copy? What are those id values?

    Would love to get this working if possible!

    Thanks!

  • Daniel Wood 27/03/2019 9:35am (6 years ago)

    hi Justin,

    I'd be happy to give you a hand getting it to work if you have an example file already that you're working on you can email it to me daniel@teamdf.com and I can look at it for you, otherwise if not let me know and I'll see if I can come up with a quick example file for you.

  • Justin 27/03/2019 4:53am (6 years ago)

    This is great! Could you show me an example of how to implement the user join table with access privileges being the data that dictates what buttons appear? I've been working at it but can't seem to get it to work.

  • Daniel 05/07/2017 9:44am (8 years ago)

    hi Barry,

    Thank you for the email. The example file is definitely unlocked, I've just had another check of it. Once you open it in FileMaker, the status area is hidden, but you can easily show it again. To do this , in the menus select View -> Status Toolbar. This allows you to navigate the layouts, and enter find mode etc. All other options are unlocked, you can access the scripts, tables, fields, relationship graph and so on all in the usual manner. If you have any other questions let me know.

  • Barry Nelthorpe 30/06/2017 2:46pm (8 years ago)

    Dear Sirs,
    I am trying to get beyond simple programming in FileMaker 16 but cannot seem to find any way of doing this. When I read your article I was impressed but could not really understand it. So I downloaded the example file thinking it would be unlocked and I could try and get a grip on the calculations and scripts. I don't see the point of locking this example.
    Yours sincerely,Barry
    PS: I am a retired senior and for me this is just a hobby to help keep me from going senile

  • Daniel Wood 13/04/2016 9:13am (9 years ago)

    hi Steven,
    The issue could be any number of things wrong with the implementation but it is very difficult for me to guess what it could be, any other info you can provide would be helpful.

  • Steven Hampson 07/04/2016 2:03am (9 years ago)

    Active segement is not working, have included the object name on each of the segments but nothing is being highlighted.

  • Simon 29/09/2015 12:14pm (9 years ago)

    Screens can be found here: https://www.dropbox.com/s/frnebsmhh1pe3ix/button_bar_screens.zip?dl=0

  • Simon 29/09/2015 12:08pm (9 years ago)

    Unfortunately I can't upload a Screen here to describe the Button-Bar-Bug.
    But theres a big Bug (or Feature?) in the Button Bar: if you have different lengths for your labels, it's not possible to fit the buttons to the content. In other words: if you have a long and a short label you can only stretch the whole button bar. The result is crab and unusable 'cause every button has the same length.

    As jonro said: "Typical of FileMaker to give us half of what we need."

  • Richard 03/09/2015 8:10am (9 years ago)

    Is there a way to have the icons change the same way? Rather and manually assigning the icons do you have a way to automatically load an icon for the button bar? I'm new to Filemaker and developing.

  • Daniel 31/05/2015 7:08pm (10 years ago)

    hi Sean, If you are using the standard FM icons then you can do so in the Inspector on the appearance tab. At the top where you choose the component of the object you want to style you can change this to Button:Icon (or in this case Button Bar:Icon). Then you can set its colour using the fill colour for the various states.

    If you add your own SVG icons then you must ensure the SVG has no fill colour data in it, otherwise you cannot change the colour via styles in FileMaker. You can find the SVG guide here - https://fmhelp.filemaker.com/docs/14/en/svg/. Obviously some knowledge of SVG's is required to follow that document, but if you want to get started with some custom svg's that can be styled try www.flaticon.com

  • Sean 31/05/2015 10:37am (10 years ago)

    How are you coloring the icons. I've tried PNG and SVG and neither are allowing me to color them.

  • jonro 29/05/2015 1:46pm (10 years ago)

    There may be a way to partially get around the icon limitation. The individual segments of the button bar can be hidden with the "Hide Object When" option in the Inspector. So, it's possible to display buttons 1, 3, and 5 on one layout and buttons 2, 4 and 6 on another. This could be abstracted in a couple of different ways. This would give us different icons and different buttons on each layout, but they would still be in numeric order,

  • jonro 29/05/2015 1:40pm (10 years ago)

    Typical of FileMaker to give us half of what we need. I've been using a button bar with repeating fields, too, and it does everything I need, including icon, text and script abstraction. I'm going to evaluate the button bar with an open mind; I really would like it to be a viable solution, especially because it is designed to work well with WebDirect and FM Go. Thanks for the article.

RSS feed for comments on this page | RSS feed for all comments

Categories(show all)

Subscribe

Tags