Note: This update is an early-access release. This chapter has not yet been updated to Vapor 4.
In the last chapter, you learned how to view categories and how to create, edit and delete acronyms. In this chapter, you’ll learn how to allow users to add categories to acronyms in a user-friendly way.
Creating acronyms with categories
The final implementation task for the web app is to allow users to manage categories on acronyms. When using the API with a REST client such as the iOS app, you send multiple requests, one per category. However, this isn’t feasible with a web browser.
The web app must accept all the information in one request and translate the request into the appropriate Fluent operations. Additionally, having to create categories before a user can select them doesn’t create a good user experience.
Open Category.swift and add the following function at the bottom of the extension below var acronyms:
static func addCategory(
_ name: String,
to acronym: Acronym,
on req: Request
) throws -> Future<Void> {
// 1
return Category.query(on: req)
.filter(\.name == name)
.first()
.flatMap(to: Void.self) { foundCategory in
if let existingCategory = foundCategory {
// 2
return acronym.categories
.attach(existingCategory, on: req)
.transform(to: ())
} else {
// 3
let category = Category(name: name)
// 4
return category.save(on: req)
.flatMap(to: Void.self) { savedCategory in
// 5
return acronym.categories
.attach(savedCategory, on: req)
.transform(to: ())
}
}
}
}
Here’s what this new function does:
Perform a query to search for a category with the provided name.
If the category exists, set up the relationship and transform the result to Void. () is shorthand for Void().
If the category doesn’t exist, create a new Category object with the provided name.
Save the new category and unwrap the returned future.
Set up the relationship and transform the result to Void.
Open WebsiteController.swift and add a new Content type at the bottom of the file to handle the new data:
struct CreateAcronymData: Content {
let userID: User.ID
let short: String
let long: String
let categories: [String]?
}
This takes the existing information required for an acronym and adds an optional array of Strings to represent the categories. This allows users to submit existing and new categories instead of only existing ones.
Next, replace createAcronymPostHandler(_:) with the following:
// 1
func createAcronymPostHandler(
_ req: Request,
data: CreateAcronymData
) throws -> Future<Response> {
// 2
let acronym = Acronym(
short: data.short,
long: data.long,
userID: data.userID)
// 3
return acronym.save(on: req)
.flatMap(to: Response.self) { acronym in
guard let id = acronym.id else {
throw Abort(.internalServerError)
}
// 4
var categorySaves: [Future<Void>] = []
// 5
for category in data.categories ?? [] {
try categorySaves.append(
Category.addCategory(category, to: acronym, on: req))
}
// 6
let redirect = req.redirect(to: "/acronyms/\(id)")
return categorySaves.flatten(on: req)
.transform(to: redirect)
}
}
Here’s what you changed:
Change the Content type of route handler to accept CreateAcronymData.
Create an Acronym object to save as it’s no longer passed into the route.
Call flatMap(to:) instead of map(to:) as you now return a Future<Response> in the closure.
Define an array of futures to store the save operations.
Loop through all the categories provided to the request and add the results of Category.addCategory(_:to:on:) to the array.
Flatten the array to complete all the Fluent operations and transform the result to a Response. Redirect the page to the new acronym’s page.
Finally, in boot(router:), replace the create acronym POST route with the following:
This changes the content type to CreateAcronymData.
You need to allow a user to specify categories when they create an acronym. Open createAcronym.leaf and, just above the <button> section, add the following:
Define a new <div> for categories that’s styled with the form-group class.
Specify a label for the input.
Define a <select> input to allow a user to specify categories. The multiple attribute lets a user specify multiple options. The name categories[] allows the form to send the categories as a URL-encoded array.
Currently the form displays no categories. Using a <select> input only allows users to select pre-defined categories. To make this a nice user-experience, you’ll use the Select2 JavaScript library.
Open base.leaf and under <link rel=stylesheet... for the Bootstrap stylesheet add the following:
#if(title == "Create An Acronym" || title == "Edit Acronym") {
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/select2/4.0.6-rc.0/css/select2.min.css" integrity="sha384-RdQbeSCGSeSdSlTMGnUr2oDJZzOuGjJAkQy1MbKMu8fZT5G0qlBajY0n0sY/hKMK" crossorigin="anonymous">
}
This adds the stylesheet for Select2 to the create and edit acronym pages. Note the complex Leaf statement. At the bottom of base.leaf, remove the first <script> tag for jQuery and replace it with the following:
Include the full jQuery library. Bootstrap only requires the slim version, but Select2 requires functionality not included in the slim version, so the full library is required.
If the page is the create or edit acronym page, include the JavaScript for Select2.
Also include the local createAcronym.js.
In Terminal, enter the following commands to create your local JavaScript file.
// 1
$.ajax({
url: "/api/categories/",
type: "GET",
contentType: "application/json; charset=utf-8"
}).then(function (response) {
var dataToReturn = [];
// 2
for (var i=0; i < response.length; i++) {
var tagToTransform = response[i];
var newTag = {
id: tagToTransform["name"],
text: tagToTransform["name"]
};
dataToReturn.push(newTag);
}
// 3
$("#categories").select2({
// 4
placeholder: "Select Categories for the Acronym",
// 5
tags: true,
// 6
tokenSeparators: [','],
// 7
data: dataToReturn
});
});
Here’s what the script does:
On page load, send a GET request to /api/categories. This gets all the categories in the TIL app.
Loop through each returned category and turn it into a JSON object and add it to dataToReturn. The JSON object looks like:
{
"id": <name of the category>,
"text": <name of the category>
}
Get the HTML element with the ID categories and call select2() on it. This enables Select2 on the <select> in the form.
Set the placeholder text on the Select2 input.
Enable tags in Select2. This allows users to dynamically create new categories that don’t exist in the input.
Set the separator for Select2. When a user types , Select2 creates a new category from the entered text. This allows users to categories with spaces.
Set the data — the options a user can choose from — to the existing categories.
Save the files, then build and run the app in Xcode. Navigate to the Create An Acronym page. The categories list allows you to input existing categories or create new ones. The list also allows you to add and remove the “tags” in a user-friendly way:
Displaying Categories
Now, open acronym.leaf. Under the “Created By” paragraph add the following:
Haey vmheiss dqu jhokezeh foxigagoar igs ozq e xagg ho aoqn uru.
Daji tto sagi efl egek JuwpitaKohrzaqheb.qnugd. Axm a woy gvupunhs ed nce pojvay of IsfopxsCognulv hoj zgi gikovofiom:
let categories: Future<[Category]>
Oj unvinnwRamcsas(_:), depleno:
let context = AcronymContext(
title: acronym.short,
acronym: acronym,
user: user)
Fidm vru verhawejb:
let categories = try acronym.categories.query(on: req).all()
let context = AcronymContext(
title: acronym.short,
acronym: acronym,
user: user,
categories: categories)
Ifein, ywoq zutmet o Qojake mi Muad, zpurh ez dokptep brax vimiuvat. Suuyw obv guc, gsab uwew mve hkeozo izgezsl mive oj kri cwumgec. Pmuope og ewkogzz jidl yuxitijueq od qvu wmifwux olp qaap ye tce ohfexrw’z bixe. Zeu’pj quo nmo ubburzm’c qayowutaaw ob yku quge:
Editing acronyms
To allow adding and editing categories when editing an acronym, open createAcronym.leaf. In the categories <div>, between the <select> and </select> tags, add the following:
Ajl e yel siwayuyd otm sferr Edkegi. Ppi luse tihitadqh nu fyu ihhujwy’c nuwi, hepr qme idwozej ilmoqtm xlumd. Hon wpc nopejefs u voqihuhw gmok et ivyedtn.
Where to go from here?
In this section, you learned how to create a full-featured web app that performs the same functions as the iOS app. You learned how to use Leaf to display different types of data and work with futures. You also learned how to accept data from web forms and provide a good user-experience for handling data.
Pda GOY ijh zitluexz suqg khu AFA ohc tbe guy isk. Qqef qexhm xopc wan ttupc exjrecuwaahl, tub qaq rewf muvcu acjfeterueyl bea yec wegbugon tbpadxupg mleg ap enwa kpoih edg eysn. Wne seg egv btuf hagrm qa tku EZO kija enq uvviv nfuehq keazz, pijc iy pki eEQ omm. Khag uzzoyp jui go vcuxe qpo fuztiferv qoqmb darayufekg. Xujki umylulocueqv kuy acah da tapohofes rp dekluhoml fuokc. Mwragbezn kpaj ot walj xqi ikhvuzagiug pfat ozm bmaxwe, nuqsiis sopoibxe ej fha iwzit heir.
In nqo gupy qabcoel ud lvu faor, duu’vm ciecc bed di edfvb iitnixgihoqoeq zo viix omkbuveyain. Vefheqytn izpini zub mfaeru ekx aksivytr ac qaqy kfi iEH efw iyw yfe beh erp. Lmaq owd’z ziyorijci, iccofiobmz gac lafku lgfhizh. Fzu giqf lgayqawc wmiy rua nis se xhanagp xitt xsu OYA omr bet oxc yuks aimwaxpamomool.
You're reading for free, with parts of this chapter shown as scrambled text. Unlock this book, and our entire catalogue of books and videos, with a kodeco.com Professional subscription.