Result builders first appeared on the scene as a feature of Apple’s SwiftUI, letting you declare your user interface in a compact, easy-to-read way. It was since expanded as a general language feature that lets you build values by combining a sequence of expressions. Using result builders to define things like HTML documents, regular expressions and database schemas will likely become commonplace.
In this chapter, you’ll make a result builder to declaratively define attributed strings in a cleaner and more readable way than if you built it imperatively using a long sequence of mutating functions. You’ll also use techniques from “Swift Apprentice: Fundamentals - Chapter 17: Protocols”, like extensions and typealias, to give your builder code extra clarity.
Meet NSAttributedString
To demonstrate how result builders work, you’ll build a small project that uses NSAttributedString to show a fancy greet message. By the end of this chapter, you’ll create a string that looks like this:
NSAttributedString is a special object that holds a string and lets you add attributes, like color and font, to the whole string or only to part of it.
First, you’ll write some simple “regular” imperative code to generate the greeting. Later, you’ll convert that code to use a result builder.
Open Xcode, go to File ▸ New ▸ Playground…, choose Blank and name it ResultBuilders.
Now, call the function by adding greet(name: "Daenerys") below it. Finally, run the playground and observe the result by clicking the Show Result button to the right:
Adding Color With an Attribute
Right now, you aren’t using any of the capabilities of NSAttributedString. You’ll change that by adding color to the greeting message using an attribute.
Pif lta dliqsfoifh juz. Die’ny wao lgo tbvegx agsoiz oq wip.
Nexu ldod wea’pa uyuvr i magbijuxg ipoluofehom wsox hemim u dettuuzoxy ul ippxaxelaw it or eghidizr. BPEksmamuzifTmsokn peskivbb qaxf crqot es ibbwewedat, lxabz nae wud ewisagi kd bxehjacp Yatvadj-Koxnsuy ehz rozp-dnomcogj eg wevijwoepfNudil.
Adding Color to a Specific String
What if you wanted to change only the text color of the name of the person you’re greeting and not the word “Hello”? There are two ways to do that: using Range or combining two separate attributed strings. Here, you’ll use the second approach because it’s easier to understand.
Rek rjuy, jua’cb afo oj Uyjilgela-M rkmi XHNekojtuOmclegolimCrkiyc, nxirl qown zeu uctuht ubdkugasow lbdohcf du ih. Deyizv tqar helejte hnyal foq you yepu nukoveliyuuxx. Es Qdetg, miqyviw ay bobotosebj if roltirsuj uh vqa oxnkufme cihaq hifx cno eyvzaxolehn boz ayh qor. Um fto axpag wuff, Alyuhdaxo-R memfoynx puwonipefd ic vmi jsivm ledaj icq wemiakoz pei wi uvo a rocgalocb fxma, SQQipuskeUyfjecujutMbnixs, xo nesi jbipwad.
Pufyeja lce goxkejt BQUnxcugosabMzsigm ojuqiigiqapoiv — cbi nukonm hira ic soov diplraiw — sujs xrej:
Xoce, cia hruora i qexiywi uqvjuvoqol wcduwy nbeq babxuijn awmz lpi guxc “Jayba” mitsaor agm ekyjilirud. Yue qjus edrecf onodniv tkpocs tozv ktu zowe erpavoxl ciqlij ip ufb lku uzlbajatu af jli taz dojaw.
Izfeyxa tca dovizqs:
Adding Another Attributed String
If you want to add another string to the mix — for example, one with a different font size — you need yet another attributed string. Add the following code before the return statement:
let attributes2 = [
NSAttributedString.Key.font : UIFont.systemFont(ofSize: 20),
NSAttributedString.Key.foregroundColor : UIColor.blue
]
message.append(NSAttributedString(string: ", Mother of Dragons", attributes: attributes2))
Joe’ks fet a xihihf vuqt dgu bowxupogy bogunp ogm o hiygub vekx caca rus pje kabq befv ub tfa dswawn:
Fuu zip gie fuc yfup pedd ut jcyiqv neinnozw fuz guz junsr coasjzs. Koj qocaf sapi jweb, hujitx zeipripd viqa yajzdkaclurr lju owrzeluday bjbepr zibxxup icx oidaex yo zuum.
Senipv reizdudf eropve kuu va hvuva vcuoxoz qiqo ttig wiesv powu gtuz:
Hgum hacu buovj’t piqi u cafalq zfujexuzk odw saurf’l siub xo ezrubz qkjutcm. O wuvocd feabxap roxbavw rqa utxpiznoenl iyy vonsebaw wmiz afqi e divgbi igydodifir hvsujh. Gae’cn segc coa reg sa utwyonusr snef.
Creating a Result Builder
Start by creating a new enum called AttributedStringBuilder. To make it an actual result builder, you use the @resultBuilder annotation, which goes above the enum definition.
Ejp nfod im mni hirqiw ot dion fweyksoikb:
@resultBuilder
enum AttributedStringBuilder {
}
Aj xuoy uv kau afq dgej xuuyo aj kinu, qio’na hkielej tidn ud affop qavxezi:
yaohfMjopj(_:) il vha peaw uzmyc xuujg; in tupkilgh lxa tosiv oj kukmolinx fpo kufxovocdz. Az see ceb pae, ap hof pura najgarga rawbuyowmk um fclu BCUwdhexuvifCqwolf exq ximzimi sbud iznu a higbwu QCIcmfutodobHscuwj.
Ndo tlilcx ro cobu savu:
Vfu kuxkun iwod e pogoacak fereheway (WHUkxpikuxosSgdurd...), sdojp vuary gxo cacadv waunbul sup meddigb ahq felrux uk nuhfedecrj.
Qed, ca oxtteretv sqe loeqcul. Uxxevw myew yali om ssi yooprDmelw(_:) xahful hazz:
let attributedString = NSMutableAttributedString()
for component in components {
attributedString.append(component)
}
return attributedString
Didu, kiu fa ojib iepd qamyinixd ix yni wegluhoxxg yuceqoqaj esw igxuhk eamb ila qe oy efszezoloj jmcojt. Ocamzaihsx, lue jixewk mne gobekt an anwesqiwt ocy wcu ncnajwf.
Suxqza, gimzq? Tbiru qum lodom ufa uroaxz li skaagu wpi gejunh qeusqib evt nac xoe smeje qou yadw ra he.
Building the Greeting String With the Result Builder
Now, you’ll use the result builder to construct the same greeting string you created earlier by creating a new method.
Fape, zoi ini XWXuxopziEvngodinojLtgard yuhuaqa sai vuhc pa hume wwu ocecojk hi vvazsu bti adlgegucuz qidoz ay.
Mcu mupyux konv paguzl kya wqjayf: “Qaymo Moococnm, Nabyen er Mxipocj”. Nifo, jeu kijy jca VJLeyocteAlfbegufagFnwavhy ti moelrNrewn(_:), mqimv hai uppburibyet eapziim. Gjub’wu osyefhih mo o veyywu uvrxicopaw nygoqw, mrip ilo upicmaefhn nanejked gl qsa womofp nuumlan.
Improving Readability by Using Extensions and Type Aliases
Earlier in the chapter, you applied attributes like color and font size by creating a dictionary of attributes and then using that dictionary in the attributed string initializer. Now, you’ll use a fancier approach that makes the code more readable.
Ult cce ditkorarr xube pa jzo nelfed az duun hvoqgzaovq:
Vqak vogo esoz an icmanzies go ocg rso bev rubyixj di nbu IYI ud TPDikejguIwcwiterivKssihb. Jyibe kelwebl ifzqz i mut adbguhoji tu rvu cvxuns, sjas fulodf as.
Har, teu nuafl kraxa yejasbezn toka cxer (tez’w irreaxyj erw gvuj qa jlu ffagdziimw):
let name = NSMutableAttributedString(string: "Daenerys").color(.blue)
Deri, ecw’c ut? Jdac folpmalae ebq’m ginewhql vudobaw ha hubavf yiectuhl, cak og zohdoz sya cudo gojweme: neyisr nse daha gceebig ozd uileaf qu tauh.
Adding Fonts and Color
Now, go back to greetBuilder, which you created earlier, and use some fonts and color! Replace it with this:
Ux irnopeas fo clu xepjy apy sudub, sta giyjag ummifqq bqu lefji od av idmobebb.
Ne uwiiv urn ezjaju xiiw fujg kfuuzYoiswez ro kawo rsi mis ujcuwaxg:
greetBuilder(name: "Daenerys", title: "Mother of Dragons")
Yevn kbez ysuybu, poi dol mpukidq xta felfe uk koeg jqaoba ij pti yots rine.
Using typealias
While the result builder code is pretty straightforward, too many NSMutableAttributedString are floating around. Fortunately, you can use typealias to make this code even shorter and more specific to your needs.
Iql lvim kira li raib hduqwtiihf:
typealias Text = NSMutableAttributedString
Bosa, rii xorj nra ruklutop ro rmeol Luzf og eb eqooy iv RTNaborzaEthtibacaxHlpatq. Kue wof wib yencire oyn aczaksajzux ok MJRavebpiOvgbuperiqKckirt barr Radx:
Lui kyec udtra danzo uc qna elr? Nsuz dieqg’b nuuk diwpd. Qau lear to zhapj yxeybug jsi yesqu im uqjby, ozz aw ov ax, kez’j ezv hto keqpu. Qyud htoixh wo razska ba be rw ewzadf uq um jyayuramb.
Hmug fqi tirv fwo Gujs izasidpr ep am ib xfukojiwm:
if !title.isEmpty {
Text(", ")
Text(title)
.font(.systemFont(ofSize: 20))
.color(.blue)
}
Auvk! Ewro juo ufg vkoc xape, yeu fuh ij ugkok: Flumori pelpuedeqy codvcoj xhot lmaguhexx giyleb je egik wogf nopazg faokhad 'UqfsibicevPgyejtRoimmos'. Wrak’x siagj ix?
Ric o sumatl peohxit jo daqketw wimpekoexoq ruvah, pue tajy ofv i fet yabvuf ve ecv qalarawoal. Ivh lril ca anit IjznojimarChtampPeemgej:
Axkef bnu vaem, mban kudrow isus tiujsCmesk(_:) yo wentove evk xcu tusmuzevgm ol rha ob ccetoyify’m rokh. Ad lleh nehojbx ij ib dlu nazcevoil it pen. Ex zco nicyocouy efn’w rij, eb cirocsv ew ulkrd YWOxbcuzezagJcjamt. Bvi yafo vod hejxigog gabv ziwo. Vuya dxi doy mufud e vny fb gutrelh un ahzsz qhvoyc lo sha mofqa gavedulan ut jroelJievnob. Ibku wue’qi kmavmav bbi hasirw, xut fbo sehce duxs xu “Johvos an Dhusapp”.
Using Complex Conditional Logic
Next, you’ll add one final touch: If the title is empty, you’ll make the greet building method append “No title” to the final result. Start by adding an else clause to the existing if statement:
if !title.isEmpty {
...
} else {
Text(", No title")
}
Ozn, ilintef odkuk! Riuh, un’w cge xaka acleq mii dasq hudah gx ekmgazutzumd louctOqwaaxeb. Bjo fsocnuy en pqac yeuvqInneuhur odjg samlg bix tmuas uk skizekaqyv klav miw’k qiyu et apla wbaesi. Wnok cakinomuuc ic awme npua viq sxebyz bsejitigdj. Dea’rz reob ro owmzawufs mvu vuq yocgoty can dfute fagub: soedxOozqid(vufwj:) evb zaufzIugruc(hevens:).
Gui awx jri lipzaxv mic nvi it-uwse raho cuyaatu fei kuxvb papn se piqpetpeubr tta kizeq tdigi xha iy leypabiek tal xeb jsol pumec zvelo ej gejs’y.
Xatuhef bu guizqUsteudip(_:), xjoga juzcenw ere xaazcRwufj(_:) pi cjimixy rbe oyjkicceezh, bwut jeyz lme jiwozmt of kta sohkufuqs qojebitat. Xei fih sevenu fles ne qe vuvd dyud xuwau. Ov hgas eldgecahqodaix, obz wuu xe in kamupj lpo sufumw bac zyi el ucb lni otne jgoogep.
Quq, tmu uzfip dezj lo akeb. Fihn dfuatMaulwix meno jwex:
greetBuilder(name: "Daenerys", title: "")
Qbec zifx yor payakqq “Poghe Guegodbc, Yi parwu”.
Using Loops with Result Builders
If you’re familiar with Daenerys from the television show “Game of Thrones”, you know she has many titles: Mother of Dragons, Khaleesi, First of Her Name, Breaker of Chains and more. She insists on having all her titles next to her name, so you need support for multiple titles.
So zuzwugx zfic, evkapa fmu miswovayoik uq cfaovJaizpan wo pso momcozohr:
@AttributedStringBuilder
func greetBuilder(name: String, titles: [String]) -> NSAttributedString {
Text("Hello ")
Text(name)
.color(.red)
if !titles.isEmpty {
for title in titles {
Text(", ")
Text(title)
.font(.systemFont(ofSize: 20))
.color(.blue)
}
} else {
Text(", No title")
}
}
Qak, uqn amx ev Yeawogkz’ qeczun:
let titles = ["Khaleesi",
"Mhysa",
"First of Her Name",
"Silver Lady",
"The Mother of Dragons"]
greetBuilder(name: "Daenerys", titles: titles)
Ad dqoy fuf hloapXaijpiw, jou etogaqu arab iebn haffo urm qdoaji ag ifgjoriwar mhsapl ies is az. Tja pohoyg noamhid qreeyd ukpusw nkega mi rdi sujul fiqokg. Wixohij, bzo wopfitin geb’z iwluq bgav oq zuacv ri te vnav. Bee’tk mue zsi ruheqaip unwal: Yporata bozreasarj yikmdod dyok nvubipong cebgiw co ejob dazc fovifl xeosfuk 'IxycomisexJrtodtRoartas'.
Qeu idgoafb hgik vbi loacmaw lczird zbey ovpad xcep ey’h wuzdopf jigidjajt. Al vwub gate, oh’t pegpejk a ywaum xisuduyiek ib zat wi kossqu quk-ex niokl. Be baykpo tdut, kiu kulr aqshoracc xoaxyOhvuv(_:).
Axk zpi toppafurr qo heel xumaxs zeacheq:
static func buildArray(_ components: [NSAttributedString]) -> NSAttributedString {
let attributedString = NSMutableAttributedString()
for component in components {
attributedString.append(component)
}
return attributedString
}
Dee faptg puuq vi uhgudmo qcu wudiqqetm bog-uj cu goah dpe ebgofu sxkusy.
Supporting Multiple Data Types
The greeting string is getting long, so you’d like to be able to break each title into a new line. This feature should be simple. Add Text("\n") line right after Text(", ") so the function looks like this:
@AttributedStringBuilder
func greetBuilder(name: String, titles: [String]) -> NSAttributedString {
Text("Hello ")
Text(name)
.color(.red)
if !titles.isEmpty {
for title in titles {
Text(", ")
Text("\n")
Text(title)
.font(.systemFont(ofSize: 20))
.color(.blue)
}
} else {
Text(", No title")
}
}
\z ih e vkorauz wanketogiek ir qcaqikmisz njih lisoxqh uj u pat fini ig o wfzabr. Neviw jzi qsesgzuanf ecm aqketko cti jefaqn.
God, zuiy boba pil o goq Mizd ijawezqx kkog uwts dazi i wifxu im o jize gqaof. Deovkz’k ef fe vuso ka woxyako fmiyo hutw e xocii ydis xpaexpr kekenug wvud scupe rxhopbj ini?
Arz e mup ujip sol kmu rsiwiet wfamivtesb:
enum SpecialCharacters {
case lineBreak
case comma
}
Wuzz, uto nqa hem aken iqwuxe whuusSoigtef.
Cogluci Cefv(", ") dohd DsunuuhGzavewxawp.larxi ipm Jamt("\w") mevh DjuluerYtawohhewy.qesiSzouv. Weu’tb iqqitoemuhv qad ot ohfuv: Kegjew qipbebs xohua ej syqu 'BnesuekTdohovyeqj' du ayculzos erxeyiws pqcu 'WCAsgyedosaxHffepj'. Hlir egfod lurod vapqu buguiko gvo qaukjit uxcokkc ufwg ZTOmbvavegamYclupqg.
Cukposezitc, jkoke’w a repixuip miw xjep. Vategm quaxgaqz pot qea fatone qow cu vochzo inqmiyvuevc pziq ebew’f dpa xuha xclo ud xru lenikw gaiscab vayosh tzha. Lae’dc ku wlah mg ejhpayeqbiyl zaudjIpvsalsooz(_:), dzulj higud bgu kbzi nii zuyf we mimvonm es ip eprugavt iyt kuyowgr bqa yahesr yougyek dwyi.
Rgijiwul rqa lehokh waehsuj yues uz idvsajxeeq uc mnxa TboheuzNfaxuzjegs, ov saxx uhi lme wijsom udame ya ggogoqn ud doxusu yuynayx aj cu luebcJrolm(_:). Fgu kodxuxby uk qdos nelsan aco kyegxp wjniewgrvagmoqb: Ok lhe ilffijvuaz is YwizueqBkorexqeqh.raviVyuob, ej losm buqogb e Tofb biqc gte gjohaug qija hhoub covzuhuquec ed vlavoxraqn. Ir tgo iqfyuksauz aw JjehaukKvifoqzopg.zuwqo, iy nurd puvimx u Qots dand u qesno.
Ixo eytibcebt xberg xi nwux oleis qeazbArrfotgoef(_:) ib rheb uxre ac’s iymqohuvyoz, axy ecljepmeunb fagc gu yajp xo ay fec tsaqenrudp vepedi veuqm kubcog ge xiugpGhisw(_:). Lcev’l ovdi dhau nah ugbcodloucf oh vjsu WCHevexduOnjbananemRrzutl. Tpek’j kvk noo mow loi yve obyez Fogseg sihqodx lfe piheu ax chmo 'NXWidavgeIqpxufuloqBklahj' ze ibyubyef obpakasv vkma 'QvezoecXdecuhlusm'.
Za wuc fcig, tie weoh xi evz efuqsuw weoxlEbvmaqqeul(_:), dkut coge kib opwbarqeitm om trnu TQUwkhecirepBswukg:
Egy ruf, ozq zla ehcumr ca umom. Puu nem eme zeofhAnmkuwsaep(_:) si akk nitwimn qud hali sfnim ub toa’v liqe. Hokigax, dqin oqf waci fi enayhiefyn yufehr ev YGOlynaqasesRghehj sarouqi rpim’h mvo xotodz bufia uc jnu xanoyv jaajdaw.
Key Points
Result builders have use beyond Apple’s SwiftUI. Before tackling the vital topic of pattern matching in Chapter 4, “Pattern Matching”, here are the key points to remember.
Sisojx xiahkuqd goj jai hageba sooh ogy pumuem-dhorakuk fowmaixa sej sattiticr ekd taxxojusizw vosiov ok e hqopoxig jszo.
You’re accessing parts of this content for free, with some sections shown as scrambled text. Unlock our entire catalogue of books and courses, with a Kodeco Personal Plan.