This tutorial demonstrates how to set up the login functionality using forms for handling user input and link it to the Appery.io database.
Introduction
In this sample app tutorial, we will demonstrate how you can create an Ionic 4 login app using forms and where the related services are linked to the Appery.io database.
Important pre-condition
First of all, create an Appery.io database and name it myDB.
Downloads and Resources
You can also try it yourself by creating from the backup:
- Download the app backup file.
- Click Create new app.
- Click From backup and select the project backup file on your drive.
- Type in the app name: Ionic 4 Form DB Login App.
- Click Create:
data:image/s3,"s3://crabby-images/230de/230de3416522da6402bbbf5b637a5dfebbd49eeb" alt="Create from Backup.jpg 1037"
Creating from backup
Defining DB credentials
Note, that if you decide to create from backup but would like to use your database, you will need to modify the default myDB service in your app.
To do it, in the Project view, unfold the Services > myDB > myDB_settings folder and replace the default database_id value with your database ID (can be found in the browser search field or in YOUR DB > Settings > API keys.
Also, if you are interested in the detailed tutorial, please follow the steps below.
Preparing DB
Create a new Appery.io database called myDB and define its Users collection like the following:
- Add three columns of String type: firstName, lastName, and gender.
- Add a column optIn of Boolean type (this option indicates whether the user agrees to receive emails).
- Then, add a row filling in the columns with some data, for example:
data:image/s3,"s3://crabby-images/12354/123543fb091d6f89e9f60b93c3bd1a7903e4af00" alt="DB.jpg 1859"
myDB Users collection
Creating App
- Go to the Apps tab and create a new Ionic 4 blank application Ionic 4 Form DB Login App.
- In the Project view, unfold the Pages folder and rename the default Screen1 to LogInPage by using the context menu.
- Click Create New > Page to add another page: ProfilePage. This is the page to be redirected to after logging in.
- Then, from the Project tree, open Project > Routing, make sure that the pages have routes and edit them like the following:
- update the Screen1 route name to LogInPage.
- update the paths, renaming screen1 to login and removing the word
page
from both paths, so that you end up with the login and profile paths:
data:image/s3,"s3://crabby-images/c4485/c4485bbc565b7df5773f423a47090485d35d85fb" alt="Routing.jpg 1908"
Routing tab
- For this app, we will also need some database user services, so in the Project view, click Create New > Database Service, select myDb database, check Login, Logout, Signup, and Me services and import them into the project:
data:image/s3,"s3://crabby-images/0671d/0671d38f0c8f4872d205a2e88f43d918c252c81c" alt="Import DB Services.jpg 1396"
Importing DB Services
The services will be instantly added to the Services folder inside the App Builder.
data:image/s3,"s3://crabby-images/cc4b3/cc4b34fc528f71327e22b12567eff9de11f45a20" alt="Sevices added.jpg 375"
myDB services added
Note
Defining myDb_signup_service created in this step is described in the related Ionic 4 Form DB Signup Sample App tutorial which explains how to extend this tutorial and describes how to add the signup page to the app.
- Also, we will need storage to save the authorization token after logging in or signing up. Open the Model and Storage > Storage asset and add a new storage variable of String type and named token:
data:image/s3,"s3://crabby-images/0456a/0456aafcc4b3b171660dee23299b9fda442b06c9" alt="Storage.jpg 1916"
Adding storage
Defining LogInPage
LogInPage UI
- Open the DESIGN panel of LogInPage and, under the PROPERTIES panel, set the Footer and Header properties to False to disable the default header and footer:
data:image/s3,"s3://crabby-images/764ef/764efcf40eb36e9740230fb6b1643eb55c5bcc6f" alt="Footer Header.jpg 1916"
- The first element to be defined for this page is the company logo, so, drag & drop an Image component to the page, click the Change button for the Image attribute to open the Media Manager window where you can upload the logo image from your drive (should be prepared beforehand) and apply it for the app:
data:image/s3,"s3://crabby-images/b8360/b8360976107f6ee90830a1ae9db2c1849092cc88" alt="MM.jpg 1886"
Working with Media Manager
- Set the image Responsive property to True so that it can be automatically adapted to different screen sizes.
- Now, set the image Class to login-logo:
data:image/s3,"s3://crabby-images/f4ba6/f4ba6cc203b64a73fcfa4abe841a3cc53298076f" alt="Image.jpg 1914"
- To see the effect, let's add the below code to the SCSS tab of LogInPage to define this class attribute:
.login-logo {
display: block;
margin: 0 auto;
width: 150px;
height: 150px;
margin-top: 45px;
}
Here is how the SCSS panel should look like:
data:image/s3,"s3://crabby-images/a1979/a19796e6f417d5810af7d57821acccb6676e337b" alt="SCSS Tab.jpg 1910"
- Now, drag & drop a Card component from PALETTE to the screen content; then, add a Text component to the Card Item Title area and define its properties:
- Container = h1
- Text = Log in to Your Account:
data:image/s3,"s3://crabby-images/4dd1e/4dd1e641fab4781b4321ac674279999564574f79" alt="Card Item Text.jpg 1918"
Note
You can also skip the steps 1-6 and start with adding the Form component as it is the element that provides all the main app logic.
- It's now time to drag & drop a Form component from PALETTE to the Card Item and define the form properties:
- Component Name = LogInForm
- FormID = loginform
- Update On = Submit:
data:image/s3,"s3://crabby-images/5d33b/5d33b2de48a123b1ae0ca5c6b38e8be7efb5384f" alt="LoginForm.jpg 1913"
- Drag & drop an Input from PALETTE inside the LogInForm component. It will be the input for username so let's update it with the following properties:
- Component Name = UsernameInput
- Placeholder = Email or Login
- Name = username (note that this property is the name of the parameter submitted with the form data)
- Required = true
- Type = Email
- Features > Pattern = ^[a-zA-Z0-9.!#$%&’+/=?^_`{|}~-]+@[a-zA-Z0-9-]+(?:.[a-zA-Z0-9-]+)$ (the HTML pattern for input validation, in our app we will allow only email as username, but you may add any validation if you need it).
- Icon > Color = success
- Icon > Slot = Start
- Icon > Style = mail
- Label > Text = clear the text field
- Control ID = usernameInp
- Control Options > Update On = Blur:
data:image/s3,"s3://crabby-images/061fb/061fb288443eb3e3c692b4037a0947864674ea8b" alt="Input1.jpg 1595"
- Drag & drop another Input into LogInForm. It should be defined as the input for password, so let's set up the following properties:
- Component Name = PasswordInput
- Placeholder = Password
- Name = password
- Required = true
- Type = Password
- Icon > Color = success
- Icon > Slot = Start
- Icon > Style = lock
- Label >Text = clear the text field
- Control ID = passInp
- Control Options > Update On = Blur:
data:image/s3,"s3://crabby-images/c5f03/c5f03319943c81a2aa5a764f37d457a4cce2b23b" alt="Input2.jpg 1595"
- Drag & drop a Button from PALETTE into LogInForm. Let's update it with the following properties to define a submit buton:
- Color = success
- Component Name = LoginSubmitButton
- Link Features > Type = Submit
- Shape = Round
- Text = Log In:
data:image/s3,"s3://crabby-images/7529b/7529b87e55bfc4253399f4ea84a74c9e39a3edd1" alt="Button.jpg 1595"
- Drag & drop a Text component from PALETTE to LogInForm below the PasswordInput, we will use it as the validation error message, so let's set up its properties:
- Color = danger
- Component Name = InvalidFormErrorText
- Text = Invalid email address or empty password
- add a new attribute: *ngIf = usernameInp.touched&&passInp.touched&&loginform.invalid (if both inputs are provided but one of them is invalid, the error message will appear and the form will not be submitted):
data:image/s3,"s3://crabby-images/1a9ee/1a9ee99a682046020b84a8c32ca18915ec301d5b" alt="Text.jpg 1909"
LogInPage Logic
Now, when the UI is done, let's add the service invocation on form submitting.
- Open the DATA panel and, by clicking the Add button, add myDb_login_service for datasource, then rename it to logIn.
- Edit the Before send mapping of logIn service mapping the Form Data (as object) property to the $ body service request, and save:
data:image/s3,"s3://crabby-images/671c9/671c937a0fc4b187a1df1df9b708fde45b5fdab6" alt="LoginPageBS.jpg 1908"
- Now, edit Success mapping, map the sesionToken response to the token storage, then click Save & Replace to save the work:
data:image/s3,"s3://crabby-images/61f51/61f51727927e9d9fe5f443788e2460fdfff4f373" alt="LoginPageSuccessMapping.jpg 1915"
- Add the Present toast action to Error event to show an error message about unsuccessful logging in and set its:
- Message = Incorrect email or password
- Color = danger
- Position = top
- Duration = 500:
data:image/s3,"s3://crabby-images/70242/70242ba90baab06a031a50f50b0d44ef551e1707" alt="Present Toast Login Page.jpg 1599"
- Add Navigate to page action to Success event. After successful login, we should be navigated to the Profile page, so select it in the action parameters and save:
data:image/s3,"s3://crabby-images/2ced8/2ced86a80e837ee1e10e16a73a3c436351b7864f" alt="Navigate Login Page.jpg 1909"
- Also, as a good practice, let's add the Present loading action (with crescent for its Spinner property) to Before send and the Dismiss loading action to Success and Error events.
Here is how the DATA panel of LoginPage with all the events defined should look like:
data:image/s3,"s3://crabby-images/2adab/2adab4e2ac0025fe5e7504dc090c29a834dcfaa7" alt="LoginEvents Tab.jpg 1596"
Now, to the last step for the login procedure.
7. Go back to the DESIGN panel, select LogInForm, unfold the EVENTS tab, and add the action Invoke service with the login datasource to Form submit event:
data:image/s3,"s3://crabby-images/d0ea4/d0ea42f7ac31edaf284611d5aec8cb76183a899c" alt="LoginForm Event.jpg 1909"
Defining ProfilePage
ProfilePage UI
Now, let's define the ProfilePage asset.
- Select the ToolbarTitle component and change its Text property to Welcome {{firstName}} {{lastName}}.
- Drag & drop a Button to the ToolbarButtons area and set its attributes as the following:
- Component Name = LogOutButton
- Icon > Color = medium
- Icon > Slot = Icon Only
- Icon > Style = log-out
- Text = clear the text field
- Drag & drop a Card component to the screen and add 4 more items into it by click the green + button on the Card element.
- Select CardItemTitle and set its Title Text to Profile.
- Then, one by one, select CardItems' Component names attributes and give them human readable Component Name: usernameItem, firstNameItem, lastNameItem, genderItem, isOptInItem.
data:image/s3,"s3://crabby-images/477ac/477ac406469e5184f77021fcce418c6224ce46d2" alt="CardItemsProfilePage.jpg 1595"
ProfilePage Logic
- Now, when the page UI is ready, we can continue to add services but first, open the CODE tab and add 2 new variables of String type: firstName and lastName (for the welcoming message in the ProfilePage header):
data:image/s3,"s3://crabby-images/51979/5197907333ae5d60edb964738c57184c1db33703" alt="CodeProfilePage.jpg 1141"
- Open the DATA tab and add myDb_logout_service as datasource, rename it to logOut, then add myDb_me_service as datasource and rename it to me:
data:image/s3,"s3://crabby-images/3115d/3115db229281954410b32691113779c2dd7f58d2" alt="DATA Profile Tab.jpg 1908"
- Edit the Before send mapping of logOut service and map the token storage property to X-Appery-Session-Token service header request:
data:image/s3,"s3://crabby-images/fa87c/fa87c75f50a0f3ffad7b627217026d43ccda6c80" alt="LogOutBS1.jpg 1916"
- Add Navigate to page action to Success event of logout. After successful logout, we should navigate to the LogInPage, so select it in the action parameters.
- After that, switch to me service and edit its Before send mapping to map the token storage property to X-Appery-Session-Token service header request:
data:image/s3,"s3://crabby-images/ae250/ae25008b7fa00a1f4151e7757ddd5c0e22cb8ef5" alt="LogOutBS.jpg 1910"
- Also, edit the Success mapping of me service and map username, firstName, lastName, gender, and optIn response parameters to the corresponding CardItem > Text properties. Also, map the firstName and lastName response parameters to the firstName and lastName variables:
data:image/s3,"s3://crabby-images/fac29/fac29b58e826a406ecb852a67374841b761a3750" alt="ProfileMeSuccess.jpg 1593"
- As a good practice, let's also add Present loading action (with Loading... for its Message) to Before send event and Dismiss loading action to the Success and Error events of both datasources:
data:image/s3,"s3://crabby-images/413d0/413d01f76293c26a6b102a0448919d353445c4c7" alt="LogOutPresentLoading.jpg 1914"
Adding Present loading action to Before send event of logOut service
Here is how the DATA panel of ProfilePage with all the events defined should look like:
data:image/s3,"s3://crabby-images/37933/379334fa2dfdac108d7020625a0cb45b8b4a0fee" alt="DATA Profile Page Tab.jpg 1597"
- Go back to the DESIGN panel, open the EVENTS tab, select the Page component, add Invoke service action with me as datasource to Before page showing event and save:
data:image/s3,"s3://crabby-images/2ad82/2ad8253f347b2c5e03119bd07a286604121b43b3" alt="BeforePageShow.jpg 1595"
Finally, select LogOutButton and add Invoke service action for its Click event with logOut defined as datasource:
data:image/s3,"s3://crabby-images/a274d/a274dc6a7985883bf0d3bbffec57004ea0d904f0" alt="LogOurButton.jpg 1916"
App testing
Now, you can test your login form behavior on the preview. Click the TEST button and wait until the app loads, don't worry if it takes some time.
Experiment entering invalid data, making mistakes in username or password, and finally, using valid user credentials to log in:
data:image/s3,"s3://crabby-images/71bf3/71bf394d1961c7ddedebf54a4dd6b4fe340598c9" alt="AppInWork.png 1983"
App in work
Please, note the different error messages firing when testing different cases with entering wrong credentials.
You can also check this alternative Appery.io video that shows how the Form component can be defined for handling user input in an Ionic 4 app: