Exercise: Create the phone number translator app

Completed

In this exercise, you construct the UI for the phone dialer app and implement the logic behind this UI.

You build a UI that takes advantage of the UI capabilities of .NET MAUI (Multi-platform Application User Interface) and the .NET MAUI Essentials package to dial the phone.

The app lets the user type text into an input field, and translates that text to numeric digits. It uses the letters that show up on a telephone keypad as the basis for translation. For example, the letters cab translate to 222 because the digit 2 has all three letters a, b, and c.

You continue with the Phoneword solution you created in the previous exercise.

Add a new C# source file to the app

  1. Open the Phoneword solution in Visual Studio if you don't already have it open.

  2. In the Solution Explorer window, right-click the Phoneword project, select Add, and select Class.

  3. In the Add New Item dialog box, name the class file PhonewordTranslator.cs, then select Add.

    A screenshot of the Add new item dialog box. The user named the class file PhonewordTranslator.cs

Add the translation logic

Replace the content of the class file with the following code and save the file. The static method ToNumber in the PhonewordTranslator class translates the number from alphanumeric text into a regular numeric phone number.

using System.Text;

namespace Phoneword;
public static class PhonewordTranslator
{
	public static string? ToNumber(string raw)
	{
		if (string.IsNullOrWhiteSpace(raw))
			return null;

		raw = raw.ToUpperInvariant();

		var newNumber = new StringBuilder();
		foreach (var c in raw)
		{
			if (" -0123456789".Contains(c))
				newNumber.Append(c);
			else
			{
				var result = TranslateToNumber(c);
				if (result != null)
					newNumber.Append(result);
				// Bad character?
				else
					return null;
			}
		}
		return newNumber.ToString();
	}

	static bool Contains(this string keyString, char c)
	{
		return keyString.IndexOf(c) >= 0;
	}

	static readonly string[] digits = {
		"ABC", "DEF", "GHI", "JKL", "MNO", "PQRS", "TUV", "WXYZ"
	};

	static int? TranslateToNumber(char c)
	{
		for (int i = 0; i < digits.Length; i++)
		{
			if (digits[i].Contains(c))
				return 2 + i;
		}
		return null;
	}
}

Create the UI

  1. Open the MainPage.xaml file in the Phoneword project.

  2. Remove the ScrollView control and its contents, leaving just the ContentPage control:

    <ContentPage xmlns="http://schemas.microsoft.com/dotnet/2021/maui"
             xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
             x:Class="Phoneword.MainPage">
    
    </ContentPage>
    
  3. Add a VerticalStackLayout control with vertical orientation and a spacing of 15 units and padding of 20 units to the ContentPage:

    <ContentPage ... >
        <VerticalStackLayout Spacing="15" Padding="20">
    
        </VerticalStackLayout>
    </ContentPage>
    
  4. Add a Label control to the StackLayout:

    <ContentPage ... >
        <VerticalStackLayout ...>
            <Label Text = "Enter a Phoneword"
                   FontSize ="20"/>
        </VerticalStackLayout>
    </ContentPage>
    
  5. Add an Entry control to the StackLayout, below the label. An Entry control provides a text box into which the user can enter data. In this code, the property x:Name gives the control a name. You reference this control in the code for the app later:

    <ContentPage ... >
        <VerticalStackLayout ...>
            <Label .../>
            <Entry x:Name = "PhoneNumberText"
                   Text = "1-555-NETMAUI" />
        </VerticalStackLayout>
    </ContentPage>
    
  6. Add two Button controls to the VerticalStackLayout, after the Entry control. Both buttons currently do nothing, and the second is disabled initially. You add the code to handle the Clicked event for these two buttons in the next task:

    <ContentPage ... >
        <VerticalStackLayout ...>
            <Label .../>
            <Entry ... />
            <Button x:Name = "TranslateButton"
                    Text = "Translate"
                    Clicked = "OnTranslate"/>
            <Button x:Name = "CallButton"
                    Text = "Call"
                    IsEnabled = "False"
                    Clicked = "OnCall"/>
        </VerticalStackLayout>
    </ContentPage>
    

Respond to the TranslateButton button tap

  1. In the Solution Explorer window, expand the MainPage.xaml entry and open the MainPage.xaml.cs code-behind file.

  2. In the MainPage class, remove the count variable and the OnCounterClicked method. The class should look like this:

    namespace Phoneword;
    
    public partial class MainPage : ContentPage
    {
        public MainPage()
        {
            InitializeComponent();
        }
    }
    
  3. Add the translatedNumber string variable and the following OnTranslate method to the MainPage class, after the constructor. The OnTranslate method retrieves the phone number from the Text property of the Entry control and passes it to the static ToNumber method of the PhonewordTranslator class that you created earlier.

    public partial class MainPage : ContentPage
    {
        ...
        string? translatedNumber;
    
        private void OnTranslate(object sender, EventArgs e)
        {
            string enteredNumber = PhoneNumberText.Text;
            translatedNumber = PhonewordTranslator.ToNumber(enteredNumber);
    
            if (!string.IsNullOrEmpty(translatedNumber))
            {
                // TODO:
            }
            else
            {
                // TODO:
            }
        }
    }
    

    Note

    You fill in the missing TODO bits of this code in the next step.

  4. In the OnTranslate method, add code to change the Text property of the Call button to append the successfully translated phone number. You can use the value you stored in the translatedNumber field. Also, enable and disable the button based on the successful translation. For example, if TranslateNumber returned null, disable the button, but if it was successful, enable it.

    private void OnTranslate(object sender, EventArgs e)
    {
        string enteredNumber = PhoneNumberText.Text;
        translatedNumber = Core.PhonewordTranslator.ToNumber(enteredNumber);
    
        if (!string.IsNullOrEmpty(translatedNumber))
        {
            CallButton.IsEnabled = true;
            CallButton.Text = "Call " + translatedNumber;
        }
        else
        {
            CallButton.IsEnabled = false;
            CallButton.Text = "Call";
        }
    }
    

Create the event method for the CallButton button

  1. Add the OnCall event handling method to the end of the MainPage class. This method makes use of asynchronous operations, so mark it as async:

    public partial class MainPage : ContentPage
    {
    
        ...
        async void OnCall(object sender, System.EventArgs e)
        {
    
        }
    }
    
  2. In the OnCall method, prompt the user, by using the Page.DisplayAlert method, to ask if they'd like to dial the number.

    The parameters to DisplayAlert are a title, a message, and two strings used for the Accept and Cancel button text. It returns a Boolean indicating whether the Accept button was pressed to dismiss the dialog box.

    async void OnCall(object sender, System.EventArgs e)
    {
        if (await this.DisplayAlert(
            "Dial a Number",
            "Would you like to call " + translatedNumber + "?",
            "Yes",
            "No"))
        {
            // TODO: dial the phone
        }
    }
    

Test the application

  1. In the Visual Studio toolbar, select the Windows Machine profile and start debugging.

  2. Tap the Translate button to convert the default text to a valid phone number. The caption on the Call button should change to Call 1-555-6386284:

    A screenshot of the Phoneword UI. The user translated the text to a valid phone number.

  3. Tap the Call button. Verify that a prompt appears asking you to confirm the operation. Select No.

    A screenshot of the PhoneWord user interface's Dial a Number prompt.

  4. Return to Visual Studio and stop debugging.

Dial the phone number

  1. In the MainPage.xaml.cs code-behind file, edit the OnCall method and replace the TODO comment with the following try/catch blocks:

    async void OnCall(object sender, System.EventArgs e)
    {
        if (await this.DisplayAlert(
            "Dial a Number",
            "Would you like to call " + translatedNumber + "?",
            "Yes",
            "No"))
        {
            try
            {
                if (PhoneDialer.Default.IsSupported && !string.IsNullOrWhiteSpace(translatedNumber))
                    PhoneDialer.Default.Open(translatedNumber);
            }
            catch (ArgumentNullException)
            {
                await DisplayAlert("Unable to dial", "Phone number was not valid.", "OK");
            }
            catch (Exception)
            {
                // Other error has occurred.
                await DisplayAlert("Unable to dial", "Phone dialing failed.", "OK");
            }
        }
    }
    

    The PhoneDialer class in the Microsoft.Maui.ApplicationModel.Communication namespace provides an abstraction of the phone dialing functionality (and others) for the Windows, Android, iOS (and iPadOS), and macOS platforms. The static Open method attempts to use the phone dialer to call the number provided as the parameter.

    The following steps show how to update the Android application manifest to enable Android to use the phone dialer. Windows, iOS, and MacCatalyst, applications follow the same general principle, except that you specify a different capability in the manifest depending on the operating system.

  2. In the Solution Explorer window, expand the Platforms folder, expand the Android folder, right-click the AndroidManifest.xml file, and select Open with > Automatic Editor Selector (XML). Select OK.

  3. Add the following XML snippet inside the manifest node, after the existing content for this node.

    <?xml version="1.0" encoding="utf-8"?>
    <manifest xmlns:android="http://schemas.android.com/apk/res/android">
        ...
        <queries>
            <intent>
                <action android:name="android.intent.action.DIAL" />
                <data android:scheme="tel"/>
            </intent>
        </queries>
    </manifest>
    
  4. Save the file.

  5. In the Visual Studio toolbar, select the Android Emulators/Pixel 7 - API 35 (or similar) profile and start debugging.

  6. When the app appears in the emulator (it can take a few minutes), enter a phone number (or accept the default) select Translate, then select Call.

  7. In the Dial a Number alert, select Yes. Verify that the Android phone dialer appears with the number you provided in the app.

    The Android phone dialer containing the number provided by the app.

  8. Return to Visual Studio and stop debugging.

Summary

In this exercise, you added a custom UI to your application by using pages and views. You also added support for placing a call by using platform-specific APIs available in Android.