DEV Community

Cover image for Elm Calculator Part 5 - Adding Decimal Support
Ryan Frazier
Ryan Frazier

Posted on • Originally published at pianomanfrazier.com on

Elm Calculator Part 5 - Adding Decimal Support

This post is part of a series. To see all published posts in the series see the tag "elm calculator book".If you wish to support my work you can purchase the book on gumroad.

This project uses Elm version 0.19

  • Part 1 - Introduction
  • Part 2 - Project Setup
  • Part 3 - Add CSS
  • Part 4 - Basic Operations
  • Part 5 - Adding Decimal Support (this post)
  • Part 6 - Supporting Negative Numbers
  • Part 7 - Add Dirty State
  • Part 8 - Support Keypad Input
  • Part 9 - Combination Key Input
  • Part 10 - Testing
  • Part 11 - Netlify Deployment

In the last chapter, we input numbers as floats and did some math to shift the numbers around. This will be a big problem when we start to introduce decimals.

Failed attempt still using floats

You'll notice that I skipped v0.4 of the app. This was my failed attempt to keep using math to manipulate the user input number.

If we try to do something similar with floats we get the following.

> 0.1 + 0.02 0.12000000000000001 : Float 
Enter fullscreen mode Exit fullscreen mode

Here is my code for reference.

Change inputNumber to String

Again like before, I like to start a new feature or refactor by changing the model.

type alias Model = { stack : List Float , currentNum : String } initialModel : Model initialModel = { stack = [] , currentNum = "0" } 
Enter fullscreen mode Exit fullscreen mode

As you can see in the model only the currentNum is a String. At some point we need to parse our input into an actual number so we can do operations on it. We'll do that when we push the number to the stack.

The way we'll parse is using String.toFloat. Let's play around with this function in the elm repl. If you have elm installed, go to a terminal and type elm repl.

As of writing this book the official Elm Guide https://guide.elm-lang.org/core_language.html has a live repl you can play with on the webpage. No installation necessary. This is helpful if you have been following along on ellie-app.

> String.toFloat <function> : String -> Maybe Float 
Enter fullscreen mode Exit fullscreen mode

Why does it return a Maybe Float? What happens if we try to give this function a string that isn't a number?

> String.toFloat "abcd" Nothing : Maybe Float > String.toFloat "123.456" Just 123.456 : Maybe Float 
Enter fullscreen mode Exit fullscreen mode

This is Elm's way of dealing with uncertainty. If it can't parse it, it will return a Nothing otherwise it will return a Just <some float>. Let's look at some other examples and then we'll explore this Maybe thing a bit more.

Again, Elm types are powerful but take some getting used to.

Parse float in JavaScript and Python

Here is the same thing in JavaScript.

» parseFloat("abcd")  NaN » parseFloat("123.456")  123.456 
Enter fullscreen mode Exit fullscreen mode

And again in Python.

>>> float("123.456") 123.456 >>> float("abcd") Traceback (most recent call last): File "<stdin>", line 1, in <module> ValueError: could not convert string to float: 'abcd' 
Enter fullscreen mode Exit fullscreen mode

No matter what language we are using, we need to deal with these kind of errors. In Elm it uses the Maybe type. JavaScript returns NaN, not a number. And Python throws a ValueError.

The Maybe type

The Maybe type is something that took me a while to understand. I had been writing quite a bit of Elm code but I just kind of ignored it. Hopefully I can make it click for you sooner.

Let's play around with the Maybe type in the elm repl.

The Maybe type is defined as the following.

type Maybe = Just a | Nothing 
Enter fullscreen mode Exit fullscreen mode

What this means is that Just can hold any value.

> Just 1 Just 1 : Maybe number > Just 1.2 Just 1.2 : Maybe Float > Just "123.456" Just "123.456" : Maybe String > Just (Just 1) Just (Just 1) : Maybe (Maybe number) > Just Nothing Just Nothing : Maybe (Maybe a) 
Enter fullscreen mode Exit fullscreen mode

And Nothing is just nothing.

> Nothing Nothing : Maybe a 
Enter fullscreen mode Exit fullscreen mode

So how is this useful?

Let's go back to parsing strings. The compiler will make sure we deal with the case our string doesn't parse into a number.

parsedNumber = String.toFloat model.currentNum 
Enter fullscreen mode Exit fullscreen mode

What is the type of parsedNumber? It's a Maybe Float. Meaning it can be Just Float or it can be Nothing. We can now pattern match on these two cases.

case parsedNumber of Nothing -> -- deal with the case it doesn't parse Just num -> -- yay it parsed! Do something with num 
Enter fullscreen mode Exit fullscreen mode

This is how Elm in practice has no runtime errors. The compiler will help us to think about uncertainty and require us to deal with it.

Now that we know about Maybe we can continue on.

Push the parsed float to the stack

Now that we can parse strings to floats we can update the model when a user clicks "Enter".

update : Msg -> Model -> Model update msg model = case msg of ... Enter -> let maybeNumber = String.toFloat model.currentNum in case maybeNumber of Nothing -> { model | error = Just "PARSE ERR" } Just num -> { model | stack = num :: model.stack , currentNum = "0" } 
Enter fullscreen mode Exit fullscreen mode

The compiler will now tell us that there is no error field in our model. So let's add that.

type alias Model = { stack : List Float , currentNum : String , error : Maybe String } initialModel : Model initialModel = { stack = [] , currentNum = "0" , error = Nothing } 
Enter fullscreen mode Exit fullscreen mode

We might not have an error, so we'll make the error a Maybe.

We also need to display the error if it exists. Let's pattern match on the error in our view to display the error.

case model.error of Nothing -> inputBox (text model.currentNum) Just err -> inputBox (span [class "error"] [text err]) 
Enter fullscreen mode Exit fullscreen mode

Input the decimal

Inputing the decimal requires some special treatment. Users should not be able to put in multiple decimals into a number.

First add the message.

type Msg = InputOperator Operator | ... | SetDecimal 
Enter fullscreen mode Exit fullscreen mode

Then handle the message in the update.

update : Msg -> Model -> Model update msg model = case msg of SetDecimal -> if String.contains "." model.currentNum then model else { model | currentNum = model.currentNum ++ "." } 
Enter fullscreen mode Exit fullscreen mode

Now we need to be able to input negative numbers and we will have a fully functioning calculator. The next chapter will add negative number support.

Top comments (0)