Make A Time Picker In Power Apps

Make A Time Picker In Power Apps

I often wonder why there is not any Time Picker control included in Power Apps. Date and Time values are found in almost every app I’ve ever made and I’m not alone. At least 14,000 people have read this forum post looking for an answer. Yet, we still have to build it ourselves. In this article, I will show you how to make a Time Picker in Power Apps as shown in the image below.




Building the Hours Gallery

A Time Picker control is something you will use over-and-over again so its best to build it as a canvas app component. Open Power Apps Studio and go to the Components screen. Create a new component called cmp_TimePicker.



The component cmp_TimePicker should have these properties.

Fill: White
Height: 430
Width: 285



Next, add a new gallery to the component called gal_TimePicker_Hours.



The gallery should have the same height as the component and 1/3 of the width of the component to allow for hours, minutes and AM/PM. Use the properties below in the gallery.

Fill: Transparent
Height: Parent.Height
TemplateSize: 60
Width: Parent.TemplateWidth/3



To populate the gallery with Hours we can put a Sequence function in the Items property. We’ll format the number with a leading zero too for consistency.

ForAll(Sequence(12), Text(Value,"[$-en-US]00"))



Now place a label called lbl_MobileTimePicker_Hours in the gallery with the code ThisItem.Value in the Text property to see the options for hours.




Styling the Component


We should style the Hours gallery before moving onto the Minutes and AM/PM galleries. That way we can copy & paste the 1st gallery and code the style only once. Set the properties of lbl_TimePicker_Hours equal to the code shown below.

Align: Center
Color: Black
Height: Parent.TemplateHeight
Text: ThisItem.Value
Width: Parent.TemplateWidth
X: 0
Y: 0



We want to change the color of the Hours gallery item to show what the user has selected. This component could be re-used in several apps so its a good idea to create new properties to control the label’s color and fill. Make two new input properties called SelectedColor and SelectedFill.




Then add this code to the properties for cmp_TimePicker.

SelectedColor: White
SelectedFill: DodgerBlue



Now that we’ve setup everything that’s needed our goal will be to make the TimePicker component behave like this.



Write this code in each respective property of lbl_TimePicker_Hours to get the desired result.

Color:
If(ThisItem.IsSelected, cmp_TimePicker.SelectedColor, Black)

HoverColor:
Self.Color

PressedColor:
Self.Color

Fill:
If(ThisItem.IsSelected, cmp_TimePicker.SelectedFill, Transparent)

HoverFill:
If(!ThisItem.IsSelected, ColorFade(cmp_TimePicker.SelectedFill, 70%))

PressedFill:
Self.HoverFill



Finally, to hide the scrollbar change the ShowScrollbar property of the gallery to false. The Hours gallery is now completed.




Minutes & AM/PM Galleries

To create the minutes gallery copy & paste the hours gallery directly beside it. Rename the gallery to gal_TimePicker_Minutes.



Replace any code in the Items property with this code to generate the minutes values from 00 to 59.

ForAll(Sequence(60), Text(Value-1,"[$-en-US]00"))


To create the AM/PM gallery copy and paste once more and rename the gallery to gal_TimePicker_AMPM.



Put this code in the Items property of the gallery instead.

["AM","PM"]



With all of the galleries created the last control to add is a label called lbl_TickerPicker_Border which will act as a border. Make sure the label is at the bottom of the tree view.



Use this code in the following properties of the label.

BorderColor: Black
BorderThickness: 4
Height: Parent.Height
Width: Parent.Width
X: 0
Y: 0




Output A Time Value

Our component needs to output a time value once a user makes their selection. To do this click and highlight all three galleries…



… then write this code in the OnSelect property to set a variable to store the time.

Set(
    varTimeValue,
    Time(
        Value(gal_TimePicker_Hours.Selected.Value)
        - If(
            Value(gal_TimePicker_Hours.Selected.Value)=12, 12)
        + If(
            gal_TimePicker_AMPM.Selected.Value = "PM",
            12,
            0
        ),
        Value(gal_TimePicker_Minutes.Selected.Value),
        0
    )
)


Now when you click on values in the gallery the time will be updated. To see how it works you can temporarily put a label inside the gallery with varTimeValue as the Text property. Delete the label once you’re done testing.



Create a new property called Value to give the ability to reference your time value in the app.



Then put this code in the Value property of cmp_TimePicker.

varTimeValue





Positioning Gallery Items

When the hours, minutes or AM/PM are clicked inside the gallery I want them to snap to the top of the component. We will need a new property called Default to accomplish this. Make sure to check the box that says Raise OnReset when value changes.



The Default value of cmp_TimePicker should be 01:00 AM unless otherwise defined. Do this by using Time(1, 0, 0) in the property.



Next comes some fairly complex code to define the selection positioning for each gallery. Put this code in the Default property of the Hours gallery.

If(
    cmp_TimePicker.Default=Blank()
    And varTimeValue=Blank(),

    // value when no default is supplied
    {Value: "01"},

    // value when default is supplied or hours were clicked
    With(
        {
            wTime: Coalesce(
                varTimeValue,
                cmp_TimePicker.Default
            )
        },
        LookUp(
            Self.AllItems.Value,
            Value = Text(
                If(Mod(Hour(wTime),12)=0,12, Mod(Hour(wTime),12)),
                "[$-en-US]00"
            )
        )
    )
)



Then write this code in the Default property of the Minutes gallery.

With(
    {
        wTime: Coalesce(
            varTimeValue,
            cmp_TimePicker.Default
        )
    },
    LookUp(
        Self.AllItems.Value,
        Value = Text(
            Minute(wTime),
            "[$-en-US]00"
        )
    )
)


And use this code in the Default property of the AMPM gallery.

With(
    {
        wTime: Coalesce(
            varTimeValue,
            cmp_TimePicker.Default
        )
    },
    If(
        Hour(varTimeValue) < 12,
        {Value: "AM"},
        {Value: "PM"}
    )
)


When the user clicks the gallery we must reset it in order to show the clicked values at the top. Put this code in the OnReset property of cmp_TimePicker to do just that.

Reset(gal_TimePicker_Hours);
Reset(gal_TimePicker_Minutes);
Reset(gal_TimePicker_AMPM);
Set(varTimeValue, Blank())




Using The Time Picker In An App

Now that we’ve completed the component lets try using it in an app. Place a new text box on the screen that we can display the time value in. I’ve added a clock icon to this text box to indicate a time value.


Add the Time Picker component and position it beside the text box.



To show the selected time inside the text input copy & paste this code into the text input’s Default property.

Text(cmp_Sample_MobileTimePicker.Value, ShortTime)



Insert a new label onto the screen to act as an overlay. When the user clicks it the component should disappear. Make sure the label is above the text input and below the Time Picker in the tree view.



Put this code in the following properties for the overlay.

OnSelect: Set(varShowTimePicker, false)
Visible: varShowTimePicker



Use this code in the OnSelect property of the Text Input.

Set(varShowTimePicker, true)



Then change the Visible property of the Time Picker to this code.

varShowTimePicker



All of your hard work has paid-off. Now you have an awesome looking reusable time picker.





Questions?

If you have any questions or feedback about Make A Time Picker In Power Apps please leave a message in the comments section below. You can post using your email address and are not required to create an account to join the discussion.

Matthew Devaney

Subscribe
Notify of
guest

75 Comments
Oldest
Newest
Inline Feedbacks
View all comments
Necdet Saritas
Necdet Saritas
3 years ago

You just rock and roll all cats:) Thanks. Great job Matthew

Mike
Mike
3 years ago

Yet another great solution Matthew.
question for you, how would you alter this component to add ok cancel buttons to set the date and hide on ok, and just hide on cancel? I’ve had challenges with buttons on components hiding the component itself as well as keeping the code with what to do with value back in the screen, external to the component,
thanks

myk
myk
3 years ago

This is great! Appreciate the efforts. However, I am facing an issue with having a time value in the variable ‘varTimeValue’. So this is not working for me. if I hover my mouse over varTimeVaue in the code snippet, it shows “varTimeValue=PowerApps encountered an internal error trying to evaluate this expression”
However, when I hover the mouse over ‘Time‘ it shows the correct values

This is the snippet where I get the above error on “OnSelect”

Set(
  varTimeValue,
  Time(
    Value(gal_Hrs.Selected.Value)
    – If(
      Value(gal_Hrs.Selected.Value)=12, 12)
    + If(
      gal_AMPM.Selected.Value = “PM”,
      12,
      0
    ),
    Value(gal_Mins.Selected.Value),
    0
  )
)

Any idea what could be wrong?

David
David
3 years ago

Was hoping to use this for TimeIn and TimeOut fields in a Gallery. But it changes the time in all the records in the Gallery. Any way to not have it do that and only populate the record in the Gallery you are calling it from? Still unbelievable that MS hasn’t put such a control in place.Your help has been filling gaps that developers need, thanks!

David
David
3 years ago

Great, thanks!

Donald
Donald
2 years ago

I’m currently using your tutorial and honestly, it’s brilliant.
Same as David, I added two text inputs (start time & end time) and would like to use the same time picker for both (basically following your steps to the letter, albeit adding a second text input).

Where can I adapt the code to output the time value selected to its corresponding field?

Thanks in advance for your help

Hussain
Hussain
2 years ago

Hi sir,
Great article as always. I am new to the PowerApps & facing some issues regarding this. Can you please tell me where should I use this Patch() function. [ OnSelect of gallery or somewhere else.]
Gallery Name : Gallery2
Text input Name : TextInput2

Thank you in advance.

André van Onselen
André van Onselen
3 years ago

This is awesome! Thanks for the info.

The only question i have is:
Why don’t use the function UpdateContext({varShowTimePicker:!varShowTimePicker}) instead of all the trouble with overlays?

Last edited 3 years ago by André van Onselen
André van Onselen
André van Onselen
3 years ago

Hi Matthew thanks again for sharing this blog.

Because it helped me a lot I wanted to share a quick best practice for adding up and down buttons. I write you from the Netherlands and here we use a slightly different time notation (24h) and a different seperator in formula’s (; instead of ,).

Step 1: Add 2 ▲ icons and put one above the hours gallery (IcHA) and one above the minutes gallery (IcMA).
Step 2: Add 2 ▼ icons and put one below the hours gallery (IcHB) and one below the minutes gallery (IcMB).
Step 3: Set the OnSelect property
IcHA:

Set(var_TimeValue;Time(Value(Gallery_Hours.Selected.Value-1);Value(Gallery_Minutes.Selected.Value);0))

IcMA:

Set(var_TimeValue;Time(If(Value(Gallery_Minutes.Selected.Value)=00;Value(Gallery_Hours.Selected.Value-1);Value(Gallery_Hours.Selected.Value));If(Value(Gallery_Minutes.Selected.Value)=00;59;Value(Gallery_Minutes.Selected.Value-1));0))

IcHB:

Set(var_TimeValue;Time(Value(Gallery_Hours.Selected.Value+1);Value(Gallery_Minutes.Selected.Value);0))

IcMB:

Set(var_TimeValue;Time(If(Value(Gallery_Minutes.Selected.Value)=59;Value(Gallery_Hours.Selected.Value+1);Value(Gallery_Hours.Selected.Value));If(Value(Gallery_Minutes.Selected.Value)=59;00;Value(Gallery_Minutes.Selected.Value+1));0))

Result: 

Lorna
Lorna
2 years ago

Where would I use the UpdateContext function? This sounds like a much better solution for me as I have very limited space.

Lorna Pirozzolo
Lorna Pirozzolo
2 years ago

Fantastic, thank you Matthew. New to PowerApps but do learn fast 🙂

Sean Corry
Sean Corry
3 years ago

Hi Matthew, Awesome stuff thank you for the clear and concise documentation. I find that client requests are typically not for granular minutes, but for 15 minute segments. Have you done this with segments for minute options, :00, :15, :30, etc. ?

Chris
Chris
3 years ago

Hi Softic, I was running into a similar issue. I created the text input inside the component instead of on the screen and made these changes.

I updated the “Default” property of the text input to: If(IsBlank(varTimeValue),TimeValue(Text(cmp_TimePicker.Default,”[$-en-US]hh:mm:ss”)),varTimeValue)
This sets the default value to a variable. If the variable is blank it uses the default of the component.

I then updated the “OnChanges” property of the text input to update this variable.
Set(varTimeValue,TimeValue(time_input.Text))

I allowed the text input to be reset. So I set “Reset” to true.

In each of the galleries, I updated the “OnSelect” property to reset the text input to the default value:
Set(varTimeValue,
Time(Value(gal_TimePicker_Hours.Selected.Value) – If(Value(gal_TimePicker_Hours.Selected.Value)=12, 12) + If(Text(gal_TimePicker_AMPM.Selected.Value) = “PM”,12,0),
Value(gal_TimePicker_Minutes.Selected.Value),0)); Reset(txt_input_time);

Here the name of my text input is “txt_input_time”. This updates the variable varTimeValue on select and then resets the text input (so that it defaults to varTimeValue). I think I was having issues with the OnSelect of the gallery, so I also added it to the OnSelect of the labels inside of the gallery (Ex: lbl_TimePicker_AMPM).

If you put the text_input inside the component like I do, you may need to update the X, Y, Width, and Height variables of the galleries and text input to be equal to the dimensions (X, Y, Width, Height) of the component itself. This helps align everything and makes size/shape more editable in the screen.

Let me know if that works,
Chris

Josh
3 years ago

Is it possible to use this to capture multiple time fields without having to create copies of all the elements on the same screen?
Example: Mon Start Time / Tues Start Time / Wed Start Time / etc

Dave
Dave
3 years ago

Heya Matthew,

thank you for this amazing timepicker 🙂 Im pretty new to PowerApps and struggling with getting the Time from the TimePicker and a Date from a DatePicker into a Sharepoint List. I got this thing in a Card within a Form and to push the values to the list I use SubmitForm. For the DateTime Value I have this Code in the Update Property of the DateTime Card:

DateValue1.SelectedDate & TimeValue(cmp_TimePicker_1.Value, DateTimeZone.UTC)

I also tried “+ Text” (this enters a wrong date into the List) and “& Time” (This shows a code error). I don’t understand why it is telling me that Time or TimeValue is wrong in this case.

Can you help me solve this?

Cheers,
Dave

Samantha
Samantha
2 years ago

Hi Matthew,

Thanks for the article. It is very helpful. How do you achieve the last part where you click outside the component and the time component pop up disappears?

Jessie
Jessie
2 years ago

Hi Matthew.

This looks amazing, great job! I was playing around this as well, but did not manage to work in my end. I have set the text property of my label inside the component to the variable varTimeValue. But when I select anything it does not show any value in the label.
Do you have any idea on this? I followed everything in the steps above.

Stevie
Stevie
6 months ago
Reply to  Jessie

I had the same issue, when I moved the code that sets varTimeValue to my text label OnSelects rather than the gallery Onselects it resolved the issue.

Jessie
Jessie
2 years ago

Hi Matthew,

This looks amazing and I did follow your steps above to create the time picker. However, for me it did not work. Even if I followed everything. When I try to select a number in the hours, minutes and AM/PM sections it does not show the value in the label I created. It looks like it does not work for the onselection property. I even assigned the variable “varTimeValue” for a label or textbox I added in the company but does not render any value after I select anything.

Sam Draper
2 years ago

Thanks!
This will streamloine an application we are currently working on

Brian GC
Brian GC
2 years ago

Hi Matthew,

This is nice! Thank you.

I tried to use the component in a new screen and in a DataCard (default template that comes with showing share point data). The selected time value is seen in the Screen clearly but not in the Data Card.

Wondering why. Any thoughts?

Brian GC
Brian GC
2 years ago

I have a share point list. I am using this data to display the list of items using data cards. For example if the list has equipment number, description and time of installation. I can see the list of my equipments using the data card template (template that come in the list of templates when we start to create an app) . The template is nothing but a collection of items that contain labels or input text boxes or any other controls. Let us say I want to modify the attribute of one of those items that contains the time of installation. I imported the time picker component and am able to select the hour, minute and am/pm flawlessly. However, I am not able to capture the selected time into a label sitting next to the time picker for that item.

while testing the component independently in a separate screen the results come as expected.

Last edited 2 years ago by Brian GC
Bernd Stoeckel
Bernd Stoeckel
2 years ago

That looks really great, thanks! Just one thing.. I seem to be unable to find the varTimeValue when I try to use it in ‘Value’ of the component. While testing, I can find it (using it in a label). I also see the variable as a global variable in the app. Only in Value of the component it returns an error and I don’t see why..

Bernd Stoeckel
Bernd Stoeckel
2 years ago
Reply to  Bernd Stoeckel

This is resolved btw:
I needed to give access to the scope of the app

Fernando Rivera
Fernando Rivera
6 months ago
Reply to  Bernd Stoeckel

You helped me so much, thx!

David
David
2 years ago

Hi Matthew,

I’ve been trying to use this time picker with a separate date picker. In the dataverse table the time shows a default date of 1970 is there a way to change this date so that it uses the date pickers date for the date?

Sayed Tufel
Sayed Tufel
2 years ago

Hi Mathew I have used your component in my app, but after submitting the form time value is not resetting to parent.default.

Please reply it is crucial for me

Pantelis
Pantelis
1 year ago

Great article and its working like a charm. I have only one question: When I am using the component inside an app, then I need to double click on each gallery (hours, minutes and AM/PM) to select an item. Any ideas?

Opal56
Opal56
1 year ago
Reply to  Pantelis

I have the same issue… a single click resets the previous click… only works on a double click.

EdB
EdB
1 year ago
Reply to  Opal56

Please check your labels in each Gallery and make sure the OnSelect property is empty or set to false. This should resolve the issue.

Maik
Maik
6 months ago
Reply to  EdB

If I do this, the value of cmp_Timer keeps empty.

EdB
EdB
1 year ago
Reply to  Opal56

Sorry, Matt, I thought that my other comment fixed the same issue I was experiencing but it doesn’t, it removes the double-clicking but breaks other things. Please delete both of these comments, sorry for the extra moderation work!

Ferds
Ferds
1 year ago

What if I want to use the time picker component inside a form with data cards?

Leandro Trindade Freire
Leandro Trindade Freire
1 year ago

Hi Mathew,
Thanks for sharing this fantastic content.
Extremely helpful and very superbly detailed.
It made my day! 

Thanks.

Leandro Trindade Freire
Leandro Trindade Freire
1 year ago

It’s adding a nice looking to my app, thanks again.

There is any command to make the texlabel show blank after clicking In a button?
like reset the time picking

Instead, show the text from time picking, and show a fresh empty text box.

Text(cmp_Sample_MobileTimePicker.Value, ShortTime)

Regards.